1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2009 Sun Microsystems
5
* Copyright (c) 2010 Jay Pipes
9
* Jay Pipes <jaypipes@gmail.com>
11
* This program is free software; you can redistribute it and/or modify
12
* it under the terms of the GNU General Public License as published by
13
* the Free Software Foundation; version 2 of the License.
15
* This program is distributed in the hope that it will be useful,
16
* but WITHOUT ANY WARRANTY; without even the implied warranty of
17
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18
* GNU General Public License for more details.
20
* You should have received a copy of the GNU General Public License
21
* along with this program; if not, write to the Free Software
22
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
28
* Implementation of various routines that can be used to convert
29
* Statement messages to other formats, including SQL strings.
34
#include "drizzled/message/statement_transform.h"
35
#include "drizzled/message/transaction.pb.h"
36
#include "drizzled/message/table.pb.h"
50
enum TransformSqlError
51
transformStatementToSql(const Statement &source,
52
vector<string> &sql_strings,
53
enum TransformSqlVariant sql_variant,
54
bool already_in_transaction)
56
TransformSqlError error= NONE;
58
switch (source.type())
60
case Statement::INSERT:
62
if (! source.has_insert_header())
64
error= MISSING_HEADER;
67
if (! source.has_insert_data())
73
const InsertHeader &insert_header= source.insert_header();
74
const InsertData &insert_data= source.insert_data();
75
size_t num_keys= insert_data.record_size();
78
if (num_keys > 1 && ! already_in_transaction)
79
sql_strings.push_back("START TRANSACTION");
81
for (x= 0; x < num_keys; ++x)
85
error= transformInsertRecordToSql(insert_header,
86
insert_data.record(x),
92
sql_strings.push_back(destination);
95
if (num_keys > 1 && ! already_in_transaction)
98
sql_strings.push_back("COMMIT");
100
sql_strings.push_back("ROLLBACK");
104
case Statement::UPDATE:
106
if (! source.has_update_header())
108
error= MISSING_HEADER;
111
if (! source.has_update_data())
117
const UpdateHeader &update_header= source.update_header();
118
const UpdateData &update_data= source.update_data();
119
size_t num_keys= update_data.record_size();
122
if (num_keys > 1 && ! already_in_transaction)
123
sql_strings.push_back("START TRANSACTION");
125
for (x= 0; x < num_keys; ++x)
129
error= transformUpdateRecordToSql(update_header,
130
update_data.record(x),
136
sql_strings.push_back(destination);
139
if (num_keys > 1 && ! already_in_transaction)
142
sql_strings.push_back("COMMIT");
144
sql_strings.push_back("ROLLBACK");
148
case Statement::DELETE:
150
if (! source.has_delete_header())
152
error= MISSING_HEADER;
155
if (! source.has_delete_data())
161
const DeleteHeader &delete_header= source.delete_header();
162
const DeleteData &delete_data= source.delete_data();
163
size_t num_keys= delete_data.record_size();
166
if (num_keys > 1 && ! already_in_transaction)
167
sql_strings.push_back("START TRANSACTION");
169
for (x= 0; x < num_keys; ++x)
173
error= transformDeleteRecordToSql(delete_header,
174
delete_data.record(x),
180
sql_strings.push_back(destination);
183
if (num_keys > 1 && ! already_in_transaction)
186
sql_strings.push_back("COMMIT");
188
sql_strings.push_back("ROLLBACK");
192
case Statement::CREATE_TABLE:
194
assert(source.has_create_table_statement());
196
error= transformCreateTableStatementToSql(source.create_table_statement(),
199
sql_strings.push_back(destination);
202
case Statement::TRUNCATE_TABLE:
204
assert(source.has_truncate_table_statement());
206
error= transformTruncateTableStatementToSql(source.truncate_table_statement(),
209
sql_strings.push_back(destination);
212
case Statement::DROP_TABLE:
214
assert(source.has_drop_table_statement());
216
error= transformDropTableStatementToSql(source.drop_table_statement(),
219
sql_strings.push_back(destination);
222
case Statement::CREATE_SCHEMA:
224
assert(source.has_create_schema_statement());
226
error= transformCreateSchemaStatementToSql(source.create_schema_statement(),
229
sql_strings.push_back(destination);
232
case Statement::DROP_SCHEMA:
234
assert(source.has_drop_schema_statement());
236
error= transformDropSchemaStatementToSql(source.drop_schema_statement(),
239
sql_strings.push_back(destination);
242
case Statement::SET_VARIABLE:
244
assert(source.has_set_variable_statement());
246
error= transformSetVariableStatementToSql(source.set_variable_statement(),
249
sql_strings.push_back(destination);
252
case Statement::RAW_SQL:
254
sql_strings.push_back(source.sql());
260
enum TransformSqlError
261
transformInsertHeaderToSql(const InsertHeader &header,
263
enum TransformSqlVariant sql_variant)
265
char quoted_identifier= '`';
266
if (sql_variant == ANSI)
267
quoted_identifier= '"';
269
destination.assign("INSERT INTO ", 12);
270
destination.push_back(quoted_identifier);
271
destination.append(header.table_metadata().schema_name());
272
destination.push_back(quoted_identifier);
273
destination.push_back('.');
274
destination.push_back(quoted_identifier);
275
destination.append(header.table_metadata().table_name());
276
destination.push_back(quoted_identifier);
277
destination.append(" (", 2);
279
/* Add field list to SQL string... */
280
size_t num_fields= header.field_metadata_size();
283
for (x= 0; x < num_fields; ++x)
285
const FieldMetadata &field_metadata= header.field_metadata(x);
287
destination.push_back(',');
289
destination.push_back(quoted_identifier);
290
destination.append(field_metadata.name());
291
destination.push_back(quoted_identifier);
297
enum TransformSqlError
298
transformInsertRecordToSql(const InsertHeader &header,
299
const InsertRecord &record,
301
enum TransformSqlVariant sql_variant)
303
enum TransformSqlError error= transformInsertHeaderToSql(header,
307
char quoted_identifier= '`';
308
if (sql_variant == ANSI)
309
quoted_identifier= '"';
311
destination.append(") VALUES (");
313
/* Add insert values */
314
size_t num_fields= header.field_metadata_size();
316
bool should_quote_field_value= false;
318
for (x= 0; x < num_fields; ++x)
321
destination.push_back(',');
323
const FieldMetadata &field_metadata= header.field_metadata(x);
325
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
327
if (should_quote_field_value)
328
destination.push_back('\'');
330
if (field_metadata.type() == Table::Field::BLOB)
333
* We do this here because BLOB data is returned
334
* in a string correctly, but calling append()
335
* without a length will result in only the string
336
* up to a \0 being output here.
338
string raw_data(record.insert_value(x));
339
destination.append(raw_data.c_str(), raw_data.size());
343
destination.append(record.insert_value(x));
346
if (should_quote_field_value)
347
destination.push_back('\'');
349
destination.push_back(')');
354
enum TransformSqlError
355
transformInsertStatementToSql(const InsertHeader &header,
356
const InsertData &data,
358
enum TransformSqlVariant sql_variant)
360
enum TransformSqlError error= transformInsertHeaderToSql(header,
364
char quoted_identifier= '`';
365
if (sql_variant == ANSI)
366
quoted_identifier= '"';
368
destination.append(") VALUES (", 10);
370
/* Add insert values */
371
size_t num_records= data.record_size();
372
size_t num_fields= header.field_metadata_size();
374
bool should_quote_field_value= false;
376
for (x= 0; x < num_records; ++x)
379
destination.append("),(", 3);
381
for (y= 0; y < num_fields; ++y)
384
destination.push_back(',');
386
const FieldMetadata &field_metadata= header.field_metadata(y);
388
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
390
if (should_quote_field_value)
391
destination.push_back('\'');
393
if (field_metadata.type() == Table::Field::BLOB)
396
* We do this here because BLOB data is returned
397
* in a string correctly, but calling append()
398
* without a length will result in only the string
399
* up to a \0 being output here.
401
string raw_data(data.record(x).insert_value(y));
402
destination.append(raw_data.c_str(), raw_data.size());
406
destination.append(data.record(x).insert_value(y));
409
if (should_quote_field_value)
410
destination.push_back('\'');
413
destination.push_back(')');
418
enum TransformSqlError
419
transformUpdateHeaderToSql(const UpdateHeader &header,
421
enum TransformSqlVariant sql_variant)
423
char quoted_identifier= '`';
424
if (sql_variant == ANSI)
425
quoted_identifier= '"';
427
destination.assign("UPDATE ", 7);
428
destination.push_back(quoted_identifier);
429
destination.append(header.table_metadata().schema_name());
430
destination.push_back(quoted_identifier);
431
destination.push_back('.');
432
destination.push_back(quoted_identifier);
433
destination.append(header.table_metadata().table_name());
434
destination.push_back(quoted_identifier);
435
destination.append(" SET ", 5);
440
enum TransformSqlError
441
transformUpdateRecordToSql(const UpdateHeader &header,
442
const UpdateRecord &record,
444
enum TransformSqlVariant sql_variant)
446
enum TransformSqlError error= transformUpdateHeaderToSql(header,
450
char quoted_identifier= '`';
451
if (sql_variant == ANSI)
452
quoted_identifier= '"';
454
/* Add field SET list to SQL string... */
455
size_t num_set_fields= header.set_field_metadata_size();
457
bool should_quote_field_value= false;
459
for (x= 0; x < num_set_fields; ++x)
461
const FieldMetadata &field_metadata= header.set_field_metadata(x);
463
destination.push_back(',');
465
destination.push_back(quoted_identifier);
466
destination.append(field_metadata.name());
467
destination.push_back(quoted_identifier);
468
destination.push_back('=');
470
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
472
if (should_quote_field_value)
473
destination.push_back('\'');
475
if (field_metadata.type() == Table::Field::BLOB)
478
* We do this here because BLOB data is returned
479
* in a string correctly, but calling append()
480
* without a length will result in only the string
481
* up to a \0 being output here.
483
string raw_data(record.after_value(x));
484
destination.append(raw_data.c_str(), raw_data.size());
488
destination.append(record.after_value(x));
491
if (should_quote_field_value)
492
destination.push_back('\'');
495
size_t num_key_fields= header.key_field_metadata_size();
497
destination.append(" WHERE ", 7);
498
for (x= 0; x < num_key_fields; ++x)
500
const FieldMetadata &field_metadata= header.key_field_metadata(x);
503
destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
505
destination.push_back(quoted_identifier);
506
destination.append(field_metadata.name());
507
destination.push_back(quoted_identifier);
509
destination.push_back('=');
511
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
513
if (should_quote_field_value)
514
destination.push_back('\'');
516
if (field_metadata.type() == Table::Field::BLOB)
519
* We do this here because BLOB data is returned
520
* in a string correctly, but calling append()
521
* without a length will result in only the string
522
* up to a \0 being output here.
524
string raw_data(record.key_value(x));
525
destination.append(raw_data.c_str(), raw_data.size());
529
destination.append(record.key_value(x));
532
if (should_quote_field_value)
533
destination.push_back('\'');
539
enum TransformSqlError
540
transformDeleteHeaderToSql(const DeleteHeader &header,
542
enum TransformSqlVariant sql_variant)
544
char quoted_identifier= '`';
545
if (sql_variant == ANSI)
546
quoted_identifier= '"';
548
destination.assign("DELETE FROM ", 12);
549
destination.push_back(quoted_identifier);
550
destination.append(header.table_metadata().schema_name());
551
destination.push_back(quoted_identifier);
552
destination.push_back('.');
553
destination.push_back(quoted_identifier);
554
destination.append(header.table_metadata().table_name());
555
destination.push_back(quoted_identifier);
560
enum TransformSqlError
561
transformDeleteRecordToSql(const DeleteHeader &header,
562
const DeleteRecord &record,
564
enum TransformSqlVariant sql_variant)
566
enum TransformSqlError error= transformDeleteHeaderToSql(header,
569
char quoted_identifier= '`';
570
if (sql_variant == ANSI)
571
quoted_identifier= '"';
573
/* Add WHERE clause to SQL string... */
574
uint32_t num_key_fields= header.key_field_metadata_size();
576
bool should_quote_field_value= false;
578
destination.append(" WHERE ", 7);
579
for (x= 0; x < num_key_fields; ++x)
581
const FieldMetadata &field_metadata= header.key_field_metadata(x);
584
destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
586
destination.push_back(quoted_identifier);
587
destination.append(field_metadata.name());
588
destination.push_back(quoted_identifier);
590
destination.push_back('=');
592
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
594
if (should_quote_field_value)
595
destination.push_back('\'');
597
if (field_metadata.type() == Table::Field::BLOB)
600
* We do this here because BLOB data is returned
601
* in a string correctly, but calling append()
602
* without a length will result in only the string
603
* up to a \0 being output here.
605
string raw_data(record.key_value(x));
606
destination.append(raw_data.c_str(), raw_data.size());
610
destination.append(record.key_value(x));
613
if (should_quote_field_value)
614
destination.push_back('\'');
620
enum TransformSqlError
621
transformDeleteStatementToSql(const DeleteHeader &header,
622
const DeleteData &data,
624
enum TransformSqlVariant sql_variant)
626
enum TransformSqlError error= transformDeleteHeaderToSql(header,
629
char quoted_identifier= '`';
630
if (sql_variant == ANSI)
631
quoted_identifier= '"';
633
/* Add WHERE clause to SQL string... */
634
uint32_t num_key_fields= header.key_field_metadata_size();
635
uint32_t num_key_records= data.record_size();
637
bool should_quote_field_value= false;
639
destination.append(" WHERE ", 7);
640
for (x= 0; x < num_key_records; ++x)
643
destination.append(" OR ", 4); /* Always OR condition for multiple key records */
645
if (num_key_fields > 1)
646
destination.push_back('(');
648
for (y= 0; y < num_key_fields; ++y)
650
const FieldMetadata &field_metadata= header.key_field_metadata(y);
653
destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
655
destination.push_back(quoted_identifier);
656
destination.append(field_metadata.name());
657
destination.push_back(quoted_identifier);
659
destination.push_back('=');
661
should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
663
if (should_quote_field_value)
664
destination.push_back('\'');
666
if (field_metadata.type() == Table::Field::BLOB)
669
* We do this here because BLOB data is returned
670
* in a string correctly, but calling append()
671
* without a length will result in only the string
672
* up to a \0 being output here.
674
string raw_data(data.record(x).key_value(y));
675
destination.append(raw_data.c_str(), raw_data.size());
679
destination.append(data.record(x).key_value(y));
682
if (should_quote_field_value)
683
destination.push_back('\'');
685
if (num_key_fields > 1)
686
destination.push_back(')');
691
enum TransformSqlError
692
transformDropSchemaStatementToSql(const DropSchemaStatement &statement,
694
enum TransformSqlVariant sql_variant)
696
char quoted_identifier= '`';
697
if (sql_variant == ANSI)
698
quoted_identifier= '"';
700
destination.append("DROP SCHEMA ", 12);
701
destination.push_back(quoted_identifier);
702
destination.append(statement.schema_name());
703
destination.push_back(quoted_identifier);
708
enum TransformSqlError
709
transformCreateSchemaStatementToSql(const CreateSchemaStatement &statement,
711
enum TransformSqlVariant sql_variant)
713
char quoted_identifier= '`';
714
if (sql_variant == ANSI)
715
quoted_identifier= '"';
717
const Schema &schema= statement.schema();
719
destination.append("CREATE SCHEMA ", 14);
720
destination.push_back(quoted_identifier);
721
destination.append(schema.name());
722
destination.push_back(quoted_identifier);
724
if (schema.has_collation())
726
destination.append(" COLLATE ", 9);
727
destination.append(schema.collation());
733
enum TransformSqlError
734
transformDropTableStatementToSql(const DropTableStatement &statement,
736
enum TransformSqlVariant sql_variant)
738
char quoted_identifier= '`';
739
if (sql_variant == ANSI)
740
quoted_identifier= '"';
742
const TableMetadata &table_metadata= statement.table_metadata();
744
destination.append("DROP TABLE ", 11);
746
/* Add the IF EXISTS clause if necessary */
747
if (statement.has_if_exists_clause() &&
748
statement.if_exists_clause() == true)
750
destination.append("IF EXISTS ", 10);
753
destination.push_back(quoted_identifier);
754
destination.append(table_metadata.schema_name());
755
destination.push_back(quoted_identifier);
756
destination.push_back('.');
757
destination.push_back(quoted_identifier);
758
destination.append(table_metadata.table_name());
759
destination.push_back(quoted_identifier);
764
enum TransformSqlError
765
transformTruncateTableStatementToSql(const TruncateTableStatement &statement,
767
enum TransformSqlVariant sql_variant)
769
char quoted_identifier= '`';
770
if (sql_variant == ANSI)
771
quoted_identifier= '"';
773
const TableMetadata &table_metadata= statement.table_metadata();
775
destination.append("TRUNCATE TABLE ", 15);
776
destination.push_back(quoted_identifier);
777
destination.append(table_metadata.schema_name());
778
destination.push_back(quoted_identifier);
779
destination.push_back('.');
780
destination.push_back(quoted_identifier);
781
destination.append(table_metadata.table_name());
782
destination.push_back(quoted_identifier);
787
enum TransformSqlError
788
transformSetVariableStatementToSql(const SetVariableStatement &statement,
790
enum TransformSqlVariant sql_variant)
793
const FieldMetadata &variable_metadata= statement.variable_metadata();
794
bool should_quote_field_value= shouldQuoteFieldValue(variable_metadata.type());
796
destination.append("SET GLOBAL ", 11); /* Only global variables are replicated */
797
destination.append(variable_metadata.name());
798
destination.push_back('=');
800
if (should_quote_field_value)
801
destination.push_back('\'');
803
destination.append(statement.variable_value());
805
if (should_quote_field_value)
806
destination.push_back('\'');
811
enum TransformSqlError
812
transformCreateTableStatementToSql(const CreateTableStatement &statement,
814
enum TransformSqlVariant sql_variant)
816
return transformTableDefinitionToSql(statement.table(), destination, sql_variant);
819
enum TransformSqlError
820
transformTableDefinitionToSql(const Table &table,
822
enum TransformSqlVariant sql_variant, bool with_schema)
824
char quoted_identifier= '`';
825
if (sql_variant == ANSI)
826
quoted_identifier= '"';
828
destination.append("CREATE ", 7);
830
if (table.type() == Table::TEMPORARY)
831
destination.append("TEMPORARY ", 10);
833
destination.append("TABLE ", 6);
836
destination.push_back(quoted_identifier);
837
destination.append(table.schema());
838
destination.push_back(quoted_identifier);
839
destination.push_back('.');
841
destination.push_back(quoted_identifier);
842
destination.append(table.name());
843
destination.push_back(quoted_identifier);
844
destination.append(" (\n", 3);
846
enum TransformSqlError result= NONE;
847
size_t num_fields= table.field_size();
848
for (size_t x= 0; x < num_fields; ++x)
850
const Table::Field &field= table.field(x);
853
destination.append(",\n", 2);
855
result= transformFieldDefinitionToSql(field, destination, sql_variant);
861
size_t num_indexes= table.indexes_size();
864
destination.append(",\n", 2);
866
for (size_t x= 0; x < num_indexes; ++x)
868
const message::Table::Index &index= table.indexes(x);
871
destination.append(",\n", 2);
873
result= transformIndexDefinitionToSql(index, table, destination, sql_variant);
878
destination.append("\n)", 2);
880
/* Add ENGINE = " clause */
881
if (table.has_engine())
883
const Table::StorageEngine &engine= table.engine();
884
destination.append("\nENGINE = ", 10);
885
destination.append(engine.name());
887
size_t num_engine_options= engine.option_size();
888
for (size_t x= 0; x < num_engine_options; ++x)
890
const Table::StorageEngine::EngineOption &option= engine.option(x);
891
destination.push_back('\n');
892
destination.append(option.option_name());
893
destination.append(" = ", 3);
894
destination.append(option.option_value());
895
destination.push_back('\n');
899
if (table.has_options())
900
(void) transformTableOptionsToSql(table.options(), destination, sql_variant);
905
enum TransformSqlError
906
transformTableOptionsToSql(const Table::TableOptions &options,
908
enum TransformSqlVariant sql_variant)
910
if (sql_variant == ANSI)
911
return NONE; /* ANSI does not support table options... */
915
if (options.has_comment())
917
destination.append("\nCOMMENT = '", 12);
918
destination.append(options.comment());
919
destination.push_back('\'');
922
if (options.has_collation())
924
destination.append("\nCOLLATE = ", 11);
925
destination.append(options.collation());
928
if (options.has_auto_increment())
930
ss << options.auto_increment();
931
destination.append("\nAUTOINCREMENT_OFFSET = ", 24);
932
destination.append(ss.str());
936
if (options.has_row_type())
938
ss << options.row_type();
939
destination.append("\nROW_TYPE = ", 12);
940
destination.append(ss.str());
944
if (options.has_data_file_name())
946
destination.append("\nDATA_FILE_NAME = '", 19);
947
destination.append(options.data_file_name());
948
destination.push_back('\'');
951
if (options.has_index_file_name())
953
destination.append("\nINDEX_FILE_NAME = '", 20);
954
destination.append(options.index_file_name());
955
destination.push_back('\'');
958
if (options.has_max_rows())
960
ss << options.max_rows();
961
destination.append("\nMAX_ROWS = ", 12);
962
destination.append(ss.str());
966
if (options.has_min_rows())
968
ss << options.min_rows();
969
destination.append("\nMIN_ROWS = ", 12);
970
destination.append(ss.str());
974
if (options.has_auto_increment_value())
976
ss << options.auto_increment_value();
977
destination.append("\nAUTO_INCREMENT = ", 18);
978
destination.append(ss.str());
982
if (options.has_avg_row_length())
984
ss << options.avg_row_length();
985
destination.append("\nAVG_ROW_LENGTH = ", 18);
986
destination.append(ss.str());
990
if (options.has_key_block_size())
992
ss << options.key_block_size();
993
destination.append("\nKEY_BLOCK_SIZE = ", 18);
994
destination.append(ss.str());
998
if (options.has_block_size())
1000
ss << options.block_size();
1001
destination.append("\nBLOCK_SIZE = ", 14);
1002
destination.append(ss.str());
1006
if (options.has_pack_keys() &&
1007
options.pack_keys())
1008
destination.append("\nPACK_KEYS = TRUE", 17);
1009
if (options.has_pack_record() &&
1010
options.pack_record())
1011
destination.append("\nPACK_RECORD = TRUE", 19);
1012
if (options.has_checksum() &&
1014
destination.append("\nCHECKSUM = TRUE", 16);
1015
if (options.has_page_checksum() &&
1016
options.page_checksum())
1017
destination.append("\nPAGE_CHECKSUM = TRUE", 21);
1022
enum TransformSqlError
1023
transformIndexDefinitionToSql(const Table::Index &index,
1025
string &destination,
1026
enum TransformSqlVariant sql_variant)
1028
char quoted_identifier= '`';
1029
if (sql_variant == ANSI)
1030
quoted_identifier= '"';
1032
if (index.is_primary())
1033
destination.append("PRIMARY ", 8);
1034
else if (index.is_unique())
1035
destination.append("UNIQUE ", 7);
1037
destination.append("KEY ", 4);
1038
destination.push_back(quoted_identifier);
1039
destination.append(index.name());
1040
destination.push_back(quoted_identifier);
1041
destination.append(" (", 2);
1043
size_t num_parts= index.index_part_size();
1044
for (size_t x= 0; x < num_parts; ++x)
1046
const Table::Index::IndexPart &part= index.index_part(x);
1047
const Table::Field &field= table.field(part.fieldnr());
1050
destination.push_back(',');
1052
destination.push_back(quoted_identifier);
1053
destination.append(field.name());
1054
destination.push_back(quoted_identifier);
1057
* If the index part's field type is VARCHAR or TEXT
1058
* then check for a prefix length then is different
1059
* from the field's full length...
1061
if (field.type() == Table::Field::VARCHAR ||
1062
field.type() == Table::Field::BLOB)
1064
if (part.has_compare_length())
1066
size_t compare_length_in_chars= part.compare_length();
1068
/* hack: compare_length() is bytes, not chars, but
1069
* only for VARCHAR. Ass. */
1070
if (field.type() == Table::Field::VARCHAR)
1071
compare_length_in_chars/= 4;
1073
if (compare_length_in_chars != field.string_options().length())
1076
destination.push_back('(');
1077
ss << compare_length_in_chars;
1078
destination.append(ss.str());
1079
destination.push_back(')');
1084
destination.push_back(')');
1089
enum TransformSqlError
1090
transformFieldDefinitionToSql(const Table::Field &field,
1091
string &destination,
1092
enum TransformSqlVariant sql_variant)
1094
char quoted_identifier= '`';
1095
if (sql_variant == ANSI)
1096
quoted_identifier= '"';
1098
destination.push_back(quoted_identifier);
1099
destination.append(field.name());
1100
destination.push_back(quoted_identifier);
1102
Table::Field::FieldType field_type= field.type();
1106
case Table::Field::DOUBLE:
1107
destination.append(" DOUBLE", 7);
1109
case Table::Field::VARCHAR:
1111
destination.append(" VARCHAR(", 9);
1113
ss << field.string_options().length() << ")";
1114
destination.append(ss.str());
1117
case Table::Field::BLOB:
1118
destination.append(" BLOB", 5);
1120
case Table::Field::ENUM:
1122
size_t num_field_values= field.enumeration_values().field_value_size();
1123
destination.append(" ENUM(", 6);
1124
for (size_t x= 0; x < num_field_values; ++x)
1126
const string &type= field.enumeration_values().field_value(x);
1129
destination.push_back(',');
1131
destination.push_back('\'');
1132
destination.append(type);
1133
destination.push_back('\'');
1135
destination.push_back(')');
1138
case Table::Field::INTEGER:
1139
destination.append(" INT", 4);
1141
case Table::Field::BIGINT:
1142
destination.append(" BIGINT", 7);
1144
case Table::Field::DECIMAL:
1146
destination.append(" DECIMAL(", 9);
1148
ss << field.numeric_options().precision() << ",";
1149
ss << field.numeric_options().scale() << ")";
1150
destination.append(ss.str());
1153
case Table::Field::DATE:
1154
destination.append(" DATE", 5);
1156
case Table::Field::TIMESTAMP:
1157
destination.append(" TIMESTAMP", 10);
1159
case Table::Field::DATETIME:
1160
destination.append(" DATETIME", 9);
1164
if (field.type() == Table::Field::INTEGER ||
1165
field.type() == Table::Field::BIGINT)
1167
if (field.has_constraints() &&
1168
field.constraints().has_is_unsigned() &&
1169
field.constraints().is_unsigned())
1171
destination.append(" UNSIGNED", 9);
1176
if (! (field.has_constraints() &&
1177
field.constraints().is_nullable()))
1179
destination.append(" NOT", 4);
1181
destination.append(" NULL", 5);
1183
if (field.type() == Table::Field::INTEGER ||
1184
field.type() == Table::Field::BIGINT)
1186
/* AUTO_INCREMENT must be after NOT NULL */
1187
if (field.has_numeric_options() &&
1188
field.numeric_options().is_autoincrement())
1190
destination.append(" AUTO_INCREMENT", 15);
1194
if (field.type() == Table::Field::BLOB ||
1195
field.type() == Table::Field::VARCHAR)
1197
if (field.string_options().has_collation())
1199
destination.append(" COLLATE ", 9);
1200
destination.append(field.string_options().collation());
1204
if (field.options().has_default_value())
1206
destination.append(" DEFAULT ", 9);
1207
destination.push_back(quoted_identifier);
1208
destination.append(field.options().default_value());
1209
destination.push_back(quoted_identifier);
1212
if (field.options().has_default_bin_value())
1214
const string &v= field.options().default_bin_value();
1215
destination.append(" DEFAULT 0x", 11);
1216
for (size_t x= 0; x < v.length(); x++)
1218
printf("%.2x", *(v.c_str() + x));
1222
if (field.type() == Table::Field::TIMESTAMP)
1223
if (field.timestamp_options().has_auto_updates() &&
1224
field.timestamp_options().auto_updates())
1225
destination.append(" ON UPDATE CURRENT_TIMESTAMP", 28);
1227
if (field.has_comment())
1229
destination.append(" COMMENT ", 9);
1230
destination.push_back(quoted_identifier);
1231
destination.append(field.comment());
1232
destination.push_back(quoted_identifier);
1237
bool shouldQuoteFieldValue(Table::Field::FieldType in_type)
1241
case Table::Field::DOUBLE:
1242
case Table::Field::DECIMAL:
1243
case Table::Field::INTEGER:
1244
case Table::Field::BIGINT:
1245
case Table::Field::ENUM:
1252
Table::Field::FieldType internalFieldTypeToFieldProtoType(enum enum_field_types type)
1255
case DRIZZLE_TYPE_LONG:
1256
return Table::Field::INTEGER;
1257
case DRIZZLE_TYPE_DOUBLE:
1258
return Table::Field::DOUBLE;
1259
case DRIZZLE_TYPE_NULL:
1260
assert(false); /* Not a user definable type */
1261
return Table::Field::INTEGER; /* unreachable */
1262
case DRIZZLE_TYPE_TIMESTAMP:
1263
return Table::Field::TIMESTAMP;
1264
case DRIZZLE_TYPE_LONGLONG:
1265
return Table::Field::BIGINT;
1266
case DRIZZLE_TYPE_DATETIME:
1267
return Table::Field::DATETIME;
1268
case DRIZZLE_TYPE_DATE:
1269
return Table::Field::DATE;
1270
case DRIZZLE_TYPE_VARCHAR:
1271
return Table::Field::VARCHAR;
1272
case DRIZZLE_TYPE_DECIMAL:
1273
return Table::Field::DECIMAL;
1274
case DRIZZLE_TYPE_ENUM:
1275
return Table::Field::ENUM;
1276
case DRIZZLE_TYPE_BLOB:
1277
return Table::Field::BLOB;
1281
return Table::Field::INTEGER; /* unreachable */
1284
bool transactionContainsBulkSegment(const Transaction &transaction)
1286
size_t num_statements= transaction.statement_size();
1287
if (num_statements == 0)
1291
* Only INSERT, UPDATE, and DELETE statements can possibly
1292
* have bulk segments. So, we loop through the statements
1293
* checking for segment_id > 1 in those specific submessages.
1296
for (x= 0; x < num_statements; ++x)
1298
const Statement &statement= transaction.statement(x);
1299
Statement::Type type= statement.type();
1303
case Statement::INSERT:
1304
if (statement.insert_data().segment_id() > 1)
1307
case Statement::UPDATE:
1308
if (statement.update_data().segment_id() > 1)
1311
case Statement::DELETE:
1312
if (statement.delete_data().segment_id() > 1)
1322
} /* namespace message */
1323
} /* namespace drizzled */