1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008-2009 Sun Microsystems
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; either version 2 of the License, or
9
* (at your option) any later version.
11
* This program is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
* GNU General Public License for more details.
16
* You should have received a copy of the GNU General Public License
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
24
* Defines the API for dealing with temporal data inside the server.
26
* The Temporal class is the base class for all data of any temporal
27
* type. A number of derived classes define specialized classes
28
* representng various date, date-time, time, or timestamp types.
30
* All Temporal derived classes are ValueObjects. That is to say that
31
* Temporal class instances are not part of the Item hierarchy and serve
32
* <em>only</em> to represent a time or date-related piece of data.
36
* Low-level calendrical calculations are done via routines in the
39
* @see drizzled/calendar.cc
42
#ifndef DRIZZLED_TEMPORAL_H
43
#define DRIZZLED_TEMPORAL_H
45
#define DRIZZLE_MAX_SECONDS 59
46
#define DRIZZLE_MAX_MINUTES 59
47
#define DRIZZLE_MAX_HOURS 23
48
#define DRIZZLE_MAX_DAYS 31
49
#define DRIZZLE_MAX_MONTHS 12
50
#define DRIZZLE_MAX_YEARS_SQL 9999
51
#define DRIZZLE_MAX_YEARS_EPOCH 2038
52
#define DRIZZLE_MIN_SECONDS 0
53
#define DRIZZLE_MIN_MINUTES 0
54
#define DRIZZLE_MIN_HOURS 0
55
#define DRIZZLE_MIN_DAYS 1
56
#define DRIZZLE_MIN_MONTHS 1
57
#define DRIZZLE_MIN_YEARS_SQL 1
58
#define DRIZZLE_MIN_YEARS_EPOCH 1970
60
#define DRIZZLE_SECONDS_IN_MINUTE 60
61
#define DRIZZLE_SECONDS_IN_HOUR (60*60)
62
#define DRIZZLE_SECONDS_IN_DAY (60*60*24)
63
#define DRIZZLE_NANOSECONDS_IN_MICROSECOND 1000
65
#define DRIZZLE_MAX_LENGTH_DATETIME_AS_STRING 40
67
#define DRIZZLE_YY_PART_YEAR 70
69
#include "drizzled/calendar.h"
74
/* Outside forward declarations */
80
/* Forward declaration needed */
81
class TemporalInterval;
82
class TemporalIntervalYear;
83
class TemporalIntervalDayOrLess;
84
class TemporalIntervalDayOrWeek;
85
class TemporalIntervalYearMonth;
88
* Base class for all temporal data classes.
93
enum calendar _calendar;
100
time_t _epoch_seconds;
103
/** Set on some operator overloads. Indicates that an overflow occurred. */
105
/** Returns number of seconds in time components (hour + minute + second) */
106
uint64_t _cumulative_seconds_in_time() const;
107
/** Resets all temporal components to zero */
110
_years= _months= _days= _hours= _minutes=
111
_seconds= _epoch_seconds= _useconds= _nseconds= 0;
116
virtual ~Temporal() {}
118
/** Returns the calendar component. */
119
inline enum calendar calendar() const {return _calendar;}
120
/** Sets the nseconds component. */
121
inline void set_nseconds(const uint32_t nsecond) {_nseconds= nsecond;}
122
/** Returns the nanoseconds component. */
123
inline uint32_t nseconds() const {return _nseconds;}
124
/** Sets the useconds component. */
125
inline void set_useconds(const uint32_t usecond) {_useconds= usecond;}
126
/** Returns the microsseconds component. */
127
inline uint32_t useconds() const {return _useconds;}
129
* Sets the epoch_seconds component automatically,
130
* based on the temporal's components.
132
void set_epoch_seconds();
133
/** Sets the epch_seconds component manually. */
134
inline void set_epoch_seconds(const uint32_t epoch_second)
135
{_epoch_seconds= epoch_second;}
136
/** Returns the UNIX epoch seconds component. */
137
inline time_t epoch_seconds() const {return _epoch_seconds;}
138
/** Sets the seconds component. */
139
inline void set_seconds(const uint32_t second) {_seconds= second;}
140
/** Returns the seconds component. */
141
inline uint32_t seconds() const {return _seconds;}
142
/** Sets the days component. */
143
inline void set_minutes(const uint32_t minute) {_minutes= minute;}
144
/** Returns the minutes component. */
145
inline uint32_t minutes() const {return _minutes;}
146
/** Sets the hours component. */
147
inline void set_hours(const uint32_t hour) {_hours= hour;}
148
/** Returns the hours component. */
149
inline uint32_t hours() const {return _hours;}
150
/** Sets the days component. */
151
inline void set_days(const uint32_t day) {_days= day;}
152
/** Returns the days component. */
153
inline uint32_t days() const {return _days;}
154
/** Sets the months component. */
155
inline void set_months(const uint32_t month) {_months= month;}
156
/** Returns the months component. */
157
inline uint32_t months() const {return _months;}
158
/** Sets the years component. */
159
inline void set_years(const uint32_t year) {_years= year;}
160
/** Returns the years component. */
161
inline uint32_t years() const {return _years;}
162
/** Returns whether the overflow flag was set
163
* (which can occur during an overloaded operator's execution) */
164
inline bool overflow() const {return _overflow;}
166
/** Returns whether the temporal value is valid as a date. */
167
virtual bool is_valid_date() const= 0;
168
/** Returns whether the temporal value is valid as a datetime. */
169
virtual bool is_valid_datetime() const= 0;
170
/** Returns whether the temporal value is valid as a time. */
171
virtual bool is_valid_time() const= 0;
172
/** Returns whether the temporal value is valid as a UNIX timestamp. */
173
virtual bool is_valid_timestamp() const= 0;
176
* Returns whether the temporal
177
* value is valid. Each subclass defines what is
178
* valid for the range of temporal data it contains.
180
virtual bool is_valid() const= 0;
183
* All Temporal derived classes must implement
184
* conversion routines for converting to and from
185
* a string. Subclasses implement other conversion
186
* routines, but should always follow these notes:
188
* 1) Ensure that ALL from_xxx methods call is_valid()
189
* 2) Ensure that ALL to_xxx methods are void returns and
190
* do not call is_valid()
192
* This minimizes the repeated bounds-checking to
193
* just the conversion from_xxx routines.
195
friend class TemporalFormat;
198
/* Forward declaration needed */
204
* Class representing temporal components in a valid
205
* SQL date range, with no time component
207
class Date: public Temporal
210
Date() :Temporal() {}
212
* Comparison operator overloads to compare a Date against
213
* another Date value.
215
* @param Date to compare against.
217
virtual bool operator==(const Date &rhs);
218
virtual bool operator!=(const Date &rhs);
219
virtual bool operator>(const Date &rhs);
220
virtual bool operator>=(const Date &rhs);
221
virtual bool operator<(const Date &rhs);
222
virtual bool operator<=(const Date &rhs);
225
* Comparison operator overloads to compare a Date against
228
* @param DateTime to compare against.
230
virtual bool operator==(const DateTime &rhs);
231
virtual bool operator!=(const DateTime &rhs);
232
virtual bool operator>(const DateTime &rhs);
233
virtual bool operator>=(const DateTime &rhs);
234
virtual bool operator<(const DateTime &rhs);
235
virtual bool operator<=(const DateTime &rhs);
238
* Comparison operator overloads to compare this against
241
* @param Timestamp to compare against.
243
virtual bool operator==(const Timestamp &rhs);
244
virtual bool operator!=(const Timestamp &rhs);
245
virtual bool operator>(const Timestamp &rhs);
246
virtual bool operator>=(const Timestamp &rhs);
247
virtual bool operator<(const Timestamp &rhs);
248
virtual bool operator<=(const Timestamp &rhs);
251
* Operator overload for adding/subtracting another Date
252
* (or subclass) to/from this temporal. When subtracting
253
* or adding two Dates, we return a new Date instance.
255
* @param Date instance to add/subtract to/from
257
const Date operator-(const Date &rhs);
258
const Date operator+(const Date &rhs);
259
Date& operator+=(const Date &rhs);
260
Date& operator-=(const Date &rhs);
263
* Operator to add/subtract a Time from a Time.
264
* We can return a Time new temporal instance.
266
* @param Temporal instance to add/subtract to/from
268
const Date operator-(const Time &rhs);
269
const Date operator+(const Time &rhs);
270
Date& operator-=(const Time &rhs);
271
Date& operator+=(const Time &rhs);
275
* Operator overload for adding/subtracting a DateTime
276
* (or subclass) to/from this temporal. When subtracting
277
* or adding two Dates, we return a new Date instance.
279
* @param DateTime instance to add/subtract to/from
281
const Date operator-(const DateTime &rhs);
282
const Date operator+(const DateTime &rhs);
283
Date& operator+=(const DateTime &rhs);
284
Date& operator-=(const DateTime &rhs);
288
* Operator overload for adding/subtracting a TemporalInterval
289
* instance to this temporal.
291
* @param TemporalInterval instance to add/subtract to/from
293
Date& operator+=(const TemporalIntervalYear &rhs);
294
Date& operator+=(const TemporalIntervalDayOrLess &rhs);
295
Date& operator+=(const TemporalIntervalDayOrWeek &rhs);
296
Date& operator+=(const TemporalIntervalYearMonth &rhs);
297
Date& operator-=(const TemporalIntervalYear &rhs);
298
Date& operator-=(const TemporalIntervalDayOrLess &rhs);
299
Date& operator-=(const TemporalIntervalDayOrWeek &rhs);
300
Date& operator-=(const TemporalIntervalYearMonth &rhs);
304
* Operator overload for when a DateTime instance is
305
* assigned to a Date. We do a copy of the DateTime's
306
* date-related components.
308
* @param The DateTime to copy from
310
Date& operator=(const DateTime &rhs);
312
virtual bool is_valid_date() const {return is_valid();}
313
virtual bool is_valid_datetime() const {return is_valid();}
314
virtual bool is_valid_time() const {return false;}
315
virtual bool is_valid_timestamp() const
317
return is_valid() && in_unix_epoch();
320
/** Returns whether the temporal value is valid date. */
321
virtual bool is_valid() const;
322
/* Returns whether the Date (or subclass) instance is in the Unix Epoch. */
323
virtual bool in_unix_epoch() const;
326
* Fills a supplied char string with a
327
* string representation of the Date
330
* @param C-String to fill.
331
* @param Length of to C-String
332
* @returns length of string written (not including trailing '\0').
333
* If output was truncated, returns length that would have
336
virtual int to_string(char *to, size_t to_len) const;
339
* Maximum length of C-String needed to represent type
342
static const int MAX_STRING_LENGTH= 11;
345
* Attempts to populate the Date instance based
346
* on the contents of a supplied string.
348
* Returns whether the conversion was
351
* @param String to convert from
352
* @param Length of supplied string
354
virtual bool from_string(const char *from, size_t from_len);
357
* Fills a supplied 8-byte integer pointer with an
358
* integer representation of the Date
361
* @param Integer to fill.
363
virtual void to_int64_t(int64_t *to) const;
366
* Fills a supplied 4-byte integer pointer with an
367
* integer representation of the Date
370
* @param Integer to fill.
372
virtual void to_int32_t(int32_t *to) const;
375
* Attempts to populate the Date instance based
376
* on the contents of a supplied 4-byte integer.
378
* Returns whether the conversion was
381
* @param Integer to convert from
383
virtual bool from_int32_t(const int32_t from);
386
* Fills a supplied int64_t with the Julian Day Number
387
* representation of this Date.
389
* @note Julian Day Number != julian day!
391
* Julian Day Number is the monotonically increasing number
392
* of days from the start of the Julian calendar (~4713 B.C.)
394
* julian day is the ordinal day number of a day in a year.
396
* @param int64_t to fill
398
void to_julian_day_number(int64_t *to) const;
401
* Attempts to populate the Date instance based
402
* on the contents of a supplied Julian Day Number
404
* Returns whether the conversion was
407
* @param Integer to convert from
409
bool from_julian_day_number(const int64_t from);
412
* Fills a supplied tm pointer with an
413
* representation of the Date
418
virtual void to_tm(struct tm *to) const;
421
* Attempts to populate the Date instance based
422
* on the contents of a supplied pointer to struct tm
425
* Returns whether the conversion was
428
* @param Pointe rto the struct tm to convert from
430
virtual bool from_tm(const struct tm *from);
433
* Attempts to convert the Date value into
436
* @param Pointer to a time_t to convert to
438
virtual void to_time_t(time_t *to) const;
441
* Attempts to populate the Date instance based
442
* on the contents of a supplied time_t
444
* Returns whether the conversion was
447
* @param time_t to convert from
449
virtual bool from_time_t(const time_t from);
452
* Fills a supplied my_decimal with a representation of
455
* @param Pointer to the my_decimal to fill
457
virtual void to_decimal(my_decimal *to) const;
459
friend class TemporalInterval;
460
friend class Timestamp;
463
/* Forward declare needed for friendship */
467
* Class representing temporal components having only
468
* a time component, with no date structure
470
class Time: public Temporal
473
Time() :Temporal() {}
474
/* Maximum number of seconds in 23:59:59 (24 * 60 * 60) */
475
static const uint32_t MAX_CUMULATIVE_SECONDS= 86400L;
478
* Comparison operator overloads to compare a Time against
479
* another Time value.
481
* @param Time to compare against.
483
bool operator==(const Time &rhs);
484
bool operator!=(const Time &rhs);
485
bool operator>(const Time &rhs);
486
bool operator>=(const Time &rhs);
487
bool operator<(const Time &rhs);
488
bool operator<=(const Time &rhs);
490
* Operator to add/subtract a Time from a Time.
491
* We can return a Time new temporal instance.
493
* @param Temporal instance to add/subtract to/from
495
const Time operator-(const Time &rhs);
496
const Time operator+(const Time &rhs);
497
Time& operator-=(const Time &rhs);
498
Time& operator+=(const Time &rhs);
500
bool is_valid_date() const {return false;}
501
bool is_valid_datetime() const {return false;}
502
bool is_valid_time() const {return is_valid();}
503
bool is_valid_timestamp() const {return false;}
504
/** Returns whether the temporal value is valid date. */
505
bool is_valid() const;
508
* Fills a supplied char string with a
509
* string representation of the Time
512
* @param C-String to fill
513
* @param Length of to C-String
514
* @returns length of string written (not including trailing '\0').
515
* If output was truncated, returns length that would have
518
int to_string(char *to, size_t to_len) const;
521
* Maximum length of C-String needed to represent type
524
static const int MAX_STRING_LENGTH= 9;
528
* Attempts to populate the Time instance based
529
* on the contents of a supplied string.
531
* Returns whether the conversion was
534
* @param String to convert from
535
* @param Length of supplied string
537
bool from_string(const char *from, size_t from_len);
540
* Fills a supplied 4-byte integer pointer with an
541
* integer representation of the Time
544
* @param Integer to fill.
546
void to_int32_t(int32_t *to) const;
549
* Attempts to populate the Time instance based
550
* on the contents of a supplied 4-byte integer.
552
* Returns whether the conversion was
555
* @param Integer to convert from
557
bool from_int32_t(const int32_t from);
560
* Attempts to populate the Time instance based
561
* on the contents of a supplied time_t
563
* Returns whether the conversion was
568
* We can only convert *from* a time_t, not back
569
* to a time_t since it would be a lossy conversion.
571
* @param time_t to convert from
573
bool from_time_t(const time_t from);
576
* Fills a supplied my_decimal with a representation of
579
* @param Pointer to the my_decimal to fill
581
void to_decimal(my_decimal *to) const;
584
friend class DateTime;
588
* Class representing temporal components in a valid
589
* SQL datetime range, including a time component
591
class DateTime: public Date
594
DateTime() :Date() {}
596
friend class TemporalInterval;
598
/** Returns whether the DateTime (or subclass) instance
599
* is in the Unix Epoch.
601
bool in_unix_epoch() const;
602
/** Returns whether the temporal value is valid datetime. */
603
virtual bool is_valid() const;
606
* It's not possible to convert to and from a DateTime and
607
* a 4-byte integer, so let us know if we try and do it!
609
void to_int32_t(int32_t *) const {assert(0);}
610
bool from_int32_t(int32_t) {assert(0); return false;}
613
* Fills a supplied char string with a
614
* string representation of the DateTime
617
* @param C-String to fill
618
* @param Length of to C-String
619
* @returns length of string written (not including trailing '\0').
620
* If output was truncated, returns length that would have
623
virtual int to_string(char *to, size_t to_len) const;
626
* Maximum length of C-String needed to represent type
629
static const int MAX_STRING_LENGTH= 27;
632
* Attempts to populate the DateTime instance based
633
* on the contents of a supplied string.
635
* Returns whether the conversion was
638
* @param String to convert from
639
* @param Length of supplied string
641
bool from_string(const char *from, size_t from_len);
644
* Fills a supplied 8-byte integer pointer with an
645
* integer representation of the DateTime
648
* @param Integer to fill.
650
void to_int64_t(int64_t *to) const;
653
* Attempts to populate the DateTime instance based
654
* on the contents of a supplied time_t
656
* Returns whether the conversion was
659
* @param time_t to convert from
661
bool from_time_t(const time_t from);
664
* Attempts to populate the DateTime instance based
665
* on the contents of a supplied 8-byte integer.
667
* Returns whether the conversion was
670
* @param Integer to convert from
671
* @param convert if conversion to canonical representation
672
* should be attempted
674
bool from_int64_t(const int64_t from, bool convert);
676
bool from_int64_t(const int64_t from) {
677
return from_int64_t(from, true);
681
* Fills a supplied tm pointer with an
682
* representation of the DateTime
687
void to_tm(struct tm *to) const;
690
* Fills a supplied my_decimal with a representation of
691
* the DateTime value.
693
* @param Pointer to the my_decimal to fill
695
void to_decimal(my_decimal *to) const;
697
friend class Timestamp;
701
* Class representing temporal components in the UNIX epoch
703
class Timestamp: public DateTime
706
Timestamp() :DateTime() {}
709
* Comparison operator overloads to compare this against
712
* @param Timestamp to compare against.
714
bool operator==(const Date &rhs);
715
bool operator!=(const Date &rhs);
716
bool operator>(const Date &rhs);
717
bool operator>=(const Date &rhs);
718
bool operator<(const Date &rhs);
719
bool operator<=(const Date &rhs);
722
* Comparison operator overloads to compare this against
725
* @param DateTime to compare against.
727
bool operator==(const DateTime &rhs);
728
bool operator!=(const DateTime &rhs);
729
bool operator>(const DateTime &rhs);
730
bool operator>=(const DateTime &rhs);
731
bool operator<(const DateTime &rhs);
732
bool operator<=(const DateTime &rhs);
735
* Comparison operator overloads to compare this against
736
* another Timestamp value.
738
* @param Timestamp to compare against.
740
bool operator==(const Timestamp &rhs);
741
bool operator!=(const Timestamp &rhs);
742
bool operator>(const Timestamp &rhs);
743
bool operator>=(const Timestamp &rhs);
744
bool operator<(const Timestamp &rhs);
745
bool operator<=(const Timestamp &rhs);
747
bool is_valid_timestamp() const {return is_valid();}
748
/** Returns whether the temporal value is valid timestamp. */
749
virtual bool is_valid() const;
752
* Attempts to convert the Timestamp value into
755
* @param Pointer to a time_t to convert to
757
void to_time_t(time_t *to) const;
761
* Operator overload to an output stream for a Timestamp.
763
std::ostream& operator<<(std::ostream& os, const Timestamp& subject);
766
* Class representing temporal components in the UNIX epoch
767
* with an additional microsecond component.
769
class MicroTimestamp: public Timestamp
772
MicroTimestamp() :Timestamp() {}
773
/** Returns whether the temporal value is valid micro-timestamp. */
774
bool is_valid() const;
777
* Fills a supplied char string with a
778
* string representation of the MicroTimestamp
781
* @param C-String to fill
782
* @param Length of to C-String
783
* @returns length of string written (not including trailing '\0').
784
* If output was truncated, returns length that would have
787
int to_string(char *to, size_t to_len) const;
790
* Maximum length of C-String needed to represent type
793
static const int MAX_STRING_LENGTH= 27;
796
* Fills a supplied timeval pointer with an
797
* representation of the MicroTimestamp
800
* Returns whether the conversion was
803
* @param timeval to fill.
805
void to_timeval(struct timeval *to) const;
809
* Class representing temporal components in the UNIX epoch
810
* with an additional nanosecond component.
812
class NanoTimestamp: public Timestamp
815
NanoTimestamp() :Timestamp() {}
816
/** Returns whether the temporal value is valid nano-timestamp. */
817
bool is_valid() const;
820
* Fills a supplied timespec pointer with an
821
* representation of the NanoTimestamp
824
* Returns whether the conversion was
827
* @param timespec to fill.
829
void to_timespec(struct timespec *to) const;
832
} /* end namespace drizzled */
834
#endif /* DRIZZLED_TEMPORAL_H */