// code generated by "translate.go.pss" a pep script // http://bumble.sf.net/books/pars/tr/ // s.HasPrefix can be used instead of strings.HasPrefix package main import ( "fmt" "bufio" "strings" "strconv" "unicode" "io" "os" "unicode/utf8" ) // an alias for Println for brevity var pr = fmt.Println /* a machine for parsing */ type machine struct { SIZE int // how many elements in stack/tape/marks eof bool charsRead int linesRead int escape rune delimiter rune counter int work string stack []string cell int tape []string marks []string peep rune reader *bufio.Reader } // there is no special init for structures func newMachine(size int) *machine { mm := machine{SIZE: size} mm.eof = false // end of stream reached? mm.charsRead = 0 // how many chars already read mm.linesRead = 1 // how many lines already read mm.escape = '\\' mm.delimiter = '*' // push/pop delimiter (default "*") mm.counter = 0 // a counter for anything mm.work = "" // the workspace mm.stack = make([]string, 0, mm.SIZE) // stack for parse tokens mm.cell = 0 // current tape cell // slices not arrays mm.tape = make([]string, mm.SIZE, mm.SIZE) // a list of attribute for tokens mm.marks = make([]string, mm.SIZE, mm.SIZE) // marked tape cells // or dont initialse peep until "parse()" calls "setInput()" // check! this is not so simple mm.reader = bufio.NewReader(os.Stdin) var err error mm.peep, _, err = mm.reader.ReadRune() if err == io.EOF { mm.eof = true } else if err != nil { fmt.Fprintln(os.Stderr, "error:", err) os.Exit(1) } return &mm } // method syntax. // func (v * vertex) abs() float64 { ... } // multiline strings are ok ? func (mm *machine) setInput(newInput string) { print("to be implemented") } // read one utf8 character from the input stream and // update the machine. func (mm *machine) read() { var err error if mm.eof { os.Exit(0) } mm.charsRead += 1 // increment lines if mm.peep == '\n' { mm.linesRead += 1 } mm.work += string(mm.peep) // check! mm.peep, _, err = mm.reader.ReadRune() if err == io.EOF { mm.eof = true } else if err != nil { fmt.Fprintln(os.Stderr, "error:", err) os.Exit(1) } } // remove escape character: trivial method ? // check the python code for this, and the c code in machine.interp.c func (mm *machine) unescapeChar(c string) { // if mm.work = "" { return } mm.work = strings.Replace(mm.work, "\\"+c, c, -1) } // add escape character : trivial func (mm *machine) escapeChar(c string) { mm.work = strings.Replace(mm.work, c, "\\"+c, -1) } /** a helper function to count trailing escapes */ func (mm *machine) countEscapes(suffix string) int { count := 0 ss := "" if strings.HasSuffix(mm.work, suffix) { ss = strings.TrimSuffix(mm.work, suffix) } for (strings.HasSuffix(ss, string(mm.escape))) { ss = strings.TrimSuffix(ss, string(mm.escape)) count++ } return count } // reads the input stream until the workspace ends with the // given character or text, ignoring escaped characters func (mm *machine) until(suffix string) { if mm.eof { return; } // read at least one character mm.read() for true { if mm.eof { return; } // we need to count the mm.Escape chars preceding suffix // if odd, keep reading, if even, stop if strings.HasSuffix(mm.work, suffix) { if (mm.countEscapes(suffix) % 2 == 0) { return } } mm.read() } } /* increment the tape pointer (command ++) and grow the tape and marks arrays if necessary */ func (mm *machine) increment() { mm.cell++ if mm.cell >= len(mm.tape) { mm.tape = append(mm.tape, "") mm.marks = append(mm.marks, "") mm.SIZE++ } } /* pop the last token from the stack into the workspace */ func (mm *machine) pop() bool { if len(mm.stack) == 0 { return false } // no, get last element of stack // a[len(a)-1] mm.work = mm.stack[len(mm.stack)-1] + mm.work // a = a[:len(a)-1] mm.stack = mm.stack[:len(mm.stack)-1] if mm.cell > 0 { mm.cell -= 1 } return true } // push the first token from the workspace to the stack func (mm *machine) push() bool { // dont increment the tape pointer on an empty push if mm.work == "" { return false } // push first token, or else whole string if no delimiter aa := strings.SplitN(mm.work, string(mm.delimiter), 2) if len(aa) == 1 { mm.stack = append(mm.stack, mm.work) mm.work = "" } else { mm.stack = append(mm.stack, aa[0]+string(mm.delimiter)) mm.work = aa[1] } mm.increment() return true } // func (mm *machine) printState() { fmt.Printf("Stack %v Work[%s] Peep[%c] \n", mm.stack, mm.work, mm.peep) fmt.Printf("Acc:%v Esc:%c Delim:%c Chars:%v", mm.counter, mm.escape, mm.delimiter, mm.charsRead) fmt.Printf(" Lines:%v Cell:%v EOF:%v \n", mm.linesRead, mm.cell, mm.eof) for ii, vv := range mm.tape { fmt.Printf("%v [%s] \n", ii, vv) if ii > 4 { return; } } } func (mm *machine) goToMark(mark string) { markFound := false for ii := range mm.marks { if mm.marks[ii] == mark { mm.cell = ii; markFound = true; break } } if markFound == false { fmt.Printf("badmark '%s'", mark) os.Exit(1) } } // this is where the actual parsing/compiling code should go // so that it can be used by other go classes/objects. Also // should have a stream argument. func (mm *machine) parse(s string) { } /* adapt for clop and clip */ func trimLastChar(s string) string { r, size := utf8.DecodeLastRuneInString(s) if r == utf8.RuneError && (size == 0 || size == 1) { size = 0 } return s[:len(s)-size] } func (mm *machine) clip() { cc, _ := utf8.DecodeLastRuneInString(mm.work) mm.work = strings.TrimSuffix(mm.work, string(cc)) } func (mm *machine) clop() { _, size := utf8.DecodeRuneInString(mm.work) mm.work = mm.work[size:] } type fn func(rune) bool // eg unicode.IsLetter('x') /* check whether the string s only contains runes of type determined by the typeFn function */ func isInClass(typeFn fn, s string) bool { if s == "" { return false; } for _, rr := range s { //if !unicode.IsLetter(rr) { if !typeFn(rr) { return false } } return true } /* range in format 'a,z' */ func isInRange(start rune, end rune, s string) bool { if s == "" { return false; } for _, rr := range s { if (rr < start) || (rr > end) { return false } } return true } /* list of runes (unicode chars ) */ func isInList(list string, s string) bool { return strings.ContainsAny(s, list) } func main() { // This size needs to be big for some applications. Eg // calculating big palindromes. Really // it should be dynamically allocated. var size = 30000 var mm = newMachine(size); var restart = false; // the go compiler complains when modules are imported but // not used, also if vars are not used. if restart {}; unicode.IsDigit('0'); strconv.Itoa(0); for !mm.eof { /* lex block */ for true { mm.read() /* read */ // Unlike Crockfords grammar, I will just completely ignore whitespace, // but this may not be acceptable in a rigorous application. Also, I // am just using the ctype.h definition of whitespace, whatever that // may be. // make character number relative to line number if (isInList("\n", mm.work)) { mm.work = "" // clear mm.charsRead = 0 /* nochars */ if (mm.eof) { break } restart = true; break // restart } if (isInClass(unicode.IsSpace, mm.work)) { mm.work = "" // clear if (mm.eof) { break } restart = true; break // restart } if (isInRange('0','9', mm.work)) { /* while */ for isInRange('0','9', string(mm.peep)) { if mm.eof { break } mm.read() } mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "integer*" mm.push(); break } if (isInRange('a','z', mm.work) || isInRange('A','Z', mm.work)) { /* while */ for isInRange('a','z', string(mm.peep)) { if mm.eof { break } mm.read() } if (mm.work != "true" && mm.work != "false" && mm.work != "null" && mm.work != "e" && mm.work != "E") { // handle error mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "Unknown value '" mm.work += mm.tape[mm.cell] /* get */ mm.work += "' at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (character " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ").\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } mm.tape[mm.cell] = mm.work /* put */ if (mm.work == "e" || mm.work == "E") { mm.work = "" // clear mm.work += "E*" mm.push(); break } mm.work = "" // clear mm.work += "value*" mm.push(); break } if (mm.work == "\"") { // save line number for error message mm.work = "" // clear mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.until("\""); if (mm.eof) { mm.work = "" // clear mm.work += "Unterminated quote (\") char, starting at line " mm.work += mm.tape[mm.cell] /* get */ mm.work += "\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } mm.clip() mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "string*" mm.push(); break } // literal tokens if (mm.work == "." || mm.work == "," || mm.work == ":" || mm.work == "-" || mm.work == "+" || mm.work == "[" || mm.work == "]" || mm.work == "{" || mm.work == "}") { mm.tape[mm.cell] = mm.work /* put */ mm.work += "*" mm.push(); break } // here check if the workspace is empty. If not it is an error. if (mm.work != "") { mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += ", char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ": unquoted '" mm.work += mm.tape[mm.cell] /* get */ mm.work += "' character.\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } break } if restart { restart = false; continue; } // parse block for true { // This is for visualising stack reductions when debugging //unstack; add "\n"; print; clip; stack; // The parse/compile phase // -------------- // 2 tokens mm.pop(); mm.pop(); //----------- // Two token errors (not necessarily a complete list) // comma errors if (mm.work == "{*,*" || mm.work == ",*}*" || mm.work == "[*,*" || mm.work == ",*,*" || mm.work == ",*]*") { mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += ", char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += " (extra or misplaced ',' comma)\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // exponent errors (e/E must be followed by an int or signed int) if (mm.work != "E*" && strings.HasPrefix(mm.work, "E*") && !strings.HasSuffix(mm.work,"integer*") && !strings.HasSuffix(mm.work,"-*") && !strings.HasSuffix(mm.work,"+*") && !strings.HasSuffix(mm.work,"number*")) { mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced exponent 'e' or 'E' \n" mm.work += "In JSON syntax, e/E may only precede an int or signed int.\n" mm.work += "for example: 33e+01 \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // exponent errors (e/E must be followed by an int or signed int) if (mm.work != "E*" && strings.HasSuffix(mm.work, "E*") && !strings.HasPrefix(mm.work,"integer*") && !strings.HasPrefix(mm.work,"sign.integer*") && !strings.HasPrefix(mm.work,"decimal*")) { mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced exponent 'e' or 'E' \n" mm.work += "In JSON syntax, e/E may only be preceded by an int, signed int.\n" mm.work += "or decimal eg: 33e+01 \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // sign errors (+/- must be followed by an integer if (mm.work != "-*" && strings.HasPrefix(mm.work, "-*") && !strings.HasSuffix(mm.work,"integer*")) { mm.work = "" // clear mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced negative '-' sign\n" mm.work += "In JSON syntax, - may only precede a number \n" mm.work += "for example: -33.01 \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // sign errors (+/- must be followed by an integer) if (mm.work != "+*" && strings.HasPrefix(mm.work, "+*") && !strings.HasSuffix(mm.work,"integer*")) { mm.work = "" // clear mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced positive '+' sign\n" mm.work += "In JSON syntax, + may only precede a number \n" mm.work += "for example: +33.01 \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // dot errors (. must be followed by an integer) if (mm.work != ".*" && strings.HasPrefix(mm.work, ".*") && !strings.HasSuffix(mm.work,"integer*")) { mm.work = "" // clear mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced dot '.' sign\n" mm.work += "In JSON syntax, dots may only be used in decimal numbers \n" mm.work += "for example: -33.01 \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // dot errors (. must be preceded by an integer or signed integer) if (mm.work != ".*" && strings.HasSuffix(mm.work, ".*") && !strings.HasPrefix(mm.work,"integer*") && !strings.HasPrefix(mm.work,"sign.integer*")) { mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced dot '.' sign\n" mm.work += "In JSON syntax, dots may only be used in decimal numbers \n" mm.work += "for example: -33.01, but .44 is not a legal JSON number \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // eg errors "items*:*","members*:*",",*:*","[*:*","{*:*" // A colon must be preceded by a string. Using logic if (strings.HasSuffix(mm.work, ":*") && mm.work != ":*" && !strings.HasPrefix(mm.work,"string*")) { mm.work = "" // clear mm.work += "Json syntax error near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += ", char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += " (misplaced colon ':') \n" mm.work += "A ':' can only occur after a string key in an object structure \n" mm.work += "Example: {\"cancelled\":true} \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // more colon errors if (mm.work != ":*" && strings.HasPrefix(mm.work, ":*")) { if (strings.HasSuffix(mm.work, "}*") || strings.HasSuffix(mm.work, ",*") || strings.HasSuffix(mm.work, "]*")) { mm.work = "" // clear mm.work += "JSON syntax error near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += ", char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += " (misplaced colon ':' or missing value?) \n" mm.work += "A ':' only occur as part of an object member \n" mm.work += "Example: {\"cancelled\":true} \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } } // catch object member errors // also need to check that not only 1 token in on the stack // hence the !"member*" construct if (strings.HasPrefix(mm.work, "member*") || strings.HasPrefix(mm.work, "members*")) { if (mm.work != "member*" && mm.work != "members*" && !strings.HasSuffix(mm.work,",*") && !strings.HasSuffix(mm.work,"}*")) { mm.work = "" // clear mm.work += "JSON syntax error after object member near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ")\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } } // catch array errors if (strings.HasPrefix(mm.work, "items*") && mm.work != "items*" && !strings.HasSuffix(mm.work,",*") && !strings.HasSuffix(mm.work,"]*")) { mm.work = "" // clear mm.work += "Error after an array item near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ")\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } if (strings.HasPrefix(mm.work, "array*") || strings.HasPrefix(mm.work, "object*")) { if (mm.work != "array*" && mm.work != "object*" && !strings.HasSuffix(mm.work,",*") && !strings.HasSuffix(mm.work,"}*") && !strings.HasSuffix(mm.work,"]*")) { mm.work = "" // clear mm.work += "JSON syntax error near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ")\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } } // invalid string sequence if (strings.HasPrefix(mm.work, "string*")) { if (mm.work != "string*" && !strings.HasSuffix(mm.work,",*") && !strings.HasSuffix(mm.work,"]*") && !strings.HasSuffix(mm.work,"}*") && !strings.HasSuffix(mm.work,":*")) { mm.work = "" // clear mm.work += "JSON syntax error after a string near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += ")\n" fmt.Printf("%s", mm.work) // print os.Exit(0) } } // transmogrify into array item, start array if (mm.work == "[*number*" || mm.work == "[*string*" || mm.work == "[*value*" || mm.work == "[*array*" || mm.work == "[*object*") { mm.work = "" // clear mm.work += "[*items*" mm.push(); mm.push(); continue } // exponents (e-403, E+120, E04), this slightly simplifies number parsing if (mm.work == "E*sign.integer*" || mm.work == "E*integer*") { mm.work = "" // clear mm.work += "^" mm.increment() /* ++ */ mm.work += mm.tape[mm.cell] /* get */ if mm.cell > 0 { mm.cell-- } /* -- */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "exponent*" mm.push(); continue } // JSON scientific format (23e-10, -201E+33) if (mm.work == "integer*exponent*" || mm.work == "sign.integer*exponent*") { mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ // enforce multidigit zero rules // But is "0e44" legal JSON number syntax? That would seem odd // if it is. if (strings.HasPrefix(mm.work, "+")) { mm.work = "" // clear mm.work += "JSON syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): \n" mm.work += "In JSON syntax, the number part may not have a positive sign \n" mm.work += "eg: +0.12e34 (error!) \n" mm.work += "eg: 0.12e+34 (OK!) \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } if (strings.HasPrefix(mm.work, "-")) { mm.clop() } if (mm.work != "0" && strings.HasPrefix(mm.work, "0")) { mm.work = "" // clear mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): \n" mm.work += "In JSON syntax, multidigit numbers must begin with 1-9 \n" mm.work += "eg: -0234.01E+9 (error) \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ mm.increment() /* ++ */ mm.work += mm.tape[mm.cell] /* get */ if mm.cell > 0 { mm.cell-- } /* -- */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "number*" mm.push(); continue } // JSON scientific format (-0.23e10, 10.2E+33) if (mm.work == "decimal*exponent*") { mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ mm.increment() /* ++ */ mm.work += mm.tape[mm.cell] /* get */ if mm.cell > 0 { mm.cell-- } /* -- */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "number*" mm.push(); continue } // where does a number terminate, this is the problem // It terminates at the tokens ,* }* ]* and maybe space but // this script doesnt have a space* token. if (mm.work == "sign.integer*,*" || mm.work == "integer*,*") { mm.work = "" // clear mm.work += "number*,*" mm.push(); mm.push(); continue } // transmog if (mm.work == "sign.integer*]*" || mm.work == "integer*]*") { mm.work = "" // clear mm.work += "items*]*" mm.push(); mm.push(); continue } if (mm.work == "sign.integer*}*" || mm.work == "integer*}*") { mm.work = "" // clear mm.work += "number*}*" mm.push(); mm.push(); continue } // convert decimals to numbers with token lookahead if (mm.work == "decimal*}*" || mm.work == "decimal*]*" || mm.work == "decimal*,*") { /* replace */ mm.work = strings.Replace(mm.work, "decimal*", "number*", -1) mm.push(); mm.push(); continue } // signed numbers if (mm.work == "-*integer*" || mm.work == "+*integer*") { mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ mm.increment() /* ++ */ mm.work += mm.tape[mm.cell] /* get */ if mm.cell > 0 { mm.cell-- } /* -- */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "sign.integer*" mm.push(); continue } // signed numbers if (mm.work == "-*integer*" || mm.work == "+*integer*") { mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ mm.increment() /* ++ */ mm.work += mm.tape[mm.cell] /* get */ if mm.cell > 0 { mm.cell-- } /* -- */ mm.tape[mm.cell] = mm.work /* put */ mm.work = "" // clear mm.work += "sign.integer*" mm.push(); continue } // empty arrays are legal json if (mm.work == "[*]*") { mm.work = "" // clear mm.work += "array*" mm.push(); continue } // empty objects are legal json if (mm.work == "{*}*") { mm.work = "" // clear mm.work += "object*" mm.push(); continue } // -------------- // 3 tokens mm.pop(); //--------------- // Some three token errors // Object errors // A negative logic doesnt work because of the lookahead required for numbers if (mm.work == "{*string*}*" || mm.work == "{*integer*}*" || mm.work == "{*sign.integer*}*" || mm.work == "{*array*}*" || mm.work == "{*object*}*" || mm.work == "{*value*}*" || mm.work == "{*decimal*}*") { mm.work = "" // clear mm.work += "Json syntax error near line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += ", char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += " (misplaced brace '}' or bad object) \n" mm.work += "A '}' can only occur to terminate an object structure \n" mm.work += "Example: {\"hour\":21.00, \"cancelled\":true} \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } // transmogrify number into array item if (mm.work == "[*number*,*") { mm.work = "" // clear mm.work += "[*items*,*" mm.push(); mm.push(); mm.push(); continue } // decimal numbers eg -4.334 or +4.3 or 0.1 if (mm.work == "sign.integer*.*integer*") { mm.work = "" // clear mm.work += mm.tape[mm.cell] /* get */ if (strings.HasPrefix(mm.work, "+")) { //error, no positive signed decimals in JSON mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): misplaced positive '+' sign\n" mm.work += "In JSON syntax, decimal numbers are not positively signed\n" mm.work += "eg: +33.01 (error) \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } if (strings.HasPrefix(mm.work, "-0") && mm.work != "-0") { mm.work += "Json syntax error at line " mm.work += strconv.Itoa(mm.linesRead) /* lines */ mm.work += " (char " mm.work += strconv.Itoa(mm.charsRead) /* chars */ mm.work += "): \n" mm.work += "In JSON syntax, multidigit numbers must begin with 1-9 \n" mm.work += "eg: -0234.01E+9 (error) \n" fmt.Printf("%s", mm.work) // print os.Exit(0) } mm.work = "" // clear mm.work += "decimal*" mm.push(); continue } // decimal numbers eg -4.334 or +4.3 or 0.1 if (mm.work == "integer*.*integer*") { mm.work = "" // clear mm.work += "decimal*" mm.push(); continue } // arrays, if (mm.work == "[*items*]*" || mm.work == "[*number*]*") { mm.work = "" // clear mm.work += "array*" mm.push(); continue } // if (mm.work == "items*,*string*" || mm.work == "items*,*value*" || mm.work == "items*,*array*" || mm.work == "items*,*object*" || mm.work == "items*,*number*") { mm.work = "" // clear mm.work += "items*" mm.push(); continue } // object members //"string*:*integer*", if (mm.work == "string*:*number*" || mm.work == "string*:*string*" || mm.work == "string*:*value*" || mm.work == "string*:*object*" || mm.work == "string*:*array*") { mm.work = "" // clear mm.work += "member*" mm.push(); continue } // multiple elements of an object if (mm.work == "member*,*member*" || mm.work == "members*,*member*") { mm.work = "" // clear mm.work += "members*" mm.push(); continue } // if (mm.work == "{*members*}*" || mm.work == "{*member*}*") { mm.work = "" // clear mm.work += "object*" mm.push(); continue } mm.pop(); // -------------- // 4 tokens if (mm.work == "items*,*items*,*" || mm.work == "items*,*number*,*") { mm.work = "" // clear mm.work += "items*,*" mm.push(); mm.push(); continue } // numbers require a lookahead token, unfortunately if (mm.work == "string*:*number*,*") { mm.work = "" // clear mm.work += "member*,*" mm.push(); mm.push(); continue } // numbers require a lookahead token, unfortunately if (mm.work == "string*:*number*}*") { mm.work = "" // clear mm.work += "member*}*" mm.push(); mm.push(); continue } // multiple elements of an object with lookahead if (mm.work == "member*,*member*,*" || mm.work == "members*,*member*,*") { mm.work = "" // clear mm.work += "members*,*" mm.push(); mm.push(); continue } // multiple elements of an object with lookahead if (mm.work == "member*,*member*}*" || mm.work == "members*,*member*}*") { mm.work = "" // clear mm.work += "members*}*" mm.push(); mm.push(); continue } mm.pop(); // -------------- // 5 tokens // need this clumsy rule for numbers which get resolved when // a ] is seen. This is the lookahead if (mm.work == "[*items*,*items*]*" || mm.work == "[*items*,*number*]*") { mm.work = "" // clear mm.work += "array*" mm.push(); continue } mm.push(); mm.push(); mm.push(); mm.push(); mm.push(); if (mm.eof) { for mm.pop() {} /* unstack */ if (mm.work == "object*" || mm.work == "array*" || mm.work == "value*" || mm.work == "string*" || mm.work == "integer*" || mm.work == "decimal*" || mm.work == "number*") { for mm.push() {} /* stack */ mm.work += "(Appears to be) valid JSON syntax. Top level structure was '" fmt.Printf("%s", mm.work) // print mm.work = "" // clear mm.pop(); mm.clip() mm.work += "'\n" fmt.Printf("%s", mm.work) // print mm.work = "" // clear os.Exit(0) } for mm.push() {} /* stack */ mm.work += "(maybe) Invalid JSON \n" mm.work += "The parse stack was \n" fmt.Printf("%s", mm.work) // print mm.work = "" // clear for mm.pop() {} /* unstack */ mm.work += "\n" fmt.Printf("%s", mm.work) // print } break } // parse } } // end of generated 'go' code