datetime.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. /*!
  2. \file lib/db/dbmi_base/datetime.c
  3. \brief DBMI Library (base) - datetime conversions
  4. (C) 1999-2009, 2011 by the GRASS Development Team
  5. This program is free software under the GNU General Public License
  6. (>=v2). Read the file COPYING that comes with GRASS for details.
  7. \author Joel Jones (CERL/UIUC), Radim Blazek
  8. \author Doxygenized by Martin Landa <landa.martin gmail.com> (2011)
  9. */
  10. #include <stdio.h>
  11. #include <string.h>
  12. #include <grass/dbmi.h>
  13. static char ds = '-';
  14. static char ts = ':';
  15. /*!
  16. \brief Convert datetime value into string
  17. \param value pointer to dbValue
  18. \param sqltype SQL data type
  19. \param[out] string pointer to dbString
  20. \return DB_OK on success
  21. */
  22. int db_convert_value_datetime_into_string(dbValue *value, int sqltype,
  23. dbString *string)
  24. {
  25. int to, from;
  26. int year, month, day, hour, minute;
  27. double seconds;
  28. char *xs;
  29. char buf[64];
  30. if (db_test_value_datetime_current(value))
  31. return db_set_string(string, "CURRENT");
  32. *buf = 0;
  33. year = db_get_value_year(value);
  34. month = db_get_value_month(value);
  35. day = db_get_value_day(value);
  36. hour = db_get_value_hour(value);
  37. minute = db_get_value_minute(value);
  38. seconds = db_get_value_seconds(value);
  39. if (seconds < 10.0)
  40. xs = "0";
  41. else
  42. xs = "";
  43. db_interval_range(sqltype, &from, &to);
  44. switch (from) {
  45. case DB_YEAR:
  46. switch (to) {
  47. case DB_YEAR:
  48. sprintf(buf, "%d", year);
  49. break;
  50. case DB_MONTH:
  51. sprintf(buf, "%d%c%02d", year, ds, month);
  52. break;
  53. case DB_DAY:
  54. sprintf(buf, "%d%c%02d%c%02d", year, ds, month, ds, day);
  55. break;
  56. case DB_HOUR:
  57. sprintf(buf, "%d%c%02d%c%02d %02d", year, ds, month, ds, day, hour);
  58. break;
  59. case DB_MINUTE:
  60. sprintf(buf, "%d%c%02d%c%02d %02d%c%02d",
  61. year, ds, month, ds, day, hour, ts, minute);
  62. break;
  63. case DB_SECOND:
  64. case DB_FRACTION:
  65. sprintf(buf, "%d%c%02d%c%02d %02d%c%02d%c%s%.10g",
  66. year, ds, month, ds, day, hour, ts, minute, ts, xs,
  67. seconds);
  68. break;
  69. }
  70. break;
  71. case DB_MONTH:
  72. switch (to) {
  73. case DB_MONTH:
  74. sprintf(buf, "%d", month);
  75. break;
  76. case DB_DAY:
  77. sprintf(buf, "%02d%c%02d", month, ds, day);
  78. break;
  79. case DB_HOUR:
  80. sprintf(buf, "%02d%c%02d %02d", month, ds, day, hour);
  81. break;
  82. case DB_MINUTE:
  83. sprintf(buf, "%02d%c%02d %02d%c%02d", month, ds, day, hour, ts, minute);
  84. break;
  85. case DB_SECOND:
  86. case DB_FRACTION:
  87. sprintf(buf, "%02d%c%02d %02d%c%02d%c%s%.10g",
  88. month, ds, day, hour, ts, minute, ts, xs, seconds);
  89. break;
  90. }
  91. break;
  92. case DB_DAY:
  93. switch (to) {
  94. case DB_DAY:
  95. sprintf(buf, "%02d", day);
  96. break;
  97. case DB_HOUR:
  98. sprintf(buf, "%02d %02d", day, hour);
  99. break;
  100. case DB_MINUTE:
  101. sprintf(buf, "%02d %02d%c%02d", day, hour, ts, minute);
  102. break;
  103. case DB_SECOND:
  104. case DB_FRACTION:
  105. sprintf(buf, "%02d %02d%c%02d%c%s%.10g",
  106. day, hour, ts, minute, ts, xs, seconds);
  107. break;
  108. }
  109. break;
  110. case DB_HOUR:
  111. switch (to) {
  112. case DB_HOUR:
  113. sprintf(buf, "%02d", hour);
  114. break;
  115. case DB_MINUTE:
  116. sprintf(buf, "%02d%c%02d", hour, ts, minute);
  117. break;
  118. case DB_SECOND:
  119. case DB_FRACTION:
  120. sprintf(buf, "%02d%c%02d%c%s%.10g", hour, ts, minute, ts, xs,
  121. seconds);
  122. break;
  123. }
  124. break;
  125. case DB_MINUTE:
  126. switch (to) {
  127. case DB_MINUTE:
  128. sprintf(buf, "%02d", minute);
  129. break;
  130. case DB_SECOND:
  131. case DB_FRACTION:
  132. sprintf(buf, "%02d%c%s%.10g", minute, ts, xs, seconds);
  133. break;
  134. }
  135. break;
  136. case DB_SECOND:
  137. case DB_FRACTION:
  138. switch (to) {
  139. case DB_SECOND:
  140. case DB_FRACTION:
  141. sprintf(buf, "%g", seconds);
  142. break;
  143. }
  144. break;
  145. default:
  146. switch (sqltype) {
  147. case DB_SQL_TYPE_DATE:
  148. sprintf(buf, "%d%c%02d%c%02d", year, ds, month, ds, day);
  149. break;
  150. case DB_SQL_TYPE_TIME:
  151. sprintf(buf, "%02d%c%02d%c%s%.10g",
  152. hour, ts, minute, ts, xs, seconds);
  153. break;
  154. case DB_SQL_TYPE_TIMESTAMP:
  155. sprintf(buf, "%d%c%02d%c%02d %02d%c%02d%c%s%.10g",
  156. year, ds, month, ds, day, hour, ts, minute, ts, xs,
  157. seconds);
  158. break;
  159. }
  160. }
  161. return db_set_string(string, buf);
  162. }
  163. /*!
  164. \brief Convert datetime string to value
  165. The format of <em>buf</em> must be as follows
  166. - buf == "CURRENT" in a case-insignificant fashion value is marked as current
  167. - sqltype == DB_SQL_TYPE_DATE
  168. "year*month*day"
  169. - sqltype == DB_SQL_TYPE_TIME
  170. "hour*minute*second"
  171. - sqltype == DB_SQL_TYPE_TIMESTAMP
  172. "year*month*day hour*minute*second"
  173. - otherwise the to and from markings in sqltype are used,
  174. where "*" represents any non-whitespace character
  175. \param buf input string buffer
  176. \param sqltype SQL data type
  177. \param[out] value pointer to dbValue to be set
  178. \return DB_OK
  179. */
  180. int db_convert_Cstring_to_value_datetime(const char *buf, int sqltype,
  181. dbValue *value)
  182. {
  183. int from, to;
  184. int year, month, day, hour, minute;
  185. double seconds;
  186. year = month = day = 0;
  187. hour = minute = 0;
  188. seconds = 0;
  189. if (db_nocase_compare(buf, "CURRENT")) {
  190. db_set_value_datetime_current(value);
  191. return DB_OK;
  192. }
  193. db_interval_range(sqltype, &from, &to);
  194. switch (from) {
  195. case DB_YEAR:
  196. switch (to) {
  197. case DB_YEAR:
  198. sscanf(buf, "%d", &year);
  199. break;
  200. case DB_MONTH:
  201. sscanf(buf, "%d%*c%d", &year, &month);
  202. break;
  203. case DB_DAY:
  204. sscanf(buf, "%d%*c%d%*c%d", &year, &month, &day);
  205. break;
  206. case DB_HOUR:
  207. sscanf(buf, "%d%*c%d%*c%d %d", &year, &month, &day, &hour);
  208. break;
  209. case DB_MINUTE:
  210. sscanf(buf, "%d%*c%d%*c%d %d%*c%d",
  211. &year, &month, &day, &hour, &minute);
  212. break;
  213. case DB_SECOND:
  214. case DB_FRACTION:
  215. sscanf(buf, "%d%*c%d%*c%d %d%*c%d%*c%lf",
  216. &year, &month, &day, &hour, &minute, &seconds);
  217. break;
  218. }
  219. break;
  220. case DB_MONTH:
  221. switch (to) {
  222. case DB_MONTH:
  223. sscanf(buf, "%d", &month);
  224. break;
  225. case DB_DAY:
  226. sscanf(buf, "%d%*c%d", &month, &day);
  227. break;
  228. case DB_HOUR:
  229. sscanf(buf, "%d%*c%d %d", &month, &day, &hour);
  230. break;
  231. case DB_MINUTE:
  232. sscanf(buf, "%d%*c%d %d%*c%d", &month, &day, &hour, &minute);
  233. break;
  234. case DB_SECOND:
  235. case DB_FRACTION:
  236. sscanf(buf, "%d%*c%d %d%*c%d%*c%lf",
  237. &month, &day, &hour, &minute, &seconds);
  238. break;
  239. }
  240. break;
  241. case DB_DAY:
  242. switch (to) {
  243. case DB_DAY:
  244. sscanf(buf, "%d", &day);
  245. break;
  246. case DB_HOUR:
  247. sscanf(buf, "%d %d", &day, &hour);
  248. break;
  249. case DB_MINUTE:
  250. sscanf(buf, "%d %d%*c%d", &day, &hour, &minute);
  251. break;
  252. case DB_SECOND:
  253. case DB_FRACTION:
  254. sscanf(buf, "%d %d%*c%d%*c%lf", &day, &hour, &minute, &seconds);
  255. break;
  256. }
  257. break;
  258. case DB_HOUR:
  259. switch (to) {
  260. case DB_HOUR:
  261. sscanf(buf, "%d", &hour);
  262. break;
  263. case DB_MINUTE:
  264. sscanf(buf, "%d%*c%d", &hour, &minute);
  265. break;
  266. case DB_SECOND:
  267. case DB_FRACTION:
  268. sscanf(buf, "%d%*c%d%*c%lf", &hour, &minute, &seconds);
  269. break;
  270. }
  271. break;
  272. case DB_MINUTE:
  273. switch (to) {
  274. case DB_MINUTE:
  275. sscanf(buf, "%d", &minute);
  276. break;
  277. case DB_SECOND:
  278. case DB_FRACTION:
  279. sscanf(buf, "%d%*c%lf", &minute, &seconds);
  280. break;
  281. }
  282. break;
  283. case DB_SECOND:
  284. case DB_FRACTION:
  285. sscanf(buf, "%lf", &seconds);
  286. break;
  287. default:
  288. switch (sqltype) {
  289. case DB_SQL_TYPE_DATE:
  290. sscanf(buf, "%d%*c%d%*c%d", &year, &month, &day);
  291. break;
  292. case DB_SQL_TYPE_TIME:
  293. sscanf(buf, "%d%*c%d%*c%lf", &hour, &minute, &seconds);
  294. break;
  295. case DB_SQL_TYPE_TIMESTAMP:
  296. sscanf(buf, "%d%*c%d%*c%d %d%*c%d%*c%lf",
  297. &year, &month, &day, &hour, &minute, &seconds);
  298. break;
  299. }
  300. }
  301. db_set_value_year(value, year);
  302. db_set_value_month(value, month);
  303. db_set_value_day(value, day);
  304. db_set_value_hour(value, hour);
  305. db_set_value_minute(value, minute);
  306. db_set_value_seconds(value, seconds);
  307. return DB_OK;
  308. }