refactor scan multiline for reuse

This commit is contained in:
Michael Yang 2023-07-27 09:55:48 -07:00
parent dbb3174cbc
commit f5ac8ddfb4

View file

@ -4,6 +4,7 @@ import (
"bufio" "bufio"
"bytes" "bytes"
"errors" "errors"
"fmt"
"io" "io"
"log" "log"
) )
@ -63,26 +64,46 @@ func Parse(reader io.Reader) ([]Command, error) {
} }
func scanModelfile(data []byte, atEOF bool) (advance int, token []byte, err error) { func scanModelfile(data []byte, atEOF bool) (advance int, token []byte, err error) {
const multilineString = `"""` advance, token, err = scan([]byte(`"""`), []byte(`"""`), data, atEOF)
if err != nil {
return 0, nil, err
}
if advance > 0 && token != nil {
return advance, token, nil
}
advance, token, err = scan([]byte(`"`), []byte(`"`), data, atEOF)
if err != nil {
return 0, nil, err
}
if advance > 0 && token != nil {
return advance, token, nil
}
return bufio.ScanLines(data, atEOF)
}
func scan(openBytes, closeBytes, data []byte, atEOF bool) (advance int, token []byte, err error) {
newline := bytes.IndexByte(data, '\n') newline := bytes.IndexByte(data, '\n')
if start := bytes.Index(data, []byte(multilineString)); start >= 0 && start < newline { if start := bytes.Index(data, openBytes); start >= 0 && start < newline {
end := bytes.Index(data[start+len(multilineString):], []byte(multilineString)) end := bytes.Index(data[start+len(openBytes):], closeBytes)
if end < 0 { if end < 0 {
if atEOF { if atEOF {
return 0, nil, errors.New("unterminated multiline string: " + multilineString) return 0, nil, fmt.Errorf("unterminated %s: expecting %s", openBytes, closeBytes)
} else { } else {
return 0, nil, nil return 0, nil, nil
} }
} }
n := start + len(multilineString) + end + len(multilineString) n := start + len(openBytes) + end + len(closeBytes)
newData := data[:start] newData := data[:start]
newData = append(newData, data[start+len(multilineString):n-len(multilineString)]...) newData = append(newData, data[start+len(openBytes):n-len(closeBytes)]...)
return n, newData, nil return n, newData, nil
} }
return bufio.ScanLines(data, atEOF) return 0, nil, nil
} }