common_date.cpp 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | http://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Gustavo Lopes <cataphract@php.net> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #include "../intl_cppshims.h"
  17. #include <unicode/calendar.h>
  18. extern "C" {
  19. #include "../php_intl.h"
  20. #define USE_CALENDAR_POINTER 1
  21. #include "../calendar/calendar_class.h"
  22. #include <ext/date/php_date.h>
  23. }
  24. #ifndef INFINITY
  25. #define INFINITY (DBL_MAX+DBL_MAX)
  26. #endif
  27. #ifndef NAN
  28. #define NAN (INFINITY-INFINITY)
  29. #endif
  30. /* {{{ timezone_convert_datetimezone
  31. * The timezone in DateTime and DateTimeZone is not unified. */
  32. U_CFUNC TimeZone *timezone_convert_datetimezone(int type,
  33. void *object,
  34. int is_datetime,
  35. intl_error *outside_error,
  36. const char *func TSRMLS_DC)
  37. {
  38. char *id = NULL,
  39. offset_id[] = "GMT+00:00";
  40. int id_len = 0;
  41. char *message;
  42. TimeZone *timeZone;
  43. switch (type) {
  44. case TIMELIB_ZONETYPE_ID:
  45. id = is_datetime
  46. ? ((php_date_obj*)object)->time->tz_info->name
  47. : ((php_timezone_obj*)object)->tzi.tz->name;
  48. id_len = strlen(id);
  49. break;
  50. case TIMELIB_ZONETYPE_OFFSET: {
  51. int offset_mins = is_datetime
  52. ? -((php_date_obj*)object)->time->z
  53. : -(int)((php_timezone_obj*)object)->tzi.utc_offset,
  54. hours = offset_mins / 60,
  55. minutes = offset_mins - hours * 60;
  56. minutes *= minutes > 0 ? 1 : -1;
  57. if (offset_mins <= -24 * 60 || offset_mins >= 24 * 60) {
  58. spprintf(&message, 0, "%s: object has an time zone offset "
  59. "that's too large", func);
  60. intl_errors_set(outside_error, U_ILLEGAL_ARGUMENT_ERROR,
  61. message, 1 TSRMLS_CC);
  62. efree(message);
  63. return NULL;
  64. }
  65. id = offset_id;
  66. id_len = slprintf(id, sizeof(offset_id), "GMT%+03d:%02d",
  67. hours, minutes);
  68. break;
  69. }
  70. case TIMELIB_ZONETYPE_ABBR:
  71. id = is_datetime
  72. ? ((php_date_obj*)object)->time->tz_abbr
  73. : ((php_timezone_obj*)object)->tzi.z.abbr;
  74. id_len = strlen(id);
  75. break;
  76. }
  77. UnicodeString s = UnicodeString(id, id_len, US_INV);
  78. timeZone = TimeZone::createTimeZone(s);
  79. #if U_ICU_VERSION_MAJOR_NUM >= 49
  80. if (*timeZone == TimeZone::getUnknown()) {
  81. #else
  82. UnicodeString resultingId;
  83. timeZone->getID(resultingId);
  84. if (resultingId == UnicodeString("Etc/Unknown", -1, US_INV)
  85. || resultingId == UnicodeString("GMT", -1, US_INV)) {
  86. #endif
  87. spprintf(&message, 0, "%s: time zone id '%s' "
  88. "extracted from ext/date DateTimeZone not recognized", func, id);
  89. intl_errors_set(outside_error, U_ILLEGAL_ARGUMENT_ERROR,
  90. message, 1 TSRMLS_CC);
  91. efree(message);
  92. delete timeZone;
  93. return NULL;
  94. }
  95. return timeZone;
  96. }
  97. /* }}} */
  98. U_CFUNC int intl_datetime_decompose(zval *z, double *millis, TimeZone **tz,
  99. intl_error *err, const char *func TSRMLS_DC)
  100. {
  101. zval retval;
  102. zval *zfuncname;
  103. char *message;
  104. if (err && U_FAILURE(err->code)) {
  105. return FAILURE;
  106. }
  107. if (millis) {
  108. *millis = NAN;
  109. }
  110. if (tz) {
  111. *tz = NULL;
  112. }
  113. if (millis) {
  114. INIT_ZVAL(retval);
  115. MAKE_STD_ZVAL(zfuncname);
  116. ZVAL_STRING(zfuncname, "getTimestamp", 1);
  117. if (call_user_function(NULL, &(z), zfuncname, &retval, 0, NULL TSRMLS_CC)
  118. != SUCCESS || Z_TYPE(retval) != IS_LONG) {
  119. spprintf(&message, 0, "%s: error calling ::getTimeStamp() on the "
  120. "object", func);
  121. intl_errors_set(err, U_INTERNAL_PROGRAM_ERROR,
  122. message, 1 TSRMLS_CC);
  123. efree(message);
  124. zval_ptr_dtor(&zfuncname);
  125. return FAILURE;
  126. }
  127. *millis = U_MILLIS_PER_SECOND * (double)Z_LVAL(retval);
  128. zval_ptr_dtor(&zfuncname);
  129. }
  130. if (tz) {
  131. php_date_obj *datetime;
  132. datetime = (php_date_obj*)zend_object_store_get_object(z TSRMLS_CC);
  133. if (!datetime->time) {
  134. spprintf(&message, 0, "%s: the DateTime object is not properly "
  135. "initialized", func);
  136. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  137. message, 1 TSRMLS_CC);
  138. efree(message);
  139. return FAILURE;
  140. }
  141. if (!datetime->time->is_localtime) {
  142. *tz = TimeZone::getGMT()->clone();
  143. } else {
  144. *tz = timezone_convert_datetimezone(datetime->time->zone_type,
  145. datetime, 1, NULL, func TSRMLS_CC);
  146. if (*tz == NULL) {
  147. spprintf(&message, 0, "%s: could not convert DateTime's "
  148. "time zone", func);
  149. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  150. message, 1 TSRMLS_CC);
  151. efree(message);
  152. return FAILURE;
  153. }
  154. }
  155. }
  156. return SUCCESS;
  157. }
  158. U_CFUNC double intl_zval_to_millis(zval *z, intl_error *err, const char *func TSRMLS_DC)
  159. {
  160. double rv = NAN;
  161. long lv;
  162. int type;
  163. char *message;
  164. if (err && U_FAILURE(err->code)) {
  165. return NAN;
  166. }
  167. switch (Z_TYPE_P(z)) {
  168. case IS_STRING:
  169. type = is_numeric_string(Z_STRVAL_P(z), Z_STRLEN_P(z), &lv, &rv, 0);
  170. if (type == IS_DOUBLE) {
  171. rv *= U_MILLIS_PER_SECOND;
  172. } else if (type == IS_LONG) {
  173. rv = U_MILLIS_PER_SECOND * (double)lv;
  174. } else {
  175. spprintf(&message, 0, "%s: string '%s' is not numeric, "
  176. "which would be required for it to be a valid date", func,
  177. Z_STRVAL_P(z));
  178. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  179. message, 1 TSRMLS_CC);
  180. efree(message);
  181. }
  182. break;
  183. case IS_LONG:
  184. rv = U_MILLIS_PER_SECOND * (double)Z_LVAL_P(z);
  185. break;
  186. case IS_DOUBLE:
  187. rv = U_MILLIS_PER_SECOND * Z_DVAL_P(z);
  188. break;
  189. case IS_OBJECT:
  190. if (instanceof_function(Z_OBJCE_P(z), php_date_get_date_ce() TSRMLS_CC)) {
  191. intl_datetime_decompose(z, &rv, NULL, err, func TSRMLS_CC);
  192. } else if (instanceof_function(Z_OBJCE_P(z), Calendar_ce_ptr TSRMLS_CC)) {
  193. Calendar_object *co = (Calendar_object *)
  194. zend_object_store_get_object(z TSRMLS_CC );
  195. if (co->ucal == NULL) {
  196. spprintf(&message, 0, "%s: IntlCalendar object is not properly "
  197. "constructed", func);
  198. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  199. message, 1 TSRMLS_CC);
  200. efree(message);
  201. } else {
  202. UErrorCode status = UErrorCode();
  203. rv = (double)co->ucal->getTime(status);
  204. if (U_FAILURE(status)) {
  205. spprintf(&message, 0, "%s: call to internal "
  206. "Calendar::getTime() has failed", func);
  207. intl_errors_set(err, status, message, 1 TSRMLS_CC);
  208. efree(message);
  209. }
  210. }
  211. } else {
  212. /* TODO: try with cast(), get() to obtain a number */
  213. spprintf(&message, 0, "%s: invalid object type for date/time "
  214. "(only IntlCalendar and DateTime permitted)", func);
  215. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  216. message, 1 TSRMLS_CC);
  217. efree(message);
  218. }
  219. break;
  220. default:
  221. spprintf(&message, 0, "%s: invalid PHP type for date", func);
  222. intl_errors_set(err, U_ILLEGAL_ARGUMENT_ERROR,
  223. message, 1 TSRMLS_CC);
  224. efree(message);
  225. break;
  226. }
  227. return rv;
  228. }