@ -3,14 +3,48 @@ package main
import (
import (
"./cm"
"./cm"
"bufio"
"bufio"
"errors"
"fmt"
"fmt"
"io"
"io"
"net"
"net"
"strings"
)
)
const MAX_SEXP_LINES = 40
var manager = cm . New ( )
var manager = cm . New ( )
var errSexpTooLong = errors . New ( "error: sexp is too long" )
func depth ( s string ) int {
n := 0
for _ , r := range s {
switch r {
case '(' :
n += 1
case ')' :
n -= 1
}
}
return n
}
func tcpInterpreter ( conn net . Conn , userinput chan string , out chan interface { } , errors chan error ) {
func tcpInterpreter ( conn net . Conn , userinput chan string , out chan interface { } , errors chan error ) {
lines := make ( [ ] string , 0 , MAX_SEXP_LINES )
currentDepth := 0
addLine := func ( line string ) error {
if len ( lines ) >= MAX_SEXP_LINES {
return errSexpTooLong
}
lines = append ( lines , line )
return nil
}
errorMode := false
skipLine := func ( line string ) {
currentDepth += depth ( line )
if currentDepth == 0 {
errorMode = false
}
}
for {
for {
select {
select {
case v := <- out :
case v := <- out :
@ -18,10 +52,35 @@ func tcpInterpreter(conn net.Conn, userinput chan string, out chan interface{},
case err := <- errors :
case err := <- errors :
fmt . Fprintf ( conn , "error: %v" , err )
fmt . Fprintf ( conn , "error: %v" , err )
case line := <- userinput :
case line := <- userinput :
tokens := make ( chan token , 32 )
if errorMode {
// this is probably dumb
skipLine ( line )
go lexs ( line + "\n" , tokens )
break
go evalall ( tokens , out , errors , universe )
}
lineDepth := depth ( line )
currentDepth += lineDepth
if len ( lines ) == 0 && lineDepth == 0 {
tokens := make ( chan token , 32 )
go lexs ( line + "\n" , tokens )
go evalall ( tokens , out , errors , universe )
break
}
if err := addLine ( line ) ; err != nil {
fmt . Fprintf ( conn , "error in addLine: %v" , err )
errorMode = true
lines = lines [ : 0 ]
currentDepth = 0
break
}
if currentDepth == 0 {
runnable := strings . Join ( lines , " " )
lines = lines [ : 0 ]
tokens := make ( chan token , 32 )
go lexs ( runnable + "\n" , tokens )
go evalall ( tokens , out , errors , universe )
}
}
}
}
}
}
}