1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2010 Brian Aker
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; either version 2 of the License, or
9
* (at your option) any later version.
11
* This program is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
* GNU General Public License for more details.
16
* You should have received a copy of the GNU General Public License
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
23
#include <drizzled/parser.h>
31
Helper to resolve the SQL:2003 Syntax exception 1) in <in predicate>.
32
See SQL:2003, Part 2, section 8.4 <in predicate>, Note 184, page 383.
33
This function returns the proper item for the SQL expression
34
<code>left [NOT] IN ( expr )</code>
35
@param session the current thread
36
@param left the in predicand
37
@param equal true for IN predicates, false for NOT IN predicates
38
@param expr first and only expression of the in value list
39
@return an expression representing the IN predicate.
42
Item* handle_sql2003_note184_exception(Session *session, Item* left, bool equal, Item *expr)
45
Relevant references for this issue:
46
- SQL:2003, Part 2, section 8.4 <in predicate>, page 383,
47
- SQL:2003, Part 2, section 7.2 <row value expression>, page 296,
48
- SQL:2003, Part 2, section 6.3 <value expression primary>, page 174,
49
- SQL:2003, Part 2, section 7.15 <subquery>, page 370,
50
- SQL:2003 Feature F561, "Full value expressions".
52
The exception in SQL:2003 Note 184 means:
53
Item_singlerow_subselect, which corresponds to a <scalar subquery>,
54
should be re-interpreted as an Item_in_subselect, which corresponds
55
to a <table subquery> when used inside an <in predicate>.
57
Our reading of Note 184 is reccursive, so that all:
59
- IN ((( <subquery> )))
60
- IN '('^N <subquery> ')'^N
62
should be interpreted as a <table subquery>, no matter how deep in the
63
expression the <subquery> is.
68
if (expr->type() == Item::SUBSELECT_ITEM)
70
Item_subselect *expr2 = (Item_subselect*) expr;
72
if (expr2->substype() == Item_subselect::SINGLEROW_SUBS)
74
Item_singlerow_subselect *expr3 = (Item_singlerow_subselect*) expr2;
75
Select_Lex *subselect;
78
Implement the mandated change, by altering the semantic tree:
79
left IN Item_singlerow_subselect(subselect)
82
which is represented as
83
Item_in_subselect(left, subselect)
85
subselect= expr3->invalidate_and_restore_select_lex();
86
result= new (session->mem_root) Item_in_subselect(left, subselect);
89
result = negate_expression(session, result);
96
result= new (session->mem_root) Item_func_eq(left, expr);
98
result= new (session->mem_root) Item_func_ne(left, expr);
104
@brief Creates a new Select_Lex for a UNION branch.
106
Sets up and initializes a Select_Lex structure for a query once the parser
107
discovers a UNION token. The current Select_Lex is pushed on the stack and
108
the new Select_Lex becomes the current one..=
110
@lex The parser state.
112
@is_union_distinct True if the union preceding the new select statement
115
@return <code>false</code> if successful, <code>true</code> if an error was
116
reported. In the latter case parsing should stop.
118
bool add_select_to_union_list(Session *session, LEX *lex, bool is_union_distinct)
122
/* Only the last SELECT can have INTO...... */
123
my_error(ER_WRONG_USAGE, MYF(0), "UNION", "INTO");
126
if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE)
128
my_parse_error(session->m_lip);
131
/* This counter shouldn't be incremented for UNION parts */
133
if (new_select(lex, 0))
136
lex->current_select->linkage=UNION_TYPE;
137
if (is_union_distinct) /* UNION DISTINCT - remember position */
138
lex->current_select->master_unit()->union_distinct=
144
@brief Initializes a Select_Lex for a query within parentheses (aka
147
@return false if successful, true if an error was reported. In the latter
148
case parsing should stop.
150
bool setup_select_in_parentheses(Session *session, LEX *lex)
152
Select_Lex * sel= lex->current_select;
153
if (sel->set_braces(1))
155
my_parse_error(session->m_lip);
158
if (sel->linkage == UNION_TYPE &&
159
!sel->master_unit()->first_select()->braces &&
160
sel->master_unit()->first_select()->linkage ==
163
my_parse_error(session->m_lip);
166
if (sel->linkage == UNION_TYPE &&
167
sel->olap != UNSPECIFIED_OLAP_TYPE &&
168
sel->master_unit()->fake_select_lex)
170
my_error(ER_WRONG_USAGE, MYF(0), "CUBE/ROLLUP", "ORDER BY");
173
/* select in braces, can't contain global parameters */
174
if (sel->master_unit()->fake_select_lex)
175
sel->master_unit()->global_parameters=
176
sel->master_unit()->fake_select_lex;
180
Item* reserved_keyword_function(Session *session, const std::string &name, List<Item> *item_list)
182
const plugin::Function *udf= plugin::Function::get(name.c_str(), name.length());
187
item= Create_udf_func::s_singleton.create(session, udf, item_list);
189
my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "FUNCTION", name.c_str());
196
@brief Push an error message into MySQL error stack with line
197
and position information.
199
This function provides semantic action implementers with a way
200
to push the famous "You have a syntax error near..." error
201
message into the error stack, which is normally produced only if
202
a parse error is discovered internally by the Bison generated
205
void my_parse_error(Lex_input_stream *lip)
209
const char *yytext= lip->get_tok_start();
210
/* Push an error into the error stack */
211
my_printf_error(ER_PARSE_ERROR, ER(ER_PARSE_ERROR), MYF(0), ER(ER_SYNTAX_ERROR),
212
(yytext ? yytext : ""),
216
void my_parse_error(const char *message)
218
my_printf_error(ER_PARSE_ERROR_UNKNOWN, ER(ER_PARSE_ERROR_UNKNOWN), MYF(0), message);
221
bool check_reserved_words(LEX_STRING *name)
223
if (!my_strcasecmp(system_charset_info, name->str, "GLOBAL") ||
224
!my_strcasecmp(system_charset_info, name->str, "LOCAL") ||
225
!my_strcasecmp(system_charset_info, name->str, "SESSION"))
233
@brief Bison callback to report a syntax/OOM error
235
This function is invoked by the bison-generated parser
236
when a syntax error, a parse error or an out-of-memory
237
condition occurs. This function is not invoked when the
238
parser is requested to abort by semantic action code
239
by means of YYABORT or YYACCEPT macros. This is why these
240
macros should not be used (use DRIZZLE_YYABORT/DRIZZLE_YYACCEPT
243
The parser will abort immediately after invoking this callback.
245
This function is not for use in semantic actions and is internal to
246
the parser, as it performs some pre-return cleanup.
247
In semantic actions, please use parser::my_parse_error or my_error to
248
push an error into the error stack and DRIZZLE_YYABORT
249
to abort from the parser.
251
void errorOn(drizzled::Session *session, const char *s)
253
/* "parse error" changed into "syntax error" between bison 1.75 and 1.875 */
254
if (strcmp(s,"parse error") == 0 || strcmp(s,"syntax error") == 0)
256
parser::my_parse_error(session->m_lip);
260
parser::my_parse_error(s);
264
bool buildOrderBy(LEX *lex)
266
Select_Lex *sel= lex->current_select;
267
Select_Lex_Unit *unit= sel-> master_unit();
269
if (sel->linkage != GLOBAL_OPTIONS_TYPE &&
270
sel->olap != UNSPECIFIED_OLAP_TYPE &&
271
(sel->linkage != UNION_TYPE || sel->braces))
273
my_error(ER_WRONG_USAGE, MYF(0),
274
"CUBE/ROLLUP", "ORDER BY");
278
if (lex->sql_command != SQLCOM_ALTER_TABLE && !unit->fake_select_lex)
281
A query of the of the form (SELECT ...) ORDER BY order_list is
282
executed in the same way as the query
283
SELECT ... ORDER BY order_list
284
unless the SELECT construct contains ORDER BY or LIMIT clauses.
285
Otherwise we create a fake Select_Lex if it has not been created
288
Select_Lex *first_sl= unit->first_select();
289
if (!unit->is_union() &&
290
(first_sl->order_list.elements ||
291
first_sl->select_limit) &&
292
unit->add_fake_select_lex(lex->session))
301
void buildEngineOption(LEX *lex, const char *key, const LEX_STRING &value)
303
message::Engine::Option *opt= lex->table()->mutable_engine()->add_options();
305
opt->set_state(value.str, value.length);
308
void buildEngineOption(LEX *lex, const char *key, uint64_t value)
310
drizzled::message::Engine::Option *opt= lex->table()->mutable_engine()->add_options();
312
opt->set_state(boost::lexical_cast<std::string>(value));
315
void buildSchemaOption(LEX *lex, const char *key, const LEX_STRING &value)
317
statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
318
message::Engine::Option *opt= statement->schema_message.mutable_engine()->add_options();
320
opt->set_state(value.str, value.length);
323
void buildSchemaOption(LEX *lex, const char *key, uint64_t value)
325
statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
326
message::Engine::Option *opt= statement->schema_message.mutable_engine()->add_options();
328
opt->set_state(boost::lexical_cast<std::string>(value));
331
bool checkFieldIdent(LEX *lex, const LEX_STRING &schema_name, const LEX_STRING &table_name)
333
TableList *table= reinterpret_cast<TableList*>(lex->current_select->table_list.first);
335
if (schema_name.length)
337
if (my_strcasecmp(table_alias_charset, schema_name.str, table->getSchemaName()))
339
my_error(ER_WRONG_DB_NAME, MYF(0), schema_name.str);
344
if (my_strcasecmp(table_alias_charset, table_name.str,
345
table->getTableName()))
347
my_error(ER_WRONG_TABLE_NAME, MYF(0), table_name.str);
354
Item *buildIdent(LEX *lex,
355
const LEX_STRING &schema_name,
356
const LEX_STRING &table_name,
357
const LEX_STRING &field_name)
359
Select_Lex *sel= lex->current_select;
361
if (table_name.length and sel->no_table_names_allowed)
363
my_error(ER_TABLENAME_NOT_ALLOWED_HERE,
364
MYF(0), table_name.str, lex->session->where());
367
Item *item= (sel->parsing_place != IN_HAVING or
368
sel->get_in_sum_expr() > 0) ?
369
(Item*) new Item_field(lex->current_context(), schema_name.str, table_name.str, field_name.str) :
370
(Item*) new Item_ref(lex->current_context(), schema_name.str, table_name.str, field_name.str);
375
Item *buildTableWild(LEX *lex, const LEX_STRING &schema_name, const LEX_STRING &table_name)
377
Select_Lex *sel= lex->current_select;
378
Item *item= new Item_field(lex->current_context(), schema_name.str, table_name.str, "*");
384
void buildCreateFieldIdent(LEX *lex)
386
statement::CreateTable *statement= (statement::CreateTable *)lex->statement;
387
lex->length= lex->dec=0;
389
statement->default_value= statement->on_update_value= 0;
390
statement->comment= null_lex_str;
392
statement->column_format= COLUMN_FORMAT_TYPE_DEFAULT;
394
message::AlterTable &alter_proto= ((statement::CreateTable *)lex->statement)->alter_info.alter_proto;
395
lex->setField(alter_proto.add_added_field());
398
void storeAlterColumnPosition(LEX *lex, const char *position)
400
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
402
lex->last_field->after=const_cast<char*> (position);
403
statement->alter_info.flags.set(ALTER_COLUMN_ORDER);
406
bool buildCollation(LEX *lex, const CHARSET_INFO *arg)
408
statement::CreateTable *statement= (statement::CreateTable *)lex->statement;
410
HA_CREATE_INFO *cinfo= &statement->create_info();
411
if ((cinfo->used_fields & HA_CREATE_USED_DEFAULT_CHARSET) &&
412
cinfo->default_table_charset && arg &&
413
!my_charset_same(cinfo->default_table_charset, arg))
415
my_error(ER_COLLATION_CHARSET_MISMATCH, MYF(0),
416
arg->name, cinfo->default_table_charset->csname);
419
statement->create_info().default_table_charset= arg;
420
statement->create_info().used_fields|= HA_CREATE_USED_DEFAULT_CHARSET;
425
void buildKey(LEX *lex, Key::Keytype type_par, const lex_string_t &name_arg)
427
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
428
Key *key= new Key(type_par, name_arg, &statement->key_create_info, 0,
430
statement->alter_info.key_list.push_back(key);
431
lex->col_list.clear(); /* Alloced by memory::sql_alloc */
434
void buildForeignKey(LEX *lex, const lex_string_t &name_arg, drizzled::Table_ident *table)
436
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
437
Key *key= new Foreign_key(name_arg, lex->col_list,
440
statement->fk_delete_opt,
441
statement->fk_update_opt,
442
statement->fk_match_option);
444
statement->alter_info.key_list.push_back(key);
445
key= new Key(Key::MULTIPLE, name_arg,
446
&default_key_create_info, 1,
448
statement->alter_info.key_list.push_back(key);
449
lex->col_list.clear(); /* Alloced by memory::sql_alloc */
450
/* Only used for ALTER TABLE. Ignored otherwise. */
451
statement->alter_info.flags.set(ALTER_FOREIGN_KEY);
454
drizzled::enum_field_types buildIntegerColumn(LEX *lex, drizzled::enum_field_types final_type, const bool is_unsigned)
456
lex->length=(char*) 0; /* use default length */
460
final_type= DRIZZLE_TYPE_LONGLONG;
465
assert (final_type == DRIZZLE_TYPE_LONG or final_type == DRIZZLE_TYPE_LONGLONG);
466
// We update the type for unsigned types
469
lex->field()->set_type(message::Table::Field::BIGINT);
470
lex->field()->mutable_constraints()->set_is_unsigned(true);
472
else if (final_type == DRIZZLE_TYPE_LONG)
474
lex->field()->set_type(message::Table::Field::INTEGER);
476
else if (final_type == DRIZZLE_TYPE_LONGLONG)
478
lex->field()->set_type(message::Table::Field::BIGINT);
485
drizzled::enum_field_types buildSerialColumn(LEX *lex)
487
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
488
statement->alter_info.flags.set(ALTER_ADD_INDEX);
490
lex->type|= (AUTO_INCREMENT_FLAG | NOT_NULL_FLAG | UNIQUE_FLAG | UNSIGNED_FLAG);
494
lex->field()->mutable_constraints()->set_is_notnull(true);
495
lex->field()->mutable_constraints()->set_is_unsigned(true);
497
lex->field()->set_type(message::Table::Field::BIGINT);
500
return DRIZZLE_TYPE_LONGLONG;
503
drizzled::enum_field_types buildVarcharColumn(LEX *lex, const char *length)
505
lex->length= const_cast<char *>(length);
509
lex->field()->set_type(message::Table::Field::VARCHAR);
511
message::Table::Field::StringFieldOptions *string_field_options;
513
string_field_options= lex->field()->mutable_string_options();
515
string_field_options->set_length(atoi(length));
518
return DRIZZLE_TYPE_VARCHAR;
521
drizzled::enum_field_types buildDecimalColumn(LEX *lex)
524
lex->field()->set_type(message::Table::Field::DECIMAL);
526
return DRIZZLE_TYPE_DECIMAL;
529
drizzled::enum_field_types buildVarbinaryColumn(LEX *lex, const char *length)
531
lex->length= const_cast<char *>(length);
532
lex->charset= &my_charset_bin;
536
lex->field()->set_type(message::Table::Field::VARCHAR);
538
message::Table::Field::StringFieldOptions *string_field_options;
540
string_field_options= lex->field()->mutable_string_options();
542
string_field_options->set_length(atoi(length));
543
string_field_options->set_collation_id(my_charset_bin.number);
544
string_field_options->set_collation(my_charset_bin.name);
547
return DRIZZLE_TYPE_VARCHAR;
550
drizzled::enum_field_types buildBlobColumn(LEX *lex)
552
lex->charset=&my_charset_bin;
553
lex->length=(char*) 0; /* use default length */
557
lex->field()->set_type(message::Table::Field::BLOB);
558
message::Table::Field::StringFieldOptions *string_field_options;
560
string_field_options= lex->field()->mutable_string_options();
561
string_field_options->set_collation_id(my_charset_bin.number);
562
string_field_options->set_collation(my_charset_bin.name);
565
return DRIZZLE_TYPE_BLOB;
568
drizzled::enum_field_types buildBooleanColumn(LEX *lex)
571
lex->field()->set_type(message::Table::Field::BOOLEAN);
573
return DRIZZLE_TYPE_BOOLEAN;
576
drizzled::enum_field_types buildUuidColumn(LEX *lex)
579
lex->field()->set_type(message::Table::Field::UUID);
581
return DRIZZLE_TYPE_UUID;
584
drizzled::enum_field_types buildDoubleColumn(LEX *lex)
588
lex->field()->set_type(message::Table::Field::DOUBLE);
591
return DRIZZLE_TYPE_DOUBLE;
594
drizzled::enum_field_types buildTimestampColumn(LEX *lex, const char *length)
598
lex->field()->set_type(message::Table::Field::EPOCH);
603
lex->length= const_cast<char *>(length);
604
return DRIZZLE_TYPE_MICROTIME;
609
return DRIZZLE_TYPE_TIMESTAMP;
612
void buildKeyOnColumn(LEX *lex)
614
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
616
lex->type|= UNIQUE_KEY_FLAG;
617
statement->alter_info.flags.set(ALTER_ADD_INDEX);
621
lex->field()->mutable_constraints()->set_is_unique(true);
625
void buildAutoOnColumn(LEX *lex)
627
lex->type|= AUTO_INCREMENT_FLAG | NOT_NULL_FLAG;
631
lex->field()->mutable_constraints()->set_is_notnull(true);
635
void buildPrimaryOnColumn(LEX *lex)
637
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
639
lex->type|= PRI_KEY_FLAG | NOT_NULL_FLAG;
640
statement->alter_info.flags.set(ALTER_ADD_INDEX);
644
lex->field()->mutable_constraints()->set_is_notnull(true);
648
void buildReplicationOption(LEX *lex, bool arg)
650
statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
651
message::ReplicationOptions *options= statement->schema_message.mutable_replication_options();
652
options->set_dont_replicate(arg);
655
void buildAddAlterDropIndex(LEX *lex, const char *name, bool is_foreign_key)
657
statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
659
statement->alter_info.flags.set(ALTER_DROP_INDEX);
662
statement->alter_info.flags.set(ALTER_FOREIGN_KEY);
663
statement->alter_info.drop_list.push_back(new AlterDrop(AlterDrop::FOREIGN_KEY, name));
667
statement->alter_info.drop_list.push_back(new AlterDrop(AlterDrop::KEY, name));
671
} // namespace parser
672
} // namespace drizzled