You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
408 lines
11 KiB
408 lines
11 KiB
package engine
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/cockroachdb/cockroachdb-parser/pkg/sql/parser"
|
|
"github.com/cockroachdb/cockroachdb-parser/pkg/sql/sem/tree"
|
|
)
|
|
|
|
// CockroachSQLParser wraps CockroachDB's PostgreSQL-compatible SQL parser for use in SeaweedFS
|
|
type CockroachSQLParser struct{}
|
|
|
|
// NewCockroachSQLParser creates a new instance of the CockroachDB SQL parser wrapper
|
|
func NewCockroachSQLParser() *CockroachSQLParser {
|
|
return &CockroachSQLParser{}
|
|
}
|
|
|
|
// ParseSQL parses a SQL statement using CockroachDB's parser
|
|
func (p *CockroachSQLParser) ParseSQL(sql string) (Statement, error) {
|
|
// Parse using CockroachDB's parser
|
|
stmts, err := parser.Parse(sql)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("CockroachDB parser error: %v", err)
|
|
}
|
|
|
|
if len(stmts) != 1 {
|
|
return nil, fmt.Errorf("expected exactly one statement, got %d", len(stmts))
|
|
}
|
|
|
|
stmt := stmts[0].AST
|
|
|
|
// Convert CockroachDB AST to SeaweedFS AST format
|
|
switch s := stmt.(type) {
|
|
case *tree.Select:
|
|
return p.convertSelectStatement(s)
|
|
default:
|
|
return nil, fmt.Errorf("unsupported statement type: %T", s)
|
|
}
|
|
}
|
|
|
|
// convertSelectStatement converts CockroachDB's Select AST to SeaweedFS format
|
|
func (p *CockroachSQLParser) convertSelectStatement(crdbSelect *tree.Select) (*SelectStatement, error) {
|
|
selectClause, ok := crdbSelect.Select.(*tree.SelectClause)
|
|
if !ok {
|
|
return nil, fmt.Errorf("expected SelectClause, got %T", crdbSelect.Select)
|
|
}
|
|
|
|
seaweedSelect := &SelectStatement{
|
|
SelectExprs: make([]SelectExpr, 0, len(selectClause.Exprs)),
|
|
From: []TableExpr{},
|
|
}
|
|
|
|
// Convert SELECT expressions
|
|
for _, expr := range selectClause.Exprs {
|
|
seaweedExpr, err := p.convertSelectExpr(expr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert select expression: %v", err)
|
|
}
|
|
seaweedSelect.SelectExprs = append(seaweedSelect.SelectExprs, seaweedExpr)
|
|
}
|
|
|
|
// Convert FROM clause
|
|
if len(selectClause.From.Tables) > 0 {
|
|
for _, fromExpr := range selectClause.From.Tables {
|
|
seaweedTableExpr, err := p.convertFromExpr(fromExpr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert FROM clause: %v", err)
|
|
}
|
|
seaweedSelect.From = append(seaweedSelect.From, seaweedTableExpr)
|
|
}
|
|
}
|
|
|
|
// Convert WHERE clause if present
|
|
if selectClause.Where != nil {
|
|
whereExpr, err := p.convertExpr(selectClause.Where.Expr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert WHERE clause: %v", err)
|
|
}
|
|
seaweedSelect.Where = &WhereClause{
|
|
Expr: whereExpr,
|
|
}
|
|
}
|
|
|
|
// Convert LIMIT and OFFSET clauses if present
|
|
if crdbSelect.Limit != nil {
|
|
limitClause := &LimitClause{}
|
|
|
|
// Convert LIMIT (Count)
|
|
if crdbSelect.Limit.Count != nil {
|
|
countExpr, err := p.convertExpr(crdbSelect.Limit.Count)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert LIMIT clause: %v", err)
|
|
}
|
|
limitClause.Rowcount = countExpr
|
|
}
|
|
|
|
// Convert OFFSET
|
|
if crdbSelect.Limit.Offset != nil {
|
|
offsetExpr, err := p.convertExpr(crdbSelect.Limit.Offset)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert OFFSET clause: %v", err)
|
|
}
|
|
limitClause.Offset = offsetExpr
|
|
}
|
|
|
|
seaweedSelect.Limit = limitClause
|
|
}
|
|
|
|
return seaweedSelect, nil
|
|
}
|
|
|
|
// convertSelectExpr converts CockroachDB SelectExpr to SeaweedFS format
|
|
func (p *CockroachSQLParser) convertSelectExpr(expr tree.SelectExpr) (SelectExpr, error) {
|
|
// Handle star expressions (SELECT *)
|
|
if _, isStar := expr.Expr.(tree.UnqualifiedStar); isStar {
|
|
return &StarExpr{}, nil
|
|
}
|
|
|
|
// CockroachDB's SelectExpr is a struct, not an interface, so handle it directly
|
|
seaweedExpr := &AliasedExpr{}
|
|
|
|
// Convert the main expression
|
|
convertedExpr, err := p.convertExpr(expr.Expr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert expression: %v", err)
|
|
}
|
|
seaweedExpr.Expr = convertedExpr
|
|
|
|
// Convert alias if present
|
|
if expr.As != "" {
|
|
seaweedExpr.As = aliasValue(expr.As)
|
|
}
|
|
|
|
return seaweedExpr, nil
|
|
}
|
|
|
|
// convertExpr converts CockroachDB expressions to SeaweedFS format
|
|
func (p *CockroachSQLParser) convertExpr(expr tree.Expr) (ExprNode, error) {
|
|
switch e := expr.(type) {
|
|
case *tree.FuncExpr:
|
|
// Function call
|
|
seaweedFunc := &FuncExpr{
|
|
Name: stringValue(strings.ToUpper(e.Func.String())), // Convert to uppercase for consistency
|
|
Exprs: make([]SelectExpr, 0, len(e.Exprs)),
|
|
}
|
|
|
|
// Convert function arguments
|
|
for _, arg := range e.Exprs {
|
|
// Special case: Handle star expressions in function calls like COUNT(*)
|
|
if _, isStar := arg.(tree.UnqualifiedStar); isStar {
|
|
seaweedFunc.Exprs = append(seaweedFunc.Exprs, &StarExpr{})
|
|
} else {
|
|
convertedArg, err := p.convertExpr(arg)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert function argument: %v", err)
|
|
}
|
|
seaweedFunc.Exprs = append(seaweedFunc.Exprs, &AliasedExpr{Expr: convertedArg})
|
|
}
|
|
}
|
|
|
|
return seaweedFunc, nil
|
|
|
|
case *tree.BinaryExpr:
|
|
// Arithmetic/binary operations (including string concatenation ||)
|
|
seaweedArith := &ArithmeticExpr{
|
|
Operator: e.Operator.String(),
|
|
}
|
|
|
|
// Convert left operand
|
|
left, err := p.convertExpr(e.Left)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert left operand: %v", err)
|
|
}
|
|
seaweedArith.Left = left
|
|
|
|
// Convert right operand
|
|
right, err := p.convertExpr(e.Right)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert right operand: %v", err)
|
|
}
|
|
seaweedArith.Right = right
|
|
|
|
return seaweedArith, nil
|
|
|
|
case *tree.ComparisonExpr:
|
|
// Comparison operations (=, >, <, >=, <=, !=, etc.) used in WHERE clauses
|
|
seaweedComp := &ComparisonExpr{
|
|
Operator: e.Operator.String(),
|
|
}
|
|
|
|
// Convert left operand
|
|
left, err := p.convertExpr(e.Left)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert comparison left operand: %v", err)
|
|
}
|
|
seaweedComp.Left = left
|
|
|
|
// Convert right operand
|
|
right, err := p.convertExpr(e.Right)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert comparison right operand: %v", err)
|
|
}
|
|
seaweedComp.Right = right
|
|
|
|
return seaweedComp, nil
|
|
|
|
case *tree.StrVal:
|
|
// String literal
|
|
return &SQLVal{
|
|
Type: StrVal,
|
|
Val: []byte(string(e.RawString())),
|
|
}, nil
|
|
|
|
case *tree.NumVal:
|
|
// Numeric literal
|
|
valStr := e.String()
|
|
if strings.Contains(valStr, ".") {
|
|
return &SQLVal{
|
|
Type: FloatVal,
|
|
Val: []byte(valStr),
|
|
}, nil
|
|
} else {
|
|
return &SQLVal{
|
|
Type: IntVal,
|
|
Val: []byte(valStr),
|
|
}, nil
|
|
}
|
|
|
|
case *tree.UnresolvedName:
|
|
// Column name
|
|
return &ColName{
|
|
Name: stringValue(e.String()),
|
|
}, nil
|
|
|
|
case *tree.AndExpr:
|
|
// AND expression
|
|
left, err := p.convertExpr(e.Left)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert AND left operand: %v", err)
|
|
}
|
|
right, err := p.convertExpr(e.Right)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert AND right operand: %v", err)
|
|
}
|
|
return &AndExpr{
|
|
Left: left,
|
|
Right: right,
|
|
}, nil
|
|
|
|
case *tree.OrExpr:
|
|
// OR expression
|
|
left, err := p.convertExpr(e.Left)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert OR left operand: %v", err)
|
|
}
|
|
right, err := p.convertExpr(e.Right)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert OR right operand: %v", err)
|
|
}
|
|
return &OrExpr{
|
|
Left: left,
|
|
Right: right,
|
|
}, nil
|
|
|
|
case *tree.Tuple:
|
|
// Tuple expression for IN clauses: (value1, value2, value3)
|
|
tupleValues := make(ValTuple, 0, len(e.Exprs))
|
|
for _, tupleExpr := range e.Exprs {
|
|
convertedExpr, err := p.convertExpr(tupleExpr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert tuple element: %v", err)
|
|
}
|
|
tupleValues = append(tupleValues, convertedExpr)
|
|
}
|
|
return tupleValues, nil
|
|
|
|
case *tree.CastExpr:
|
|
// Handle INTERVAL expressions: INTERVAL '1 hour'
|
|
// CockroachDB represents these as cast expressions
|
|
if p.isIntervalCast(e) {
|
|
// Extract the string value being cast to interval
|
|
if strVal, ok := e.Expr.(*tree.StrVal); ok {
|
|
return &IntervalExpr{
|
|
Value: string(strVal.RawString()),
|
|
}, nil
|
|
}
|
|
return nil, fmt.Errorf("invalid INTERVAL expression: expected string literal")
|
|
}
|
|
// For non-interval casts, just convert the inner expression
|
|
return p.convertExpr(e.Expr)
|
|
|
|
case *tree.RangeCond:
|
|
// Handle BETWEEN expressions: column BETWEEN value1 AND value2
|
|
seaweedBetween := &BetweenExpr{
|
|
Not: e.Not, // Handle NOT BETWEEN
|
|
}
|
|
|
|
// Convert the left operand (the expression being tested)
|
|
left, err := p.convertExpr(e.Left)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert BETWEEN left operand: %v", err)
|
|
}
|
|
seaweedBetween.Left = left
|
|
|
|
// Convert the FROM operand (lower bound)
|
|
from, err := p.convertExpr(e.From)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert BETWEEN from operand: %v", err)
|
|
}
|
|
seaweedBetween.From = from
|
|
|
|
// Convert the TO operand (upper bound)
|
|
to, err := p.convertExpr(e.To)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert BETWEEN to operand: %v", err)
|
|
}
|
|
seaweedBetween.To = to
|
|
|
|
return seaweedBetween, nil
|
|
|
|
case *tree.IsNullExpr:
|
|
// Handle IS NULL expressions: column IS NULL
|
|
expr, err := p.convertExpr(e.Expr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert IS NULL expression: %v", err)
|
|
}
|
|
|
|
return &IsNullExpr{
|
|
Expr: expr,
|
|
}, nil
|
|
|
|
case *tree.IsNotNullExpr:
|
|
// Handle IS NOT NULL expressions: column IS NOT NULL
|
|
expr, err := p.convertExpr(e.Expr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert IS NOT NULL expression: %v", err)
|
|
}
|
|
|
|
return &IsNotNullExpr{
|
|
Expr: expr,
|
|
}, nil
|
|
|
|
default:
|
|
return nil, fmt.Errorf("unsupported expression type: %T", e)
|
|
}
|
|
}
|
|
|
|
// convertFromExpr converts CockroachDB FROM expressions to SeaweedFS format
|
|
func (p *CockroachSQLParser) convertFromExpr(expr tree.TableExpr) (TableExpr, error) {
|
|
switch e := expr.(type) {
|
|
case *tree.TableName:
|
|
// Simple table name
|
|
tableName := TableName{
|
|
Name: stringValue(e.Table()),
|
|
}
|
|
|
|
// Extract database qualifier if present
|
|
|
|
if e.Schema() != "" {
|
|
tableName.Qualifier = stringValue(e.Schema())
|
|
}
|
|
|
|
return &AliasedTableExpr{
|
|
Expr: tableName,
|
|
}, nil
|
|
|
|
case *tree.AliasedTableExpr:
|
|
// Handle aliased table expressions (which is what CockroachDB uses for qualified names)
|
|
if tableName, ok := e.Expr.(*tree.TableName); ok {
|
|
seaweedTableName := TableName{
|
|
Name: stringValue(tableName.Table()),
|
|
}
|
|
|
|
// Extract database qualifier if present
|
|
if tableName.Schema() != "" {
|
|
seaweedTableName.Qualifier = stringValue(tableName.Schema())
|
|
}
|
|
|
|
return &AliasedTableExpr{
|
|
Expr: seaweedTableName,
|
|
}, nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("unsupported expression in AliasedTableExpr: %T", e.Expr)
|
|
|
|
default:
|
|
return nil, fmt.Errorf("unsupported table expression type: %T", e)
|
|
}
|
|
}
|
|
|
|
// isIntervalCast checks if a CastExpr is casting to an INTERVAL type
|
|
func (p *CockroachSQLParser) isIntervalCast(castExpr *tree.CastExpr) bool {
|
|
// Check if the target type is an interval type
|
|
// CockroachDB represents interval types in the Type field
|
|
// We need to check if it's an interval type by examining the type structure
|
|
if castExpr.Type != nil {
|
|
// Try to detect interval type by examining the AST structure
|
|
// Since we can't easily access the type string, we'll be more conservative
|
|
// and assume any cast expression on a string literal could be an interval
|
|
if _, ok := castExpr.Expr.(*tree.StrVal); ok {
|
|
// This is likely an INTERVAL expression since CockroachDB
|
|
// represents INTERVAL '1 hour' as casting a string to interval type
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|