~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to drizzled/message/statement_transform.cc

  • Committer: Jay Pipes
  • Date: 2009-02-21 16:00:06 UTC
  • mto: (907.1.1 trunk-with-temporal)
  • mto: This revision was merged to the branch mainline in revision 908.
  • Revision ID: jpipes@serialcoder-20090221160006-vnk3wt4qbcz62eru
Removes the TIME column type and related time functions.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
 
 *  vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3
 
 *
4
 
 *  Copyright (C) 2009 Sun Microsystems
5
 
 *  Copyright (c) 2010 Jay Pipes
6
 
 *
7
 
 *  Authors:
8
 
 *
9
 
 *    Jay Pipes <jaypipes@gmail.com>
10
 
 *
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.
14
 
 *
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.
19
 
 *
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
23
 
 */
24
 
 
25
 
/**
26
 
 * @file
27
 
 *
28
 
 * Implementation of various routines that can be used to convert
29
 
 * Statement messages to other formats, including SQL strings.
30
 
 */
31
 
 
32
 
#include "config.h"
33
 
 
34
 
#include "drizzled/message/statement_transform.h"
35
 
#include "drizzled/message/transaction.pb.h"
36
 
#include "drizzled/message/table.pb.h"
37
 
 
38
 
#include <string>
39
 
#include <vector>
40
 
#include <sstream>
41
 
 
42
 
using namespace std;
43
 
 
44
 
namespace drizzled
45
 
{
46
 
 
47
 
namespace message
48
 
{
49
 
 
50
 
enum TransformSqlError
51
 
transformStatementToSql(const Statement &source,
52
 
                        vector<string> &sql_strings,
53
 
                        enum TransformSqlVariant sql_variant,
54
 
                        bool already_in_transaction)
55
 
{
56
 
  TransformSqlError error= NONE;
57
 
 
58
 
  switch (source.type())
59
 
  {
60
 
  case Statement::INSERT:
61
 
    {
62
 
      if (! source.has_insert_header())
63
 
      {
64
 
        error= MISSING_HEADER;
65
 
        return error;
66
 
      }
67
 
      if (! source.has_insert_data())
68
 
      {
69
 
        error= MISSING_DATA;
70
 
        return error;
71
 
      }
72
 
 
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();
76
 
      size_t x;
77
 
 
78
 
      if (num_keys > 1 && ! already_in_transaction)
79
 
        sql_strings.push_back("START TRANSACTION");
80
 
 
81
 
      for (x= 0; x < num_keys; ++x)
82
 
      {
83
 
        string destination;
84
 
 
85
 
        error= transformInsertRecordToSql(insert_header,
86
 
                                          insert_data.record(x),
87
 
                                          destination,
88
 
                                          sql_variant);
89
 
        if (error != NONE)
90
 
          break;
91
 
 
92
 
        sql_strings.push_back(destination);
93
 
      }
94
 
 
95
 
      if (num_keys > 1 && ! already_in_transaction)
96
 
      {
97
 
        if (error == NONE)
98
 
          sql_strings.push_back("COMMIT");
99
 
        else
100
 
          sql_strings.push_back("ROLLBACK");
101
 
      }
102
 
    }
103
 
    break;
104
 
  case Statement::UPDATE:
105
 
    {
106
 
      if (! source.has_update_header())
107
 
      {
108
 
        error= MISSING_HEADER;
109
 
        return error;
110
 
      }
111
 
      if (! source.has_update_data())
112
 
      {
113
 
        error= MISSING_DATA;
114
 
        return error;
115
 
      }
116
 
 
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();
120
 
      size_t x;
121
 
 
122
 
      if (num_keys > 1 && ! already_in_transaction)
123
 
        sql_strings.push_back("START TRANSACTION");
124
 
 
125
 
      for (x= 0; x < num_keys; ++x)
126
 
      {
127
 
        string destination;
128
 
 
129
 
        error= transformUpdateRecordToSql(update_header,
130
 
                                          update_data.record(x),
131
 
                                          destination,
132
 
                                          sql_variant);
133
 
        if (error != NONE)
134
 
          break;
135
 
 
136
 
        sql_strings.push_back(destination);
137
 
      }
138
 
 
139
 
      if (num_keys > 1 && ! already_in_transaction)
140
 
      {
141
 
        if (error == NONE)
142
 
          sql_strings.push_back("COMMIT");
143
 
        else
144
 
          sql_strings.push_back("ROLLBACK");
145
 
      }
146
 
    }
147
 
    break;
148
 
  case Statement::DELETE:
149
 
    {
150
 
      if (! source.has_delete_header())
151
 
      {
152
 
        error= MISSING_HEADER;
153
 
        return error;
154
 
      }
155
 
      if (! source.has_delete_data())
156
 
      {
157
 
        error= MISSING_DATA;
158
 
        return error;
159
 
      }
160
 
 
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();
164
 
      size_t x;
165
 
 
166
 
      if (num_keys > 1 && ! already_in_transaction)
167
 
        sql_strings.push_back("START TRANSACTION");
168
 
 
169
 
      for (x= 0; x < num_keys; ++x)
170
 
      {
171
 
        string destination;
172
 
 
173
 
        error= transformDeleteRecordToSql(delete_header,
174
 
                                          delete_data.record(x),
175
 
                                          destination,
176
 
                                          sql_variant);
177
 
        if (error != NONE)
178
 
          break;
179
 
 
180
 
        sql_strings.push_back(destination);
181
 
      }
182
 
 
183
 
      if (num_keys > 1 && ! already_in_transaction)
184
 
      {
185
 
        if (error == NONE)
186
 
          sql_strings.push_back("COMMIT");
187
 
        else
188
 
          sql_strings.push_back("ROLLBACK");
189
 
      }
190
 
    }
191
 
    break;
192
 
  case Statement::CREATE_TABLE:
193
 
    {
194
 
      assert(source.has_create_table_statement());
195
 
      string destination;
196
 
      error= transformCreateTableStatementToSql(source.create_table_statement(),
197
 
                                                destination,
198
 
                                                sql_variant);
199
 
      sql_strings.push_back(destination);
200
 
    }
201
 
    break;
202
 
  case Statement::TRUNCATE_TABLE:
203
 
    {
204
 
      assert(source.has_truncate_table_statement());
205
 
      string destination;
206
 
      error= transformTruncateTableStatementToSql(source.truncate_table_statement(),
207
 
                                                  destination,
208
 
                                                  sql_variant);
209
 
      sql_strings.push_back(destination);
210
 
    }
211
 
    break;
212
 
  case Statement::DROP_TABLE:
213
 
    {
214
 
      assert(source.has_drop_table_statement());
215
 
      string destination;
216
 
      error= transformDropTableStatementToSql(source.drop_table_statement(),
217
 
                                              destination,
218
 
                                              sql_variant);
219
 
      sql_strings.push_back(destination);
220
 
    }
221
 
    break;
222
 
  case Statement::CREATE_SCHEMA:
223
 
    {
224
 
      assert(source.has_create_schema_statement());
225
 
      string destination;
226
 
      error= transformCreateSchemaStatementToSql(source.create_schema_statement(),
227
 
                                                 destination,
228
 
                                                 sql_variant);
229
 
      sql_strings.push_back(destination);
230
 
    }
231
 
    break;
232
 
  case Statement::DROP_SCHEMA:
233
 
    {
234
 
      assert(source.has_drop_schema_statement());
235
 
      string destination;
236
 
      error= transformDropSchemaStatementToSql(source.drop_schema_statement(),
237
 
                                               destination,
238
 
                                               sql_variant);
239
 
      sql_strings.push_back(destination);
240
 
    }
241
 
    break;
242
 
  case Statement::SET_VARIABLE:
243
 
    {
244
 
      assert(source.has_set_variable_statement());
245
 
      string destination;
246
 
      error= transformSetVariableStatementToSql(source.set_variable_statement(),
247
 
                                                destination,
248
 
                                                sql_variant);
249
 
      sql_strings.push_back(destination);
250
 
    }
251
 
    break;
252
 
  case Statement::RAW_SQL:
253
 
  default:
254
 
    sql_strings.push_back(source.sql());
255
 
    break;
256
 
  }
257
 
  return error;
258
 
}
259
 
 
260
 
enum TransformSqlError
261
 
transformInsertHeaderToSql(const InsertHeader &header,
262
 
                           string &destination,
263
 
                           enum TransformSqlVariant sql_variant)
264
 
{
265
 
  char quoted_identifier= '`';
266
 
  if (sql_variant == ANSI)
267
 
    quoted_identifier= '"';
268
 
 
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);
278
 
 
279
 
  /* Add field list to SQL string... */
280
 
  size_t num_fields= header.field_metadata_size();
281
 
  size_t x;
282
 
 
283
 
  for (x= 0; x < num_fields; ++x)
284
 
  {
285
 
    const FieldMetadata &field_metadata= header.field_metadata(x);
286
 
    if (x != 0)
287
 
      destination.push_back(',');
288
 
    
289
 
    destination.push_back(quoted_identifier);
290
 
    destination.append(field_metadata.name());
291
 
    destination.push_back(quoted_identifier);
292
 
  }
293
 
 
294
 
  return NONE;
295
 
}
296
 
 
297
 
enum TransformSqlError
298
 
transformInsertRecordToSql(const InsertHeader &header,
299
 
                           const InsertRecord &record,
300
 
                           string &destination,
301
 
                           enum TransformSqlVariant sql_variant)
302
 
{
303
 
  enum TransformSqlError error= transformInsertHeaderToSql(header,
304
 
                                                           destination,
305
 
                                                           sql_variant);
306
 
 
307
 
  char quoted_identifier= '`';
308
 
  if (sql_variant == ANSI)
309
 
    quoted_identifier= '"';
310
 
 
311
 
  destination.append(") VALUES (");
312
 
 
313
 
  /* Add insert values */
314
 
  size_t num_fields= header.field_metadata_size();
315
 
  size_t x;
316
 
  bool should_quote_field_value= false;
317
 
  
318
 
  for (x= 0; x < num_fields; ++x)
319
 
  {
320
 
    if (x != 0)
321
 
      destination.push_back(',');
322
 
 
323
 
    const FieldMetadata &field_metadata= header.field_metadata(x);
324
 
 
325
 
    should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
326
 
 
327
 
    if (should_quote_field_value)
328
 
      destination.push_back('\'');
329
 
 
330
 
    if (field_metadata.type() == Table::Field::BLOB)
331
 
    {
332
 
      /* 
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.
337
 
        */
338
 
      string raw_data(record.insert_value(x));
339
 
      destination.append(raw_data.c_str(), raw_data.size());
340
 
    }
341
 
    else
342
 
    {
343
 
      destination.append(record.insert_value(x));
344
 
    }
345
 
 
346
 
    if (should_quote_field_value)
347
 
      destination.push_back('\'');
348
 
  }
349
 
  destination.push_back(')');
350
 
 
351
 
  return error;
352
 
}
353
 
 
354
 
enum TransformSqlError
355
 
transformInsertStatementToSql(const InsertHeader &header,
356
 
                              const InsertData &data,
357
 
                              string &destination,
358
 
                              enum TransformSqlVariant sql_variant)
359
 
{
360
 
  enum TransformSqlError error= transformInsertHeaderToSql(header,
361
 
                                                           destination,
362
 
                                                           sql_variant);
363
 
 
364
 
  char quoted_identifier= '`';
365
 
  if (sql_variant == ANSI)
366
 
    quoted_identifier= '"';
367
 
 
368
 
  destination.append(") VALUES (", 10);
369
 
 
370
 
  /* Add insert values */
371
 
  size_t num_records= data.record_size();
372
 
  size_t num_fields= header.field_metadata_size();
373
 
  size_t x, y;
374
 
  bool should_quote_field_value= false;
375
 
  
376
 
  for (x= 0; x < num_records; ++x)
377
 
  {
378
 
    if (x != 0)
379
 
      destination.append("),(", 3);
380
 
 
381
 
    for (y= 0; y < num_fields; ++y)
382
 
    {
383
 
      if (y != 0)
384
 
        destination.push_back(',');
385
 
 
386
 
      const FieldMetadata &field_metadata= header.field_metadata(y);
387
 
      
388
 
      should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
389
 
 
390
 
      if (should_quote_field_value)
391
 
        destination.push_back('\'');
392
 
 
393
 
      if (field_metadata.type() == Table::Field::BLOB)
394
 
      {
395
 
        /* 
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.
400
 
         */
401
 
        string raw_data(data.record(x).insert_value(y));
402
 
        destination.append(raw_data.c_str(), raw_data.size());
403
 
      }
404
 
      else
405
 
      {
406
 
        destination.append(data.record(x).insert_value(y));
407
 
      }
408
 
 
409
 
      if (should_quote_field_value)
410
 
        destination.push_back('\'');
411
 
    }
412
 
  }
413
 
  destination.push_back(')');
414
 
 
415
 
  return error;
416
 
}
417
 
 
418
 
enum TransformSqlError
419
 
transformUpdateHeaderToSql(const UpdateHeader &header,
420
 
                           string &destination,
421
 
                           enum TransformSqlVariant sql_variant)
422
 
{
423
 
  char quoted_identifier= '`';
424
 
  if (sql_variant == ANSI)
425
 
    quoted_identifier= '"';
426
 
 
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);
436
 
 
437
 
  return NONE;
438
 
}
439
 
 
440
 
enum TransformSqlError
441
 
transformUpdateRecordToSql(const UpdateHeader &header,
442
 
                           const UpdateRecord &record,
443
 
                           string &destination,
444
 
                           enum TransformSqlVariant sql_variant)
445
 
{
446
 
  enum TransformSqlError error= transformUpdateHeaderToSql(header,
447
 
                                                           destination,
448
 
                                                           sql_variant);
449
 
 
450
 
  char quoted_identifier= '`';
451
 
  if (sql_variant == ANSI)
452
 
    quoted_identifier= '"';
453
 
 
454
 
  /* Add field SET list to SQL string... */
455
 
  size_t num_set_fields= header.set_field_metadata_size();
456
 
  size_t x;
457
 
  bool should_quote_field_value= false;
458
 
 
459
 
  for (x= 0; x < num_set_fields; ++x)
460
 
  {
461
 
    const FieldMetadata &field_metadata= header.set_field_metadata(x);
462
 
    if (x != 0)
463
 
      destination.push_back(',');
464
 
    
465
 
    destination.push_back(quoted_identifier);
466
 
    destination.append(field_metadata.name());
467
 
    destination.push_back(quoted_identifier);
468
 
    destination.push_back('=');
469
 
 
470
 
    should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
471
 
 
472
 
    if (should_quote_field_value)
473
 
      destination.push_back('\'');
474
 
 
475
 
    if (field_metadata.type() == Table::Field::BLOB)
476
 
    {
477
 
      /* 
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.
482
 
       */
483
 
      string raw_data(record.after_value(x));
484
 
      destination.append(raw_data.c_str(), raw_data.size());
485
 
    }
486
 
    else
487
 
    {
488
 
      destination.append(record.after_value(x));
489
 
    }
490
 
 
491
 
    if (should_quote_field_value)
492
 
      destination.push_back('\'');
493
 
  }
494
 
 
495
 
  size_t num_key_fields= header.key_field_metadata_size();
496
 
 
497
 
  destination.append(" WHERE ", 7);
498
 
  for (x= 0; x < num_key_fields; ++x) 
499
 
  {
500
 
    const FieldMetadata &field_metadata= header.key_field_metadata(x);
501
 
    
502
 
    if (x != 0)
503
 
      destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
504
 
 
505
 
    destination.push_back(quoted_identifier);
506
 
    destination.append(field_metadata.name());
507
 
    destination.push_back(quoted_identifier);
508
 
 
509
 
    destination.push_back('=');
510
 
 
511
 
    should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
512
 
 
513
 
    if (should_quote_field_value)
514
 
      destination.push_back('\'');
515
 
 
516
 
    if (field_metadata.type() == Table::Field::BLOB)
517
 
    {
518
 
      /* 
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.
523
 
       */
524
 
      string raw_data(record.key_value(x));
525
 
      destination.append(raw_data.c_str(), raw_data.size());
526
 
    }
527
 
    else
528
 
    {
529
 
      destination.append(record.key_value(x));
530
 
    }
531
 
 
532
 
    if (should_quote_field_value)
533
 
      destination.push_back('\'');
534
 
  }
535
 
 
536
 
  return error;
537
 
}
538
 
 
539
 
enum TransformSqlError
540
 
transformDeleteHeaderToSql(const DeleteHeader &header,
541
 
                           string &destination,
542
 
                           enum TransformSqlVariant sql_variant)
543
 
{
544
 
  char quoted_identifier= '`';
545
 
  if (sql_variant == ANSI)
546
 
    quoted_identifier= '"';
547
 
 
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);
556
 
 
557
 
  return NONE;
558
 
}
559
 
 
560
 
enum TransformSqlError
561
 
transformDeleteRecordToSql(const DeleteHeader &header,
562
 
                           const DeleteRecord &record,
563
 
                           string &destination,
564
 
                           enum TransformSqlVariant sql_variant)
565
 
{
566
 
  enum TransformSqlError error= transformDeleteHeaderToSql(header,
567
 
                                                           destination,
568
 
                                                           sql_variant);
569
 
  char quoted_identifier= '`';
570
 
  if (sql_variant == ANSI)
571
 
    quoted_identifier= '"';
572
 
 
573
 
  /* Add WHERE clause to SQL string... */
574
 
  uint32_t num_key_fields= header.key_field_metadata_size();
575
 
  uint32_t x;
576
 
  bool should_quote_field_value= false;
577
 
 
578
 
  destination.append(" WHERE ", 7);
579
 
  for (x= 0; x < num_key_fields; ++x) 
580
 
  {
581
 
    const FieldMetadata &field_metadata= header.key_field_metadata(x);
582
 
    
583
 
    if (x != 0)
584
 
      destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
585
 
 
586
 
    destination.push_back(quoted_identifier);
587
 
    destination.append(field_metadata.name());
588
 
    destination.push_back(quoted_identifier);
589
 
 
590
 
    destination.push_back('=');
591
 
 
592
 
    should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
593
 
 
594
 
    if (should_quote_field_value)
595
 
      destination.push_back('\'');
596
 
 
597
 
    if (field_metadata.type() == Table::Field::BLOB)
598
 
    {
599
 
      /* 
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.
604
 
       */
605
 
      string raw_data(record.key_value(x));
606
 
      destination.append(raw_data.c_str(), raw_data.size());
607
 
    }
608
 
    else
609
 
    {
610
 
      destination.append(record.key_value(x));
611
 
    }
612
 
 
613
 
    if (should_quote_field_value)
614
 
      destination.push_back('\'');
615
 
  }
616
 
 
617
 
  return error;
618
 
}
619
 
 
620
 
enum TransformSqlError
621
 
transformDeleteStatementToSql(const DeleteHeader &header,
622
 
                              const DeleteData &data,
623
 
                              string &destination,
624
 
                              enum TransformSqlVariant sql_variant)
625
 
{
626
 
  enum TransformSqlError error= transformDeleteHeaderToSql(header,
627
 
                                                           destination,
628
 
                                                           sql_variant);
629
 
  char quoted_identifier= '`';
630
 
  if (sql_variant == ANSI)
631
 
    quoted_identifier= '"';
632
 
 
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();
636
 
  uint32_t x, y;
637
 
  bool should_quote_field_value= false;
638
 
 
639
 
  destination.append(" WHERE ", 7);
640
 
  for (x= 0; x < num_key_records; ++x)
641
 
  {
642
 
    if (x != 0)
643
 
      destination.append(" OR ", 4); /* Always OR condition for multiple key records */
644
 
 
645
 
    if (num_key_fields > 1)
646
 
      destination.push_back('(');
647
 
 
648
 
    for (y= 0; y < num_key_fields; ++y) 
649
 
    {
650
 
      const FieldMetadata &field_metadata= header.key_field_metadata(y);
651
 
      
652
 
      if (y != 0)
653
 
        destination.append(" AND ", 5); /* Always AND condition with a multi-column PK */
654
 
 
655
 
      destination.push_back(quoted_identifier);
656
 
      destination.append(field_metadata.name());
657
 
      destination.push_back(quoted_identifier);
658
 
 
659
 
      destination.push_back('=');
660
 
 
661
 
      should_quote_field_value= shouldQuoteFieldValue(field_metadata.type());
662
 
 
663
 
      if (should_quote_field_value)
664
 
        destination.push_back('\'');
665
 
 
666
 
      if (field_metadata.type() == Table::Field::BLOB)
667
 
      {
668
 
        /* 
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.
673
 
         */
674
 
        string raw_data(data.record(x).key_value(y));
675
 
        destination.append(raw_data.c_str(), raw_data.size());
676
 
      }
677
 
      else
678
 
      {
679
 
        destination.append(data.record(x).key_value(y));
680
 
      }
681
 
 
682
 
      if (should_quote_field_value)
683
 
        destination.push_back('\'');
684
 
    }
685
 
    if (num_key_fields > 1)
686
 
      destination.push_back(')');
687
 
  }
688
 
  return error;
689
 
}
690
 
 
691
 
enum TransformSqlError
692
 
transformDropSchemaStatementToSql(const DropSchemaStatement &statement,
693
 
                                  string &destination,
694
 
                                  enum TransformSqlVariant sql_variant)
695
 
{
696
 
  char quoted_identifier= '`';
697
 
  if (sql_variant == ANSI)
698
 
    quoted_identifier= '"';
699
 
 
700
 
  destination.append("DROP SCHEMA ", 12);
701
 
  destination.push_back(quoted_identifier);
702
 
  destination.append(statement.schema_name());
703
 
  destination.push_back(quoted_identifier);
704
 
 
705
 
  return NONE;
706
 
}
707
 
 
708
 
enum TransformSqlError
709
 
transformCreateSchemaStatementToSql(const CreateSchemaStatement &statement,
710
 
                                    string &destination,
711
 
                                    enum TransformSqlVariant sql_variant)
712
 
{
713
 
  char quoted_identifier= '`';
714
 
  if (sql_variant == ANSI)
715
 
    quoted_identifier= '"';
716
 
 
717
 
  const Schema &schema= statement.schema();
718
 
 
719
 
  destination.append("CREATE SCHEMA ", 14);
720
 
  destination.push_back(quoted_identifier);
721
 
  destination.append(schema.name());
722
 
  destination.push_back(quoted_identifier);
723
 
 
724
 
  if (schema.has_collation())
725
 
  {
726
 
    destination.append(" COLLATE ", 9);
727
 
    destination.append(schema.collation());
728
 
  }
729
 
 
730
 
  return NONE;
731
 
}
732
 
 
733
 
enum TransformSqlError
734
 
transformDropTableStatementToSql(const DropTableStatement &statement,
735
 
                                 string &destination,
736
 
                                 enum TransformSqlVariant sql_variant)
737
 
{
738
 
  char quoted_identifier= '`';
739
 
  if (sql_variant == ANSI)
740
 
    quoted_identifier= '"';
741
 
 
742
 
  const TableMetadata &table_metadata= statement.table_metadata();
743
 
 
744
 
  destination.append("DROP TABLE ", 11);
745
 
 
746
 
  /* Add the IF EXISTS clause if necessary */
747
 
  if (statement.has_if_exists_clause() &&
748
 
      statement.if_exists_clause() == true)
749
 
  {
750
 
    destination.append("IF EXISTS ", 10);
751
 
  }
752
 
 
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);
760
 
 
761
 
  return NONE;
762
 
}
763
 
 
764
 
enum TransformSqlError
765
 
transformTruncateTableStatementToSql(const TruncateTableStatement &statement,
766
 
                                     string &destination,
767
 
                                     enum TransformSqlVariant sql_variant)
768
 
{
769
 
  char quoted_identifier= '`';
770
 
  if (sql_variant == ANSI)
771
 
    quoted_identifier= '"';
772
 
 
773
 
  const TableMetadata &table_metadata= statement.table_metadata();
774
 
 
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);
783
 
 
784
 
  return NONE;
785
 
}
786
 
 
787
 
enum TransformSqlError
788
 
transformSetVariableStatementToSql(const SetVariableStatement &statement,
789
 
                                   string &destination,
790
 
                                   enum TransformSqlVariant sql_variant)
791
 
{
792
 
  (void) sql_variant;
793
 
  const FieldMetadata &variable_metadata= statement.variable_metadata();
794
 
  bool should_quote_field_value= shouldQuoteFieldValue(variable_metadata.type());
795
 
 
796
 
  destination.append("SET GLOBAL ", 11); /* Only global variables are replicated */
797
 
  destination.append(variable_metadata.name());
798
 
  destination.push_back('=');
799
 
 
800
 
  if (should_quote_field_value)
801
 
    destination.push_back('\'');
802
 
  
803
 
  destination.append(statement.variable_value());
804
 
 
805
 
  if (should_quote_field_value)
806
 
    destination.push_back('\'');
807
 
 
808
 
  return NONE;
809
 
}
810
 
 
811
 
enum TransformSqlError
812
 
transformCreateTableStatementToSql(const CreateTableStatement &statement,
813
 
                                   string &destination,
814
 
                                   enum TransformSqlVariant sql_variant)
815
 
{
816
 
  return transformTableDefinitionToSql(statement.table(), destination, sql_variant);
817
 
}
818
 
 
819
 
enum TransformSqlError
820
 
transformTableDefinitionToSql(const Table &table,
821
 
                              string &destination,
822
 
                              enum TransformSqlVariant sql_variant, bool with_schema)
823
 
{
824
 
  char quoted_identifier= '`';
825
 
  if (sql_variant == ANSI)
826
 
    quoted_identifier= '"';
827
 
 
828
 
  destination.append("CREATE ", 7);
829
 
 
830
 
  if (table.type() == Table::TEMPORARY)
831
 
    destination.append("TEMPORARY ", 10);
832
 
  
833
 
  destination.append("TABLE ", 6);
834
 
  if (with_schema)
835
 
  {
836
 
    destination.push_back(quoted_identifier);
837
 
    destination.append(table.schema());
838
 
    destination.push_back(quoted_identifier);
839
 
    destination.push_back('.');
840
 
  }
841
 
  destination.push_back(quoted_identifier);
842
 
  destination.append(table.name());
843
 
  destination.push_back(quoted_identifier);
844
 
  destination.append(" (\n", 3);
845
 
 
846
 
  enum TransformSqlError result= NONE;
847
 
  size_t num_fields= table.field_size();
848
 
  for (size_t x= 0; x < num_fields; ++x)
849
 
  {
850
 
    const Table::Field &field= table.field(x);
851
 
 
852
 
    if (x != 0)
853
 
      destination.append(",\n", 2);
854
 
 
855
 
    result= transformFieldDefinitionToSql(field, destination, sql_variant);
856
 
    
857
 
    if (result != NONE)
858
 
      return result;
859
 
  }
860
 
 
861
 
  size_t num_indexes= table.indexes_size();
862
 
  
863
 
  if (num_indexes > 0)
864
 
    destination.append(",\n", 2);
865
 
 
866
 
  for (size_t x= 0; x < num_indexes; ++x)
867
 
  {
868
 
    const message::Table::Index &index= table.indexes(x);
869
 
 
870
 
    if (x != 0)
871
 
      destination.append(",\n", 2);
872
 
 
873
 
    result= transformIndexDefinitionToSql(index, table, destination, sql_variant);
874
 
    
875
 
    if (result != NONE)
876
 
      return result;
877
 
  }
878
 
  destination.append("\n)", 2);
879
 
 
880
 
  /* Add ENGINE = " clause */
881
 
  if (table.has_engine())
882
 
  {
883
 
    const Table::StorageEngine &engine= table.engine();
884
 
    destination.append("\nENGINE = ", 10);
885
 
    destination.append(engine.name());
886
 
 
887
 
    size_t num_engine_options= engine.option_size();
888
 
    for (size_t x= 0; x < num_engine_options; ++x)
889
 
    {
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');
896
 
    }
897
 
  }
898
 
 
899
 
  if (table.has_options())
900
 
    (void) transformTableOptionsToSql(table.options(), destination, sql_variant);
901
 
 
902
 
  return NONE;
903
 
}
904
 
 
905
 
enum TransformSqlError
906
 
transformTableOptionsToSql(const Table::TableOptions &options,
907
 
                           string &destination,
908
 
                           enum TransformSqlVariant sql_variant)
909
 
{
910
 
  if (sql_variant == ANSI)
911
 
    return NONE; /* ANSI does not support table options... */
912
 
 
913
 
  stringstream ss;
914
 
 
915
 
  if (options.has_comment())
916
 
  {
917
 
    destination.append("\nCOMMENT = '", 12);
918
 
    destination.append(options.comment());
919
 
    destination.push_back('\'');
920
 
  }
921
 
 
922
 
  if (options.has_collation())
923
 
  {
924
 
    destination.append("\nCOLLATE = ", 11);
925
 
    destination.append(options.collation());
926
 
  }
927
 
 
928
 
  if (options.has_auto_increment())
929
 
  {
930
 
    ss << options.auto_increment();
931
 
    destination.append("\nAUTOINCREMENT_OFFSET = ", 24);
932
 
    destination.append(ss.str());
933
 
    ss.clear();
934
 
  }
935
 
  
936
 
  if (options.has_row_type())
937
 
  {
938
 
    ss << options.row_type();
939
 
    destination.append("\nROW_TYPE = ", 12);
940
 
    destination.append(ss.str());
941
 
    ss.clear();
942
 
  }
943
 
 
944
 
  if (options.has_data_file_name())
945
 
  {
946
 
    destination.append("\nDATA_FILE_NAME = '", 19);
947
 
    destination.append(options.data_file_name());
948
 
    destination.push_back('\'');
949
 
  }
950
 
 
951
 
  if (options.has_index_file_name())
952
 
  {
953
 
    destination.append("\nINDEX_FILE_NAME = '", 20);
954
 
    destination.append(options.index_file_name());
955
 
    destination.push_back('\'');
956
 
  }
957
 
 
958
 
  if (options.has_max_rows())
959
 
  {
960
 
    ss << options.max_rows();
961
 
    destination.append("\nMAX_ROWS = ", 12);
962
 
    destination.append(ss.str());
963
 
    ss.clear();
964
 
  }
965
 
 
966
 
  if (options.has_min_rows())
967
 
  {
968
 
    ss << options.min_rows();
969
 
    destination.append("\nMIN_ROWS = ", 12);
970
 
    destination.append(ss.str());
971
 
    ss.clear();
972
 
  }
973
 
 
974
 
  if (options.has_auto_increment_value())
975
 
  {
976
 
    ss << options.auto_increment_value();
977
 
    destination.append("\nAUTO_INCREMENT = ", 18);
978
 
    destination.append(ss.str());
979
 
    ss.clear();
980
 
  }
981
 
 
982
 
  if (options.has_avg_row_length())
983
 
  {
984
 
    ss << options.avg_row_length();
985
 
    destination.append("\nAVG_ROW_LENGTH = ", 18);
986
 
    destination.append(ss.str());
987
 
    ss.clear();
988
 
  }
989
 
 
990
 
  if (options.has_key_block_size())
991
 
  {
992
 
    ss << options.key_block_size();
993
 
    destination.append("\nKEY_BLOCK_SIZE = ", 18);
994
 
    destination.append(ss.str());
995
 
    ss.clear();
996
 
  }
997
 
 
998
 
  if (options.has_block_size())
999
 
  {
1000
 
    ss << options.block_size();
1001
 
    destination.append("\nBLOCK_SIZE = ", 14);
1002
 
    destination.append(ss.str());
1003
 
    ss.clear();
1004
 
  }
1005
 
 
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() &&
1013
 
      options.checksum())
1014
 
    destination.append("\nCHECKSUM = TRUE", 16);
1015
 
  if (options.has_page_checksum() &&
1016
 
      options.page_checksum())
1017
 
    destination.append("\nPAGE_CHECKSUM = TRUE", 21);
1018
 
 
1019
 
  return NONE;
1020
 
}
1021
 
 
1022
 
enum TransformSqlError
1023
 
transformIndexDefinitionToSql(const Table::Index &index,
1024
 
                              const Table &table,
1025
 
                              string &destination,
1026
 
                              enum TransformSqlVariant sql_variant)
1027
 
{
1028
 
  char quoted_identifier= '`';
1029
 
  if (sql_variant == ANSI)
1030
 
    quoted_identifier= '"';
1031
 
 
1032
 
  if (index.is_primary())
1033
 
    destination.append("PRIMARY ", 8);
1034
 
  else if (index.is_unique())
1035
 
    destination.append("UNIQUE ", 7);
1036
 
 
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);
1042
 
  
1043
 
  size_t num_parts= index.index_part_size();
1044
 
  for (size_t x= 0; x < num_parts; ++x)
1045
 
  {
1046
 
    const Table::Index::IndexPart &part= index.index_part(x);
1047
 
    const Table::Field &field= table.field(part.fieldnr());
1048
 
 
1049
 
    if (x != 0)
1050
 
      destination.push_back(',');
1051
 
    
1052
 
    destination.push_back(quoted_identifier);
1053
 
    destination.append(field.name());
1054
 
    destination.push_back(quoted_identifier);
1055
 
 
1056
 
    /* 
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...
1060
 
     */
1061
 
    if (field.type() == Table::Field::VARCHAR ||
1062
 
        field.type() == Table::Field::BLOB)
1063
 
    {
1064
 
      if (part.has_compare_length())
1065
 
      {
1066
 
        size_t compare_length_in_chars= part.compare_length();
1067
 
        
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;
1072
 
 
1073
 
        if (compare_length_in_chars != field.string_options().length())
1074
 
        {
1075
 
          stringstream ss;
1076
 
          destination.push_back('(');
1077
 
          ss << compare_length_in_chars;
1078
 
          destination.append(ss.str());
1079
 
          destination.push_back(')');
1080
 
        }
1081
 
      }
1082
 
    }
1083
 
  }
1084
 
  destination.push_back(')');
1085
 
 
1086
 
  return NONE;
1087
 
}
1088
 
 
1089
 
enum TransformSqlError
1090
 
transformFieldDefinitionToSql(const Table::Field &field,
1091
 
                              string &destination,
1092
 
                              enum TransformSqlVariant sql_variant)
1093
 
{
1094
 
  char quoted_identifier= '`';
1095
 
  if (sql_variant == ANSI)
1096
 
    quoted_identifier= '"';
1097
 
 
1098
 
  destination.push_back(quoted_identifier);
1099
 
  destination.append(field.name());
1100
 
  destination.push_back(quoted_identifier);
1101
 
 
1102
 
  Table::Field::FieldType field_type= field.type();
1103
 
 
1104
 
  switch (field_type)
1105
 
  {
1106
 
    case Table::Field::DOUBLE:
1107
 
    destination.append(" DOUBLE", 7);
1108
 
    break;
1109
 
  case Table::Field::VARCHAR:
1110
 
    {
1111
 
      destination.append(" VARCHAR(", 9);
1112
 
      stringstream ss;
1113
 
      ss << field.string_options().length() << ")";
1114
 
      destination.append(ss.str());
1115
 
    }
1116
 
    break;
1117
 
  case Table::Field::BLOB:
1118
 
    destination.append(" BLOB", 5);
1119
 
    break;
1120
 
  case Table::Field::ENUM:
1121
 
    {
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)
1125
 
      {
1126
 
        const string &type= field.enumeration_values().field_value(x);
1127
 
 
1128
 
        if (x != 0)
1129
 
          destination.push_back(',');
1130
 
 
1131
 
        destination.push_back('\'');
1132
 
        destination.append(type);
1133
 
        destination.push_back('\'');
1134
 
      }
1135
 
      destination.push_back(')');
1136
 
      break;
1137
 
    }
1138
 
  case Table::Field::INTEGER:
1139
 
    destination.append(" INT", 4);
1140
 
    break;
1141
 
  case Table::Field::BIGINT:
1142
 
    destination.append(" BIGINT", 7);
1143
 
    break;
1144
 
  case Table::Field::DECIMAL:
1145
 
    {
1146
 
      destination.append(" DECIMAL(", 9);
1147
 
      stringstream ss;
1148
 
      ss << field.numeric_options().precision() << ",";
1149
 
      ss << field.numeric_options().scale() << ")";
1150
 
      destination.append(ss.str());
1151
 
    }
1152
 
    break;
1153
 
  case Table::Field::DATE:
1154
 
    destination.append(" DATE", 5);
1155
 
    break;
1156
 
  case Table::Field::TIMESTAMP:
1157
 
    destination.append(" TIMESTAMP",  10);
1158
 
    break;
1159
 
  case Table::Field::DATETIME:
1160
 
    destination.append(" DATETIME",  9);
1161
 
    break;
1162
 
  }
1163
 
 
1164
 
  if (field.type() == Table::Field::INTEGER || 
1165
 
      field.type() == Table::Field::BIGINT)
1166
 
  {
1167
 
    if (field.has_constraints() &&
1168
 
        field.constraints().has_is_unsigned() &&
1169
 
        field.constraints().is_unsigned())
1170
 
    {
1171
 
      destination.append(" UNSIGNED", 9);
1172
 
    }
1173
 
  }
1174
 
 
1175
 
 
1176
 
  if (! (field.has_constraints() &&
1177
 
         field.constraints().is_nullable()))
1178
 
  {
1179
 
    destination.append(" NOT", 4);
1180
 
  }
1181
 
  destination.append(" NULL", 5);
1182
 
 
1183
 
  if (field.type() == Table::Field::INTEGER || 
1184
 
      field.type() == Table::Field::BIGINT)
1185
 
  {
1186
 
    /* AUTO_INCREMENT must be after NOT NULL */
1187
 
    if (field.has_numeric_options() &&
1188
 
        field.numeric_options().is_autoincrement())
1189
 
    {
1190
 
      destination.append(" AUTO_INCREMENT", 15);
1191
 
    }
1192
 
  }
1193
 
 
1194
 
  if (field.type() == Table::Field::BLOB ||
1195
 
      field.type() == Table::Field::VARCHAR)
1196
 
  {
1197
 
    if (field.string_options().has_collation())
1198
 
    {
1199
 
      destination.append(" COLLATE ", 9);
1200
 
      destination.append(field.string_options().collation());
1201
 
    }
1202
 
  }
1203
 
 
1204
 
  if (field.options().has_default_value())
1205
 
  {
1206
 
    destination.append(" DEFAULT ", 9);
1207
 
    destination.push_back(quoted_identifier);
1208
 
    destination.append(field.options().default_value());
1209
 
    destination.push_back(quoted_identifier);
1210
 
  }
1211
 
 
1212
 
  if (field.options().has_default_bin_value())
1213
 
  {
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++)
1217
 
    {
1218
 
      printf("%.2x", *(v.c_str() + x));
1219
 
    }
1220
 
  }
1221
 
 
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);
1226
 
 
1227
 
  if (field.has_comment())
1228
 
  {
1229
 
    destination.append(" COMMENT ", 9);
1230
 
    destination.push_back(quoted_identifier);
1231
 
    destination.append(field.comment());
1232
 
    destination.push_back(quoted_identifier);
1233
 
  }
1234
 
  return NONE;
1235
 
}
1236
 
 
1237
 
bool shouldQuoteFieldValue(Table::Field::FieldType in_type)
1238
 
{
1239
 
  switch (in_type)
1240
 
  {
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:
1246
 
    return false;
1247
 
  default:
1248
 
    return true;
1249
 
  } 
1250
 
}
1251
 
 
1252
 
Table::Field::FieldType internalFieldTypeToFieldProtoType(enum enum_field_types type)
1253
 
{
1254
 
  switch (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;
1278
 
  }
1279
 
 
1280
 
  assert(false);
1281
 
  return Table::Field::INTEGER; /* unreachable */
1282
 
}
1283
 
 
1284
 
bool transactionContainsBulkSegment(const Transaction &transaction)
1285
 
{
1286
 
  size_t num_statements= transaction.statement_size();
1287
 
  if (num_statements == 0)
1288
 
    return false;
1289
 
 
1290
 
  /*
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.
1294
 
   */
1295
 
  size_t x;
1296
 
  for (x= 0; x < num_statements; ++x)
1297
 
  {
1298
 
    const Statement &statement= transaction.statement(x);
1299
 
    Statement::Type type= statement.type();
1300
 
 
1301
 
    switch (type)
1302
 
    {
1303
 
      case Statement::INSERT:
1304
 
        if (statement.insert_data().segment_id() > 1)
1305
 
          return true;
1306
 
        break;
1307
 
      case Statement::UPDATE:
1308
 
        if (statement.update_data().segment_id() > 1)
1309
 
          return true;
1310
 
        break;
1311
 
      case Statement::DELETE:
1312
 
        if (statement.delete_data().segment_id() > 1)
1313
 
          return true;
1314
 
        break;
1315
 
      default:
1316
 
        break;
1317
 
    }
1318
 
  }
1319
 
  return false;
1320
 
}
1321
 
 
1322
 
} /* namespace message */
1323
 
} /* namespace drizzled */