18
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22
#include "drizzled/field/datetime.h"
23
#include "drizzled/error.h"
24
#include "drizzled/table.h"
25
#include "drizzled/temporal.h"
26
#include "drizzled/session.h"
21
#ifdef USE_PRAGMA_IMPLEMENTATION
22
#pragma implementation // gcc: Class implementation
25
#include <drizzled/server_includes.h>
26
#include <drizzled/field/datetime.h>
37
28
/****************************************************************************
39
30
** In string context: YYYY-MM-DD HH:MM:DD
40
31
** In number context: YYYYMMDDHHMMDD
32
** Stored as a 8 byte unsigned int. Should sometimes be change to a 6 byte int.
41
33
****************************************************************************/
43
35
int Field_datetime::store(const char *from,
45
const CHARSET_INFO * const )
47
ASSERT_COLUMN_MARKED_FOR_WRITE;
49
* Try to create a DateTime from the supplied string. Throw an error
50
* if unable to create a valid DateTime.
53
if (! temporal.from_string(from, (size_t) len))
55
my_error(ER_INVALID_DATETIME_VALUE, MYF(ME_FATALERROR), from);
58
/* Create the stored integer format. @TODO This should go away. Should be up to engine... */
60
temporal.to_int64_t(&int_value);
62
#ifdef WORDS_BIGENDIAN
63
if (getTable() && getTable()->s->db_low_byte_first)
65
int8store(ptr, int_value);
69
int64_tstore(ptr, int_value);
73
int Field_datetime::store(double from)
75
ASSERT_COLUMN_MARKED_FOR_WRITE;
76
if (from < 0.0 || from > 99991231235959.0)
78
/* Convert the double to a string using stringstream */
81
ss.precision(18); /* 18 places should be fine for error display of double input. */
82
ss << from; ss >> tmp;
84
my_error(ER_INVALID_DATETIME_VALUE, MYF(ME_FATALERROR), tmp.c_str());
87
return Field_datetime::store((int64_t) rint(from), false);
90
int Field_datetime::store(int64_t from, bool)
92
ASSERT_COLUMN_MARKED_FOR_WRITE;
94
* Try to create a DateTime from the supplied integer. Throw an error
95
* if unable to create a valid DateTime.
98
if (! temporal.from_int64_t(from))
100
/* Convert the integer to a string using stringstream */
101
std::stringstream ss;
103
ss << from; ss >> tmp;
105
my_error(ER_INVALID_DATETIME_VALUE, MYF(ME_FATALERROR), tmp.c_str());
110
* Because "from" may be a silly MySQL-like "datetime number" (like, oh, 101)
111
* we must here get the value of the DateTime as its *real* int64_t, after
112
* the conversion above has been done...yuck. God, save us.
115
temporal.to_int64_t(&int_value);
117
#ifdef WORDS_BIGENDIAN
118
if (getTable() && getTable()->s->db_low_byte_first)
120
int8store(ptr, int_value);
124
int64_tstore(ptr, int_value);
128
int Field_datetime::store_time(DRIZZLE_TIME *ltime, enum enum_drizzle_timestamp_type)
132
temporal.set_years(ltime->year);
133
temporal.set_months(ltime->month);
134
temporal.set_days(ltime->day);
135
temporal.set_hours(ltime->hour);
136
temporal.set_minutes(ltime->minute);
137
temporal.set_seconds(ltime->second);
139
if (! temporal.is_valid())
141
char tmp_string[MAX_DATE_STRING_REP_LENGTH];
142
size_t tmp_string_len;
144
tmp_string_len= temporal.to_string(tmp_string, MAX_DATE_STRING_REP_LENGTH);
145
assert(tmp_string_len < MAX_DATE_STRING_REP_LENGTH);
146
my_error(ER_INVALID_DATETIME_VALUE, MYF(ME_FATALERROR), tmp_string);
151
temporal.to_int64_t(&int_value);
153
#ifdef WORDS_BIGENDIAN
154
if (getTable() && getTable()->s->db_low_byte_first)
156
int8store(ptr, int_value);
160
int64_tstore(ptr, int_value);
37
const CHARSET_INFO * const cs __attribute__((unused)))
39
DRIZZLE_TIME time_tmp;
42
enum enum_drizzle_timestamp_type func_res;
43
THD *thd= table ? table->in_use : current_thd;
45
func_res= str_to_datetime(from, len, &time_tmp,
47
(thd->variables.sql_mode &
48
(MODE_NO_ZERO_DATE | MODE_INVALID_DATES))),
50
if ((int) func_res > (int) DRIZZLE_TIMESTAMP_ERROR)
51
tmp= TIME_to_uint64_t_datetime(&time_tmp);
53
error= 1; // Fix if invalid zero date
56
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
57
ER_WARN_DATA_OUT_OF_RANGE,
58
from, len, DRIZZLE_TIMESTAMP_DATETIME, 1);
60
#ifdef WORDS_BIGENDIAN
61
if (table && table->s->db_low_byte_first)
67
int64_tstore(ptr,tmp);
72
int Field_datetime::store(double nr)
75
if (nr < 0.0 || nr > 99991231235959.0)
77
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
78
ER_WARN_DATA_OUT_OF_RANGE,
79
nr, DRIZZLE_TIMESTAMP_DATETIME);
83
error|= Field_datetime::store((int64_t) rint(nr), false);
88
int Field_datetime::store(int64_t nr,
89
bool unsigned_val __attribute__((unused)))
91
DRIZZLE_TIME not_used;
93
int64_t initial_nr= nr;
94
THD *thd= table ? table->in_use : current_thd;
96
nr= number_to_datetime(nr, ¬_used, (TIME_FUZZY_DATE |
97
(thd->variables.sql_mode &
99
MODE_INVALID_DATES))), &error);
108
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
109
error == 2 ? ER_WARN_DATA_OUT_OF_RANGE :
110
ER_WARN_DATA_TRUNCATED, initial_nr,
111
DRIZZLE_TIMESTAMP_DATETIME, 1);
113
#ifdef WORDS_BIGENDIAN
114
if (table && table->s->db_low_byte_first)
120
int64_tstore(ptr,nr);
125
int Field_datetime::store_time(DRIZZLE_TIME *ltime,timestamp_type time_type)
130
We don't perform range checking here since values stored in TIME
131
structure always fit into DATETIME range.
133
if (time_type == DRIZZLE_TIMESTAMP_DATE ||
134
time_type == DRIZZLE_TIMESTAMP_DATETIME)
136
tmp=((ltime->year*10000L+ltime->month*100+ltime->day)*1000000LL+
137
(ltime->hour*10000L+ltime->minute*100+ltime->second));
138
if (check_date(ltime, tmp != 0,
140
(current_thd->variables.sql_mode &
141
(MODE_NO_ZERO_DATE | MODE_INVALID_DATES))), &error))
143
char buff[MAX_DATE_STRING_REP_LENGTH];
144
String str(buff, sizeof(buff), &my_charset_utf8_general_ci);
145
make_datetime((DATE_TIME_FORMAT *) 0, ltime, &str);
146
set_datetime_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_TRUNCATED,
147
str.ptr(), str.length(), DRIZZLE_TIMESTAMP_DATETIME,1);
154
set_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_TRUNCATED, 1);
156
#ifdef WORDS_BIGENDIAN
157
if (table && table->s->db_low_byte_first)
163
int64_tstore(ptr,tmp);
167
bool Field_datetime::send_binary(Protocol *protocol)
170
Field_datetime::get_date(&tm, TIME_FUZZY_DATE);
171
return protocol->store(&tm);
164
175
double Field_datetime::val_real(void)
185
193
String *Field_datetime::val_str(String *val_buffer,
194
String *val_ptr __attribute__((unused)))
188
val_buffer->alloc(DateTime::MAX_STRING_LENGTH);
189
val_buffer->length(DateTime::MAX_STRING_LENGTH);
192
ASSERT_COLUMN_MARKED_FOR_READ;
196
val_buffer->alloc(field_length);
197
val_buffer->length(field_length);
194
203
#ifdef WORDS_BIGENDIAN
195
if (getTable() && getTable()->s->db_low_byte_first)
204
if (table && table->s->db_low_byte_first)
196
205
tmp=sint8korr(ptr);
199
208
int64_tget(tmp,ptr);
203
/* TODO: add an assert that this succeeds
204
* currently fails due to bug in allowing
205
* ALTER TABLE to add a datetime column that's
206
* not null without a default value.
208
dt.from_int64_t(tmp, false); /* NOTE: this does *NOT* attempt convertion
209
from formats such as 20090101 as
210
the stored value has already been
215
rlen= dt.to_string((char*)val_buffer->ptr(), DateTime::MAX_STRING_LENGTH);
216
assert((rlen+1) < DateTime::MAX_STRING_LENGTH);
218
val_buffer->length(rlen);
211
Avoid problem with slow int64_t arithmetic and sprintf
214
part1=(long) (tmp/1000000LL);
215
part2=(long) (tmp - (uint64_t) part1*1000000LL);
217
pos=(char*) val_buffer->ptr() + MAX_DATETIME_WIDTH;
219
*pos--= (char) ('0'+(char) (part2%10)); part2/=10;
220
*pos--= (char) ('0'+(char) (part2%10)); part3= (int) (part2 / 10);
222
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
223
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
225
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
226
*pos--= (char) ('0'+(char) part3);
228
*pos--= (char) ('0'+(char) (part1%10)); part1/=10;
229
*pos--= (char) ('0'+(char) (part1%10)); part1/=10;
231
*pos--= (char) ('0'+(char) (part1%10)); part1/=10;
232
*pos--= (char) ('0'+(char) (part1%10)); part3= (int) (part1/10);
234
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
235
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
236
*pos--= (char) ('0'+(char) (part3%10)); part3/=10;
237
*pos=(char) ('0'+(char) part3);
220
238
return val_buffer;
223
bool Field_datetime::get_date(DRIZZLE_TIME *ltime, uint32_t fuzzydate)
241
bool Field_datetime::get_date(DRIZZLE_TIME *ltime, uint fuzzydate)
225
243
int64_t tmp=Field_datetime::val_int();
226
244
uint32_t part1,part2;
227
part1=(uint32_t) (tmp/INT64_C(1000000));
228
part2=(uint32_t) (tmp - (uint64_t) part1*INT64_C(1000000));
245
part1=(uint32_t) (tmp/1000000LL);
246
part2=(uint32_t) (tmp - (uint64_t) part1*1000000LL);
230
248
ltime->time_type= DRIZZLE_TIMESTAMP_DATETIME;