~drizzle-trunk/drizzle/development

1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
 *  vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3
 *
1999.6.1 by kalebral at gmail
update Copyright strings to a more common format to help with creating the master debian copyright file
4
 *  Copyright (C) 2008-2009 Sun Microsystems, Inc.
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
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; version 2 of the License.
9
 *
10
 *  This program is distributed in the hope that it will be useful,
11
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
 *  GNU General Public License for more details.
14
 *
15
 *  You should have received a copy of the GNU General Public License
16
 *  along with this program; if not, write to the Free Software
17
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
18
 */
19
20
/**
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
21
 * @file Implementation of CreateField class
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
22
 */
23
1241.9.36 by Monty Taylor
ZOMG. I deleted drizzled/server_includes.h.
24
#include "config.h"
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
25
#include <errno.h>
1241.9.1 by Monty Taylor
Removed global.h. Fixed all the headers.
26
#include <float.h>
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
27
#include "drizzled/sql_select.h"
28
#include "drizzled/error.h"
29
#include "drizzled/field.h"
30
#include "drizzled/create_field.h"
31
#include "drizzled/field/str.h"
32
#include "drizzled/field/num.h"
33
#include "drizzled/field/blob.h"
34
#include "drizzled/field/enum.h"
35
#include "drizzled/field/null.h"
36
#include "drizzled/field/date.h"
37
#include "drizzled/field/decimal.h"
38
#include "drizzled/field/real.h"
39
#include "drizzled/field/double.h"
2007 by Brian Aker
Refactor naming for integers.
40
#include "drizzled/field/int32.h"
41
#include "drizzled/field/int64.h"
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
42
#include "drizzled/field/num.h"
1999.4.9 by Brian Aker
Created EPOCH
43
#include "drizzled/field/epoch.h"
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
44
#include "drizzled/field/datetime.h"
45
#include "drizzled/field/varstring.h"
1996.2.1 by Brian Aker
uuid type code.
46
#include "drizzled/field/uuid.h"
1079.3.2 by Stewart Smith
Replace MAX_(DATE|TIME).*_WIDTH defines in definitions.h with real (and correct) static const members to Temporal types.
47
#include "drizzled/temporal.h"
1878.3.2 by Monty Taylor
Split out show_type into its own header and made sys_var work through
48
#include "drizzled/item/string.h"
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
49
2008.2.4 by Brian Aker
Merge in additional fixes for sign, plus alter table, plus TIME on
50
#include "drizzled/display.h"
51
1067.4.1 by Nathan Williams
First few changes at converting cmin to std::min.
52
#include <algorithm>
53
54
using namespace std;
55
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
56
namespace drizzled
57
{
1067.4.1 by Nathan Williams
First few changes at converting cmin to std::min.
58
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
59
/** Create a field suitable for create of table. */
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
60
CreateField::CreateField(Field *old_field, Field *orig_field)
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
61
{
62
  field= old_field;
63
  field_name= change= old_field->field_name;
64
  length= old_field->field_length;
65
  flags= old_field->flags;
66
  unireg_check= old_field->unireg_check;
67
  pack_length= old_field->pack_length();
68
  key_length= old_field->key_length();
69
  sql_type= old_field->real_type();
70
  charset= old_field->charset(); // May be NULL ptr
71
  comment= old_field->comment;
72
  decimals= old_field->decimals();
73
74
  /* Fix if the original table had 4 byte pointer blobs */
75
  if (flags & BLOB_FLAG)
1660.1.3 by Brian Aker
Encapsulate Table in field
76
    pack_length= (pack_length - old_field->getTable()->getShare()->blob_ptr_size + portable_sizeof_char_ptr);
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
77
78
  switch (sql_type) 
79
  {
80
    case DRIZZLE_TYPE_BLOB:
81
      sql_type= DRIZZLE_TYPE_BLOB;
82
      length/= charset->mbmaxlen;
83
      key_length/= charset->mbmaxlen;
84
      break;
85
    case DRIZZLE_TYPE_ENUM:
86
    case DRIZZLE_TYPE_VARCHAR:
87
      /* This is corrected in create_length_to_internal_length */
88
      length= (length+charset->mbmaxlen-1) / charset->mbmaxlen;
89
      break;
90
    default:
91
      break;
92
  }
93
1315.2.15 by Stewart Smith
remove the now unused SET_FLAG define. We don't have the SET field type, and this flag was never set.
94
  if (flags & ENUM_FLAG)
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
95
    interval= ((Field_enum*) old_field)->typelib;
96
  else
97
    interval= 0;
98
  def= 0;
99
  char_length= length;
100
1638.10.114 by Stewart Smith
when creating a CreateField from a Field, a auto_incremen column should not have a default value of 0 as that's not magic.
101
  if (!(flags & (NO_DEFAULT_VALUE_FLAG)) &&
102
      !(flags & AUTO_INCREMENT_FLAG) &&
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
103
      old_field->ptr && orig_field &&
104
      (sql_type != DRIZZLE_TYPE_TIMESTAMP ||                /* set def only if */
1660.1.3 by Brian Aker
Encapsulate Table in field
105
       old_field->getTable()->timestamp_field != old_field ||  /* timestamp field */
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
106
       unireg_check == Field::TIMESTAMP_UN_FIELD))        /* has default val */
107
  {
108
    ptrdiff_t diff;
109
110
    /* Get the value from default_values */
1672.3.6 by Brian Aker
First pass in encapsulating row
111
    diff= (ptrdiff_t) (orig_field->getTable()->getDefaultValues() - orig_field->getTable()->getInsertRecord());
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
112
    orig_field->move_field_offset(diff);	// Points now at default_values
113
    if (! orig_field->is_real_null())
114
    {
115
      char buff[MAX_FIELD_WIDTH], *pos;
116
      String tmp(buff, sizeof(buff), charset), *res;
1996.2.1 by Brian Aker
uuid type code.
117
      res= orig_field->val_str_internal(&tmp);
1253.1.6 by Monty Taylor
Moved mem_root functions into drizzled::memory:: namespace.
118
      pos= (char*) memory::sql_strmake(res->ptr(), res->length());
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
119
      def= new Item_string(pos, res->length(), charset);
120
    }
1672.3.6 by Brian Aker
First pass in encapsulating row
121
    orig_field->move_field_offset(-diff);	// Back to getInsertRecord()
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
122
  }
123
}
124
125
/**
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
126
  Convert CreateField::length from number of characters to number of bytes.
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
127
*/
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
128
void CreateField::create_length_to_internal_length(void)
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
129
{
130
  switch (sql_type) 
131
  {
132
    case DRIZZLE_TYPE_BLOB:
133
    case DRIZZLE_TYPE_VARCHAR:
134
      length*= charset->mbmaxlen;
135
      key_length= length;
136
      pack_length= calc_pack_length(sql_type, length);
137
      break;
138
    case DRIZZLE_TYPE_ENUM:
139
      /* Pack_length already calculated in ::init() */
140
      length*= charset->mbmaxlen;
141
      key_length= pack_length;
142
      break;
1211.1.1 by Brian Aker
Updating with my change to to DECIMAL from NEWDECIMAL and Stewart's update
143
    case DRIZZLE_TYPE_DECIMAL:
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
144
      key_length= pack_length=
145
        my_decimal_get_binary_size(my_decimal_length_to_precision(length,
146
                  decimals,
147
                  flags &
148
                  UNSIGNED_FLAG),
149
          decimals);
150
      break;
151
    default:
152
      key_length= pack_length= calc_pack_length(sql_type, length);
153
      break;
154
  }
155
}
156
157
/**
158
  Init for a tmp table field. To be extended if need be.
159
*/
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
160
void CreateField::init_for_tmp_table(enum_field_types sql_type_arg,
1119.9.10 by Jay Pipes
Removes FIELDFLAG_MAYBE_NULL and f_maybe_null() macro.
161
                                     uint32_t length_arg,
162
                                     uint32_t decimals_arg,
1119.9.18 by Stewart Smith
Fix maybe_null for CreateField created for temp table in item/sum.cc (Item_sum_distinct)
163
                                     bool maybe_null)
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
164
{
165
  field_name= "";
166
  sql_type= sql_type_arg;
167
  char_length= length= length_arg;;
168
  unireg_check= Field::NONE;
169
  interval= 0;
170
  charset= &my_charset_bin;
1119.9.16 by Stewart Smith
don't store decimal/double scale in pack_flag, instead use the numeric option scale field in the table proto. This removes f_decimals() macro and the bits in the pack_flag for scale.
171
  decimals= decimals_arg & FIELDFLAG_MAX_DEC;
1119.9.18 by Stewart Smith
Fix maybe_null for CreateField created for temp table in item/sum.cc (Item_sum_distinct)
172
173
  if (! maybe_null)
174
    flags= NOT_NULL_FLAG;
175
  else
176
    flags= 0;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
177
}
178
1052.2.7 by Nathan Williams
Merged trunk and resolved conflicts.
179
bool CreateField::init(Session *,
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
180
                        char *fld_name,
181
                        enum_field_types fld_type,
182
                        char *fld_length,
183
                        char *fld_decimals,
184
                        uint32_t fld_type_modifier,
185
                        Item *fld_default_value,
186
                        Item *fld_on_update_value,
187
                        LEX_STRING *fld_comment,
188
                        char *fld_change,
189
                        List<String> *fld_interval_list,
190
                        const CHARSET_INFO * const fld_charset,
191
                        uint32_t,
192
                        enum column_format_type column_format_in)
193
                        
194
{
195
  uint32_t sign_len= 0;
196
  uint32_t allowed_type_modifier= 0;
197
  uint32_t max_field_charlength= MAX_FIELD_CHARLENGTH;
198
199
  field= 0;
200
  field_name= fld_name;
201
  def= fld_default_value;
202
  flags= fld_type_modifier;
203
  flags|= (((uint32_t)column_format_in & COLUMN_FORMAT_MASK) << COLUMN_FORMAT_FLAGS);
204
  unireg_check= (fld_type_modifier & AUTO_INCREMENT_FLAG ?
205
                 Field::NEXT_NUMBER : Field::NONE);
206
  decimals= fld_decimals ? (uint32_t)atoi(fld_decimals) : 0;
207
  if (decimals >= NOT_FIXED_DEC)
208
  {
209
    my_error(ER_TOO_BIG_SCALE, MYF(0), decimals, fld_name,
210
             NOT_FIXED_DEC-1);
211
    return(true);
212
  }
213
214
  sql_type= fld_type;
215
  length= 0;
216
  change= fld_change;
217
  interval= 0;
218
  pack_length= key_length= 0;
219
  charset= fld_charset;
1101.1.24 by Monty Taylor
Reverted my change to interval_list
220
  interval_list.empty();
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
221
222
  comment= *fld_comment;
223
224
  /*
225
    Set NO_DEFAULT_VALUE_FLAG if this field doesn't have a default value and
226
    it is NOT NULL, not an AUTO_INCREMENT field and not a TIMESTAMP.
227
  */
228
  if (!fld_default_value && !(fld_type_modifier & AUTO_INCREMENT_FLAG) &&
229
      (fld_type_modifier & NOT_NULL_FLAG) && fld_type != DRIZZLE_TYPE_TIMESTAMP)
230
    flags|= NO_DEFAULT_VALUE_FLAG;
231
232
  if (fld_length && !(length= (uint32_t) atoi(fld_length)))
971.6.11 by Eric Day
Removed purecov messages.
233
    fld_length= 0;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
234
  sign_len= fld_type_modifier & UNSIGNED_FLAG ? 0 : 1;
235
236
  switch (fld_type) 
237
  {
238
    case DRIZZLE_TYPE_LONG:
239
      if (!fld_length)
240
        length= MAX_INT_WIDTH+sign_len;
241
      allowed_type_modifier= AUTO_INCREMENT_FLAG;
242
      break;
243
    case DRIZZLE_TYPE_LONGLONG:
244
      if (!fld_length)
245
        length= MAX_BIGINT_WIDTH;
246
      allowed_type_modifier= AUTO_INCREMENT_FLAG;
247
      break;
248
    case DRIZZLE_TYPE_NULL:
249
      break;
1211.1.1 by Brian Aker
Updating with my change to to DECIMAL from NEWDECIMAL and Stewart's update
250
    case DRIZZLE_TYPE_DECIMAL:
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
251
      my_decimal_trim(&length, &decimals);
252
      if (length > DECIMAL_MAX_PRECISION)
253
      {
254
        my_error(ER_TOO_BIG_PRECISION, MYF(0), length, fld_name,
255
                DECIMAL_MAX_PRECISION);
256
        return(true);
257
      }
258
      if (length < decimals)
259
      {
260
        my_error(ER_M_BIGGER_THAN_D, MYF(0), fld_name);
261
        return(true);
262
      }
263
      length= my_decimal_precision_to_length(length, decimals, fld_type_modifier & UNSIGNED_FLAG);
264
      pack_length= my_decimal_get_binary_size(length, decimals);
265
      break;
266
    case DRIZZLE_TYPE_VARCHAR:
267
      /*
268
        Long VARCHAR's are automaticly converted to blobs in mysql_prepare_table
269
        if they don't have a default value
270
      */
271
      max_field_charlength= MAX_FIELD_VARCHARLENGTH;
272
      break;
273
    case DRIZZLE_TYPE_BLOB:
274
      if (fld_default_value)
275
      {
276
        /* Allow empty as default value. */
277
        String str,*res;
278
        res= fld_default_value->val_str(&str);
279
        if (res->length())
280
        {
281
          my_error(ER_BLOB_CANT_HAVE_DEFAULT, MYF(0), fld_name);
282
          return(true);
283
        }
284
285
      }
286
      flags|= BLOB_FLAG;
287
      break;
288
    case DRIZZLE_TYPE_DOUBLE:
289
      allowed_type_modifier= AUTO_INCREMENT_FLAG;
290
      if (!fld_length && !fld_decimals)
291
      {
292
        length= DBL_DIG+7;
293
        decimals= NOT_FIXED_DEC;
294
      }
295
      if (length < decimals &&
296
          decimals != NOT_FIXED_DEC)
297
      {
298
        my_error(ER_M_BIGGER_THAN_D, MYF(0), fld_name);
299
        return(true);
300
      }
301
      break;
302
    case DRIZZLE_TYPE_TIMESTAMP:
303
      if (!fld_length)
304
      {
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
305
        length= DateTime::MAX_STRING_LENGTH;
1079.3.2 by Stewart Smith
Replace MAX_(DATE|TIME).*_WIDTH defines in definitions.h with real (and correct) static const members to Temporal types.
306
      }
307
308
      /* This assert() should be correct due to absence of length
309
         specifiers for timestamp. Previous manipulation also wasn't
310
         ever called (from examining lcov)
311
      */
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
312
      assert(length == (uint32_t)DateTime::MAX_STRING_LENGTH);
1079.3.2 by Stewart Smith
Replace MAX_(DATE|TIME).*_WIDTH defines in definitions.h with real (and correct) static const members to Temporal types.
313
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
314
      flags|= UNSIGNED_FLAG;
315
      if (fld_default_value)
316
      {
317
        /* Grammar allows only NOW() value for ON UPDATE clause */
318
        if (fld_default_value->type() == Item::FUNC_ITEM &&
319
            ((Item_func*)fld_default_value)->functype() == Item_func::NOW_FUNC)
320
        {
321
          unireg_check= (fld_on_update_value ? Field::TIMESTAMP_DNUN_FIELD:
322
                                              Field::TIMESTAMP_DN_FIELD);
323
          /*
324
            We don't need default value any longer moreover it is dangerous.
325
            Everything handled by unireg_check further.
326
          */
327
          def= 0;
328
        }
329
        else
330
          unireg_check= (fld_on_update_value ? Field::TIMESTAMP_UN_FIELD:
331
                                              Field::NONE);
332
      }
333
      else
334
      {
335
        /*
336
          If we have default TIMESTAMP NOT NULL column without explicit DEFAULT
337
          or ON UPDATE values then for the sake of compatiblity we should treat
338
          this column as having DEFAULT NOW() ON UPDATE NOW() (when we don't
339
          have another TIMESTAMP column with auto-set option before this one)
340
          or DEFAULT 0 (in other cases).
341
          So here we are setting TIMESTAMP_OLD_FIELD only temporary, and will
342
          replace this value by TIMESTAMP_DNUN_FIELD or NONE later when
343
          information about all TIMESTAMP fields in table will be availiable.
344
345
          If we have TIMESTAMP NULL column without explicit DEFAULT value
346
          we treat it as having DEFAULT NULL attribute.
347
        */
348
        unireg_check= (fld_on_update_value ? Field::TIMESTAMP_UN_FIELD :
349
                      (flags & NOT_NULL_FLAG ? Field::TIMESTAMP_OLD_FIELD :
350
                                                Field::NONE));
351
      }
352
      break;
353
    case DRIZZLE_TYPE_DATE:
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
354
      length= Date::MAX_STRING_LENGTH;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
355
      break;
1996.2.1 by Brian Aker
uuid type code.
356
    case DRIZZLE_TYPE_UUID:
357
      length= field::Uuid::max_string_length();
358
      break;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
359
    case DRIZZLE_TYPE_DATETIME:
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
360
      length= DateTime::MAX_STRING_LENGTH;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
361
      break;
1999.4.7 by Brian Aker
Merge in first pass of TIME type (closer to EPOCH time).
362
    case DRIZZLE_TYPE_TIME:
363
      length= DateTime::MAX_STRING_LENGTH;
364
      break;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
365
    case DRIZZLE_TYPE_ENUM:
366
      {
367
        /* Should be safe. */
1782.4.4 by Brian Aker
Fix enum at being an intefer (which is what PG did, and it saves on
368
        pack_length= 4;
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
369
370
        List_iterator<String> it(*fld_interval_list);
371
        String *tmp;
372
        while ((tmp= it++))
1101.1.24 by Monty Taylor
Reverted my change to interval_list
373
          interval_list.push_back(tmp);
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
374
        length= 1;
375
        break;
376
    }
377
  }
378
  /* Remember the value of length */
379
  char_length= length;
380
381
  if (!(flags & BLOB_FLAG) &&
382
      ((length > max_field_charlength &&
383
        fld_type != DRIZZLE_TYPE_ENUM &&
384
        (fld_type != DRIZZLE_TYPE_VARCHAR || fld_default_value)) ||
385
       (!length && fld_type != DRIZZLE_TYPE_VARCHAR)))
386
  {
387
    my_error((fld_type == DRIZZLE_TYPE_VARCHAR) ?  ER_TOO_BIG_FIELDLENGTH : ER_TOO_BIG_DISPLAYWIDTH,
388
              MYF(0),
971.6.11 by Eric Day
Removed purecov messages.
389
              fld_name, max_field_charlength);
1055.2.10 by Jay Pipes
Moves Create_field implementation out into its own implementation file.
390
    return true;
391
  }
392
  fld_type_modifier&= AUTO_INCREMENT_FLAG;
393
  if ((~allowed_type_modifier) & fld_type_modifier)
394
  {
395
    my_error(ER_WRONG_FIELD_SPEC, MYF(0), fld_name);
396
    return true;
397
  }
398
399
  return false; /* success */
400
}
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
401
2008.2.4 by Brian Aker
Merge in additional fixes for sign, plus alter table, plus TIME on
402
std::ostream& operator<<(std::ostream& output, const CreateField &field)
403
{
404
  output << "CreateField:(";
405
  output <<  field.field_name;
406
  output << ", ";
407
  output << drizzled::display::type(field.type());
408
  output << ", { ";
409
410
  if (field.flags & NOT_NULL_FLAG)
411
    output << " NOT_NULL";
412
413
  if (field.flags & PRI_KEY_FLAG)
414
    output << ", PRIMARY KEY";
415
416
  if (field.flags & UNIQUE_KEY_FLAG)
417
    output << ", UNIQUE KEY";
418
419
  if (field.flags & MULTIPLE_KEY_FLAG)
420
    output << ", MULTIPLE KEY";
421
422
  if (field.flags & BLOB_FLAG)
423
    output << ", BLOB";
424
425
  if (field.flags & UNSIGNED_FLAG)
426
    output << ", UNSIGNED";
427
428
  if (field.flags & BINARY_FLAG)
429
    output << ", BINARY";
430
  output << "}, ";
431
  if (field.field)
432
    output << *field.field;
433
  output << ")";
434
435
  return output;  // for multiple << operators.
436
}
437
1280.1.10 by Monty Taylor
Put everything in drizzled into drizzled namespace.
438
} /* namespace drizzled */