18
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21
#ifdef USE_PRAGMA_IMPLEMENTATION
22
#pragma implementation // gcc: Class implementation
22
25
#include <drizzled/server_includes.h>
23
26
#include <drizzled/field/timestamp.h>
24
#include <drizzled/error.h>
25
#include <drizzled/tztime.h>
26
#include <drizzled/table.h>
27
#include <drizzled/session.h>
29
#include "drizzled/temporal.h"
33
TIMESTAMP type holds datetime values in range from 1970-01-01 00:00:01 UTC to
34
2038-01-01 00:00:00 UTC stored as number of seconds since Unix
29
TIMESTAMP type holds datetime values in range from 1970-01-01 00:00:01 UTC to
30
2038-01-01 00:00:00 UTC stored as number of seconds since Unix
37
Up to one of timestamps columns in the table can be automatically
33
Up to one of timestamps columns in the table can be automatically
38
34
set on row update and/or have NOW() as default value.
39
TABLE::timestamp_field points to Field object for such timestamp with
35
TABLE::timestamp_field points to Field object for such timestamp with
40
36
auto-set-on-update. TABLE::time_stamp holds offset in record + 1 for this
41
37
field, and is used by handler code which performs updates required.
43
39
Actually SQL-99 says that we should allow niladic functions (like NOW())
44
as defaults for any field. Current limitations (only NOW() and only
45
for one TIMESTAMP field) are because of restricted binary .frm format
40
as defaults for any field. Current limitations (only NOW() and only
41
for one TIMESTAMP field) are because of restricted binary .frm format
46
42
and should go away in the future.
48
44
Also because of this limitation of binary .frm format we use 5 different
49
45
unireg_check values with TIMESTAMP field to distinguish various cases of
50
46
DEFAULT or ON UPDATE values. These values are:
52
48
TIMESTAMP_OLD_FIELD - old timestamp, if there was not any fields with
53
auto-set-on-update (or now() as default) in this table before, then this
54
field has NOW() as default and is updated when row changes, else it is
49
auto-set-on-update (or now() as default) in this table before, then this
50
field has NOW() as default and is updated when row changes, else it is
55
51
field which has 0 as default value and is not automatically updated.
56
52
TIMESTAMP_DN_FIELD - field with NOW() as default but not set on update
57
53
automatically (TIMESTAMP DEFAULT NOW())
58
TIMESTAMP_UN_FIELD - field which is set on update automatically but has not
59
NOW() as default (but it may has 0 or some other const timestamp as
54
TIMESTAMP_UN_FIELD - field which is set on update automatically but has not
55
NOW() as default (but it may has 0 or some other const timestamp as
60
56
default) (TIMESTAMP ON UPDATE NOW()).
61
TIMESTAMP_DNUN_FIELD - field which has now() as default and is auto-set on
57
TIMESTAMP_DNUN_FIELD - field which has now() as default and is auto-set on
62
58
update. (TIMESTAMP DEFAULT NOW() ON UPDATE NOW())
63
NONE - field which is not auto-set on update with some other than NOW()
59
NONE - field which is not auto-set on update with some other than NOW()
64
60
default value (TIMESTAMP DEFAULT 0).
66
Note that TIMESTAMP_OLD_FIELDs are never created explicitly now, they are
67
left only for preserving ability to read old tables. Such fields replaced
68
with their newer analogs in CREATE TABLE and in SHOW CREATE TABLE. This is
69
because we want to prefer NONE unireg_check before TIMESTAMP_OLD_FIELD for
70
"TIMESTAMP DEFAULT 'Const'" field. (Old timestamps allowed such
71
specification too but ignored default value for first timestamp, which of
62
Note that TIMESTAMP_OLD_FIELDs are never created explicitly now, they are
63
left only for preserving ability to read old tables. Such fields replaced
64
with their newer analogs in CREATE TABLE and in SHOW CREATE TABLE. This is
65
because we want to prefer NONE unireg_check before TIMESTAMP_OLD_FIELD for
66
"TIMESTAMP DEFAULT 'Const'" field. (Old timestamps allowed such
67
specification too but ignored default value for first timestamp, which of
72
68
course is non-standard.) In most cases user won't notice any change, only
73
69
exception is different behavior of old/new timestamps during ALTER TABLE.
75
72
Field_timestamp::Field_timestamp(unsigned char *ptr_arg,
73
uint32_t len_arg __attribute__((unused)),
77
74
unsigned char *null_ptr_arg, unsigned char null_bit_arg,
78
75
enum utype unireg_check_arg,
79
76
const char *field_name_arg,
81
78
const CHARSET_INFO * const cs)
83
drizzled::DateTime::MAX_STRING_LENGTH - 1 /* no \0 */,
84
null_ptr_arg, null_bit_arg,
79
:Field_str(ptr_arg, MAX_DATETIME_WIDTH, null_ptr_arg, null_bit_arg,
85
80
unireg_check_arg, field_name_arg, cs)
87
82
/* For 4.0 MYD and 4.0 InnoDB compatibility */
147
144
int Field_timestamp::store(const char *from,
149
const CHARSET_INFO * const )
151
drizzled::Timestamp temporal;
153
ASSERT_COLUMN_MARKED_FOR_WRITE;
155
if (! temporal.from_string(from, (size_t) len))
157
my_error(ER_INVALID_UNIX_TIMESTAMP_VALUE, MYF(ME_FATALERROR), from);
162
temporal.to_time_t(&tmp);
164
store_timestamp(tmp);
168
int Field_timestamp::store(double from)
170
ASSERT_COLUMN_MARKED_FOR_WRITE;
172
if (from < 0 || from > 99991231235959.0)
174
/* Convert the double to a string using stringstream */
175
std::stringstream ss;
177
ss.precision(18); /* 18 places should be fine for error display of double input. */
178
ss << from; ss >> tmp;
180
my_error(ER_INVALID_UNIX_TIMESTAMP_VALUE, MYF(ME_FATALERROR), tmp.c_str());
183
return Field_timestamp::store((int64_t) rint(from), false);
186
int Field_timestamp::store(int64_t from, bool)
188
ASSERT_COLUMN_MARKED_FOR_WRITE;
191
* Try to create a DateTime from the supplied integer. Throw an error
192
* if unable to create a valid DateTime.
194
drizzled::Timestamp temporal;
195
if (! temporal.from_int64_t(from))
197
/* Convert the integer to a string using stringstream */
198
std::stringstream ss;
200
ss << from; ss >> tmp;
202
my_error(ER_INVALID_UNIX_TIMESTAMP_VALUE, MYF(ME_FATALERROR), tmp.c_str());
207
temporal.to_time_t(&tmp);
209
store_timestamp(tmp);
146
const CHARSET_INFO * const cs __attribute__((unused)))
151
bool have_smth_to_conv;
152
bool in_dst_time_gap;
153
THD *thd= table ? table->in_use : current_thd;
155
/* We don't want to store invalid or fuzzy datetime values in TIMESTAMP */
156
have_smth_to_conv= (str_to_datetime(from, len, &l_time, 1, &error) >
157
DRIZZLE_TIMESTAMP_ERROR);
159
if (error || !have_smth_to_conv)
162
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_TRUNCATED,
163
from, len, DRIZZLE_TIMESTAMP_DATETIME, 1);
166
/* Only convert a correct date (not a zero date) */
167
if (have_smth_to_conv && l_time.month)
169
if (!(tmp= TIME_to_timestamp(thd, &l_time, &in_dst_time_gap)))
171
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
172
ER_WARN_DATA_OUT_OF_RANGE,
173
from, len, DRIZZLE_TIMESTAMP_DATETIME, !error);
176
else if (in_dst_time_gap)
178
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
179
ER_WARN_INVALID_TIMESTAMP,
180
from, len, DRIZZLE_TIMESTAMP_DATETIME, !error);
184
store_timestamp(tmp);
189
int Field_timestamp::store(double nr)
192
if (nr < 0 || nr > 99991231235959.0)
194
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
195
ER_WARN_DATA_OUT_OF_RANGE,
196
nr, DRIZZLE_TIMESTAMP_DATETIME);
197
nr= 0; // Avoid overflow on buff
200
error|= Field_timestamp::store((int64_t) rint(nr), false);
205
int Field_timestamp::store(int64_t nr,
206
bool unsigned_val __attribute__((unused)))
209
my_time_t timestamp= 0;
211
bool in_dst_time_gap;
212
THD *thd= table ? table->in_use : current_thd;
214
/* We don't want to store invalid or fuzzy datetime values in TIMESTAMP */
215
int64_t tmp= number_to_datetime(nr, &l_time, (thd->variables.sql_mode &
216
MODE_NO_ZERO_DATE), &error);
217
if (tmp == INT64_C(-1))
224
if (!(timestamp= TIME_to_timestamp(thd, &l_time, &in_dst_time_gap)))
226
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
227
ER_WARN_DATA_OUT_OF_RANGE,
228
nr, DRIZZLE_TIMESTAMP_DATETIME, 1);
233
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
234
ER_WARN_INVALID_TIMESTAMP,
235
nr, DRIZZLE_TIMESTAMP_DATETIME, 1);
239
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
240
ER_WARN_DATA_TRUNCATED,
241
nr, DRIZZLE_TIMESTAMP_DATETIME, 1);
243
store_timestamp(timestamp);
213
247
double Field_timestamp::val_real(void)
218
252
int64_t Field_timestamp::val_int(void)
222
ASSERT_COLUMN_MARKED_FOR_READ;
255
DRIZZLE_TIME time_tmp;
256
THD *thd= table ? table->in_use : current_thd;
258
thd->time_zone_used= 1;
224
259
#ifdef WORDS_BIGENDIAN
225
260
if (table && table->s->db_low_byte_first)
226
temp= uint4korr(ptr);
231
drizzled::Timestamp temporal;
232
(void) temporal.from_time_t((time_t) temp);
234
/* We must convert into a "timestamp-formatted integer" ... */
236
temporal.to_int64_t(&result);
266
if (temp == 0L) // No time
267
return(0); /* purecov: inspected */
269
thd->variables.time_zone->gmt_sec_to_TIME(&time_tmp, (my_time_t)temp);
271
return time_tmp.year * INT64_C(10000000000) +
272
time_tmp.month * INT64_C(100000000) +
273
time_tmp.day * 1000000 + time_tmp.hour * 10000 +
274
time_tmp.minute * 100 + time_tmp.second;
240
String *Field_timestamp::val_str(String *val_buffer, String *)
278
String *Field_timestamp::val_str(String *val_buffer, String *val_ptr)
280
uint32_t temp, temp2;
281
DRIZZLE_TIME time_tmp;
282
THD *thd= table ? table->in_use : current_thd;
244
int to_len= field_length + 1;
246
val_buffer->alloc(to_len);
247
to= (char *) val_buffer->ptr();
285
val_buffer->alloc(field_length+1);
286
to= (char*) val_buffer->ptr();
287
val_buffer->length(field_length);
289
thd->time_zone_used= 1;
249
290
#ifdef WORDS_BIGENDIAN
250
291
if (table && table->s->db_low_byte_first)
251
temp= uint4korr(ptr);
256
val_buffer->set_charset(&my_charset_bin); /* Safety */
258
drizzled::Timestamp temporal;
259
(void) temporal.from_time_t((time_t) temp);
262
rlen= temporal.to_string(to, to_len);
263
assert(rlen < to_len);
265
val_buffer->length(rlen);
298
{ /* Zero time is "000000" */
299
val_ptr->set(STRING_WITH_LEN("0000-00-00 00:00:00"), &my_charset_bin);
302
val_buffer->set_charset(&my_charset_bin); // Safety
304
thd->variables.time_zone->gmt_sec_to_TIME(&time_tmp,(my_time_t)temp);
306
temp= time_tmp.year % 100;
307
if (temp < YY_PART_YEAR - 1)
317
temp2=temp/10; temp=temp-temp2*10;
318
*to++= (char) ('0'+(char) (temp2));
319
*to++= (char) ('0'+(char) (temp));
322
temp2=temp/10; temp=temp-temp2*10;
323
*to++= (char) ('0'+(char) (temp2));
324
*to++= (char) ('0'+(char) (temp));
327
temp2=temp/10; temp=temp-temp2*10;
328
*to++= (char) ('0'+(char) (temp2));
329
*to++= (char) ('0'+(char) (temp));
332
temp2=temp/10; temp=temp-temp2*10;
333
*to++= (char) ('0'+(char) (temp2));
334
*to++= (char) ('0'+(char) (temp));
336
temp=time_tmp.minute;
337
temp2=temp/10; temp=temp-temp2*10;
338
*to++= (char) ('0'+(char) (temp2));
339
*to++= (char) ('0'+(char) (temp));
341
temp=time_tmp.second;
342
temp2=temp/10; temp=temp-temp2*10;
343
*to++= (char) ('0'+(char) (temp2));
344
*to++= (char) ('0'+(char) (temp));
266
346
return val_buffer;
269
bool Field_timestamp::get_date(DRIZZLE_TIME *ltime, uint32_t)
350
bool Field_timestamp::get_date(DRIZZLE_TIME *ltime, uint32_t fuzzydate)
353
THD *thd= table ? table->in_use : current_thd;
354
thd->time_zone_used= 1;
273
355
#ifdef WORDS_BIGENDIAN
274
356
if (table && table->s->db_low_byte_first)
275
temp= uint4korr(ptr);
280
memset(ltime, 0, sizeof(*ltime));
282
drizzled::Timestamp temporal;
283
(void) temporal.from_time_t((time_t) temp);
285
/* @TODO Goodbye the below code when DRIZZLE_TIME is finally gone.. */
287
ltime->time_type= DRIZZLE_TIMESTAMP_DATETIME;
288
ltime->year= temporal.years();
289
ltime->month= temporal.months();
290
ltime->day= temporal.days();
291
ltime->hour= temporal.hours();
292
ltime->minute= temporal.minutes();
293
ltime->second= temporal.seconds();
362
{ /* Zero time is "000000" */
363
if (fuzzydate & TIME_NO_ZERO_DATE)
365
memset(ltime, 0, sizeof(*ltime));
369
thd->variables.time_zone->gmt_sec_to_TIME(ltime, (my_time_t)temp);
342
428
void Field_timestamp::sql_type(String &res) const
344
430
res.set_ascii(STRING_WITH_LEN("timestamp"));
347
434
void Field_timestamp::set_time()
349
Session *session= table ? table->in_use : current_session;
350
long tmp= (long) session->query_start();
436
THD *thd= table ? table->in_use : current_thd;
437
long tmp= (long) thd->query_start();
352
439
store_timestamp(tmp);
355
void Field_timestamp::set_default()
357
if (table->timestamp_field == this &&
358
unireg_check != TIMESTAMP_UN_FIELD)
361
Field::set_default();
364
long Field_timestamp::get_timestamp(bool *null_value)
366
if ((*null_value= is_null()))
368
#ifdef WORDS_BIGENDIAN
369
if (table && table->s->db_low_byte_first)
370
return sint4korr(ptr);
377
void Field_timestamp::store_timestamp(time_t timestamp)
379
#ifdef WORDS_BIGENDIAN
380
if (table && table->s->db_low_byte_first)
382
int4store(ptr,timestamp);
386
longstore(ptr,(uint32_t) timestamp);