-
-
Notifications
You must be signed in to change notification settings - Fork 13
/
analyzer_test.go
148 lines (143 loc) · 4.68 KB
/
analyzer_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
package zetasql_test
import (
"testing"
"github.com/goccy/go-zetasql"
ast "github.com/goccy/go-zetasql/resolved_ast"
"github.com/goccy/go-zetasql/types"
)
func TestAnalyzer(t *testing.T) {
const tableName = "z_table"
catalog := types.NewSimpleCatalog("z_catalog")
catalog.AddTable(
types.NewSimpleTable(tableName, []types.Column{
types.NewSimpleColumn(tableName, "col1", types.Int64Type()),
types.NewSimpleColumn(tableName, "col2", types.StringType()),
}),
)
catalog.AddZetaSQLBuiltinFunctions(nil)
langOpt := zetasql.NewLanguageOptions()
langOpt.SetNameResolutionMode(zetasql.NameResolutionDefault)
langOpt.SetProductMode(types.ProductExternal)
langOpt.SetEnabledLanguageFeatures([]zetasql.LanguageFeature{
zetasql.FeatureNamedArguments,
zetasql.FeatureNumericType,
zetasql.FeatureTablesample,
zetasql.FeatureTimestampNanos,
zetasql.FeatureV11HavingInAggregate,
zetasql.FeatureV11NullHandlingModifierInAggregate,
zetasql.FeatureV11OrderByCollate,
zetasql.FeatureV11SelectStarExceptReplace,
zetasql.FeatureV12SafeFunctionCall,
zetasql.FeatureJsonType,
zetasql.FeatureJsonArrayFunctions,
zetasql.FeatureJsonStrictNumberParsing,
})
langOpt.SetSupportedStatementKinds([]ast.Kind{
ast.QueryStmt, ast.InsertStmt, ast.UpdateStmt, ast.DeleteStmt,
})
opt := zetasql.NewAnalyzerOptions()
opt.SetAllowUndeclaredParameters(true)
opt.SetLanguage(langOpt)
opt.SetParseLocationRecordType(zetasql.ParseLocationRecordFullNodeScope)
out, err := zetasql.AnalyzeStatement("SELECT * FROM z_table WHERE col1 = 1000", catalog, opt)
if err != nil {
t.Fatal(err)
}
stmt := out.Statement()
if err := ast.Walk(stmt, func(n ast.Node) error {
locRange := n.ParseLocationRange()
if locRange != nil {
t.Logf("%T[%s] %s", n, locRange, n.DebugString())
} else {
t.Logf("%T %s", n, n.DebugString())
}
return nil
}); err != nil {
t.Fatal(err)
}
query := stmt.(*ast.QueryStmtNode)
outputColumns := query.OutputColumnList()
if len(outputColumns) != 2 {
t.Fatal("failed to get output column list")
}
col1 := outputColumns[0].Column()
col2 := outputColumns[1].Column()
if col1.Name() != "col1" || col2.Name() != "col2" {
t.Fatalf("failed to get column name")
}
if col1.Type().TypeName(0) != "INT64" || col2.Type().TypeName(0) != "STRING" {
t.Fatalf("failed to get column type")
}
project := query.Query().(*ast.ProjectScanNode)
filter := project.InputScan().(*ast.FilterScanNode)
if filter.InputScan().(*ast.TableScanNode).Table().Name() != "z_table" {
t.Fatal("failed to get table name")
}
funcCall := filter.FilterExpr().(*ast.FunctionCallNode)
fn := funcCall.Function()
if !fn.IsZetaSQLBuiltin() || fn.Name() != "$equal" {
t.Fatalf("failed to get function: %s", fn.Name())
}
fnArgs := funcCall.ArgumentList()
if len(fnArgs) != 2 {
t.Fatalf("failed to get function arguments: %d", len(fnArgs))
}
if fnArgs[0].(*ast.ColumnRefNode).Column().Name() != "col1" {
t.Fatal("failed to get function argument column name")
}
if fnArgs[1].(*ast.LiteralNode).Value().Int64Value() != 1000 {
t.Fatal("failed to get function argument value")
}
}
func TestAnalyzeMultiStatements(t *testing.T) {
const tableName = "table"
catalog := types.NewSimpleCatalog("catalog")
catalog.AddTable(
types.NewSimpleTable(tableName, []types.Column{
types.NewSimpleColumn(tableName, "col1", types.Int64Type()),
types.NewSimpleColumn(tableName, "col2", types.StringType()),
}),
)
langOpt := zetasql.NewLanguageOptions()
langOpt.SetNameResolutionMode(zetasql.NameResolutionDefault)
langOpt.SetProductMode(types.ProductExternal)
langOpt.EnableMaximumLanguageFeatures()
langOpt.SetSupportedStatementKinds([]ast.Kind{ast.CreateFunctionStmt, ast.QueryStmt})
catalog.AddZetaSQLBuiltinFunctions(langOpt.BuiltinFunctionOptions())
opt := zetasql.NewAnalyzerOptions()
opt.SetAllowUndeclaredParameters(true)
opt.SetLanguage(langOpt)
query := `
CREATE TEMP FUNCTION Add(x INT64, y INT64) AS (x + y);
SELECT Add(3, 4);
`
loc := zetasql.NewParseResumeLocation(query)
out, isEnd, err := zetasql.AnalyzeNextStatement(loc, catalog, opt)
if err != nil {
t.Fatal(err)
}
if isEnd {
t.Fatalf("failed to read multiple statement")
}
fnNode, ok := out.Statement().(*ast.CreateFunctionStmtNode)
if !ok {
t.Fatalf("failed to get create function statement node: %T", out.Statement())
}
fn := types.NewFunction(
[]string{"Add"},
"",
types.ScalarMode,
[]*types.FunctionSignature{fnNode.Signature()},
)
catalog.AddFunction(fn)
out, isEnd, err = zetasql.AnalyzeNextStatement(loc, catalog, opt)
if err != nil {
t.Fatal(err)
}
if !isEnd {
t.Fatalf("failed to get isEnd flag")
}
if _, ok := out.Statement().(*ast.QueryStmtNode); !ok {
t.Fatalf("failed to get query statement node: %T", out.Statement())
}
}