surrealpatch/sql/select.go

209 lines
3.8 KiB
Go
Raw Normal View History

2016-02-26 17:27:07 +00:00
// Copyright © 2016 Abcum Ltd
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package sql
func (p *parser) parseSelectStatement(explain bool) (stmt *SelectStatement, err error) {
2016-02-26 17:27:07 +00:00
2016-05-23 12:32:02 +00:00
stmt = &SelectStatement{}
2016-02-26 17:27:07 +00:00
2016-05-23 12:32:02 +00:00
stmt.EX = explain
stmt.KV = p.c.Get("KV").(string)
stmt.NS = p.c.Get("NS").(string)
stmt.DB = p.c.Get("DB").(string)
2016-02-26 17:27:07 +00:00
2016-05-23 12:32:02 +00:00
if stmt.Expr, err = p.parseExpr(); err != nil {
2016-02-26 17:27:07 +00:00
return nil, err
}
_, _, err = p.shouldBe(FROM)
if err != nil {
return nil, err
}
2016-05-23 12:32:02 +00:00
if stmt.What, err = p.parseWhat(); err != nil {
2016-02-26 17:27:07 +00:00
return nil, err
}
2016-05-23 12:32:02 +00:00
if stmt.Cond, err = p.parseCond(); err != nil {
2016-02-26 17:27:07 +00:00
return nil, err
}
if stmt.Group, err = p.parseGroup(); err != nil {
return nil, err
}
if stmt.Order, err = p.parseOrder(); err != nil {
return nil, err
}
if stmt.Limit, err = p.parseLimit(); err != nil {
return nil, err
}
if stmt.Start, err = p.parseStart(); err != nil {
return nil, err
}
if stmt.Version, err = p.parseVersion(); err != nil {
return nil, err
}
2016-09-06 13:30:59 +00:00
if stmt.Echo, err = p.parseEcho(); err != nil {
return nil, err
}
2016-02-26 17:27:07 +00:00
if _, _, err = p.shouldBe(EOF, SEMICOLON); err != nil {
return nil, err
}
return stmt, nil
}
func (p *parser) parseGroup() (mul []*Group, err error) {
2016-02-26 17:27:07 +00:00
// Remove the GROUP keyword
if _, _, exi := p.mightBe(GROUP); !exi {
return nil, nil
}
// Next token might be BY
_, _, _ = p.mightBe(BY)
2016-07-04 10:37:37 +00:00
return
2016-02-26 17:27:07 +00:00
}
func (p *parser) parseOrder() (mul []*Order, err error) {
2016-02-26 17:27:07 +00:00
var tok Token
var lit string
var exi bool
// Remove the ORDER keyword
if _, _, exi := p.mightBe(ORDER); !exi {
return nil, nil
}
// Next token might be BY
_, _, _ = p.mightBe(BY)
for {
2016-05-23 12:32:02 +00:00
one := &Order{}
2016-02-26 17:27:07 +00:00
2016-05-23 12:32:02 +00:00
tok, lit, err = p.shouldBe(IDENT, ID)
2016-02-26 17:27:07 +00:00
if err != nil {
return nil, &ParseError{Found: lit, Expected: []string{"field name"}}
}
2016-05-23 12:32:02 +00:00
one.Expr, err = p.declare(tok, lit)
2016-05-23 12:32:02 +00:00
if err != nil {
return nil, err
}
2016-02-26 17:27:07 +00:00
tok, lit, exi = p.mightBe(ASC, DESC)
if !exi {
2016-09-06 13:30:59 +00:00
tok, lit = ASC, "ASC"
2016-02-26 17:27:07 +00:00
}
one.Dir, err = p.declare(tok, lit)
2016-05-23 12:32:02 +00:00
if err != nil {
return nil, err
}
mul = append(mul, one)
2016-02-26 17:27:07 +00:00
// If the next token is not a comma then break the loop.
if _, _, exi := p.mightBe(COMMA); !exi {
break
}
}
2016-07-04 10:37:37 +00:00
return
2016-02-26 17:27:07 +00:00
}
func (p *parser) parseLimit() (Expr, error) {
2016-02-26 17:27:07 +00:00
// Remove the LIMIT keyword
if _, _, exi := p.mightBe(LIMIT); !exi {
return nil, nil
}
// Next token might be BY
_, _, _ = p.mightBe(BY)
2016-07-04 10:37:37 +00:00
tok, lit, err := p.shouldBe(NUMBER)
2016-02-26 17:27:07 +00:00
if err != nil {
return nil, &ParseError{Found: lit, Expected: []string{"limit number"}}
}
return p.declare(tok, lit)
2016-02-26 17:27:07 +00:00
}
func (p *parser) parseStart() (Expr, error) {
2016-02-26 17:27:07 +00:00
// Remove the START keyword
if _, _, exi := p.mightBe(START); !exi {
return nil, nil
}
// Next token might be AT
_, _, _ = p.mightBe(AT)
// Next token might be @
_, _, exi := p.mightBe(EAT)
if exi == false {
2016-05-24 12:51:52 +00:00
val, err := p.parseNumber()
if err != nil {
return nil, err
2016-02-26 17:27:07 +00:00
}
2016-05-24 12:51:52 +00:00
return val, nil
2016-02-26 17:27:07 +00:00
}
if exi == true {
2016-05-24 12:51:52 +00:00
p.unscan()
val, err := p.parseThing()
2016-02-26 17:27:07 +00:00
if err != nil {
return nil, err
}
2016-05-24 12:51:52 +00:00
return val, nil
2016-02-26 17:27:07 +00:00
}
return nil, nil
}
func (p *parser) parseVersion() (Expr, error) {
2016-02-26 17:27:07 +00:00
// Remove the VERSION keyword
if _, _, exi := p.mightBe(VERSION); !exi {
return nil, nil
}
2016-05-23 12:32:02 +00:00
tok, lit, err := p.shouldBe(DATE, TIME)
2016-02-26 17:27:07 +00:00
if err != nil {
return nil, &ParseError{Found: lit, Expected: []string{"timestamp"}}
}
return p.declare(tok, lit)
2016-02-26 17:27:07 +00:00
}