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.
3160 lines
58 KiB
3160 lines
58 KiB
7 years ago
|
/*
|
||
|
Copyright 2017 Google Inc.
|
||
|
|
||
|
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 sqlparser
|
||
|
|
||
|
func setParseTree(yylex interface{}, stmt Statement) {
|
||
|
yylex.(*Tokenizer).ParseTree = stmt
|
||
|
}
|
||
|
|
||
|
func setAllowComments(yylex interface{}, allow bool) {
|
||
|
yylex.(*Tokenizer).AllowComments = allow
|
||
|
}
|
||
|
|
||
|
func setDDL(yylex interface{}, ddl *DDL) {
|
||
|
yylex.(*Tokenizer).partialDDL = ddl
|
||
|
}
|
||
|
|
||
|
func incNesting(yylex interface{}) bool {
|
||
|
yylex.(*Tokenizer).nesting++
|
||
|
if yylex.(*Tokenizer).nesting == 200 {
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func decNesting(yylex interface{}) {
|
||
|
yylex.(*Tokenizer).nesting--
|
||
|
}
|
||
|
|
||
|
// forceEOF forces the lexer to end prematurely. Not all SQL statements
|
||
|
// are supported by the Parser, thus calling forceEOF will make the lexer
|
||
|
// return EOF early.
|
||
|
func forceEOF(yylex interface{}) {
|
||
|
yylex.(*Tokenizer).ForceEOF = true
|
||
|
}
|
||
|
|
||
|
%}
|
||
|
|
||
|
%union {
|
||
|
empty struct{}
|
||
|
statement Statement
|
||
|
selStmt SelectStatement
|
||
|
ddl *DDL
|
||
|
ins *Insert
|
||
|
byt byte
|
||
|
bytes []byte
|
||
|
bytes2 [][]byte
|
||
|
str string
|
||
|
strs []string
|
||
|
selectExprs SelectExprs
|
||
|
selectExpr SelectExpr
|
||
|
columns Columns
|
||
|
partitions Partitions
|
||
|
colName *ColName
|
||
|
tableExprs TableExprs
|
||
|
tableExpr TableExpr
|
||
|
joinCondition JoinCondition
|
||
|
tableName TableName
|
||
|
tableNames TableNames
|
||
|
indexHints *IndexHints
|
||
|
expr Expr
|
||
|
exprs Exprs
|
||
|
boolVal BoolVal
|
||
|
colTuple ColTuple
|
||
|
values Values
|
||
|
valTuple ValTuple
|
||
|
subquery *Subquery
|
||
|
whens []*When
|
||
|
when *When
|
||
|
orderBy OrderBy
|
||
|
order *Order
|
||
|
limit *Limit
|
||
|
updateExprs UpdateExprs
|
||
|
setExprs SetExprs
|
||
|
updateExpr *UpdateExpr
|
||
|
setExpr *SetExpr
|
||
|
colIdent ColIdent
|
||
|
tableIdent TableIdent
|
||
|
convertType *ConvertType
|
||
|
aliasedTableName *AliasedTableExpr
|
||
|
TableSpec *TableSpec
|
||
|
columnType ColumnType
|
||
|
colKeyOpt ColumnKeyOption
|
||
|
optVal *SQLVal
|
||
|
LengthScaleOption LengthScaleOption
|
||
|
columnDefinition *ColumnDefinition
|
||
|
indexDefinition *IndexDefinition
|
||
|
indexInfo *IndexInfo
|
||
|
indexOption *IndexOption
|
||
|
indexOptions []*IndexOption
|
||
|
indexColumn *IndexColumn
|
||
|
indexColumns []*IndexColumn
|
||
|
partDefs []*PartitionDefinition
|
||
|
partDef *PartitionDefinition
|
||
|
partSpec *PartitionSpec
|
||
|
vindexParam VindexParam
|
||
|
vindexParams []VindexParam
|
||
|
showFilter *ShowFilter
|
||
|
}
|
||
|
|
||
|
%token LEX_ERROR
|
||
|
%left <bytes> UNION
|
||
|
%token <bytes> SELECT STREAM INSERT UPDATE DELETE FROM WHERE GROUP HAVING ORDER BY LIMIT OFFSET FOR
|
||
|
%token <bytes> ALL DISTINCT AS EXISTS ASC DESC INTO DUPLICATE KEY DEFAULT SET LOCK KEYS
|
||
|
%token <bytes> VALUES LAST_INSERT_ID
|
||
|
%token <bytes> NEXT VALUE SHARE MODE
|
||
|
%token <bytes> SQL_NO_CACHE SQL_CACHE
|
||
|
%left <bytes> JOIN STRAIGHT_JOIN LEFT RIGHT INNER OUTER CROSS NATURAL USE FORCE
|
||
|
%left <bytes> ON USING
|
||
|
%token <empty> '(' ',' ')'
|
||
|
%token <bytes> ID HEX STRING INTEGRAL FLOAT HEXNUM VALUE_ARG LIST_ARG COMMENT COMMENT_KEYWORD BIT_LITERAL
|
||
|
%token <bytes> NULL TRUE FALSE
|
||
|
|
||
|
// Precedence dictated by mysql. But the vitess grammar is simplified.
|
||
|
// Some of these operators don't conflict in our situation. Nevertheless,
|
||
|
// it's better to have these listed in the correct order. Also, we don't
|
||
|
// support all operators yet.
|
||
|
%left <bytes> OR
|
||
|
%left <bytes> AND
|
||
|
%right <bytes> NOT '!'
|
||
|
%left <bytes> BETWEEN CASE WHEN THEN ELSE END
|
||
|
%left <bytes> '=' '<' '>' LE GE NE NULL_SAFE_EQUAL IS LIKE REGEXP IN
|
||
|
%left <bytes> '|'
|
||
|
%left <bytes> '&'
|
||
|
%left <bytes> SHIFT_LEFT SHIFT_RIGHT
|
||
|
%left <bytes> '+' '-'
|
||
|
%left <bytes> '*' '/' DIV '%' MOD
|
||
|
%left <bytes> '^'
|
||
|
%right <bytes> '~' UNARY
|
||
|
%left <bytes> COLLATE
|
||
|
%right <bytes> BINARY UNDERSCORE_BINARY
|
||
|
%right <bytes> INTERVAL
|
||
|
%nonassoc <bytes> '.'
|
||
|
|
||
|
// There is no need to define precedence for the JSON
|
||
|
// operators because the syntax is restricted enough that
|
||
|
// they don't cause conflicts.
|
||
|
%token <empty> JSON_EXTRACT_OP JSON_UNQUOTE_EXTRACT_OP
|
||
|
|
||
|
// DDL Tokens
|
||
|
%token <bytes> CREATE ALTER DROP RENAME ANALYZE ADD
|
||
|
%token <bytes> SCHEMA TABLE INDEX VIEW TO IGNORE IF UNIQUE PRIMARY COLUMN CONSTRAINT SPATIAL FULLTEXT FOREIGN KEY_BLOCK_SIZE
|
||
|
%token <bytes> SHOW DESCRIBE EXPLAIN DATE ESCAPE REPAIR OPTIMIZE TRUNCATE
|
||
|
%token <bytes> MAXVALUE PARTITION REORGANIZE LESS THAN PROCEDURE TRIGGER
|
||
|
%token <bytes> VINDEX VINDEXES
|
||
|
%token <bytes> STATUS VARIABLES
|
||
|
|
||
|
// Transaction Tokens
|
||
|
%token <bytes> BEGIN START TRANSACTION COMMIT ROLLBACK
|
||
|
|
||
|
// Type Tokens
|
||
|
%token <bytes> BIT TINYINT SMALLINT MEDIUMINT INT INTEGER BIGINT INTNUM
|
||
|
%token <bytes> REAL DOUBLE FLOAT_TYPE DECIMAL NUMERIC
|
||
|
%token <bytes> TIME TIMESTAMP DATETIME YEAR
|
||
|
%token <bytes> CHAR VARCHAR BOOL CHARACTER VARBINARY NCHAR
|
||
|
%token <bytes> TEXT TINYTEXT MEDIUMTEXT LONGTEXT
|
||
|
%token <bytes> BLOB TINYBLOB MEDIUMBLOB LONGBLOB JSON ENUM
|
||
|
%token <bytes> GEOMETRY POINT LINESTRING POLYGON GEOMETRYCOLLECTION MULTIPOINT MULTILINESTRING MULTIPOLYGON
|
||
|
|
||
|
// Type Modifiers
|
||
|
%token <bytes> NULLX AUTO_INCREMENT APPROXNUM SIGNED UNSIGNED ZEROFILL
|
||
|
|
||
|
// Supported SHOW tokens
|
||
|
%token <bytes> DATABASES TABLES VITESS_KEYSPACES VITESS_SHARDS VITESS_TABLETS VSCHEMA_TABLES EXTENDED FULL PROCESSLIST
|
||
|
|
||
|
// SET tokens
|
||
|
%token <bytes> NAMES CHARSET GLOBAL SESSION ISOLATION LEVEL READ WRITE ONLY REPEATABLE COMMITTED UNCOMMITTED SERIALIZABLE
|
||
|
|
||
|
// Functions
|
||
|
%token <bytes> CURRENT_TIMESTAMP DATABASE CURRENT_DATE
|
||
|
%token <bytes> CURRENT_TIME LOCALTIME LOCALTIMESTAMP
|
||
|
%token <bytes> UTC_DATE UTC_TIME UTC_TIMESTAMP
|
||
|
%token <bytes> REPLACE
|
||
|
%token <bytes> CONVERT CAST
|
||
|
%token <bytes> SUBSTR SUBSTRING
|
||
|
%token <bytes> GROUP_CONCAT SEPARATOR
|
||
|
|
||
|
// Match
|
||
|
%token <bytes> MATCH AGAINST BOOLEAN LANGUAGE WITH QUERY EXPANSION
|
||
|
|
||
|
// MySQL reserved words that are unused by this grammar will map to this token.
|
||
|
%token <bytes> UNUSED
|
||
|
|
||
|
%type <statement> command
|
||
|
%type <selStmt> select_statement base_select union_lhs union_rhs
|
||
|
%type <statement> stream_statement insert_statement update_statement delete_statement set_statement
|
||
|
%type <statement> create_statement alter_statement rename_statement drop_statement truncate_statement
|
||
|
%type <ddl> create_table_prefix
|
||
|
%type <statement> analyze_statement show_statement use_statement other_statement
|
||
|
%type <statement> begin_statement commit_statement rollback_statement
|
||
|
%type <bytes2> comment_opt comment_list
|
||
|
%type <str> union_op insert_or_replace
|
||
|
%type <str> distinct_opt straight_join_opt cache_opt match_option separator_opt
|
||
|
%type <expr> like_escape_opt
|
||
|
%type <selectExprs> select_expression_list select_expression_list_opt
|
||
|
%type <selectExpr> select_expression
|
||
|
%type <expr> expression
|
||
|
%type <tableExprs> from_opt table_references
|
||
|
%type <tableExpr> table_reference table_factor join_table
|
||
|
%type <joinCondition> join_condition join_condition_opt on_expression_opt
|
||
|
%type <tableNames> table_name_list
|
||
|
%type <str> inner_join outer_join straight_join natural_join
|
||
|
%type <tableName> table_name into_table_name
|
||
|
%type <aliasedTableName> aliased_table_name
|
||
|
%type <indexHints> index_hint_list
|
||
|
%type <expr> where_expression_opt
|
||
|
%type <expr> condition
|
||
|
%type <boolVal> boolean_value
|
||
|
%type <str> compare
|
||
|
%type <ins> insert_data
|
||
|
%type <expr> value value_expression num_val
|
||
|
%type <expr> function_call_keyword function_call_nonkeyword function_call_generic function_call_conflict
|
||
|
%type <str> is_suffix
|
||
|
%type <colTuple> col_tuple
|
||
|
%type <exprs> expression_list
|
||
|
%type <values> tuple_list
|
||
|
%type <valTuple> row_tuple tuple_or_empty
|
||
|
%type <expr> tuple_expression
|
||
|
%type <subquery> subquery
|
||
|
%type <colName> column_name
|
||
|
%type <whens> when_expression_list
|
||
|
%type <when> when_expression
|
||
|
%type <expr> expression_opt else_expression_opt
|
||
|
%type <exprs> group_by_opt
|
||
|
%type <expr> having_opt
|
||
|
%type <orderBy> order_by_opt order_list
|
||
|
%type <order> order
|
||
|
%type <str> asc_desc_opt
|
||
|
%type <limit> limit_opt
|
||
|
%type <str> lock_opt
|
||
|
%type <columns> ins_column_list column_list
|
||
|
%type <partitions> opt_partition_clause partition_list
|
||
|
%type <updateExprs> on_dup_opt
|
||
|
%type <updateExprs> update_list
|
||
|
%type <setExprs> set_list transaction_chars
|
||
|
%type <bytes> charset_or_character_set
|
||
|
%type <updateExpr> update_expression
|
||
|
%type <setExpr> set_expression transaction_char isolation_level
|
||
|
%type <bytes> for_from
|
||
|
%type <str> ignore_opt default_opt
|
||
|
%type <str> extended_opt full_opt from_database_opt tables_or_processlist
|
||
|
%type <showFilter> like_or_where_opt
|
||
|
%type <byt> exists_opt
|
||
|
%type <empty> not_exists_opt non_add_drop_or_rename_operation to_opt index_opt constraint_opt
|
||
|
%type <bytes> reserved_keyword non_reserved_keyword
|
||
|
%type <colIdent> sql_id reserved_sql_id col_alias as_ci_opt using_opt
|
||
|
%type <expr> charset_value
|
||
|
%type <tableIdent> table_id reserved_table_id table_alias as_opt_id
|
||
|
%type <empty> as_opt
|
||
|
%type <empty> force_eof ddl_force_eof
|
||
|
%type <str> charset
|
||
|
%type <str> set_session_or_global show_session_or_global
|
||
|
%type <convertType> convert_type
|
||
|
%type <columnType> column_type
|
||
|
%type <columnType> int_type decimal_type numeric_type time_type char_type spatial_type
|
||
|
%type <optVal> length_opt column_default_opt column_comment_opt on_update_opt
|
||
|
%type <str> charset_opt collate_opt
|
||
|
%type <boolVal> unsigned_opt zero_fill_opt
|
||
|
%type <LengthScaleOption> float_length_opt decimal_length_opt
|
||
|
%type <boolVal> null_opt auto_increment_opt
|
||
|
%type <colKeyOpt> column_key_opt
|
||
|
%type <strs> enum_values
|
||
|
%type <columnDefinition> column_definition
|
||
|
%type <indexDefinition> index_definition
|
||
|
%type <str> index_or_key
|
||
|
%type <str> equal_opt
|
||
|
%type <TableSpec> table_spec table_column_list
|
||
|
%type <str> table_option_list table_option table_opt_value
|
||
|
%type <indexInfo> index_info
|
||
|
%type <indexColumn> index_column
|
||
|
%type <indexColumns> index_column_list
|
||
|
%type <indexOption> index_option
|
||
|
%type <indexOptions> index_option_list
|
||
|
%type <partDefs> partition_definitions
|
||
|
%type <partDef> partition_definition
|
||
|
%type <partSpec> partition_operation
|
||
|
%type <vindexParam> vindex_param
|
||
|
%type <vindexParams> vindex_param_list vindex_params_opt
|
||
|
%type <colIdent> vindex_type vindex_type_opt
|
||
|
%type <bytes> alter_object_type
|
||
|
|
||
|
%start any_command
|
||
|
|
||
|
%%
|
||
|
|
||
|
any_command:
|
||
|
command semicolon_opt
|
||
|
{
|
||
|
setParseTree(yylex, $1)
|
||
|
}
|
||
|
|
||
|
semicolon_opt:
|
||
|
/*empty*/ {}
|
||
|
| ';' {}
|
||
|
|
||
|
command:
|
||
|
select_statement
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| stream_statement
|
||
|
| insert_statement
|
||
|
| update_statement
|
||
|
| delete_statement
|
||
|
| set_statement
|
||
|
| create_statement
|
||
|
| alter_statement
|
||
|
| rename_statement
|
||
|
| drop_statement
|
||
|
| truncate_statement
|
||
|
| analyze_statement
|
||
|
| show_statement
|
||
|
| use_statement
|
||
|
| begin_statement
|
||
|
| commit_statement
|
||
|
| rollback_statement
|
||
|
| other_statement
|
||
|
|
||
|
select_statement:
|
||
|
base_select order_by_opt limit_opt lock_opt
|
||
|
{
|
||
|
sel := $1.(*Select)
|
||
|
sel.OrderBy = $2
|
||
|
sel.Limit = $3
|
||
|
sel.Lock = $4
|
||
|
$$ = sel
|
||
|
}
|
||
|
| union_lhs union_op union_rhs order_by_opt limit_opt lock_opt
|
||
|
{
|
||
|
$$ = &Union{Type: $2, Left: $1, Right: $3, OrderBy: $4, Limit: $5, Lock: $6}
|
||
|
}
|
||
|
| SELECT comment_opt cache_opt NEXT num_val for_from table_name
|
||
|
{
|
||
|
$$ = &Select{Comments: Comments($2), Cache: $3, SelectExprs: SelectExprs{Nextval{Expr: $5}}, From: TableExprs{&AliasedTableExpr{Expr: $7}}}
|
||
|
}
|
||
|
|
||
|
stream_statement:
|
||
|
STREAM comment_opt select_expression FROM table_name
|
||
|
{
|
||
|
$$ = &Stream{Comments: Comments($2), SelectExpr: $3, Table: $5}
|
||
|
}
|
||
|
|
||
|
// base_select is an unparenthesized SELECT with no order by clause or beyond.
|
||
|
base_select:
|
||
|
SELECT comment_opt cache_opt distinct_opt straight_join_opt select_expression_list from_opt where_expression_opt group_by_opt having_opt
|
||
|
{
|
||
|
$$ = &Select{Comments: Comments($2), Cache: $3, Distinct: $4, Hints: $5, SelectExprs: $6, From: $7, Where: NewWhere(WhereStr, $8), GroupBy: GroupBy($9), Having: NewWhere(HavingStr, $10)}
|
||
|
}
|
||
|
|
||
|
union_lhs:
|
||
|
select_statement
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| openb select_statement closeb
|
||
|
{
|
||
|
$$ = &ParenSelect{Select: $2}
|
||
|
}
|
||
|
|
||
|
union_rhs:
|
||
|
base_select
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| openb select_statement closeb
|
||
|
{
|
||
|
$$ = &ParenSelect{Select: $2}
|
||
|
}
|
||
|
|
||
|
|
||
|
insert_statement:
|
||
|
insert_or_replace comment_opt ignore_opt into_table_name opt_partition_clause insert_data on_dup_opt
|
||
|
{
|
||
|
// insert_data returns a *Insert pre-filled with Columns & Values
|
||
|
ins := $6
|
||
|
ins.Action = $1
|
||
|
ins.Comments = $2
|
||
|
ins.Ignore = $3
|
||
|
ins.Table = $4
|
||
|
ins.Partitions = $5
|
||
|
ins.OnDup = OnDup($7)
|
||
|
$$ = ins
|
||
|
}
|
||
|
| insert_or_replace comment_opt ignore_opt into_table_name opt_partition_clause SET update_list on_dup_opt
|
||
|
{
|
||
|
cols := make(Columns, 0, len($7))
|
||
|
vals := make(ValTuple, 0, len($8))
|
||
|
for _, updateList := range $7 {
|
||
|
cols = append(cols, updateList.Name.Name)
|
||
|
vals = append(vals, updateList.Expr)
|
||
|
}
|
||
|
$$ = &Insert{Action: $1, Comments: Comments($2), Ignore: $3, Table: $4, Partitions: $5, Columns: cols, Rows: Values{vals}, OnDup: OnDup($8)}
|
||
|
}
|
||
|
|
||
|
insert_or_replace:
|
||
|
INSERT
|
||
|
{
|
||
|
$$ = InsertStr
|
||
|
}
|
||
|
| REPLACE
|
||
|
{
|
||
|
$$ = ReplaceStr
|
||
|
}
|
||
|
|
||
|
update_statement:
|
||
|
UPDATE comment_opt table_references SET update_list where_expression_opt order_by_opt limit_opt
|
||
|
{
|
||
|
$$ = &Update{Comments: Comments($2), TableExprs: $3, Exprs: $5, Where: NewWhere(WhereStr, $6), OrderBy: $7, Limit: $8}
|
||
|
}
|
||
|
|
||
|
delete_statement:
|
||
|
DELETE comment_opt FROM table_name opt_partition_clause where_expression_opt order_by_opt limit_opt
|
||
|
{
|
||
|
$$ = &Delete{Comments: Comments($2), TableExprs: TableExprs{&AliasedTableExpr{Expr:$4}}, Partitions: $5, Where: NewWhere(WhereStr, $6), OrderBy: $7, Limit: $8}
|
||
|
}
|
||
|
| DELETE comment_opt FROM table_name_list USING table_references where_expression_opt
|
||
|
{
|
||
|
$$ = &Delete{Comments: Comments($2), Targets: $4, TableExprs: $6, Where: NewWhere(WhereStr, $7)}
|
||
|
}
|
||
|
| DELETE comment_opt table_name_list from_or_using table_references where_expression_opt
|
||
|
{
|
||
|
$$ = &Delete{Comments: Comments($2), Targets: $3, TableExprs: $5, Where: NewWhere(WhereStr, $6)}
|
||
|
}
|
||
|
|
||
|
from_or_using:
|
||
|
FROM {}
|
||
|
| USING {}
|
||
|
|
||
|
table_name_list:
|
||
|
table_name
|
||
|
{
|
||
|
$$ = TableNames{$1}
|
||
|
}
|
||
|
| table_name_list ',' table_name
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
opt_partition_clause:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| PARTITION openb partition_list closeb
|
||
|
{
|
||
|
$$ = $3
|
||
|
}
|
||
|
|
||
|
set_statement:
|
||
|
SET comment_opt set_list
|
||
|
{
|
||
|
$$ = &Set{Comments: Comments($2), Exprs: $3}
|
||
|
}
|
||
|
| SET comment_opt set_session_or_global set_list
|
||
|
{
|
||
|
$$ = &Set{Comments: Comments($2), Scope: $3, Exprs: $4}
|
||
|
}
|
||
|
| SET comment_opt set_session_or_global TRANSACTION transaction_chars
|
||
|
{
|
||
|
$$ = &Set{Comments: Comments($2), Scope: $3, Exprs: $5}
|
||
|
}
|
||
|
| SET comment_opt TRANSACTION transaction_chars
|
||
|
{
|
||
|
$$ = &Set{Comments: Comments($2), Exprs: $4}
|
||
|
}
|
||
|
|
||
|
transaction_chars:
|
||
|
transaction_char
|
||
|
{
|
||
|
$$ = SetExprs{$1}
|
||
|
}
|
||
|
| transaction_chars ',' transaction_char
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
transaction_char:
|
||
|
ISOLATION LEVEL isolation_level
|
||
|
{
|
||
|
$$ = $3
|
||
|
}
|
||
|
| READ WRITE
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_read_only"), Expr: NewIntVal([]byte("0"))}
|
||
|
}
|
||
|
| READ ONLY
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_read_only"), Expr: NewIntVal([]byte("1"))}
|
||
|
}
|
||
|
|
||
|
isolation_level:
|
||
|
REPEATABLE READ
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_isolation"), Expr: NewStrVal([]byte("repeatable read"))}
|
||
|
}
|
||
|
| READ COMMITTED
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_isolation"), Expr: NewStrVal([]byte("read committed"))}
|
||
|
}
|
||
|
| READ UNCOMMITTED
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_isolation"), Expr: NewStrVal([]byte("read uncommitted"))}
|
||
|
}
|
||
|
| SERIALIZABLE
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent("tx_isolation"), Expr: NewStrVal([]byte("serializable"))}
|
||
|
}
|
||
|
|
||
|
set_session_or_global:
|
||
|
SESSION
|
||
|
{
|
||
|
$$ = SessionStr
|
||
|
}
|
||
|
| GLOBAL
|
||
|
{
|
||
|
$$ = GlobalStr
|
||
|
}
|
||
|
|
||
|
create_statement:
|
||
|
create_table_prefix table_spec
|
||
|
{
|
||
|
$1.TableSpec = $2
|
||
|
$$ = $1
|
||
|
}
|
||
|
| CREATE constraint_opt INDEX ID using_opt ON table_name ddl_force_eof
|
||
|
{
|
||
|
// Change this to an alter statement
|
||
|
$$ = &DDL{Action: AlterStr, Table: $7, NewName:$7}
|
||
|
}
|
||
|
| CREATE VIEW table_name ddl_force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: CreateStr, NewName: $3.ToViewName()}
|
||
|
}
|
||
|
| CREATE OR REPLACE VIEW table_name ddl_force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: CreateStr, NewName: $5.ToViewName()}
|
||
|
}
|
||
|
| CREATE VINDEX sql_id vindex_type_opt vindex_params_opt
|
||
|
{
|
||
|
$$ = &DDL{Action: CreateVindexStr, VindexSpec: &VindexSpec{
|
||
|
Name: $3,
|
||
|
Type: $4,
|
||
|
Params: $5,
|
||
|
}}
|
||
|
}
|
||
|
| CREATE DATABASE not_exists_opt ID ddl_force_eof
|
||
|
{
|
||
|
$$ = &DBDDL{Action: CreateStr, DBName: string($4)}
|
||
|
}
|
||
|
| CREATE SCHEMA not_exists_opt ID ddl_force_eof
|
||
|
{
|
||
|
$$ = &DBDDL{Action: CreateStr, DBName: string($4)}
|
||
|
}
|
||
|
|
||
|
vindex_type_opt:
|
||
|
{
|
||
|
$$ = NewColIdent("")
|
||
|
}
|
||
|
| USING vindex_type
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
vindex_type:
|
||
|
ID
|
||
|
{
|
||
|
$$ = NewColIdent(string($1))
|
||
|
}
|
||
|
|
||
|
vindex_params_opt:
|
||
|
{
|
||
|
var v []VindexParam
|
||
|
$$ = v
|
||
|
}
|
||
|
| WITH vindex_param_list
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
vindex_param_list:
|
||
|
vindex_param
|
||
|
{
|
||
|
$$ = make([]VindexParam, 0, 4)
|
||
|
$$ = append($$, $1)
|
||
|
}
|
||
|
| vindex_param_list ',' vindex_param
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
vindex_param:
|
||
|
reserved_sql_id '=' table_opt_value
|
||
|
{
|
||
|
$$ = VindexParam{Key: $1, Val: $3}
|
||
|
}
|
||
|
|
||
|
create_table_prefix:
|
||
|
CREATE TABLE not_exists_opt table_name
|
||
|
{
|
||
|
$$ = &DDL{Action: CreateStr, NewName: $4}
|
||
|
setDDL(yylex, $$)
|
||
|
}
|
||
|
|
||
|
table_spec:
|
||
|
'(' table_column_list ')' table_option_list
|
||
|
{
|
||
|
$$ = $2
|
||
|
$$.Options = $4
|
||
|
}
|
||
|
|
||
|
table_column_list:
|
||
|
column_definition
|
||
|
{
|
||
|
$$ = &TableSpec{}
|
||
|
$$.AddColumn($1)
|
||
|
}
|
||
|
| table_column_list ',' column_definition
|
||
|
{
|
||
|
$$.AddColumn($3)
|
||
|
}
|
||
|
| table_column_list ',' index_definition
|
||
|
{
|
||
|
$$.AddIndex($3)
|
||
|
}
|
||
|
|
||
|
column_definition:
|
||
|
ID column_type null_opt column_default_opt on_update_opt auto_increment_opt column_key_opt column_comment_opt
|
||
|
{
|
||
|
$2.NotNull = $3
|
||
|
$2.Default = $4
|
||
|
$2.OnUpdate = $5
|
||
|
$2.Autoincrement = $6
|
||
|
$2.KeyOpt = $7
|
||
|
$2.Comment = $8
|
||
|
$$ = &ColumnDefinition{Name: NewColIdent(string($1)), Type: $2}
|
||
|
}
|
||
|
column_type:
|
||
|
numeric_type unsigned_opt zero_fill_opt
|
||
|
{
|
||
|
$$ = $1
|
||
|
$$.Unsigned = $2
|
||
|
$$.Zerofill = $3
|
||
|
}
|
||
|
| char_type
|
||
|
| time_type
|
||
|
| spatial_type
|
||
|
|
||
|
numeric_type:
|
||
|
int_type length_opt
|
||
|
{
|
||
|
$$ = $1
|
||
|
$$.Length = $2
|
||
|
}
|
||
|
| decimal_type
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
|
||
|
int_type:
|
||
|
BIT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| TINYINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| SMALLINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| MEDIUMINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| INT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| INTEGER
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| BIGINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
|
||
|
decimal_type:
|
||
|
REAL float_length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
| DOUBLE float_length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
| FLOAT_TYPE float_length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
| DECIMAL decimal_length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
| NUMERIC decimal_length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
|
||
|
time_type:
|
||
|
DATE
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| TIME length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| TIMESTAMP length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| DATETIME length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| YEAR
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
|
||
|
char_type:
|
||
|
CHAR length_opt charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2, Charset: $3, Collate: $4}
|
||
|
}
|
||
|
| VARCHAR length_opt charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2, Charset: $3, Collate: $4}
|
||
|
}
|
||
|
| BINARY length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| VARBINARY length_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| TEXT charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Charset: $2, Collate: $3}
|
||
|
}
|
||
|
| TINYTEXT charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Charset: $2, Collate: $3}
|
||
|
}
|
||
|
| MEDIUMTEXT charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Charset: $2, Collate: $3}
|
||
|
}
|
||
|
| LONGTEXT charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), Charset: $2, Collate: $3}
|
||
|
}
|
||
|
| BLOB
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| TINYBLOB
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| MEDIUMBLOB
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| LONGBLOB
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| JSON
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| ENUM '(' enum_values ')' charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), EnumValues: $3, Charset: $5, Collate: $6}
|
||
|
}
|
||
|
// need set_values / SetValues ?
|
||
|
| SET '(' enum_values ')' charset_opt collate_opt
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1), EnumValues: $3, Charset: $5, Collate: $6}
|
||
|
}
|
||
|
|
||
|
spatial_type:
|
||
|
GEOMETRY
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| POINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| LINESTRING
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| POLYGON
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| GEOMETRYCOLLECTION
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| MULTIPOINT
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| MULTILINESTRING
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
| MULTIPOLYGON
|
||
|
{
|
||
|
$$ = ColumnType{Type: string($1)}
|
||
|
}
|
||
|
|
||
|
enum_values:
|
||
|
STRING
|
||
|
{
|
||
|
$$ = make([]string, 0, 4)
|
||
|
$$ = append($$, "'" + string($1) + "'")
|
||
|
}
|
||
|
| enum_values ',' STRING
|
||
|
{
|
||
|
$$ = append($1, "'" + string($3) + "'")
|
||
|
}
|
||
|
|
||
|
length_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| '(' INTEGRAL ')'
|
||
|
{
|
||
|
$$ = NewIntVal($2)
|
||
|
}
|
||
|
|
||
|
float_length_opt:
|
||
|
{
|
||
|
$$ = LengthScaleOption{}
|
||
|
}
|
||
|
| '(' INTEGRAL ',' INTEGRAL ')'
|
||
|
{
|
||
|
$$ = LengthScaleOption{
|
||
|
Length: NewIntVal($2),
|
||
|
Scale: NewIntVal($4),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
decimal_length_opt:
|
||
|
{
|
||
|
$$ = LengthScaleOption{}
|
||
|
}
|
||
|
| '(' INTEGRAL ')'
|
||
|
{
|
||
|
$$ = LengthScaleOption{
|
||
|
Length: NewIntVal($2),
|
||
|
}
|
||
|
}
|
||
|
| '(' INTEGRAL ',' INTEGRAL ')'
|
||
|
{
|
||
|
$$ = LengthScaleOption{
|
||
|
Length: NewIntVal($2),
|
||
|
Scale: NewIntVal($4),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
unsigned_opt:
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
| UNSIGNED
|
||
|
{
|
||
|
$$ = BoolVal(true)
|
||
|
}
|
||
|
|
||
|
zero_fill_opt:
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
| ZEROFILL
|
||
|
{
|
||
|
$$ = BoolVal(true)
|
||
|
}
|
||
|
|
||
|
// Null opt returns false to mean NULL (i.e. the default) and true for NOT NULL
|
||
|
null_opt:
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
| NULL
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
| NOT NULL
|
||
|
{
|
||
|
$$ = BoolVal(true)
|
||
|
}
|
||
|
|
||
|
column_default_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| DEFAULT STRING
|
||
|
{
|
||
|
$$ = NewStrVal($2)
|
||
|
}
|
||
|
| DEFAULT INTEGRAL
|
||
|
{
|
||
|
$$ = NewIntVal($2)
|
||
|
}
|
||
|
| DEFAULT FLOAT
|
||
|
{
|
||
|
$$ = NewFloatVal($2)
|
||
|
}
|
||
|
| DEFAULT NULL
|
||
|
{
|
||
|
$$ = NewValArg($2)
|
||
|
}
|
||
|
| DEFAULT CURRENT_TIMESTAMP
|
||
|
{
|
||
|
$$ = NewValArg($2)
|
||
|
}
|
||
|
| DEFAULT BIT_LITERAL
|
||
|
{
|
||
|
$$ = NewBitVal($2)
|
||
|
}
|
||
|
|
||
|
on_update_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| ON UPDATE CURRENT_TIMESTAMP
|
||
|
{
|
||
|
$$ = NewValArg($3)
|
||
|
}
|
||
|
|
||
|
auto_increment_opt:
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
| AUTO_INCREMENT
|
||
|
{
|
||
|
$$ = BoolVal(true)
|
||
|
}
|
||
|
|
||
|
charset_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| CHARACTER SET ID
|
||
|
{
|
||
|
$$ = string($3)
|
||
|
}
|
||
|
| CHARACTER SET BINARY
|
||
|
{
|
||
|
$$ = string($3)
|
||
|
}
|
||
|
|
||
|
collate_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| COLLATE ID
|
||
|
{
|
||
|
$$ = string($2)
|
||
|
}
|
||
|
|
||
|
column_key_opt:
|
||
|
{
|
||
|
$$ = colKeyNone
|
||
|
}
|
||
|
| PRIMARY KEY
|
||
|
{
|
||
|
$$ = colKeyPrimary
|
||
|
}
|
||
|
| KEY
|
||
|
{
|
||
|
$$ = colKey
|
||
|
}
|
||
|
| UNIQUE KEY
|
||
|
{
|
||
|
$$ = colKeyUniqueKey
|
||
|
}
|
||
|
| UNIQUE
|
||
|
{
|
||
|
$$ = colKeyUnique
|
||
|
}
|
||
|
|
||
|
column_comment_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| COMMENT_KEYWORD STRING
|
||
|
{
|
||
|
$$ = NewStrVal($2)
|
||
|
}
|
||
|
|
||
|
index_definition:
|
||
|
index_info '(' index_column_list ')' index_option_list
|
||
|
{
|
||
|
$$ = &IndexDefinition{Info: $1, Columns: $3, Options: $5}
|
||
|
}
|
||
|
| index_info '(' index_column_list ')'
|
||
|
{
|
||
|
$$ = &IndexDefinition{Info: $1, Columns: $3}
|
||
|
}
|
||
|
|
||
|
index_option_list:
|
||
|
index_option
|
||
|
{
|
||
|
$$ = []*IndexOption{$1}
|
||
|
}
|
||
|
| index_option_list index_option
|
||
|
{
|
||
|
$$ = append($$, $2)
|
||
|
}
|
||
|
|
||
|
index_option:
|
||
|
USING ID
|
||
|
{
|
||
|
$$ = &IndexOption{Name: string($1), Using: string($2)}
|
||
|
}
|
||
|
| KEY_BLOCK_SIZE equal_opt INTEGRAL
|
||
|
{
|
||
|
// should not be string
|
||
|
$$ = &IndexOption{Name: string($1), Value: NewIntVal($3)}
|
||
|
}
|
||
|
| COMMENT_KEYWORD STRING
|
||
|
{
|
||
|
$$ = &IndexOption{Name: string($1), Value: NewStrVal($2)}
|
||
|
}
|
||
|
|
||
|
equal_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| '='
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
|
||
|
index_info:
|
||
|
PRIMARY KEY
|
||
|
{
|
||
|
$$ = &IndexInfo{Type: string($1) + " " + string($2), Name: NewColIdent("PRIMARY"), Primary: true, Unique: true}
|
||
|
}
|
||
|
| SPATIAL index_or_key ID
|
||
|
{
|
||
|
$$ = &IndexInfo{Type: string($1) + " " + string($2), Name: NewColIdent(string($3)), Spatial: true, Unique: false}
|
||
|
}
|
||
|
| UNIQUE index_or_key ID
|
||
|
{
|
||
|
$$ = &IndexInfo{Type: string($1) + " " + string($2), Name: NewColIdent(string($3)), Unique: true}
|
||
|
}
|
||
|
| UNIQUE ID
|
||
|
{
|
||
|
$$ = &IndexInfo{Type: string($1), Name: NewColIdent(string($2)), Unique: true}
|
||
|
}
|
||
|
| index_or_key ID
|
||
|
{
|
||
|
$$ = &IndexInfo{Type: string($1), Name: NewColIdent(string($2)), Unique: false}
|
||
|
}
|
||
|
|
||
|
index_or_key:
|
||
|
INDEX
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
| KEY
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
|
||
|
index_column_list:
|
||
|
index_column
|
||
|
{
|
||
|
$$ = []*IndexColumn{$1}
|
||
|
}
|
||
|
| index_column_list ',' index_column
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
index_column:
|
||
|
sql_id length_opt
|
||
|
{
|
||
|
$$ = &IndexColumn{Column: $1, Length: $2}
|
||
|
}
|
||
|
|
||
|
table_option_list:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| table_option
|
||
|
{
|
||
|
$$ = " " + string($1)
|
||
|
}
|
||
|
| table_option_list ',' table_option
|
||
|
{
|
||
|
$$ = string($1) + ", " + string($3)
|
||
|
}
|
||
|
|
||
|
// rather than explicitly parsing the various keywords for table options,
|
||
|
// just accept any number of keywords, IDs, strings, numbers, and '='
|
||
|
table_option:
|
||
|
table_opt_value
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| table_option table_opt_value
|
||
|
{
|
||
|
$$ = $1 + " " + $2
|
||
|
}
|
||
|
| table_option '=' table_opt_value
|
||
|
{
|
||
|
$$ = $1 + "=" + $3
|
||
|
}
|
||
|
|
||
|
table_opt_value:
|
||
|
reserved_sql_id
|
||
|
{
|
||
|
$$ = $1.String()
|
||
|
}
|
||
|
| STRING
|
||
|
{
|
||
|
$$ = "'" + string($1) + "'"
|
||
|
}
|
||
|
| INTEGRAL
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
|
||
|
alter_statement:
|
||
|
ALTER ignore_opt TABLE table_name non_add_drop_or_rename_operation force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $4, NewName: $4}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name ADD alter_object_type force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $4, NewName: $4}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name DROP alter_object_type force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $4, NewName: $4}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name ADD VINDEX sql_id '(' column_list ')' vindex_type_opt vindex_params_opt
|
||
|
{
|
||
|
$$ = &DDL{
|
||
|
Action: AddColVindexStr,
|
||
|
Table: $4,
|
||
|
VindexSpec: &VindexSpec{
|
||
|
Name: $7,
|
||
|
Type: $11,
|
||
|
Params: $12,
|
||
|
},
|
||
|
VindexCols: $9,
|
||
|
}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name DROP VINDEX sql_id
|
||
|
{
|
||
|
$$ = &DDL{
|
||
|
Action: DropColVindexStr,
|
||
|
Table: $4,
|
||
|
VindexSpec: &VindexSpec{
|
||
|
Name: $7,
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name RENAME to_opt table_name
|
||
|
{
|
||
|
// Change this to a rename statement
|
||
|
$$ = &DDL{Action: RenameStr, Table: $4, NewName: $7}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name RENAME index_opt force_eof
|
||
|
{
|
||
|
// Rename an index can just be an alter
|
||
|
$$ = &DDL{Action: AlterStr, Table: $4, NewName: $4}
|
||
|
}
|
||
|
| ALTER VIEW table_name ddl_force_eof
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $3.ToViewName(), NewName: $3.ToViewName()}
|
||
|
}
|
||
|
| ALTER ignore_opt TABLE table_name partition_operation
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $4, PartitionSpec: $5}
|
||
|
}
|
||
|
|
||
|
alter_object_type:
|
||
|
COLUMN
|
||
|
| CONSTRAINT
|
||
|
| FOREIGN
|
||
|
| FULLTEXT
|
||
|
| ID
|
||
|
| INDEX
|
||
|
| KEY
|
||
|
| PRIMARY
|
||
|
| SPATIAL
|
||
|
| PARTITION
|
||
|
| UNIQUE
|
||
|
|
||
|
partition_operation:
|
||
|
REORGANIZE PARTITION sql_id INTO openb partition_definitions closeb
|
||
|
{
|
||
|
$$ = &PartitionSpec{Action: ReorganizeStr, Name: $3, Definitions: $6}
|
||
|
}
|
||
|
|
||
|
partition_definitions:
|
||
|
partition_definition
|
||
|
{
|
||
|
$$ = []*PartitionDefinition{$1}
|
||
|
}
|
||
|
| partition_definitions ',' partition_definition
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
partition_definition:
|
||
|
PARTITION sql_id VALUES LESS THAN openb value_expression closeb
|
||
|
{
|
||
|
$$ = &PartitionDefinition{Name: $2, Limit: $7}
|
||
|
}
|
||
|
| PARTITION sql_id VALUES LESS THAN openb MAXVALUE closeb
|
||
|
{
|
||
|
$$ = &PartitionDefinition{Name: $2, Maxvalue: true}
|
||
|
}
|
||
|
|
||
|
rename_statement:
|
||
|
RENAME TABLE table_name TO table_name
|
||
|
{
|
||
|
$$ = &DDL{Action: RenameStr, Table: $3, NewName: $5}
|
||
|
}
|
||
|
|
||
|
drop_statement:
|
||
|
DROP TABLE exists_opt table_name
|
||
|
{
|
||
|
var exists bool
|
||
|
if $3 != 0 {
|
||
|
exists = true
|
||
|
}
|
||
|
$$ = &DDL{Action: DropStr, Table: $4, IfExists: exists}
|
||
|
}
|
||
|
| DROP INDEX ID ON table_name ddl_force_eof
|
||
|
{
|
||
|
// Change this to an alter statement
|
||
|
$$ = &DDL{Action: AlterStr, Table: $5, NewName: $5}
|
||
|
}
|
||
|
| DROP VIEW exists_opt table_name ddl_force_eof
|
||
|
{
|
||
|
var exists bool
|
||
|
if $3 != 0 {
|
||
|
exists = true
|
||
|
}
|
||
|
$$ = &DDL{Action: DropStr, Table: $4.ToViewName(), IfExists: exists}
|
||
|
}
|
||
|
| DROP DATABASE exists_opt ID
|
||
|
{
|
||
|
$$ = &DBDDL{Action: DropStr, DBName: string($4)}
|
||
|
}
|
||
|
| DROP SCHEMA exists_opt ID
|
||
|
{
|
||
|
$$ = &DBDDL{Action: DropStr, DBName: string($4)}
|
||
|
}
|
||
|
|
||
|
truncate_statement:
|
||
|
TRUNCATE TABLE table_name
|
||
|
{
|
||
|
$$ = &DDL{Action: TruncateStr, Table: $3}
|
||
|
}
|
||
|
| TRUNCATE table_name
|
||
|
{
|
||
|
$$ = &DDL{Action: TruncateStr, Table: $2}
|
||
|
}
|
||
|
analyze_statement:
|
||
|
ANALYZE TABLE table_name
|
||
|
{
|
||
|
$$ = &DDL{Action: AlterStr, Table: $3, NewName: $3}
|
||
|
}
|
||
|
|
||
|
show_statement:
|
||
|
SHOW BINARY ID ddl_force_eof /* SHOW BINARY LOGS */
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CHARACTER SET ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CREATE DATABASE ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
/* Rule to handle SHOW CREATE EVENT, SHOW CREATE FUNCTION, etc. */
|
||
|
| SHOW CREATE ID ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CREATE PROCEDURE ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CREATE TABLE ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CREATE TRIGGER ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW CREATE VIEW ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2) + " " + string($3)}
|
||
|
}
|
||
|
| SHOW DATABASES ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW INDEX ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW KEYS ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW PROCEDURE ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW show_session_or_global STATUS ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Scope: $2, Type: string($3)}
|
||
|
}
|
||
|
| SHOW TABLE ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW extended_opt full_opt tables_or_processlist from_database_opt like_or_where_opt
|
||
|
{
|
||
|
// this is ugly, but I couldn't find a better way for now
|
||
|
if $4 == "processlist" {
|
||
|
$$ = &Show{Type: $4}
|
||
|
} else {
|
||
|
showTablesOpt := &ShowTablesOpt{Extended: $2, Full:$3, DbName:$5, Filter:$6}
|
||
|
$$ = &Show{Type: $4, ShowTablesOpt: showTablesOpt}
|
||
|
}
|
||
|
}
|
||
|
| SHOW show_session_or_global VARIABLES ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Scope: $2, Type: string($3)}
|
||
|
}
|
||
|
| SHOW VINDEXES
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW VINDEXES ON table_name
|
||
|
{
|
||
|
$$ = &Show{Type: string($2), OnTable: $4}
|
||
|
}
|
||
|
| SHOW VITESS_KEYSPACES
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW VITESS_SHARDS
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW VITESS_TABLETS
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
| SHOW VSCHEMA_TABLES
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
/*
|
||
|
* Catch-all for show statements without vitess keywords:
|
||
|
*
|
||
|
* SHOW BINARY LOGS
|
||
|
* SHOW INVALID
|
||
|
*/
|
||
|
| SHOW ID ddl_force_eof
|
||
|
{
|
||
|
$$ = &Show{Type: string($2)}
|
||
|
}
|
||
|
|
||
|
tables_or_processlist:
|
||
|
TABLES
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
| PROCESSLIST
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
|
||
|
extended_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| EXTENDED
|
||
|
{
|
||
|
$$ = "extended "
|
||
|
}
|
||
|
|
||
|
full_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| FULL
|
||
|
{
|
||
|
$$ = "full "
|
||
|
}
|
||
|
|
||
|
from_database_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| FROM table_id
|
||
|
{
|
||
|
$$ = $2.v
|
||
|
}
|
||
|
| IN table_id
|
||
|
{
|
||
|
$$ = $2.v
|
||
|
}
|
||
|
|
||
|
like_or_where_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| LIKE STRING
|
||
|
{
|
||
|
$$ = &ShowFilter{Like:string($2)}
|
||
|
}
|
||
|
| WHERE expression
|
||
|
{
|
||
|
$$ = &ShowFilter{Filter:$2}
|
||
|
}
|
||
|
|
||
|
show_session_or_global:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| SESSION
|
||
|
{
|
||
|
$$ = SessionStr
|
||
|
}
|
||
|
| GLOBAL
|
||
|
{
|
||
|
$$ = GlobalStr
|
||
|
}
|
||
|
|
||
|
use_statement:
|
||
|
USE table_id
|
||
|
{
|
||
|
$$ = &Use{DBName: $2}
|
||
|
}
|
||
|
| USE
|
||
|
{
|
||
|
$$ = &Use{DBName:TableIdent{v:""}}
|
||
|
}
|
||
|
|
||
|
begin_statement:
|
||
|
BEGIN
|
||
|
{
|
||
|
$$ = &Begin{}
|
||
|
}
|
||
|
| START TRANSACTION
|
||
|
{
|
||
|
$$ = &Begin{}
|
||
|
}
|
||
|
|
||
|
commit_statement:
|
||
|
COMMIT
|
||
|
{
|
||
|
$$ = &Commit{}
|
||
|
}
|
||
|
|
||
|
rollback_statement:
|
||
|
ROLLBACK
|
||
|
{
|
||
|
$$ = &Rollback{}
|
||
|
}
|
||
|
|
||
|
other_statement:
|
||
|
DESC force_eof
|
||
|
{
|
||
|
$$ = &OtherRead{}
|
||
|
}
|
||
|
| DESCRIBE force_eof
|
||
|
{
|
||
|
$$ = &OtherRead{}
|
||
|
}
|
||
|
| EXPLAIN force_eof
|
||
|
{
|
||
|
$$ = &OtherRead{}
|
||
|
}
|
||
|
| REPAIR force_eof
|
||
|
{
|
||
|
$$ = &OtherAdmin{}
|
||
|
}
|
||
|
| OPTIMIZE force_eof
|
||
|
{
|
||
|
$$ = &OtherAdmin{}
|
||
|
}
|
||
|
|
||
|
comment_opt:
|
||
|
{
|
||
|
setAllowComments(yylex, true)
|
||
|
}
|
||
|
comment_list
|
||
|
{
|
||
|
$$ = $2
|
||
|
setAllowComments(yylex, false)
|
||
|
}
|
||
|
|
||
|
comment_list:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| comment_list COMMENT
|
||
|
{
|
||
|
$$ = append($1, $2)
|
||
|
}
|
||
|
|
||
|
union_op:
|
||
|
UNION
|
||
|
{
|
||
|
$$ = UnionStr
|
||
|
}
|
||
|
| UNION ALL
|
||
|
{
|
||
|
$$ = UnionAllStr
|
||
|
}
|
||
|
| UNION DISTINCT
|
||
|
{
|
||
|
$$ = UnionDistinctStr
|
||
|
}
|
||
|
|
||
|
cache_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| SQL_NO_CACHE
|
||
|
{
|
||
|
$$ = SQLNoCacheStr
|
||
|
}
|
||
|
| SQL_CACHE
|
||
|
{
|
||
|
$$ = SQLCacheStr
|
||
|
}
|
||
|
|
||
|
distinct_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| DISTINCT
|
||
|
{
|
||
|
$$ = DistinctStr
|
||
|
}
|
||
|
|
||
|
straight_join_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| STRAIGHT_JOIN
|
||
|
{
|
||
|
$$ = StraightJoinHint
|
||
|
}
|
||
|
|
||
|
select_expression_list_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| select_expression_list
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
|
||
|
select_expression_list:
|
||
|
select_expression
|
||
|
{
|
||
|
$$ = SelectExprs{$1}
|
||
|
}
|
||
|
| select_expression_list ',' select_expression
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
select_expression:
|
||
|
'*'
|
||
|
{
|
||
|
$$ = &StarExpr{}
|
||
|
}
|
||
|
| expression as_ci_opt
|
||
|
{
|
||
|
$$ = &AliasedExpr{Expr: $1, As: $2}
|
||
|
}
|
||
|
| table_id '.' '*'
|
||
|
{
|
||
|
$$ = &StarExpr{TableName: TableName{Name: $1}}
|
||
|
}
|
||
|
| table_id '.' reserved_table_id '.' '*'
|
||
|
{
|
||
|
$$ = &StarExpr{TableName: TableName{Qualifier: $1, Name: $3}}
|
||
|
}
|
||
|
|
||
|
as_ci_opt:
|
||
|
{
|
||
|
$$ = ColIdent{}
|
||
|
}
|
||
|
| col_alias
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| AS col_alias
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
col_alias:
|
||
|
sql_id
|
||
|
| STRING
|
||
|
{
|
||
|
$$ = NewColIdent(string($1))
|
||
|
}
|
||
|
|
||
|
from_opt:
|
||
|
{
|
||
|
$$ = TableExprs{&AliasedTableExpr{Expr:TableName{Name: NewTableIdent("dual")}}}
|
||
|
}
|
||
|
| FROM table_references
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
table_references:
|
||
|
table_reference
|
||
|
{
|
||
|
$$ = TableExprs{$1}
|
||
|
}
|
||
|
| table_references ',' table_reference
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
table_reference:
|
||
|
table_factor
|
||
|
| join_table
|
||
|
|
||
|
table_factor:
|
||
|
aliased_table_name
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| subquery as_opt table_id
|
||
|
{
|
||
|
$$ = &AliasedTableExpr{Expr:$1, As: $3}
|
||
|
}
|
||
|
| openb table_references closeb
|
||
|
{
|
||
|
$$ = &ParenTableExpr{Exprs: $2}
|
||
|
}
|
||
|
|
||
|
aliased_table_name:
|
||
|
table_name as_opt_id index_hint_list
|
||
|
{
|
||
|
$$ = &AliasedTableExpr{Expr:$1, As: $2, Hints: $3}
|
||
|
}
|
||
|
| table_name PARTITION openb partition_list closeb as_opt_id index_hint_list
|
||
|
{
|
||
|
$$ = &AliasedTableExpr{Expr:$1, Partitions: $4, As: $6, Hints: $7}
|
||
|
}
|
||
|
|
||
|
column_list:
|
||
|
sql_id
|
||
|
{
|
||
|
$$ = Columns{$1}
|
||
|
}
|
||
|
| column_list ',' sql_id
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
partition_list:
|
||
|
sql_id
|
||
|
{
|
||
|
$$ = Partitions{$1}
|
||
|
}
|
||
|
| partition_list ',' sql_id
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
|
||
|
// There is a grammar conflict here:
|
||
|
// 1: INSERT INTO a SELECT * FROM b JOIN c ON b.i = c.i
|
||
|
// 2: INSERT INTO a SELECT * FROM b JOIN c ON DUPLICATE KEY UPDATE a.i = 1
|
||
|
// When yacc encounters the ON clause, it cannot determine which way to
|
||
|
// resolve. The %prec override below makes the parser choose the
|
||
|
// first construct, which automatically makes the second construct a
|
||
|
// syntax error. This is the same behavior as MySQL.
|
||
|
join_table:
|
||
|
table_reference inner_join table_factor join_condition_opt
|
||
|
{
|
||
|
$$ = &JoinTableExpr{LeftExpr: $1, Join: $2, RightExpr: $3, Condition: $4}
|
||
|
}
|
||
|
| table_reference straight_join table_factor on_expression_opt
|
||
|
{
|
||
|
$$ = &JoinTableExpr{LeftExpr: $1, Join: $2, RightExpr: $3, Condition: $4}
|
||
|
}
|
||
|
| table_reference outer_join table_reference join_condition
|
||
|
{
|
||
|
$$ = &JoinTableExpr{LeftExpr: $1, Join: $2, RightExpr: $3, Condition: $4}
|
||
|
}
|
||
|
| table_reference natural_join table_factor
|
||
|
{
|
||
|
$$ = &JoinTableExpr{LeftExpr: $1, Join: $2, RightExpr: $3}
|
||
|
}
|
||
|
|
||
|
join_condition:
|
||
|
ON expression
|
||
|
{ $$ = JoinCondition{On: $2} }
|
||
|
| USING '(' column_list ')'
|
||
|
{ $$ = JoinCondition{Using: $3} }
|
||
|
|
||
|
join_condition_opt:
|
||
|
%prec JOIN
|
||
|
{ $$ = JoinCondition{} }
|
||
|
| join_condition
|
||
|
{ $$ = $1 }
|
||
|
|
||
|
on_expression_opt:
|
||
|
%prec JOIN
|
||
|
{ $$ = JoinCondition{} }
|
||
|
| ON expression
|
||
|
{ $$ = JoinCondition{On: $2} }
|
||
|
|
||
|
as_opt:
|
||
|
{ $$ = struct{}{} }
|
||
|
| AS
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
as_opt_id:
|
||
|
{
|
||
|
$$ = NewTableIdent("")
|
||
|
}
|
||
|
| table_alias
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| AS table_alias
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
table_alias:
|
||
|
table_id
|
||
|
| STRING
|
||
|
{
|
||
|
$$ = NewTableIdent(string($1))
|
||
|
}
|
||
|
|
||
|
inner_join:
|
||
|
JOIN
|
||
|
{
|
||
|
$$ = JoinStr
|
||
|
}
|
||
|
| INNER JOIN
|
||
|
{
|
||
|
$$ = JoinStr
|
||
|
}
|
||
|
| CROSS JOIN
|
||
|
{
|
||
|
$$ = JoinStr
|
||
|
}
|
||
|
|
||
|
straight_join:
|
||
|
STRAIGHT_JOIN
|
||
|
{
|
||
|
$$ = StraightJoinStr
|
||
|
}
|
||
|
|
||
|
outer_join:
|
||
|
LEFT JOIN
|
||
|
{
|
||
|
$$ = LeftJoinStr
|
||
|
}
|
||
|
| LEFT OUTER JOIN
|
||
|
{
|
||
|
$$ = LeftJoinStr
|
||
|
}
|
||
|
| RIGHT JOIN
|
||
|
{
|
||
|
$$ = RightJoinStr
|
||
|
}
|
||
|
| RIGHT OUTER JOIN
|
||
|
{
|
||
|
$$ = RightJoinStr
|
||
|
}
|
||
|
|
||
|
natural_join:
|
||
|
NATURAL JOIN
|
||
|
{
|
||
|
$$ = NaturalJoinStr
|
||
|
}
|
||
|
| NATURAL outer_join
|
||
|
{
|
||
|
if $2 == LeftJoinStr {
|
||
|
$$ = NaturalLeftJoinStr
|
||
|
} else {
|
||
|
$$ = NaturalRightJoinStr
|
||
|
}
|
||
|
}
|
||
|
|
||
|
into_table_name:
|
||
|
INTO table_name
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
| table_name
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
|
||
|
table_name:
|
||
|
table_id
|
||
|
{
|
||
|
$$ = TableName{Name: $1}
|
||
|
}
|
||
|
| table_id '.' reserved_table_id
|
||
|
{
|
||
|
$$ = TableName{Qualifier: $1, Name: $3}
|
||
|
}
|
||
|
|
||
|
index_hint_list:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| USE INDEX openb column_list closeb
|
||
|
{
|
||
|
$$ = &IndexHints{Type: UseStr, Indexes: $4}
|
||
|
}
|
||
|
| IGNORE INDEX openb column_list closeb
|
||
|
{
|
||
|
$$ = &IndexHints{Type: IgnoreStr, Indexes: $4}
|
||
|
}
|
||
|
| FORCE INDEX openb column_list closeb
|
||
|
{
|
||
|
$$ = &IndexHints{Type: ForceStr, Indexes: $4}
|
||
|
}
|
||
|
|
||
|
where_expression_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| WHERE expression
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
expression:
|
||
|
condition
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| expression AND expression
|
||
|
{
|
||
|
$$ = &AndExpr{Left: $1, Right: $3}
|
||
|
}
|
||
|
| expression OR expression
|
||
|
{
|
||
|
$$ = &OrExpr{Left: $1, Right: $3}
|
||
|
}
|
||
|
| NOT expression
|
||
|
{
|
||
|
$$ = &NotExpr{Expr: $2}
|
||
|
}
|
||
|
| expression IS is_suffix
|
||
|
{
|
||
|
$$ = &IsExpr{Operator: $3, Expr: $1}
|
||
|
}
|
||
|
| value_expression
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| DEFAULT default_opt
|
||
|
{
|
||
|
$$ = &Default{ColName: $2}
|
||
|
}
|
||
|
|
||
|
default_opt:
|
||
|
/* empty */
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| openb ID closeb
|
||
|
{
|
||
|
$$ = string($2)
|
||
|
}
|
||
|
|
||
|
boolean_value:
|
||
|
TRUE
|
||
|
{
|
||
|
$$ = BoolVal(true)
|
||
|
}
|
||
|
| FALSE
|
||
|
{
|
||
|
$$ = BoolVal(false)
|
||
|
}
|
||
|
|
||
|
condition:
|
||
|
value_expression compare value_expression
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: $2, Right: $3}
|
||
|
}
|
||
|
| value_expression IN col_tuple
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: InStr, Right: $3}
|
||
|
}
|
||
|
| value_expression NOT IN col_tuple
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: NotInStr, Right: $4}
|
||
|
}
|
||
|
| value_expression LIKE value_expression like_escape_opt
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: LikeStr, Right: $3, Escape: $4}
|
||
|
}
|
||
|
| value_expression NOT LIKE value_expression like_escape_opt
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: NotLikeStr, Right: $4, Escape: $5}
|
||
|
}
|
||
|
| value_expression REGEXP value_expression
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: RegexpStr, Right: $3}
|
||
|
}
|
||
|
| value_expression NOT REGEXP value_expression
|
||
|
{
|
||
|
$$ = &ComparisonExpr{Left: $1, Operator: NotRegexpStr, Right: $4}
|
||
|
}
|
||
|
| value_expression BETWEEN value_expression AND value_expression
|
||
|
{
|
||
|
$$ = &RangeCond{Left: $1, Operator: BetweenStr, From: $3, To: $5}
|
||
|
}
|
||
|
| value_expression NOT BETWEEN value_expression AND value_expression
|
||
|
{
|
||
|
$$ = &RangeCond{Left: $1, Operator: NotBetweenStr, From: $4, To: $6}
|
||
|
}
|
||
|
| EXISTS subquery
|
||
|
{
|
||
|
$$ = &ExistsExpr{Subquery: $2}
|
||
|
}
|
||
|
|
||
|
is_suffix:
|
||
|
NULL
|
||
|
{
|
||
|
$$ = IsNullStr
|
||
|
}
|
||
|
| NOT NULL
|
||
|
{
|
||
|
$$ = IsNotNullStr
|
||
|
}
|
||
|
| TRUE
|
||
|
{
|
||
|
$$ = IsTrueStr
|
||
|
}
|
||
|
| NOT TRUE
|
||
|
{
|
||
|
$$ = IsNotTrueStr
|
||
|
}
|
||
|
| FALSE
|
||
|
{
|
||
|
$$ = IsFalseStr
|
||
|
}
|
||
|
| NOT FALSE
|
||
|
{
|
||
|
$$ = IsNotFalseStr
|
||
|
}
|
||
|
|
||
|
compare:
|
||
|
'='
|
||
|
{
|
||
|
$$ = EqualStr
|
||
|
}
|
||
|
| '<'
|
||
|
{
|
||
|
$$ = LessThanStr
|
||
|
}
|
||
|
| '>'
|
||
|
{
|
||
|
$$ = GreaterThanStr
|
||
|
}
|
||
|
| LE
|
||
|
{
|
||
|
$$ = LessEqualStr
|
||
|
}
|
||
|
| GE
|
||
|
{
|
||
|
$$ = GreaterEqualStr
|
||
|
}
|
||
|
| NE
|
||
|
{
|
||
|
$$ = NotEqualStr
|
||
|
}
|
||
|
| NULL_SAFE_EQUAL
|
||
|
{
|
||
|
$$ = NullSafeEqualStr
|
||
|
}
|
||
|
|
||
|
like_escape_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| ESCAPE value_expression
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
col_tuple:
|
||
|
row_tuple
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| subquery
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| LIST_ARG
|
||
|
{
|
||
|
$$ = ListArg($1)
|
||
|
}
|
||
|
|
||
|
subquery:
|
||
|
openb select_statement closeb
|
||
|
{
|
||
|
$$ = &Subquery{$2}
|
||
|
}
|
||
|
|
||
|
expression_list:
|
||
|
expression
|
||
|
{
|
||
|
$$ = Exprs{$1}
|
||
|
}
|
||
|
| expression_list ',' expression
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
value_expression:
|
||
|
value
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| boolean_value
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| column_name
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| tuple_expression
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| subquery
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| value_expression '&' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: BitAndStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '|' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: BitOrStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '^' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: BitXorStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '+' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: PlusStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '-' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: MinusStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '*' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: MultStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '/' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: DivStr, Right: $3}
|
||
|
}
|
||
|
| value_expression DIV value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: IntDivStr, Right: $3}
|
||
|
}
|
||
|
| value_expression '%' value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: ModStr, Right: $3}
|
||
|
}
|
||
|
| value_expression MOD value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: ModStr, Right: $3}
|
||
|
}
|
||
|
| value_expression SHIFT_LEFT value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: ShiftLeftStr, Right: $3}
|
||
|
}
|
||
|
| value_expression SHIFT_RIGHT value_expression
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: ShiftRightStr, Right: $3}
|
||
|
}
|
||
|
| column_name JSON_EXTRACT_OP value
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: JSONExtractOp, Right: $3}
|
||
|
}
|
||
|
| column_name JSON_UNQUOTE_EXTRACT_OP value
|
||
|
{
|
||
|
$$ = &BinaryExpr{Left: $1, Operator: JSONUnquoteExtractOp, Right: $3}
|
||
|
}
|
||
|
| value_expression COLLATE charset
|
||
|
{
|
||
|
$$ = &CollateExpr{Expr: $1, Charset: $3}
|
||
|
}
|
||
|
| BINARY value_expression %prec UNARY
|
||
|
{
|
||
|
$$ = &UnaryExpr{Operator: BinaryStr, Expr: $2}
|
||
|
}
|
||
|
| UNDERSCORE_BINARY value_expression %prec UNARY
|
||
|
{
|
||
|
$$ = &UnaryExpr{Operator: UBinaryStr, Expr: $2}
|
||
|
}
|
||
|
| '+' value_expression %prec UNARY
|
||
|
{
|
||
|
if num, ok := $2.(*SQLVal); ok && num.Type == IntVal {
|
||
|
$$ = num
|
||
|
} else {
|
||
|
$$ = &UnaryExpr{Operator: UPlusStr, Expr: $2}
|
||
|
}
|
||
|
}
|
||
|
| '-' value_expression %prec UNARY
|
||
|
{
|
||
|
if num, ok := $2.(*SQLVal); ok && num.Type == IntVal {
|
||
|
// Handle double negative
|
||
|
if num.Val[0] == '-' {
|
||
|
num.Val = num.Val[1:]
|
||
|
$$ = num
|
||
|
} else {
|
||
|
$$ = NewIntVal(append([]byte("-"), num.Val...))
|
||
|
}
|
||
|
} else {
|
||
|
$$ = &UnaryExpr{Operator: UMinusStr, Expr: $2}
|
||
|
}
|
||
|
}
|
||
|
| '~' value_expression
|
||
|
{
|
||
|
$$ = &UnaryExpr{Operator: TildaStr, Expr: $2}
|
||
|
}
|
||
|
| '!' value_expression %prec UNARY
|
||
|
{
|
||
|
$$ = &UnaryExpr{Operator: BangStr, Expr: $2}
|
||
|
}
|
||
|
| INTERVAL value_expression sql_id
|
||
|
{
|
||
|
// This rule prevents the usage of INTERVAL
|
||
|
// as a function. If support is needed for that,
|
||
|
// we'll need to revisit this. The solution
|
||
|
// will be non-trivial because of grammar conflicts.
|
||
|
$$ = &IntervalExpr{Expr: $2, Unit: $3.String()}
|
||
|
}
|
||
|
| function_call_generic
|
||
|
| function_call_keyword
|
||
|
| function_call_nonkeyword
|
||
|
| function_call_conflict
|
||
|
|
||
|
/*
|
||
|
Regular function calls without special token or syntax, guaranteed to not
|
||
|
introduce side effects due to being a simple identifier
|
||
|
*/
|
||
|
function_call_generic:
|
||
|
sql_id openb select_expression_list_opt closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: $1, Exprs: $3}
|
||
|
}
|
||
|
| sql_id openb DISTINCT select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: $1, Distinct: true, Exprs: $4}
|
||
|
}
|
||
|
| table_id '.' reserved_sql_id openb select_expression_list_opt closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Qualifier: $1, Name: $3, Exprs: $5}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Function calls using reserved keywords, with dedicated grammar rules
|
||
|
as a result
|
||
|
*/
|
||
|
function_call_keyword:
|
||
|
LEFT openb select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("left"), Exprs: $3}
|
||
|
}
|
||
|
| RIGHT openb select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("right"), Exprs: $3}
|
||
|
}
|
||
|
| CONVERT openb expression ',' convert_type closeb
|
||
|
{
|
||
|
$$ = &ConvertExpr{Expr: $3, Type: $5}
|
||
|
}
|
||
|
| CAST openb expression AS convert_type closeb
|
||
|
{
|
||
|
$$ = &ConvertExpr{Expr: $3, Type: $5}
|
||
|
}
|
||
|
| CONVERT openb expression USING charset closeb
|
||
|
{
|
||
|
$$ = &ConvertUsingExpr{Expr: $3, Type: $5}
|
||
|
}
|
||
|
| SUBSTR openb column_name ',' value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: nil}
|
||
|
}
|
||
|
| SUBSTR openb column_name ',' value_expression ',' value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: $7}
|
||
|
}
|
||
|
| SUBSTR openb column_name FROM value_expression FOR value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: $7}
|
||
|
}
|
||
|
| SUBSTRING openb column_name ',' value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: nil}
|
||
|
}
|
||
|
| SUBSTRING openb column_name ',' value_expression ',' value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: $7}
|
||
|
}
|
||
|
| SUBSTRING openb column_name FROM value_expression FOR value_expression closeb
|
||
|
{
|
||
|
$$ = &SubstrExpr{Name: $3, From: $5, To: $7}
|
||
|
}
|
||
|
| MATCH openb select_expression_list closeb AGAINST openb value_expression match_option closeb
|
||
|
{
|
||
|
$$ = &MatchExpr{Columns: $3, Expr: $7, Option: $8}
|
||
|
}
|
||
|
| GROUP_CONCAT openb distinct_opt select_expression_list order_by_opt separator_opt closeb
|
||
|
{
|
||
|
$$ = &GroupConcatExpr{Distinct: $3, Exprs: $4, OrderBy: $5, Separator: $6}
|
||
|
}
|
||
|
| CASE expression_opt when_expression_list else_expression_opt END
|
||
|
{
|
||
|
$$ = &CaseExpr{Expr: $2, Whens: $3, Else: $4}
|
||
|
}
|
||
|
| VALUES openb column_name closeb
|
||
|
{
|
||
|
$$ = &ValuesFuncExpr{Name: $3}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Function calls using non reserved keywords but with special syntax forms.
|
||
|
Dedicated grammar rules are needed because of the special syntax
|
||
|
*/
|
||
|
function_call_nonkeyword:
|
||
|
CURRENT_TIMESTAMP func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("current_timestamp")}
|
||
|
}
|
||
|
| UTC_TIMESTAMP func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("utc_timestamp")}
|
||
|
}
|
||
|
| UTC_TIME func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("utc_time")}
|
||
|
}
|
||
|
| UTC_DATE func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("utc_date")}
|
||
|
}
|
||
|
// now
|
||
|
| LOCALTIME func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("localtime")}
|
||
|
}
|
||
|
// now
|
||
|
| LOCALTIMESTAMP func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("localtimestamp")}
|
||
|
}
|
||
|
// curdate
|
||
|
| CURRENT_DATE func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("current_date")}
|
||
|
}
|
||
|
// curtime
|
||
|
| CURRENT_TIME func_datetime_precision_opt
|
||
|
{
|
||
|
$$ = &FuncExpr{Name:NewColIdent("current_time")}
|
||
|
}
|
||
|
|
||
|
func_datetime_precision_opt:
|
||
|
/* empty */
|
||
|
| openb closeb
|
||
|
|
||
|
/*
|
||
|
Function calls using non reserved keywords with *normal* syntax forms. Because
|
||
|
the names are non-reserved, they need a dedicated rule so as not to conflict
|
||
|
*/
|
||
|
function_call_conflict:
|
||
|
IF openb select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("if"), Exprs: $3}
|
||
|
}
|
||
|
| DATABASE openb select_expression_list_opt closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("database"), Exprs: $3}
|
||
|
}
|
||
|
| MOD openb select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("mod"), Exprs: $3}
|
||
|
}
|
||
|
| REPLACE openb select_expression_list closeb
|
||
|
{
|
||
|
$$ = &FuncExpr{Name: NewColIdent("replace"), Exprs: $3}
|
||
|
}
|
||
|
|
||
|
match_option:
|
||
|
/*empty*/
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| IN BOOLEAN MODE
|
||
|
{
|
||
|
$$ = BooleanModeStr
|
||
|
}
|
||
|
| IN NATURAL LANGUAGE MODE
|
||
|
{
|
||
|
$$ = NaturalLanguageModeStr
|
||
|
}
|
||
|
| IN NATURAL LANGUAGE MODE WITH QUERY EXPANSION
|
||
|
{
|
||
|
$$ = NaturalLanguageModeWithQueryExpansionStr
|
||
|
}
|
||
|
| WITH QUERY EXPANSION
|
||
|
{
|
||
|
$$ = QueryExpansionStr
|
||
|
}
|
||
|
|
||
|
charset:
|
||
|
ID
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
| STRING
|
||
|
{
|
||
|
$$ = string($1)
|
||
|
}
|
||
|
|
||
|
convert_type:
|
||
|
BINARY length_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| CHAR length_opt charset_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2, Charset: $3, Operator: CharacterSetStr}
|
||
|
}
|
||
|
| CHAR length_opt ID
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2, Charset: string($3)}
|
||
|
}
|
||
|
| DATE
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
| DATETIME length_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| DECIMAL decimal_length_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
$$.Length = $2.Length
|
||
|
$$.Scale = $2.Scale
|
||
|
}
|
||
|
| JSON
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
| NCHAR length_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| SIGNED
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
| SIGNED INTEGER
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
| TIME length_opt
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1), Length: $2}
|
||
|
}
|
||
|
| UNSIGNED
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
| UNSIGNED INTEGER
|
||
|
{
|
||
|
$$ = &ConvertType{Type: string($1)}
|
||
|
}
|
||
|
|
||
|
expression_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| expression
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
|
||
|
separator_opt:
|
||
|
{
|
||
|
$$ = string("")
|
||
|
}
|
||
|
| SEPARATOR STRING
|
||
|
{
|
||
|
$$ = " separator '"+string($2)+"'"
|
||
|
}
|
||
|
|
||
|
when_expression_list:
|
||
|
when_expression
|
||
|
{
|
||
|
$$ = []*When{$1}
|
||
|
}
|
||
|
| when_expression_list when_expression
|
||
|
{
|
||
|
$$ = append($1, $2)
|
||
|
}
|
||
|
|
||
|
when_expression:
|
||
|
WHEN expression THEN expression
|
||
|
{
|
||
|
$$ = &When{Cond: $2, Val: $4}
|
||
|
}
|
||
|
|
||
|
else_expression_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| ELSE expression
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
column_name:
|
||
|
sql_id
|
||
|
{
|
||
|
$$ = &ColName{Name: $1}
|
||
|
}
|
||
|
| table_id '.' reserved_sql_id
|
||
|
{
|
||
|
$$ = &ColName{Qualifier: TableName{Name: $1}, Name: $3}
|
||
|
}
|
||
|
| table_id '.' reserved_table_id '.' reserved_sql_id
|
||
|
{
|
||
|
$$ = &ColName{Qualifier: TableName{Qualifier: $1, Name: $3}, Name: $5}
|
||
|
}
|
||
|
|
||
|
value:
|
||
|
STRING
|
||
|
{
|
||
|
$$ = NewStrVal($1)
|
||
|
}
|
||
|
| HEX
|
||
|
{
|
||
|
$$ = NewHexVal($1)
|
||
|
}
|
||
|
| BIT_LITERAL
|
||
|
{
|
||
|
$$ = NewBitVal($1)
|
||
|
}
|
||
|
| INTEGRAL
|
||
|
{
|
||
|
$$ = NewIntVal($1)
|
||
|
}
|
||
|
| FLOAT
|
||
|
{
|
||
|
$$ = NewFloatVal($1)
|
||
|
}
|
||
|
| HEXNUM
|
||
|
{
|
||
|
$$ = NewHexNum($1)
|
||
|
}
|
||
|
| VALUE_ARG
|
||
|
{
|
||
|
$$ = NewValArg($1)
|
||
|
}
|
||
|
| NULL
|
||
|
{
|
||
|
$$ = &NullVal{}
|
||
|
}
|
||
|
|
||
|
num_val:
|
||
|
sql_id
|
||
|
{
|
||
|
// TODO(sougou): Deprecate this construct.
|
||
|
if $1.Lowered() != "value" {
|
||
|
yylex.Error("expecting value after next")
|
||
|
return 1
|
||
|
}
|
||
|
$$ = NewIntVal([]byte("1"))
|
||
|
}
|
||
|
| INTEGRAL VALUES
|
||
|
{
|
||
|
$$ = NewIntVal($1)
|
||
|
}
|
||
|
| VALUE_ARG VALUES
|
||
|
{
|
||
|
$$ = NewValArg($1)
|
||
|
}
|
||
|
|
||
|
group_by_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| GROUP BY expression_list
|
||
|
{
|
||
|
$$ = $3
|
||
|
}
|
||
|
|
||
|
having_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| HAVING expression
|
||
|
{
|
||
|
$$ = $2
|
||
|
}
|
||
|
|
||
|
order_by_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| ORDER BY order_list
|
||
|
{
|
||
|
$$ = $3
|
||
|
}
|
||
|
|
||
|
order_list:
|
||
|
order
|
||
|
{
|
||
|
$$ = OrderBy{$1}
|
||
|
}
|
||
|
| order_list ',' order
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
order:
|
||
|
expression asc_desc_opt
|
||
|
{
|
||
|
$$ = &Order{Expr: $1, Direction: $2}
|
||
|
}
|
||
|
|
||
|
asc_desc_opt:
|
||
|
{
|
||
|
$$ = AscScr
|
||
|
}
|
||
|
| ASC
|
||
|
{
|
||
|
$$ = AscScr
|
||
|
}
|
||
|
| DESC
|
||
|
{
|
||
|
$$ = DescScr
|
||
|
}
|
||
|
|
||
|
limit_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| LIMIT expression
|
||
|
{
|
||
|
$$ = &Limit{Rowcount: $2}
|
||
|
}
|
||
|
| LIMIT expression ',' expression
|
||
|
{
|
||
|
$$ = &Limit{Offset: $2, Rowcount: $4}
|
||
|
}
|
||
|
| LIMIT expression OFFSET expression
|
||
|
{
|
||
|
$$ = &Limit{Offset: $4, Rowcount: $2}
|
||
|
}
|
||
|
|
||
|
lock_opt:
|
||
|
{
|
||
|
$$ = ""
|
||
|
}
|
||
|
| FOR UPDATE
|
||
|
{
|
||
|
$$ = ForUpdateStr
|
||
|
}
|
||
|
| LOCK IN SHARE MODE
|
||
|
{
|
||
|
$$ = ShareModeStr
|
||
|
}
|
||
|
|
||
|
// insert_data expands all combinations into a single rule.
|
||
|
// This avoids a shift/reduce conflict while encountering the
|
||
|
// following two possible constructs:
|
||
|
// insert into t1(a, b) (select * from t2)
|
||
|
// insert into t1(select * from t2)
|
||
|
// Because the rules are together, the parser can keep shifting
|
||
|
// the tokens until it disambiguates a as sql_id and select as keyword.
|
||
|
insert_data:
|
||
|
VALUES tuple_list
|
||
|
{
|
||
|
$$ = &Insert{Rows: $2}
|
||
|
}
|
||
|
| select_statement
|
||
|
{
|
||
|
$$ = &Insert{Rows: $1}
|
||
|
}
|
||
|
| openb select_statement closeb
|
||
|
{
|
||
|
// Drop the redundant parenthesis.
|
||
|
$$ = &Insert{Rows: $2}
|
||
|
}
|
||
|
| openb ins_column_list closeb VALUES tuple_list
|
||
|
{
|
||
|
$$ = &Insert{Columns: $2, Rows: $5}
|
||
|
}
|
||
|
| openb ins_column_list closeb select_statement
|
||
|
{
|
||
|
$$ = &Insert{Columns: $2, Rows: $4}
|
||
|
}
|
||
|
| openb ins_column_list closeb openb select_statement closeb
|
||
|
{
|
||
|
// Drop the redundant parenthesis.
|
||
|
$$ = &Insert{Columns: $2, Rows: $5}
|
||
|
}
|
||
|
|
||
|
ins_column_list:
|
||
|
sql_id
|
||
|
{
|
||
|
$$ = Columns{$1}
|
||
|
}
|
||
|
| sql_id '.' sql_id
|
||
|
{
|
||
|
$$ = Columns{$3}
|
||
|
}
|
||
|
| ins_column_list ',' sql_id
|
||
|
{
|
||
|
$$ = append($$, $3)
|
||
|
}
|
||
|
| ins_column_list ',' sql_id '.' sql_id
|
||
|
{
|
||
|
$$ = append($$, $5)
|
||
|
}
|
||
|
|
||
|
on_dup_opt:
|
||
|
{
|
||
|
$$ = nil
|
||
|
}
|
||
|
| ON DUPLICATE KEY UPDATE update_list
|
||
|
{
|
||
|
$$ = $5
|
||
|
}
|
||
|
|
||
|
tuple_list:
|
||
|
tuple_or_empty
|
||
|
{
|
||
|
$$ = Values{$1}
|
||
|
}
|
||
|
| tuple_list ',' tuple_or_empty
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
tuple_or_empty:
|
||
|
row_tuple
|
||
|
{
|
||
|
$$ = $1
|
||
|
}
|
||
|
| openb closeb
|
||
|
{
|
||
|
$$ = ValTuple{}
|
||
|
}
|
||
|
|
||
|
row_tuple:
|
||
|
openb expression_list closeb
|
||
|
{
|
||
|
$$ = ValTuple($2)
|
||
|
}
|
||
|
|
||
|
tuple_expression:
|
||
|
row_tuple
|
||
|
{
|
||
|
if len($1) == 1 {
|
||
|
$$ = &ParenExpr{$1[0]}
|
||
|
} else {
|
||
|
$$ = $1
|
||
|
}
|
||
|
}
|
||
|
|
||
|
update_list:
|
||
|
update_expression
|
||
|
{
|
||
|
$$ = UpdateExprs{$1}
|
||
|
}
|
||
|
| update_list ',' update_expression
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
update_expression:
|
||
|
column_name '=' expression
|
||
|
{
|
||
|
$$ = &UpdateExpr{Name: $1, Expr: $3}
|
||
|
}
|
||
|
|
||
|
set_list:
|
||
|
set_expression
|
||
|
{
|
||
|
$$ = SetExprs{$1}
|
||
|
}
|
||
|
| set_list ',' set_expression
|
||
|
{
|
||
|
$$ = append($1, $3)
|
||
|
}
|
||
|
|
||
|
set_expression:
|
||
|
reserved_sql_id '=' ON
|
||
|
{
|
||
|
$$ = &SetExpr{Name: $1, Expr: NewStrVal([]byte("on"))}
|
||
|
}
|
||
|
| reserved_sql_id '=' expression
|
||
|
{
|
||
|
$$ = &SetExpr{Name: $1, Expr: $3}
|
||
|
}
|
||
|
| charset_or_character_set charset_value collate_opt
|
||
|
{
|
||
|
$$ = &SetExpr{Name: NewColIdent(string($1)), Expr: $2}
|
||
|
}
|
||
|
|
||
|
charset_or_character_set:
|
||
|
CHARSET
|
||
|
| CHARACTER SET
|
||
|
{
|
||
|
$$ = []byte("charset")
|
||
|
}
|
||
|
| NAMES
|
||
|
|
||
|
charset_value:
|
||
|
sql_id
|
||
|
{
|
||
|
$$ = NewStrVal([]byte($1.String()))
|
||
|
}
|
||
|
| STRING
|
||
|
{
|
||
|
$$ = NewStrVal($1)
|
||
|
}
|
||
|
| DEFAULT
|
||
|
{
|
||
|
$$ = &Default{}
|
||
|
}
|
||
|
|
||
|
for_from:
|
||
|
FOR
|
||
|
| FROM
|
||
|
|
||
|
exists_opt:
|
||
|
{ $$ = 0 }
|
||
|
| IF EXISTS
|
||
|
{ $$ = 1 }
|
||
|
|
||
|
not_exists_opt:
|
||
|
{ $$ = struct{}{} }
|
||
|
| IF NOT EXISTS
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
ignore_opt:
|
||
|
{ $$ = "" }
|
||
|
| IGNORE
|
||
|
{ $$ = IgnoreStr }
|
||
|
|
||
|
non_add_drop_or_rename_operation:
|
||
|
ALTER
|
||
|
{ $$ = struct{}{} }
|
||
|
| AUTO_INCREMENT
|
||
|
{ $$ = struct{}{} }
|
||
|
| CHARACTER
|
||
|
{ $$ = struct{}{} }
|
||
|
| COMMENT_KEYWORD
|
||
|
{ $$ = struct{}{} }
|
||
|
| DEFAULT
|
||
|
{ $$ = struct{}{} }
|
||
|
| ORDER
|
||
|
{ $$ = struct{}{} }
|
||
|
| CONVERT
|
||
|
{ $$ = struct{}{} }
|
||
|
| PARTITION
|
||
|
{ $$ = struct{}{} }
|
||
|
| UNUSED
|
||
|
{ $$ = struct{}{} }
|
||
|
| ID
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
to_opt:
|
||
|
{ $$ = struct{}{} }
|
||
|
| TO
|
||
|
{ $$ = struct{}{} }
|
||
|
| AS
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
index_opt:
|
||
|
INDEX
|
||
|
{ $$ = struct{}{} }
|
||
|
| KEY
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
constraint_opt:
|
||
|
{ $$ = struct{}{} }
|
||
|
| UNIQUE
|
||
|
{ $$ = struct{}{} }
|
||
|
| sql_id
|
||
|
{ $$ = struct{}{} }
|
||
|
|
||
|
using_opt:
|
||
|
{ $$ = ColIdent{} }
|
||
|
| USING sql_id
|
||
|
{ $$ = $2 }
|
||
|
|
||
|
sql_id:
|
||
|
ID
|
||
|
{
|
||
|
$$ = NewColIdent(string($1))
|
||
|
}
|
||
|
| non_reserved_keyword
|
||
|
{
|
||
|
$$ = NewColIdent(string($1))
|
||
|
}
|
||
|
|
||
|
reserved_sql_id:
|
||
|
sql_id
|
||
|
| reserved_keyword
|
||
|
{
|
||
|
$$ = NewColIdent(string($1))
|
||
|
}
|
||
|
|
||
|
table_id:
|
||
|
ID
|
||
|
{
|
||
|
$$ = NewTableIdent(string($1))
|
||
|
}
|
||
|
| non_reserved_keyword
|
||
|
{
|
||
|
$$ = NewTableIdent(string($1))
|
||
|
}
|
||
|
|
||
|
reserved_table_id:
|
||
|
table_id
|
||
|
| reserved_keyword
|
||
|
{
|
||
|
$$ = NewTableIdent(string($1))
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
These are not all necessarily reserved in MySQL, but some are.
|
||
|
|
||
|
These are more importantly reserved because they may conflict with our grammar.
|
||
|
If you want to move one that is not reserved in MySQL (i.e. ESCAPE) to the
|
||
|
non_reserved_keywords, you'll need to deal with any conflicts.
|
||
|
|
||
|
Sorted alphabetically
|
||
|
*/
|
||
|
reserved_keyword:
|
||
|
ADD
|
||
|
| AND
|
||
|
| AS
|
||
|
| ASC
|
||
|
| AUTO_INCREMENT
|
||
|
| BETWEEN
|
||
|
| BINARY
|
||
|
| BY
|
||
|
| CASE
|
||
|
| COLLATE
|
||
|
| CONVERT
|
||
|
| CREATE
|
||
|
| CROSS
|
||
|
| CURRENT_DATE
|
||
|
| CURRENT_TIME
|
||
|
| CURRENT_TIMESTAMP
|
||
|
| SUBSTR
|
||
|
| SUBSTRING
|
||
|
| DATABASE
|
||
|
| DATABASES
|
||
|
| DEFAULT
|
||
|
| DELETE
|
||
|
| DESC
|
||
|
| DESCRIBE
|
||
|
| DISTINCT
|
||
|
| DIV
|
||
|
| DROP
|
||
|
| ELSE
|
||
|
| END
|
||
|
| ESCAPE
|
||
|
| EXISTS
|
||
|
| EXPLAIN
|
||
|
| FALSE
|
||
|
| FOR
|
||
|
| FORCE
|
||
|
| FROM
|
||
|
| GROUP
|
||
|
| HAVING
|
||
|
| IF
|
||
|
| IGNORE
|
||
|
| IN
|
||
|
| INDEX
|
||
|
| INNER
|
||
|
| INSERT
|
||
|
| INTERVAL
|
||
|
| INTO
|
||
|
| IS
|
||
|
| JOIN
|
||
|
| KEY
|
||
|
| LEFT
|
||
|
| LIKE
|
||
|
| LIMIT
|
||
|
| LOCALTIME
|
||
|
| LOCALTIMESTAMP
|
||
|
| LOCK
|
||
|
| MATCH
|
||
|
| MAXVALUE
|
||
|
| MOD
|
||
|
| NATURAL
|
||
|
| NEXT // next should be doable as non-reserved, but is not due to the special `select next num_val` query that vitess supports
|
||
|
| NOT
|
||
|
| NULL
|
||
|
| ON
|
||
|
| OR
|
||
|
| ORDER
|
||
|
| OUTER
|
||
|
| REGEXP
|
||
|
| RENAME
|
||
|
| REPLACE
|
||
|
| RIGHT
|
||
|
| SCHEMA
|
||
|
| SELECT
|
||
|
| SEPARATOR
|
||
|
| SET
|
||
|
| SHOW
|
||
|
| STRAIGHT_JOIN
|
||
|
| TABLE
|
||
|
| TABLES
|
||
|
| THEN
|
||
|
| TO
|
||
|
| TRUE
|
||
|
| UNION
|
||
|
| UNIQUE
|
||
|
| UPDATE
|
||
|
| USE
|
||
|
| USING
|
||
|
| UTC_DATE
|
||
|
| UTC_TIME
|
||
|
| UTC_TIMESTAMP
|
||
|
| VALUES
|
||
|
| WHEN
|
||
|
| WHERE
|
||
|
|
||
|
/*
|
||
|
These are non-reserved Vitess, because they don't cause conflicts in the grammar.
|
||
|
Some of them may be reserved in MySQL. The good news is we backtick quote them
|
||
|
when we rewrite the query, so no issue should arise.
|
||
|
|
||
|
Sorted alphabetically
|
||
|
*/
|
||
|
non_reserved_keyword:
|
||
|
AGAINST
|
||
|
| BEGIN
|
||
|
| BIGINT
|
||
|
| BIT
|
||
|
| BLOB
|
||
|
| BOOL
|
||
|
| CHAR
|
||
|
| CHARACTER
|
||
|
| CHARSET
|
||
|
| COMMENT_KEYWORD
|
||
|
| COMMIT
|
||
|
| COMMITTED
|
||
|
| DATE
|
||
|
| DATETIME
|
||
|
| DECIMAL
|
||
|
| DOUBLE
|
||
|
| DUPLICATE
|
||
|
| ENUM
|
||
|
| EXPANSION
|
||
|
| FLOAT_TYPE
|
||
|
| FOREIGN
|
||
|
| FULLTEXT
|
||
|
| GEOMETRY
|
||
|
| GEOMETRYCOLLECTION
|
||
|
| GLOBAL
|
||
|
| INT
|
||
|
| INTEGER
|
||
|
| ISOLATION
|
||
|
| JSON
|
||
|
| KEY_BLOCK_SIZE
|
||
|
| KEYS
|
||
|
| LANGUAGE
|
||
|
| LAST_INSERT_ID
|
||
|
| LESS
|
||
|
| LEVEL
|
||
|
| LINESTRING
|
||
|
| LONGBLOB
|
||
|
| LONGTEXT
|
||
|
| MEDIUMBLOB
|
||
|
| MEDIUMINT
|
||
|
| MEDIUMTEXT
|
||
|
| MODE
|
||
|
| MULTILINESTRING
|
||
|
| MULTIPOINT
|
||
|
| MULTIPOLYGON
|
||
|
| NAMES
|
||
|
| NCHAR
|
||
|
| NUMERIC
|
||
|
| OFFSET
|
||
|
| ONLY
|
||
|
| OPTIMIZE
|
||
|
| PARTITION
|
||
|
| POINT
|
||
|
| POLYGON
|
||
|
| PRIMARY
|
||
|
| PROCEDURE
|
||
|
| QUERY
|
||
|
| READ
|
||
|
| REAL
|
||
|
| REORGANIZE
|
||
|
| REPAIR
|
||
|
| REPEATABLE
|
||
|
| ROLLBACK
|
||
|
| SESSION
|
||
|
| SERIALIZABLE
|
||
|
| SHARE
|
||
|
| SIGNED
|
||
|
| SMALLINT
|
||
|
| SPATIAL
|
||
|
| START
|
||
|
| STATUS
|
||
|
| TEXT
|
||
|
| THAN
|
||
|
| TIME
|
||
|
| TIMESTAMP
|
||
|
| TINYBLOB
|
||
|
| TINYINT
|
||
|
| TINYTEXT
|
||
|
| TRANSACTION
|
||
|
| TRIGGER
|
||
|
| TRUNCATE
|
||
|
| UNCOMMITTED
|
||
|
| UNSIGNED
|
||
|
| UNUSED
|
||
|
| VARBINARY
|
||
|
| VARCHAR
|
||
|
| VARIABLES
|
||
|
| VIEW
|
||
|
| VINDEX
|
||
|
| VINDEXES
|
||
|
| VITESS_KEYSPACES
|
||
|
| VITESS_SHARDS
|
||
|
| VITESS_TABLETS
|
||
|
| VSCHEMA_TABLES
|
||
|
| WITH
|
||
|
| WRITE
|
||
|
| YEAR
|
||
|
| ZEROFILL
|
||
|
|
||
|
openb:
|
||
|
'('
|
||
|
{
|
||
|
if incNesting(yylex) {
|
||
|
yylex.Error("max nesting level reached")
|
||
|
return 1
|
||
|
}
|
||
|
}
|
||
|
|
||
|
closeb:
|
||
|
')'
|
||
|
{
|
||
|
decNesting(yylex)
|
||
|
}
|
||
|
|
||
|
force_eof:
|
||
|
{
|
||
|
forceEOF(yylex)
|
||
|
}
|
||
|
|
||
|
ddl_force_eof:
|
||
|
{
|
||
|
forceEOF(yylex)
|
||
|
}
|
||
|
| openb
|
||
|
{
|
||
|
forceEOF(yylex)
|
||
|
}
|
||
|
| reserved_sql_id
|
||
|
{
|
||
|
forceEOF(yylex)
|
||
|
}
|