parser: WIP

Signed-off-by: Alejandro Mery <amery@jpi.io>
This commit is contained in:
2023-08-28 23:46:05 +00:00
parent ab175180d6
commit 22b372cdb0
4 changed files with 132 additions and 0 deletions
+58
View File
@@ -0,0 +1,58 @@
package parser
import "asciigoat.org/core/lexer"
// Run parses the source
func (p *Parser) Run() error {
p.setDefaults()
p.pos.Reset()
return lexer.Run(p.lexStart)
}
func (p *Parser) lexStart() (lexer.StateFn, error) {
for {
r, _, err := p.src.ReadRune()
switch {
case err != nil:
return p.emitError("", err)
case r == '\n':
// new line
r2, _, err := p.src.ReadRune()
switch {
case err == nil && r2 == '\r':
// MAC new line
case err == nil:
// UNIX new line
p.src.UnreadRune()
}
p.src.Discard()
p.pos.StepLine()
case r == '\r':
// new line
r2, _, err := p.src.ReadRune()
switch {
case err == nil && r2 == '\n':
// DOS new line
case err == nil:
// incomplete CRLN
p.src.UnreadRune()
return p.emitInvalidRune(r2)
}
p.src.Discard()
p.pos.StepLine()
case p.IsSpace(r):
// whitespace
p.src.Discard()
p.pos.Step()
case p.IsCommentStart(r):
// switch to comment lexer
p.src.UnreadRune()
return p.lexComment, nil
}
}
}
func (p *Parser) lexComment() (lexer.StateFn, error)
+19
View File
@@ -0,0 +1,19 @@
package parser
import "asciigoat.org/core/lexer"
func (p *Parser) emitError(content string, err error) (lexer.StateFn, error) {
err2 := p.OnError(p.pos, content, err)
switch {
case err2 != nil:
// return wrapped error
return nil, err2
default:
// return original error
return nil, err
}
}
func (p *Parser) emitInvalidRune(r rune) (lexer.StateFn, error) {
return p.emitError(string([]rune{r}), lexer.ErrUnacceptableRune)
}
+17
View File
@@ -0,0 +1,17 @@
package parser
import "unicode"
func (*Parser) IsSpace(r rune) bool {
return unicode.IsSpace(r)
}
func (*Parser) IsCommentStart(r rune) bool {
// TODO: use Parser options
switch r {
case ';', '#':
return true
default:
return false
}
}
+38
View File
@@ -10,6 +10,44 @@ import (
// Parser parses a dosini-style document
type Parser struct {
src *lexer.Reader
pos lexer.Position
OnSection func(pos lexer.Position, name, subname string, hasSubname bool) error
OnField func(pos lexer.Position, key, value string) error
OnComment func(pos lexer.Position, comment string) error
OnError func(pos lexer.Position, content string, err error) error
}
func defaultOnSection(_ lexer.Position, _, _ string, _ bool) error { return nil }
func defaultOnField(_ lexer.Position, _, _ string) error { return nil }
func defaultOnComment(_ lexer.Position, _ string) error { return nil }
func defaultOnError(pos lexer.Position, content string, err error) error {
return &lexer.Error{
Line: pos.Line,
Column: pos.Column,
Content: content,
Err: err,
}
}
func (p *Parser) setDefaults() {
if p.OnSection == nil {
p.OnSection = defaultOnSection
}
if p.OnField == nil {
p.OnField = defaultOnField
}
if p.OnComment == nil {
p.OnComment = defaultOnComment
}
if p.OnError == nil {
p.OnError = defaultOnError
}
}
// NewParser creates a dosini-style parser using