Compare commits
11 Commits
main
...
dev-amery-
Author | SHA1 | Date |
---|---|---|
Alejandro Mery | cbd3f705f5 | 1 year ago |
Alejandro Mery | c9f206c9aa | 1 year ago |
Alejandro Mery | 9912146d21 | 1 year ago |
Alejandro Mery | 71ab4a58c1 | 1 year ago |
Alejandro Mery | 8e838c3566 | 1 year ago |
Alejandro Mery | ff0c7d1b9f | 1 year ago |
Alejandro Mery | fabd192e3d | 1 year ago |
Alejandro Mery | 37f3efebfb | 1 year ago |
Alejandro Mery | 506fff8725 | 1 year ago |
Alejandro Mery | d75b2dbc78 | 1 year ago |
Alejandro Mery | 46ba96d6b4 | 1 year ago |
13 changed files with 149 additions and 263 deletions
@ -0,0 +1,7 @@ |
|||||||
|
{ |
||||||
|
"cSpell.words": [ |
||||||
|
"asciigoat", |
||||||
|
"Subname", |
||||||
|
"unescapes" |
||||||
|
] |
||||||
|
} |
@ -0,0 +1,49 @@ |
|||||||
|
package ini |
||||||
|
|
||||||
|
import ( |
||||||
|
"bytes" |
||||||
|
"io" |
||||||
|
"strings" |
||||||
|
|
||||||
|
"asciigoat.org/core" |
||||||
|
"asciigoat.org/ini/parser" |
||||||
|
) |
||||||
|
|
||||||
|
// Decoder ...
|
||||||
|
type Decoder struct { |
||||||
|
io.Closer |
||||||
|
|
||||||
|
p *parser.Parser |
||||||
|
} |
||||||
|
|
||||||
|
// Decode ...
|
||||||
|
func (dec *Decoder) Decode() error { |
||||||
|
defer dec.Close() |
||||||
|
|
||||||
|
return dec.p.Run() |
||||||
|
} |
||||||
|
|
||||||
|
// NewDecoder creates a Decoder over the provided [io.Reader]
|
||||||
|
func NewDecoder(r io.Reader) *Decoder { |
||||||
|
rc := core.NewReadCloser(r) |
||||||
|
switch { |
||||||
|
case rc == nil: |
||||||
|
return nil |
||||||
|
default: |
||||||
|
dec := &Decoder{ |
||||||
|
p: parser.NewParser(rc), |
||||||
|
Closer: rc, |
||||||
|
} |
||||||
|
return dec |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// NewDecoderBytes creates a Decoder over a provided bytes array
|
||||||
|
func NewDecoderBytes(b []byte) *Decoder { |
||||||
|
return NewDecoder(bytes.NewReader(b)) |
||||||
|
} |
||||||
|
|
||||||
|
// NewDecoderString creates a Decoder over a provided string of data
|
||||||
|
func NewDecoderString(s string) *Decoder { |
||||||
|
return NewDecoder(strings.NewReader(s)) |
||||||
|
} |
@ -1,88 +0,0 @@ |
|||||||
package parser |
|
||||||
|
|
||||||
import ( |
|
||||||
"strings" |
|
||||||
|
|
||||||
"asciigoat.org/core/lexer" |
|
||||||
) |
|
||||||
|
|
||||||
type commaArrayParser struct { |
|
||||||
TextParser |
|
||||||
|
|
||||||
out []string |
|
||||||
} |
|
||||||
|
|
||||||
func (p *commaArrayParser) lexStart() (lexer.StateFn, error) { |
|
||||||
for { |
|
||||||
r, _, err := p.ReadRune() |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// EOF
|
|
||||||
return nil, err |
|
||||||
case r == RuneQuotes: |
|
||||||
// Quoted Value
|
|
||||||
return p.lexQuotedString, nil |
|
||||||
case IsNewLine(r): |
|
||||||
// new lines are acceptable when parsing a string for
|
|
||||||
// comma delimited arrays. but make sure we discard it
|
|
||||||
// complete
|
|
||||||
p.UnreadRune() |
|
||||||
p.AcceptNewLine() |
|
||||||
p.Discard() |
|
||||||
case lexer.IsSpace(r): |
|
||||||
// discard whitespace outside quotes
|
|
||||||
p.Discard() |
|
||||||
default: |
|
||||||
p.UnreadRune() |
|
||||||
return p.lexWord, nil |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
func (p *commaArrayParser) lexWord() (lexer.StateFn, error) { |
|
||||||
for { |
|
||||||
r, _, err := p.ReadRune() |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// done. store what we got and move on
|
|
||||||
_, s := p.Emit() |
|
||||||
p.out = append(p.out, s) |
|
||||||
return nil, err |
|
||||||
case r == ',': |
|
||||||
// done
|
|
||||||
_, s := p.Emit() |
|
||||||
// remove comma, trim and append to output
|
|
||||||
s = strings.TrimRightFunc(s[:len(s)-1], IsSpace) |
|
||||||
p.out = append(p.out, s) |
|
||||||
return p.lexStart, nil |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
func (p *commaArrayParser) lexQuotedString() (lexer.StateFn, error) { |
|
||||||
s, err := lexQuotedString(&p.TextParser) |
|
||||||
if err != nil { |
|
||||||
return nil, err |
|
||||||
} |
|
||||||
|
|
||||||
p.Discard() |
|
||||||
p.out = append(p.out, s) |
|
||||||
return p.lexStart, nil |
|
||||||
} |
|
||||||
|
|
||||||
func (p *commaArrayParser) Run() ([]string, error) { |
|
||||||
err := lexer.Run(p.lexStart) |
|
||||||
|
|
||||||
return p.out, err |
|
||||||
} |
|
||||||
|
|
||||||
// SplitCommaArray splits comma separated strings, removing whitespace
|
|
||||||
// and respecting quoted literals.
|
|
||||||
func SplitCommaArray(s string) ([]string, error) { |
|
||||||
if s != "" { |
|
||||||
var p commaArrayParser |
|
||||||
p.InitString(s) |
|
||||||
return p.Run() |
|
||||||
} |
|
||||||
return nil, nil |
|
||||||
} |
|
@ -1,43 +0,0 @@ |
|||||||
package parser |
|
||||||
|
|
||||||
import ( |
|
||||||
"io/fs" |
|
||||||
|
|
||||||
"asciigoat.org/core/lexer" |
|
||||||
) |
|
||||||
|
|
||||||
// NewError creates a lexer.Error using a lexer.Position
|
|
||||||
func NewError(pos lexer.Position, content, hint string, err error) *lexer.Error { |
|
||||||
return &lexer.Error{ |
|
||||||
Line: pos.Line, |
|
||||||
Column: pos.Column, |
|
||||||
Content: content, |
|
||||||
Hint: hint, |
|
||||||
Err: err, |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// ErrPlusPosition returns a copy of the given [lexer.Error]
|
|
||||||
// offsetting the Line/Column information.
|
|
||||||
func ErrPlusPosition(pos lexer.Position, e *lexer.Error) *lexer.Error { |
|
||||||
pos.Add(lexer.Position{ |
|
||||||
Line: e.Line, |
|
||||||
Column: e.Column, |
|
||||||
}) |
|
||||||
|
|
||||||
return NewError(pos, e.Content, e.Hint, e.Err) |
|
||||||
} |
|
||||||
|
|
||||||
// NewErrIncompleteQuotedString returns a [lexer.Error]
|
|
||||||
// indicating the quoted string being parsed wasn't correctly
|
|
||||||
// terminated
|
|
||||||
func NewErrIncompleteQuotedString(p *TextParser) *lexer.Error { |
|
||||||
return newErrIncomplete(p, "incomplete quoted string") |
|
||||||
} |
|
||||||
|
|
||||||
func newErrIncomplete(p *TextParser, hint string) *lexer.Error { |
|
||||||
pos, s := p.Emit() |
|
||||||
pos.Add(GetPositionalLength(s)) |
|
||||||
|
|
||||||
return NewError(pos, s, hint, fs.ErrInvalid) |
|
||||||
} |
|
@ -1,97 +0,0 @@ |
|||||||
package parser |
|
||||||
|
|
||||||
import ( |
|
||||||
"strings" |
|
||||||
|
|
||||||
"asciigoat.org/core/lexer" |
|
||||||
) |
|
||||||
|
|
||||||
// AcceptQuotedString consumes a quoted string from the source
|
|
||||||
// and returns it unquoted and unescaped
|
|
||||||
func (p *TextParser) AcceptQuotedString() (string, bool, error) { |
|
||||||
r, _, err := p.ReadRune() |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// nothing here
|
|
||||||
return "", false, err |
|
||||||
case r != RuneQuotes: |
|
||||||
// not for us
|
|
||||||
p.UnreadRune() |
|
||||||
return "", false, nil |
|
||||||
default: |
|
||||||
// let's roll
|
|
||||||
s, err := lexQuotedString(p) |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// bad quoted string
|
|
||||||
return "", false, err |
|
||||||
default: |
|
||||||
// success
|
|
||||||
return s, true, nil |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
func lexQuotedString(p *TextParser) (string, *lexer.Error) { |
|
||||||
for { |
|
||||||
r, _, err := p.ReadRune() |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// incomplete
|
|
||||||
return "", NewErrIncompleteQuotedString(p) |
|
||||||
case r == RuneQuotes: |
|
||||||
// end, remove quotes and process escaped characters
|
|
||||||
return lexReturnUnescapedQuotedString(p) |
|
||||||
case r == RuneEscape: |
|
||||||
// escaped, take another
|
|
||||||
_, _, err := p.ReadRune() |
|
||||||
if err != nil { |
|
||||||
// incomplete
|
|
||||||
return "", NewErrIncompleteQuotedString(p) |
|
||||||
} |
|
||||||
case IsNewLine(r): |
|
||||||
// new lines within quoted values are acceptable
|
|
||||||
p.UnreadRune() |
|
||||||
p.AcceptNewLine() |
|
||||||
default: |
|
||||||
// continue
|
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
func lexReturnUnescapedQuotedString(p *TextParser) (string, *lexer.Error) { |
|
||||||
// remove quotes
|
|
||||||
s := p.String() |
|
||||||
l := len(s) |
|
||||||
s = s[1 : l-1] |
|
||||||
|
|
||||||
if strings.ContainsRune(s, RuneEscape) { |
|
||||||
// TODO: implement unescaping
|
|
||||||
err := NewError(p.Position(), s, "escaped characters", lexer.ErrNotImplemented) |
|
||||||
return "", err |
|
||||||
} |
|
||||||
|
|
||||||
return s, nil |
|
||||||
} |
|
||||||
|
|
||||||
// Unquoted removes quotes the content and unescapes the content
|
|
||||||
func Unquoted(s string) (string, error) { |
|
||||||
var p TextParser |
|
||||||
if s == "" { |
|
||||||
return "", nil |
|
||||||
} |
|
||||||
|
|
||||||
p.InitString(s) |
|
||||||
unquoted, ok, err := p.AcceptQuotedString() |
|
||||||
switch { |
|
||||||
case err != nil: |
|
||||||
// bad string
|
|
||||||
return "", err |
|
||||||
case ok: |
|
||||||
// success
|
|
||||||
return unquoted, nil |
|
||||||
default: |
|
||||||
// not quoted
|
|
||||||
return s, nil |
|
||||||
} |
|
||||||
} |
|
Loading…
Reference in new issue