2021-08-12 20:03:24 +01:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"strconv"
|
|
|
|
|
2021-08-25 14:34:33 +01:00
|
|
|
"github.com/uptrace/bun/internal"
|
2021-08-12 20:03:24 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Parser struct {
|
|
|
|
b []byte
|
|
|
|
i int
|
|
|
|
}
|
|
|
|
|
|
|
|
func New(b []byte) *Parser {
|
|
|
|
return &Parser{
|
|
|
|
b: b,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewString(s string) *Parser {
|
2021-08-25 14:34:33 +01:00
|
|
|
return New(internal.Bytes(s))
|
2021-08-12 20:03:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Valid() bool {
|
|
|
|
return p.i < len(p.b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Bytes() []byte {
|
|
|
|
return p.b[p.i:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Read() byte {
|
|
|
|
if p.Valid() {
|
|
|
|
c := p.b[p.i]
|
|
|
|
p.Advance()
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Peek() byte {
|
|
|
|
if p.Valid() {
|
|
|
|
return p.b[p.i]
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Advance() {
|
|
|
|
p.i++
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Skip(skip byte) bool {
|
|
|
|
if p.Peek() == skip {
|
|
|
|
p.Advance()
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) SkipBytes(skip []byte) bool {
|
|
|
|
if len(skip) > len(p.b[p.i:]) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if !bytes.Equal(p.b[p.i:p.i+len(skip)], skip) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
p.i += len(skip)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) ReadSep(sep byte) ([]byte, bool) {
|
|
|
|
ind := bytes.IndexByte(p.b[p.i:], sep)
|
|
|
|
if ind == -1 {
|
|
|
|
b := p.b[p.i:]
|
|
|
|
p.i = len(p.b)
|
|
|
|
return b, false
|
|
|
|
}
|
|
|
|
|
|
|
|
b := p.b[p.i : p.i+ind]
|
|
|
|
p.i += ind + 1
|
|
|
|
return b, true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) ReadIdentifier() (string, bool) {
|
|
|
|
if p.i < len(p.b) && p.b[p.i] == '(' {
|
|
|
|
s := p.i + 1
|
|
|
|
if ind := bytes.IndexByte(p.b[s:], ')'); ind != -1 {
|
|
|
|
b := p.b[s : s+ind]
|
|
|
|
p.i = s + ind + 1
|
2021-08-25 14:34:33 +01:00
|
|
|
return internal.String(b), false
|
2021-08-12 20:03:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ind := len(p.b) - p.i
|
|
|
|
var alpha bool
|
|
|
|
for i, c := range p.b[p.i:] {
|
|
|
|
if isNum(c) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if isAlpha(c) || (i > 0 && alpha && c == '_') {
|
|
|
|
alpha = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ind = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if ind == 0 {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
b := p.b[p.i : p.i+ind]
|
|
|
|
p.i += ind
|
2021-08-25 14:34:33 +01:00
|
|
|
return internal.String(b), !alpha
|
2021-08-12 20:03:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) ReadNumber() int {
|
|
|
|
ind := len(p.b) - p.i
|
|
|
|
for i, c := range p.b[p.i:] {
|
|
|
|
if !isNum(c) {
|
|
|
|
ind = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ind == 0 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
n, err := strconv.Atoi(string(p.b[p.i : p.i+ind]))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
p.i += ind
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
func isNum(c byte) bool {
|
|
|
|
return c >= '0' && c <= '9'
|
|
|
|
}
|
|
|
|
|
|
|
|
func isAlpha(c byte) bool {
|
|
|
|
return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z')
|
|
|
|
}
|