package circuitcompiler import ( "bufio" "bytes" "io" ) type OperatorSymbol int type Token int const ( ILLEGAL Token = iota WS EOF IDENT // val VAR // var CONST // const value EQ // = PLUS // + MINUS // - MULTIPLY // * DIVIDE // / EXP // ^ OUT ) var eof = rune(0) func isWhitespace(ch rune) bool { return ch == ' ' || ch == '\t' || ch == '\n' } func isLetter(ch rune) bool { return (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') } func isDigit(ch rune) bool { return (ch >= '0' && ch <= '9') } type Scanner struct { r *bufio.Reader } func NewScanner(r io.Reader) *Scanner { return &Scanner{r: bufio.NewReader(r)} } func (s *Scanner) read() rune { ch, _, err := s.r.ReadRune() if err != nil { return eof } return ch } func (s *Scanner) unread() { _ = s.r.UnreadRune() } func (s *Scanner) Scan() (tok Token, lit string) { ch := s.read() if isWhitespace(ch) { // space s.unread() return s.scanWhitespace() } else if isLetter(ch) { // letter s.unread() return s.scanIndent() } else if isDigit(ch) { s.unread() return s.scanIndent() } switch ch { case eof: return EOF, "" case '=': return EQ, "=" case '+': return PLUS, "+" case '-': return MINUS, "-" case '*': return MULTIPLY, "*" case '/': return DIVIDE, "/" case '^': return EXP, "^" } return ILLEGAL, string(ch) } func (s *Scanner) scanWhitespace() (token Token, lit string) { var buf bytes.Buffer buf.WriteRune(s.read()) for { if ch := s.read(); ch == eof { break } else if !isWhitespace(ch) { s.unread() break } else { _, _ = buf.WriteRune(ch) } } return WS, buf.String() } func (s *Scanner) scanIndent() (tok Token, lit string) { var buf bytes.Buffer buf.WriteRune(s.read()) for { if ch := s.read(); ch == eof { break } else if !isLetter(ch) && !isDigit(ch) { s.unread() break } else { _, _ = buf.WriteRune(ch) } } switch buf.String() { case "var": return VAR, buf.String() } if len(buf.String()) == 1 { return Token(rune(buf.String()[0])), buf.String() } if buf.String() == "out" { return OUT, buf.String() } return IDENT, buf.String() }