~drizzle-trunk/drizzle/development

2096.1.14 by Brian Aker
Pull in more C code from parser.
1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
 * vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3
 *
4
 *  Copyright (C) 2010 Brian Aker
5
 *
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.
10
 *
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.
15
 *
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
19
 */
20
2173.2.1 by Monty Taylor
Fixes incorrect usage of include
21
#include <config.h>
2096.1.14 by Brian Aker
Pull in more C code from parser.
22
#include <drizzled/parser.h>
2198.1.1 by Olaf van der Spek
Remove unnecessary alter* includes
23
#include <drizzled/alter_info.h>
2318.8.3 by Olaf van der Spek
CreateField includes
24
#include <drizzled/create_field.h>
2318.8.7 by Olaf van der Spek
Add const
25
#include <drizzled/item/subselect.h>
26
#include <drizzled/lex_input_stream.h>
2200.2.18 by Stewart Smith
add fields, indexes and foreign keys to drop in ALTER TABLE DROP and DROP INDEX to the AlterTable protobuf message in parser
27
#include <drizzled/message/alter_table.pb.h>
2318.8.7 by Olaf van der Spek
Add const
28
#include <drizzled/session.h>
2234.1.4 by Olaf van der Spek
Refactor includes
29
#include <drizzled/sql_lex.h>
2096.1.14 by Brian Aker
Pull in more C code from parser.
30
2198.1.1 by Olaf van der Spek
Remove unnecessary alter* includes
31
namespace drizzled {
32
namespace parser {
2096.1.14 by Brian Aker
Pull in more C code from parser.
33
34
/**
35
  Helper to resolve the SQL:2003 Syntax exception 1) in <in predicate>.
36
  See SQL:2003, Part 2, section 8.4 <in predicate>, Note 184, page 383.
37
  This function returns the proper item for the SQL expression
38
  <code>left [NOT] IN ( expr )</code>
39
  @param session the current thread
40
  @param left the in predicand
41
  @param equal true for IN predicates, false for NOT IN predicates
42
  @param expr first and only expression of the in value list
43
  @return an expression representing the IN predicate.
44
*/
45
46
Item* handle_sql2003_note184_exception(Session *session, Item* left, bool equal, Item *expr)
47
{
48
  /*
49
    Relevant references for this issue:
50
    - SQL:2003, Part 2, section 8.4 <in predicate>, page 383,
51
    - SQL:2003, Part 2, section 7.2 <row value expression>, page 296,
52
    - SQL:2003, Part 2, section 6.3 <value expression primary>, page 174,
53
    - SQL:2003, Part 2, section 7.15 <subquery>, page 370,
54
    - SQL:2003 Feature F561, "Full value expressions".
55
56
    The exception in SQL:2003 Note 184 means:
57
    Item_singlerow_subselect, which corresponds to a <scalar subquery>,
58
    should be re-interpreted as an Item_in_subselect, which corresponds
59
    to a <table subquery> when used inside an <in predicate>.
60
61
    Our reading of Note 184 is reccursive, so that all:
62
    - IN (( <subquery> ))
63
    - IN ((( <subquery> )))
64
    - IN '('^N <subquery> ')'^N
65
    - etc
66
    should be interpreted as a <table subquery>, no matter how deep in the
67
    expression the <subquery> is.
68
  */
69
70
  Item *result;
71
72
  if (expr->type() == Item::SUBSELECT_ITEM)
73
  {
74
    Item_subselect *expr2 = (Item_subselect*) expr;
75
76
    if (expr2->substype() == Item_subselect::SINGLEROW_SUBS)
77
    {
78
      Item_singlerow_subselect *expr3 = (Item_singlerow_subselect*) expr2;
79
      Select_Lex *subselect;
80
81
      /*
82
        Implement the mandated change, by altering the semantic tree:
83
          left IN Item_singlerow_subselect(subselect)
84
        is modified to
85
          left IN (subselect)
86
        which is represented as
87
          Item_in_subselect(left, subselect)
88
      */
89
      subselect= expr3->invalidate_and_restore_select_lex();
90
      result= new (session->mem_root) Item_in_subselect(left, subselect);
91
92
      if (! equal)
93
        result = negate_expression(session, result);
94
95
      return(result);
96
    }
97
  }
98
99
  if (equal)
100
    result= new (session->mem_root) Item_func_eq(left, expr);
101
  else
102
    result= new (session->mem_root) Item_func_ne(left, expr);
103
104
  return(result);
105
}
106
107
/**
108
   @brief Creates a new Select_Lex for a UNION branch.
109
110
   Sets up and initializes a Select_Lex structure for a query once the parser
111
   discovers a UNION token. The current Select_Lex is pushed on the stack and
112
   the new Select_Lex becomes the current one..=
113
114
   @lex The parser state.
115
116
   @is_union_distinct True if the union preceding the new select statement
117
   uses UNION DISTINCT.
118
119
   @return <code>false</code> if successful, <code>true</code> if an error was
120
   reported. In the latter case parsing should stop.
121
 */
122
bool add_select_to_union_list(Session *session, LEX *lex, bool is_union_distinct)
123
{
124
  if (lex->result)
125
  {
126
    /* Only the last SELECT can have  INTO...... */
127
    my_error(ER_WRONG_USAGE, MYF(0), "UNION", "INTO");
128
    return true;
129
  }
130
  if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE)
131
  {
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
132
    my_parse_error(session->m_lip);
2096.1.14 by Brian Aker
Pull in more C code from parser.
133
    return true;
134
  }
135
  /* This counter shouldn't be incremented for UNION parts */
136
  lex->nest_level--;
137
  if (new_select(lex, 0))
138
    return true;
139
  init_select(lex);
140
  lex->current_select->linkage=UNION_TYPE;
141
  if (is_union_distinct) /* UNION DISTINCT - remember position */
142
    lex->current_select->master_unit()->union_distinct=
143
      lex->current_select;
144
  return false;
145
}
146
147
/**
148
   @brief Initializes a Select_Lex for a query within parentheses (aka
149
   braces).
150
151
   @return false if successful, true if an error was reported. In the latter
152
   case parsing should stop.
153
 */
154
bool setup_select_in_parentheses(Session *session, LEX *lex)
155
{
156
  Select_Lex * sel= lex->current_select;
157
  if (sel->set_braces(1))
158
  {
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
159
    my_parse_error(session->m_lip);
2096.1.14 by Brian Aker
Pull in more C code from parser.
160
    return true;
161
  }
162
  if (sel->linkage == UNION_TYPE &&
163
      !sel->master_unit()->first_select()->braces &&
164
      sel->master_unit()->first_select()->linkage ==
165
      UNION_TYPE)
166
  {
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
167
    my_parse_error(session->m_lip);
2096.1.14 by Brian Aker
Pull in more C code from parser.
168
    return true;
169
  }
170
  if (sel->linkage == UNION_TYPE &&
171
      sel->olap != UNSPECIFIED_OLAP_TYPE &&
172
      sel->master_unit()->fake_select_lex)
173
  {
174
    my_error(ER_WRONG_USAGE, MYF(0), "CUBE/ROLLUP", "ORDER BY");
175
    return true;
176
  }
177
  /* select in braces, can't contain global parameters */
178
  if (sel->master_unit()->fake_select_lex)
179
    sel->master_unit()->global_parameters=
180
      sel->master_unit()->fake_select_lex;
181
  return false;
182
}
183
184
Item* reserved_keyword_function(Session *session, const std::string &name, List<Item> *item_list)
185
{
2198.6.15 by Brian Aker
Remove caller convert to std::string (ie we were going back and forth in
186
  const plugin::Function *udf= plugin::Function::get(name);
2096.1.14 by Brian Aker
Pull in more C code from parser.
187
188
  if (udf)
189
  {
2198.6.15 by Brian Aker
Remove caller convert to std::string (ie we were going back and forth in
190
    return Create_udf_func::s_singleton.create(session, udf, item_list);
2096.1.14 by Brian Aker
Pull in more C code from parser.
191
  }
192
2198.6.15 by Brian Aker
Remove caller convert to std::string (ie we were going back and forth in
193
  my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "FUNCTION", name.c_str());
194
195
  return NULL;
2096.1.14 by Brian Aker
Pull in more C code from parser.
196
}
197
198
/**
199
  @brief Push an error message into MySQL error stack with line
200
  and position information.
201
202
  This function provides semantic action implementers with a way
203
  to push the famous "You have a syntax error near..." error
204
  message into the error stack, which is normally produced only if
205
  a parse error is discovered internally by the Bison generated
206
  parser.
207
*/
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
208
void my_parse_error(Lex_input_stream *lip)
2096.1.14 by Brian Aker
Pull in more C code from parser.
209
{
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
210
  assert(lip);
2096.1.14 by Brian Aker
Pull in more C code from parser.
211
212
  const char *yytext= lip->get_tok_start();
213
  /* Push an error into the error stack */
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
214
  my_printf_error(ER_PARSE_ERROR,  ER(ER_PARSE_ERROR), MYF(0), ER(ER_SYNTAX_ERROR),
2096.1.14 by Brian Aker
Pull in more C code from parser.
215
                  (yytext ? yytext : ""),
216
                  lip->yylineno);
217
}
218
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
219
void my_parse_error(const char *message)
220
{
221
  my_printf_error(ER_PARSE_ERROR_UNKNOWN, ER(ER_PARSE_ERROR_UNKNOWN), MYF(0), message);
222
}
223
2371.1.2 by Brian Aker
Remove the typedef on lexkey
224
bool check_reserved_words(lex_string_t *name)
2096.1.15 by Brian Aker
Additional parser cleanup.
225
{
226
  if (!my_strcasecmp(system_charset_info, name->str, "GLOBAL") ||
227
      !my_strcasecmp(system_charset_info, name->str, "LOCAL") ||
228
      !my_strcasecmp(system_charset_info, name->str, "SESSION"))
229
    return true;
230
231
  return false;
232
}
233
234
2104.3.1 by Brian Aker
Merge in ulong typedef used.
235
/**
236
  @brief Bison callback to report a syntax/OOM error
237
238
  This function is invoked by the bison-generated parser
239
  when a syntax error, a parse error or an out-of-memory
240
  condition occurs. This function is not invoked when the
241
  parser is requested to abort by semantic action code
242
  by means of YYABORT or YYACCEPT macros. This is why these
243
  macros should not be used (use DRIZZLE_YYABORT/DRIZZLE_YYACCEPT
244
  instead).
245
246
  The parser will abort immediately after invoking this callback.
247
248
  This function is not for use in semantic actions and is internal to
249
  the parser, as it performs some pre-return cleanup.
250
  In semantic actions, please use parser::my_parse_error or my_error to
251
  push an error into the error stack and DRIZZLE_YYABORT
252
  to abort from the parser.
253
*/
2172.3.8 by Brian Aker
Modernize our call to the parser, no more casting required.
254
void errorOn(drizzled::Session *session, const char *s)
2104.3.1 by Brian Aker
Merge in ulong typedef used.
255
{
256
  /* "parse error" changed into "syntax error" between bison 1.75 and 1.875 */
257
  if (strcmp(s,"parse error") == 0 || strcmp(s,"syntax error") == 0)
2104.3.5 by Brian Aker
Fixed case where parser error message might be null.
258
  {
259
    parser::my_parse_error(session->m_lip);
260
  }
261
  else
262
  {
263
    parser::my_parse_error(s);
264
  }
2104.3.1 by Brian Aker
Merge in ulong typedef used.
265
}
266
2114.4.9 by Brian Aker
Try not to pass session quite as often
267
bool buildOrderBy(LEX *lex)
2104.3.4 by Brian Aker
Encapsulate from parser buildOrderBy()
268
{
2114.4.9 by Brian Aker
Try not to pass session quite as often
269
  Select_Lex *sel= lex->current_select;
2104.3.4 by Brian Aker
Encapsulate from parser buildOrderBy()
270
  Select_Lex_Unit *unit= sel-> master_unit();
271
272
  if (sel->linkage != GLOBAL_OPTIONS_TYPE &&
273
      sel->olap != UNSPECIFIED_OLAP_TYPE &&
274
      (sel->linkage != UNION_TYPE || sel->braces))
275
  {
276
    my_error(ER_WRONG_USAGE, MYF(0),
277
             "CUBE/ROLLUP", "ORDER BY");
278
    return false;
279
  }
280
2114.4.9 by Brian Aker
Try not to pass session quite as often
281
  if (lex->sql_command != SQLCOM_ALTER_TABLE && !unit->fake_select_lex)
2104.3.4 by Brian Aker
Encapsulate from parser buildOrderBy()
282
  {
283
    /*
284
      A query of the of the form (SELECT ...) ORDER BY order_list is
285
      executed in the same way as the query
286
      SELECT ... ORDER BY order_list
287
      unless the SELECT construct contains ORDER BY or LIMIT clauses.
288
      Otherwise we create a fake Select_Lex if it has not been created
289
      yet.
290
    */
291
    Select_Lex *first_sl= unit->first_select();
292
    if (!unit->is_union() &&
293
        (first_sl->order_list.elements ||
294
         first_sl->select_limit) &&           
2114.4.9 by Brian Aker
Try not to pass session quite as often
295
        unit->add_fake_select_lex(lex->session))
2104.3.4 by Brian Aker
Encapsulate from parser buildOrderBy()
296
    {
297
      return false;
298
    }
299
  }
300
301
  return true;
302
}
2096.1.14 by Brian Aker
Pull in more C code from parser.
303
2371.1.2 by Brian Aker
Remove the typedef on lexkey
304
void buildEngineOption(LEX *lex, const char *key, const lex_string_t &value)
2114.4.9 by Brian Aker
Try not to pass session quite as often
305
{
306
  message::Engine::Option *opt= lex->table()->mutable_engine()->add_options();
307
  opt->set_name(key);
308
  opt->set_state(value.str, value.length);
309
}
310
311
void buildEngineOption(LEX *lex, const char *key, uint64_t value)
312
{
313
  drizzled::message::Engine::Option *opt= lex->table()->mutable_engine()->add_options();
314
  opt->set_name(key);
315
  opt->set_state(boost::lexical_cast<std::string>(value));
316
}
317
2371.1.2 by Brian Aker
Remove the typedef on lexkey
318
void buildSchemaOption(LEX *lex, const char *key, const lex_string_t &value)
2114.4.9 by Brian Aker
Try not to pass session quite as often
319
{
320
  statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
321
  message::Engine::Option *opt= statement->schema_message.mutable_engine()->add_options();
322
  opt->set_name(key);
323
  opt->set_state(value.str, value.length);
324
}
325
2371.1.2 by Brian Aker
Remove the typedef on lexkey
326
void buildSchemaDefiner(LEX *lex, const lex_string_t &value)
2241.6.2 by Brian Aker
This adds the concept of a definer to a table definition.
327
{
328
  statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
329
  identifier::User user(value.str);
330
  message::set_definer(statement->schema_message, user);
331
}
332
333
void buildSchemaDefiner(LEX *lex, const identifier::User &user)
334
{
335
  statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
336
  message::set_definer(statement->schema_message, user);
337
}
338
2114.4.9 by Brian Aker
Try not to pass session quite as often
339
void buildSchemaOption(LEX *lex, const char *key, uint64_t value)
340
{
341
  statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
342
  message::Engine::Option *opt= statement->schema_message.mutable_engine()->add_options();
343
  opt->set_name(key);
344
  opt->set_state(boost::lexical_cast<std::string>(value));
345
}
346
2371.1.2 by Brian Aker
Remove the typedef on lexkey
347
bool checkFieldIdent(LEX *lex, const lex_string_t &schema_name, const lex_string_t &table_name)
2114.4.9 by Brian Aker
Try not to pass session quite as often
348
{
349
  TableList *table= reinterpret_cast<TableList*>(lex->current_select->table_list.first);
2114.4.4 by Brian Aker
Merge in field ident wrap from parser.
350
2114.4.6 by Brian Aker
Remove duplicate call by using null
351
  if (schema_name.length)
2114.4.4 by Brian Aker
Merge in field ident wrap from parser.
352
  {
2114.4.6 by Brian Aker
Remove duplicate call by using null
353
    if (my_strcasecmp(table_alias_charset, schema_name.str, table->getSchemaName()))
354
    {
355
      my_error(ER_WRONG_DB_NAME, MYF(0), schema_name.str);
356
      return false;
357
    }
2114.4.4 by Brian Aker
Merge in field ident wrap from parser.
358
  }
359
360
  if (my_strcasecmp(table_alias_charset, table_name.str,
361
                    table->getTableName()))
362
  {
363
    my_error(ER_WRONG_TABLE_NAME, MYF(0), table_name.str);
364
    return false;
365
  }
366
367
  return true;
368
}
369
2114.4.9 by Brian Aker
Try not to pass session quite as often
370
Item *buildIdent(LEX *lex,
2371.1.2 by Brian Aker
Remove the typedef on lexkey
371
                 const lex_string_t &schema_name,
372
                 const lex_string_t &table_name,
373
                 const lex_string_t &field_name)
2114.4.5 by Brian Aker
Remove duplicate code around ident check.
374
{
2114.4.9 by Brian Aker
Try not to pass session quite as often
375
  Select_Lex *sel= lex->current_select;
2114.4.5 by Brian Aker
Remove duplicate code around ident check.
376
2114.4.7 by Brian Aker
Pull simple_ident into method.
377
  if (table_name.length and sel->no_table_names_allowed)
2114.4.5 by Brian Aker
Remove duplicate code around ident check.
378
  {
379
    my_error(ER_TABLENAME_NOT_ALLOWED_HERE,
2114.4.10 by Brian Aker
Remove current_session from a couple of locations, encapsulate where in
380
             MYF(0), table_name.str, lex->session->where());
2114.4.5 by Brian Aker
Remove duplicate code around ident check.
381
  }
382
383
  Item *item= (sel->parsing_place != IN_HAVING or
384
               sel->get_in_sum_expr() > 0) ?
2114.4.9 by Brian Aker
Try not to pass session quite as often
385
    (Item*) new Item_field(lex->current_context(), schema_name.str, table_name.str, field_name.str) :
386
    (Item*) new Item_ref(lex->current_context(), schema_name.str, table_name.str, field_name.str);
2114.4.5 by Brian Aker
Remove duplicate code around ident check.
387
388
  return item;
389
}
390
2371.1.2 by Brian Aker
Remove the typedef on lexkey
391
Item *buildTableWild(LEX *lex, const lex_string_t &schema_name, const lex_string_t &table_name)
2114.4.8 by Brian Aker
Merge in additional parser encapsulation.
392
{
2114.4.9 by Brian Aker
Try not to pass session quite as often
393
  Select_Lex *sel= lex->current_select;
394
  Item *item= new Item_field(lex->current_context(), schema_name.str, table_name.str, "*");
2114.4.8 by Brian Aker
Merge in additional parser encapsulation.
395
  sel->with_wild++;
396
397
  return item;
398
}
399
2114.4.9 by Brian Aker
Try not to pass session quite as often
400
void buildCreateFieldIdent(LEX *lex)
401
{
402
  statement::CreateTable *statement= (statement::CreateTable *)lex->statement;
403
  lex->length= lex->dec=0;
404
  lex->type=0;
405
  statement->default_value= statement->on_update_value= 0;
2371.1.2 by Brian Aker
Remove the typedef on lexkey
406
  statement->comment= null_lex_string();
2114.4.9 by Brian Aker
Try not to pass session quite as often
407
  lex->charset= NULL;
408
  statement->column_format= COLUMN_FORMAT_TYPE_DEFAULT;
409
2200.2.1 by Stewart Smith
rename old alter table proto to AddedFields as it really has a dual purpose and isn't how we should structure Alter Table.
410
  message::AddedFields &added_fields_proto= ((statement::CreateTable *)lex->statement)->alter_info.added_fields_proto;
411
  lex->setField(added_fields_proto.add_added_field());
2114.4.9 by Brian Aker
Try not to pass session quite as often
412
}
413
2137.1.13 by Brian Aker
Fix bad error in warnings/errors.
414
void storeAlterColumnPosition(LEX *lex, const char *position)
415
{
416
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
417
2385.1.2 by Olaf van der Spek
Refactor Table_ident
418
  lex->last_field->after= position;
2137.1.13 by Brian Aker
Fix bad error in warnings/errors.
419
  statement->alter_info.flags.set(ALTER_COLUMN_ORDER);
420
}
421
2254 by Brian Aker
Shift CHARSET_INFO to charset_info_st
422
bool buildCollation(LEX *lex, const charset_info_st *arg)
2148.5.4 by Brian Aker
Move over the collation test to a function outside of the parser.
423
{
424
  statement::CreateTable *statement= (statement::CreateTable *)lex->statement;
425
426
  HA_CREATE_INFO *cinfo= &statement->create_info();
427
  if ((cinfo->used_fields & HA_CREATE_USED_DEFAULT_CHARSET) &&
428
      cinfo->default_table_charset && arg &&
429
      !my_charset_same(cinfo->default_table_charset, arg))
430
  {
431
    my_error(ER_COLLATION_CHARSET_MISMATCH, MYF(0),
432
             arg->name, cinfo->default_table_charset->csname);
433
    return false;
434
  }
435
  statement->create_info().default_table_charset= arg;
436
  statement->create_info().used_fields|= HA_CREATE_USED_DEFAULT_CHARSET;
437
438
  return true;
439
}
440
2148.5.5 by Brian Aker
buildKey for parser.
441
void buildKey(LEX *lex, Key::Keytype type_par, const lex_string_t &name_arg)
442
{
443
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
2385.1.2 by Olaf van der Spek
Refactor Table_ident
444
  Key *key= new Key(type_par, name_arg, &statement->key_create_info, 0, lex->col_list);
2148.5.5 by Brian Aker
buildKey for parser.
445
  statement->alter_info.key_list.push_back(key);
2179.2.1 by Olaf van der Spek
Rename List::empty to clear (std::list uses clear)
446
  lex->col_list.clear(); /* Alloced by memory::sql_alloc */
2148.5.5 by Brian Aker
buildKey for parser.
447
}
448
2148.5.6 by Brian Aker
Additional encapsulation.
449
void buildForeignKey(LEX *lex, const lex_string_t &name_arg, drizzled::Table_ident *table)
450
{
451
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
452
  Key *key= new Foreign_key(name_arg, lex->col_list,
453
                            table,
454
                            lex->ref_list,
455
                            statement->fk_delete_opt,
456
                            statement->fk_update_opt,
457
                            statement->fk_match_option);
458
459
  statement->alter_info.key_list.push_back(key);
2385.1.2 by Olaf van der Spek
Refactor Table_ident
460
  key= new Key(Key::MULTIPLE, name_arg, &default_key_create_info, 1, lex->col_list);
2148.5.6 by Brian Aker
Additional encapsulation.
461
  statement->alter_info.key_list.push_back(key);
2179.2.1 by Olaf van der Spek
Rename List::empty to clear (std::list uses clear)
462
  lex->col_list.clear(); /* Alloced by memory::sql_alloc */
2148.5.6 by Brian Aker
Additional encapsulation.
463
  /* Only used for ALTER TABLE. Ignored otherwise. */
464
  statement->alter_info.flags.set(ALTER_FOREIGN_KEY);
465
}
466
2148.5.8 by Brian Aker
Move integer building into parser code file.
467
drizzled::enum_field_types buildIntegerColumn(LEX *lex, drizzled::enum_field_types final_type, const bool is_unsigned)
468
{ 
2385.1.2 by Olaf van der Spek
Refactor Table_ident
469
  lex->length= NULL; /* use default length */
2148.5.8 by Brian Aker
Move integer building into parser code file.
470
471
  if (is_unsigned)
472
  {
473
    final_type= DRIZZLE_TYPE_LONGLONG;
474
  }
475
476
  if (lex->field())
477
  {
478
    assert (final_type == DRIZZLE_TYPE_LONG or final_type == DRIZZLE_TYPE_LONGLONG);
479
    // We update the type for unsigned types
480
    if (is_unsigned)
481
    {
482
      lex->field()->set_type(message::Table::Field::BIGINT);
483
      lex->field()->mutable_constraints()->set_is_unsigned(true);
484
    }
485
    else if (final_type == DRIZZLE_TYPE_LONG)
486
    {
487
      lex->field()->set_type(message::Table::Field::INTEGER);
488
    }
489
    else if (final_type == DRIZZLE_TYPE_LONGLONG)
490
    {
491
      lex->field()->set_type(message::Table::Field::BIGINT);
492
    }
493
  }
494
495
  return final_type;
496
}
497
2148.5.9 by Brian Aker
Merge in column varchar build.
498
drizzled::enum_field_types buildSerialColumn(LEX *lex)
499
{
2148.5.15 by Brian Aker
Fix serial to be the same in the parser.
500
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
501
  statement->alter_info.flags.set(ALTER_ADD_INDEX);
502
2148.5.9 by Brian Aker
Merge in column varchar build.
503
  lex->type|= (AUTO_INCREMENT_FLAG | NOT_NULL_FLAG | UNIQUE_FLAG | UNSIGNED_FLAG);
504
505
  if (lex->field())
506
  {
507
    lex->field()->mutable_constraints()->set_is_notnull(true);
508
    lex->field()->mutable_constraints()->set_is_unsigned(true);
509
510
    lex->field()->set_type(message::Table::Field::BIGINT);
511
  }
512
513
  return DRIZZLE_TYPE_LONGLONG;
514
}
515
516
drizzled::enum_field_types buildVarcharColumn(LEX *lex, const char *length)
517
{
2385.1.2 by Olaf van der Spek
Refactor Table_ident
518
  lex->length= length;
2148.5.9 by Brian Aker
Merge in column varchar build.
519
520
  if (lex->field())
521
  {
522
    lex->field()->set_type(message::Table::Field::VARCHAR);
523
524
    message::Table::Field::StringFieldOptions *string_field_options;
525
526
    string_field_options= lex->field()->mutable_string_options();
527
528
    string_field_options->set_length(atoi(length));
529
  }
530
531
  return DRIZZLE_TYPE_VARCHAR;
532
}
533
2148.5.14 by Brian Aker
function for decimal to be pulled out of the parser.
534
drizzled::enum_field_types buildDecimalColumn(LEX *lex)
535
{
536
  if (lex->field())
537
    lex->field()->set_type(message::Table::Field::DECIMAL);
538
539
  return DRIZZLE_TYPE_DECIMAL;
540
}
541
2148.5.10 by Brian Aker
Merge in timestamp build parser function.
542
drizzled::enum_field_types buildVarbinaryColumn(LEX *lex, const char *length)
543
{
2385.1.2 by Olaf van der Spek
Refactor Table_ident
544
  lex->length= length;
2148.5.11 by Brian Aker
Merge in varbinary build method.
545
  lex->charset= &my_charset_bin;
2148.5.10 by Brian Aker
Merge in timestamp build parser function.
546
547
  if (lex->field())
548
  {
549
    lex->field()->set_type(message::Table::Field::VARCHAR);
550
551
    message::Table::Field::StringFieldOptions *string_field_options;
552
553
    string_field_options= lex->field()->mutable_string_options();
554
555
    string_field_options->set_length(atoi(length));
556
    string_field_options->set_collation_id(my_charset_bin.number);
557
    string_field_options->set_collation(my_charset_bin.name);
558
  }
559
560
  return DRIZZLE_TYPE_VARCHAR;
561
}
562
563
drizzled::enum_field_types buildBlobColumn(LEX *lex)
564
{
565
  lex->charset=&my_charset_bin;
566
  lex->length=(char*) 0; /* use default length */
567
568
  if (lex->field())
569
  {
570
    lex->field()->set_type(message::Table::Field::BLOB);
571
    message::Table::Field::StringFieldOptions *string_field_options;
572
573
    string_field_options= lex->field()->mutable_string_options();
574
    string_field_options->set_collation_id(my_charset_bin.number);
575
    string_field_options->set_collation(my_charset_bin.name);
576
  }
577
578
  return DRIZZLE_TYPE_BLOB;
579
}
580
581
drizzled::enum_field_types buildBooleanColumn(LEX *lex)
582
{
583
  if (lex->field())
584
    lex->field()->set_type(message::Table::Field::BOOLEAN);
585
586
  return DRIZZLE_TYPE_BOOLEAN;
587
}
588
589
drizzled::enum_field_types buildUuidColumn(LEX *lex)
590
{
591
  if (lex->field())
592
    lex->field()->set_type(message::Table::Field::UUID);
593
594
  return DRIZZLE_TYPE_UUID;
595
}
596
597
drizzled::enum_field_types buildDoubleColumn(LEX *lex)
598
{
599
  if (lex->field())
600
  {
601
    lex->field()->set_type(message::Table::Field::DOUBLE);
602
  }
603
604
  return DRIZZLE_TYPE_DOUBLE;
605
}
606
607
drizzled::enum_field_types buildTimestampColumn(LEX *lex, const char *length)
608
{
609
  if (lex->field())
610
  {
611
    lex->field()->set_type(message::Table::Field::EPOCH);
612
  }
613
614
  if (length)
615
  {
2385.1.2 by Olaf van der Spek
Refactor Table_ident
616
    lex->length= length;
2148.5.10 by Brian Aker
Merge in timestamp build parser function.
617
    return DRIZZLE_TYPE_MICROTIME;
618
  }
619
620
  lex->length= NULL;
621
622
  return DRIZZLE_TYPE_TIMESTAMP;
623
}
624
2148.5.16 by Brian Aker
Adding build for unique column
625
void buildKeyOnColumn(LEX *lex)
626
{
627
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
628
629
  lex->type|= UNIQUE_KEY_FLAG;
630
  statement->alter_info.flags.set(ALTER_ADD_INDEX);
631
632
  if (lex->field())
633
  {
634
    lex->field()->mutable_constraints()->set_is_unique(true);
635
  }
636
}
637
2148.5.17 by Brian Aker
Merge up parser work for additional column builds.
638
void buildAutoOnColumn(LEX *lex)
639
{
640
  lex->type|= AUTO_INCREMENT_FLAG | NOT_NULL_FLAG;
641
642
  if (lex->field())
643
  {
644
    lex->field()->mutable_constraints()->set_is_notnull(true);
645
  }
646
}
647
648
void buildPrimaryOnColumn(LEX *lex)
649
{
650
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
651
652
  lex->type|= PRI_KEY_FLAG | NOT_NULL_FLAG;
653
  statement->alter_info.flags.set(ALTER_ADD_INDEX);
654
655
  if (lex->field())
656
  {
657
    lex->field()->mutable_constraints()->set_is_notnull(true);
658
  }
659
}
660
2172.3.3 by Brian Aker
Abstract out the replication set.
661
void buildReplicationOption(LEX *lex, bool arg)
662
{
663
  statement::CreateSchema *statement= (statement::CreateSchema *)lex->statement;
2187.7.6 by Brian Aker
This fixes the message such that the table inherits the no replication
664
  message::set_is_replicated(statement->schema_message, arg);
2172.3.3 by Brian Aker
Abstract out the replication set.
665
}
666
2172.3.4 by Brian Aker
Move out drop keys to parser method file.
667
void buildAddAlterDropIndex(LEX *lex, const char *name, bool is_foreign_key)
668
{
669
  statement::AlterTable *statement= (statement::AlterTable *)lex->statement;
670
2200.2.18 by Stewart Smith
add fields, indexes and foreign keys to drop in ALTER TABLE DROP and DROP INDEX to the AlterTable protobuf message in parser
671
  message::AlterTable::AlterTableOperation *operation;
672
  operation= lex->alter_table()->add_operations();
673
  operation->set_drop_name(name);
674
2172.3.4 by Brian Aker
Move out drop keys to parser method file.
675
  statement->alter_info.flags.set(ALTER_DROP_INDEX);
676
  if (is_foreign_key)
677
  {
678
    statement->alter_info.flags.set(ALTER_FOREIGN_KEY);
2200.2.18 by Stewart Smith
add fields, indexes and foreign keys to drop in ALTER TABLE DROP and DROP INDEX to the AlterTable protobuf message in parser
679
    operation->set_operation(message::AlterTable::AlterTableOperation::DROP_FOREIGN_KEY);
2172.3.4 by Brian Aker
Move out drop keys to parser method file.
680
  }
681
  else
682
  {
2200.2.18 by Stewart Smith
add fields, indexes and foreign keys to drop in ALTER TABLE DROP and DROP INDEX to the AlterTable protobuf message in parser
683
    operation->set_operation(message::AlterTable::AlterTableOperation::DROP_KEY);
2172.3.4 by Brian Aker
Move out drop keys to parser method file.
684
  }
685
}
686
2096.1.14 by Brian Aker
Pull in more C code from parser.
687
} // namespace parser
688
} // namespace drizzled