datetime_math.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620
  1. """!@package grass.temporal
  2. @brief GRASS Python scripting module (temporal GIS functions)
  3. Temporal GIS datetime math functions to be used in library functions and modules.
  4. (C) 2008-2011 by the GRASS Development Team
  5. This program is free software under the GNU General Public
  6. License (>=v2). Read the file COPYING that comes with GRASS
  7. for details.
  8. @author Soeren Gebbert
  9. """
  10. from datetime import datetime, date, time, timedelta
  11. import grass.script.core as core
  12. import copy
  13. DAY_IN_SECONDS = 86400
  14. SECOND_AS_DAY = 1.1574074074074073e-05
  15. ###############################################################################
  16. def relative_time_to_time_delta(value):
  17. """!Convert the double value representing days
  18. into a timedelta object.
  19. """
  20. days = int(value)
  21. seconds = value % 1
  22. seconds = round(seconds * DAY_IN_SECONDS)
  23. return timedelta(days, seconds)
  24. ###############################################################################
  25. def time_delta_to_relative_time(delta):
  26. """!Convert the time delta into a
  27. double value, representing days.
  28. """
  29. return float(delta.days) + float(delta.seconds * SECOND_AS_DAY)
  30. ###############################################################################
  31. def increment_datetime_by_string(mydate, increment, mult=1):
  32. """!Return a new datetime object incremented with the provided
  33. relative dates specified as string.
  34. Additional a multiplier can be specified to multiply the increment
  35. before adding to the provided datetime object.
  36. Usage:
  37. @code
  38. >>> dt = datetime(2001, 9, 1, 0, 0, 0)
  39. >>> string = "60 seconds, 4 minutes, 12 hours, 10 days, 1 weeks, 5 months, 1 years"
  40. >>> increment_datetime_by_string(dt, string)
  41. datetime.datetime(2003, 2, 18, 12, 5)
  42. >>> dt = datetime(2001, 11, 1, 0, 0, 0)
  43. >>> string = "1 months"
  44. >>> increment_datetime_by_string(dt, string)
  45. datetime.datetime(2001, 12, 1, 0, 0)
  46. >>> dt = datetime(2001, 11, 1, 0, 0, 0)
  47. >>> string = "13 months"
  48. >>> increment_datetime_by_string(dt, string)
  49. datetime.datetime(2002, 12, 1, 0, 0)
  50. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  51. >>> string = "72 months"
  52. >>> increment_datetime_by_string(dt, string)
  53. datetime.datetime(2007, 1, 1, 0, 0)
  54. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  55. >>> string = "72 months"
  56. >>> increment_datetime_by_string(dt, string)
  57. datetime.datetime(2007, 1, 1, 0, 0)
  58. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  59. >>> string = "5 minutes"
  60. >>> increment_datetime_by_string(dt, string)
  61. datetime.datetime(2001, 1, 1, 0, 5)
  62. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  63. >>> string = "49 hours"
  64. >>> increment_datetime_by_string(dt, string)
  65. datetime.datetime(2001, 1, 3, 1, 0)
  66. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  67. >>> string = "3600 seconds"
  68. >>> increment_datetime_by_string(dt, string)
  69. datetime.datetime(2001, 1, 1, 1, 0)
  70. >>> dt = datetime(2001, 1, 1, 0, 0, 0)
  71. >>> string = "30 days"
  72. >>> increment_datetime_by_string(dt, string)
  73. datetime.datetime(2001, 1, 31, 0, 0)
  74. @endcode
  75. @param mydate A datetime object to incremented
  76. @param increment A string providing increment information:
  77. The string may include comma separated values of type seconds,
  78. minutes, hours, days, weeks, months and years
  79. Example: Increment the datetime 2001-01-01 00:00:00
  80. with "60 seconds, 4 minutes, 12 hours, 10 days, 1 weeks, 5 months, 1 years"
  81. will result in the datetime 2003-02-18 12:05:00
  82. @param mult A multiplier, default is 1
  83. """
  84. if increment:
  85. seconds = 0
  86. minutes = 0
  87. hours = 0
  88. days = 0
  89. weeks = 0
  90. months = 0
  91. years = 0
  92. inclist = []
  93. # Split the increment string
  94. incparts = increment.split(",")
  95. for incpart in incparts:
  96. inclist.append(incpart.strip().split(" "))
  97. for inc in inclist:
  98. if len(inc) < 2:
  99. core.error(_("Wrong increment format: %s") % (increment))
  100. return None
  101. if inc[1].find("seconds") >= 0 or inc[1].find("second") >= 0:
  102. seconds = mult * int(inc[0])
  103. elif inc[1].find("minutes") >= 0 or inc[1].find("minute") >= 0:
  104. minutes = mult * int(inc[0])
  105. elif inc[1].find("hours") >= 0 or inc[1].find("hour") >= 0:
  106. hours = mult * int(inc[0])
  107. elif inc[1].find("days") >= 0 or inc[1].find("day") >= 0:
  108. days = mult * int(inc[0])
  109. elif inc[1].find("weeks") >= 0 or inc[1].find("week") >= 0:
  110. weeks = mult * int(inc[0])
  111. elif inc[1].find("months") >= 0 or inc[1].find("month") >= 0:
  112. months = mult * int(inc[0])
  113. elif inc[1].find("years") >= 0 or inc[1].find("year") >= 0:
  114. years = mult * int(inc[0])
  115. else:
  116. core.error(_("Wrong increment format: %s") % (increment))
  117. return None
  118. return increment_datetime(mydate, years, months, weeks, days, hours, minutes, seconds)
  119. return mydate
  120. ###############################################################################
  121. def increment_datetime(mydate, years=0, months=0, weeks=0, days=0, hours=0,
  122. minutes=0, seconds=0):
  123. """!Return a new datetime object incremented with the provided
  124. relative dates and times"""
  125. tdelta_seconds = timedelta(seconds=seconds)
  126. tdelta_minutes = timedelta(minutes=minutes)
  127. tdelta_hours = timedelta(hours=hours)
  128. tdelta_days = timedelta(days=days)
  129. tdelta_weeks = timedelta(weeks=weeks)
  130. tdelta_months = timedelta(0)
  131. tdelta_years = timedelta(0)
  132. if months > 0:
  133. # Compute the actual number of days in the month to add as timedelta
  134. year = mydate.year
  135. month = mydate.month
  136. all_months = int(months) + int(month)
  137. years_to_add = int(all_months / 12.001)
  138. residual_months = all_months - (years_to_add * 12)
  139. # Make a deep copy of the datetime object
  140. dt1 = copy.copy(mydate)
  141. # Make sure the month starts with a 1
  142. if residual_months == 0:
  143. residual_months = 1
  144. try:
  145. dt1 = dt1.replace(year=year + years_to_add, month=residual_months)
  146. except:
  147. core.fatal(_("Unable to increment the datetime %s. "\
  148. "Please check that the yearly or monthly increment does not result in wrong number of days."%(mydate)))
  149. tdelta_months = dt1 - mydate
  150. if years > 0:
  151. # Make a deep copy of the datetime object
  152. dt1 = copy.copy(mydate)
  153. # Compute the number of days
  154. dt1 = dt1.replace(year=mydate.year + int(years))
  155. tdelta_years = dt1 - mydate
  156. return mydate + tdelta_seconds + tdelta_minutes + tdelta_hours + \
  157. tdelta_days + tdelta_weeks + tdelta_months + tdelta_years
  158. ###############################################################################
  159. def adjust_datetime_to_granularity(mydate, granularity):
  160. """!Modify the datetime object to fit the given granularity
  161. - Years will start at the first of Januar
  162. - Months will start at the first day of the month
  163. - Days will start at the first Hour of the day
  164. - Hours will start at the first minute of an hour
  165. - Minutes will start at the first second of a minute
  166. Usage:
  167. @code
  168. >>> dt = datetime(2001, 8, 8, 12,30,30)
  169. >>> adjust_datetime_to_granularity(dt, "5 seconds")
  170. datetime.datetime(2001, 8, 8, 12, 30, 30)
  171. >>> adjust_datetime_to_granularity(dt, "20 minutes")
  172. datetime.datetime(2001, 8, 8, 12, 30)
  173. >>> adjust_datetime_to_granularity(dt, "20 minutes")
  174. datetime.datetime(2001, 8, 8, 12, 30)
  175. >>> adjust_datetime_to_granularity(dt, "3 hours")
  176. datetime.datetime(2001, 8, 8, 12, 0)
  177. >>> adjust_datetime_to_granularity(dt, "5 days")
  178. datetime.datetime(2001, 8, 8, 0, 0)
  179. >>> adjust_datetime_to_granularity(dt, "2 weeks")
  180. datetime.datetime(2001, 8, 6, 0, 0)
  181. >>> adjust_datetime_to_granularity(dt, "6 months")
  182. datetime.datetime(2001, 8, 1, 0, 0)
  183. >>> adjust_datetime_to_granularity(dt, "2 years")
  184. datetime.datetime(2001, 1, 1, 0, 0)
  185. >>> adjust_datetime_to_granularity(dt, "2 years, 3 months, 5 days, 3 hours, 3 minutes, 2 seconds")
  186. datetime.datetime(2001, 8, 8, 12, 30, 30)
  187. >>> adjust_datetime_to_granularity(dt, "3 months, 5 days, 3 minutes")
  188. datetime.datetime(2001, 8, 8, 12, 30)
  189. >>> adjust_datetime_to_granularity(dt, "3 weeks, 5 days")
  190. datetime.datetime(2001, 8, 8, 0, 0)
  191. @endcode
  192. """
  193. if granularity:
  194. has_seconds = False
  195. has_minutes = False
  196. has_hours = False
  197. has_days = False
  198. has_weeks = False
  199. has_months = False
  200. has_years = False
  201. seconds = mydate.second
  202. minutes = mydate.minute
  203. hours = mydate.hour
  204. days = mydate.day
  205. weekday = mydate.weekday()
  206. months = mydate.month
  207. years = mydate.year
  208. granlist = []
  209. # Split the increment string
  210. granparts = granularity.split(",")
  211. for granpart in granparts:
  212. granlist.append(granpart.strip().split(" "))
  213. for inc in granlist:
  214. if inc[1].find("seconds") >= 0 or inc[1].find("second") >= 0:
  215. has_seconds = True
  216. elif inc[1].find("minutes") >= 0 or inc[1].find("minute") >= 0:
  217. has_minutes = True
  218. elif inc[1].find("hours") >= 0 or inc[1].find("hour") >= 0:
  219. has_hours = True
  220. elif inc[1].find("days") >= 0 or inc[1].find("day") >= 0:
  221. has_days = True
  222. elif inc[1].find("weeks") >= 0 or inc[1].find("week") >= 0:
  223. has_weeks = True
  224. elif inc[1].find("months") >= 0 or inc[1].find("month") >= 0:
  225. has_months = True
  226. elif inc[1].find("years") >= 0 or inc[1].find("year") >= 0:
  227. has_years = True
  228. else:
  229. core.error(_("Wrong granularity format: %s") % (granularity))
  230. return None
  231. if has_seconds:
  232. pass
  233. elif has_minutes: # Start at 0 seconds
  234. seconds = 0
  235. elif has_hours: # Start at 0 minutes and seconds
  236. seconds = 0
  237. minutes = 0
  238. elif has_days: # Start at 0 hours, minutes and seconds
  239. seconds = 0
  240. minutes = 0
  241. hours = 0
  242. elif has_weeks: # Start at the first day of the week (Monday) at 00:00:00
  243. seconds = 0
  244. minutes = 0
  245. hours = 0
  246. if days > weekday:
  247. days = days - weekday # this needs to be fixed
  248. else:
  249. days = days + weekday # this needs to be fixed
  250. elif has_months: # Start at the first day of the month at 00:00:00
  251. seconds = 0
  252. minutes = 0
  253. hours = 0
  254. days = 1
  255. elif has_years: # Start at the first day of the first month at 00:00:00
  256. seconds = 0
  257. minutes = 0
  258. hours = 0
  259. days = 1
  260. months = 1
  261. dt = copy.copy(mydate)
  262. return dt.replace(year=years, month=months, day=days,
  263. hour=hours, minute=minutes, second=seconds)
  264. ###############################################################################
  265. def compute_datetime_delta(start, end):
  266. """!Return a dictionary with the accumulated delta in year, month, day,
  267. hour, minute and second
  268. Usage:
  269. @code
  270. >>> start = datetime(2001, 1, 1, 00,00,00)
  271. >>> end = datetime(2001, 1, 1, 00,00,00)
  272. >>> compute_datetime_delta(start, end)
  273. {'hour': 0, 'month': 0, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 0}
  274. >>> start = datetime(2001, 1, 1, 00,00,14)
  275. >>> end = datetime(2001, 1, 1, 00,00,44)
  276. >>> compute_datetime_delta(start, end)
  277. {'hour': 0, 'month': 0, 'second': 30, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 0}
  278. >>> start = datetime(2001, 1, 1, 00,00,44)
  279. >>> end = datetime(2001, 1, 1, 00,01,14)
  280. >>> compute_datetime_delta(start, end)
  281. {'hour': 0, 'month': 0, 'second': 30, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 1}
  282. >>> start = datetime(2001, 1, 1, 00,00,30)
  283. >>> end = datetime(2001, 1, 1, 00,05,30)
  284. >>> compute_datetime_delta(start, end)
  285. {'hour': 0, 'month': 0, 'second': 300, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 5}
  286. >>> start = datetime(2001, 1, 1, 00,00,00)
  287. >>> end = datetime(2001, 1, 1, 00,01,00)
  288. >>> compute_datetime_delta(start, end)
  289. {'hour': 0, 'month': 0, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 1}
  290. >>> start = datetime(2011,10,31, 00,45,00)
  291. >>> end = datetime(2011,10,31, 01,45,00)
  292. >>> compute_datetime_delta(start, end)
  293. {'hour': 1, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 60}
  294. >>> start = datetime(2011,10,31, 00,45,00)
  295. >>> end = datetime(2011,10,31, 01,15,00)
  296. >>> compute_datetime_delta(start, end)
  297. {'hour': 1, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 30}
  298. >>> start = datetime(2011,10,31, 00,45,00)
  299. >>> end = datetime(2011,10,31, 12,15,00)
  300. >>> compute_datetime_delta(start, end)
  301. {'hour': 12, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 690}
  302. >>> start = datetime(2011,10,31, 00,00,00)
  303. >>> end = datetime(2011,10,31, 01,00,00)
  304. >>> compute_datetime_delta(start, end)
  305. {'hour': 1, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 0}
  306. >>> start = datetime(2011,10,31, 00,00,00)
  307. >>> end = datetime(2011,11,01, 01,00,00)
  308. >>> compute_datetime_delta(start, end)
  309. {'hour': 25, 'second': 0, 'max_days': 1, 'year': 0, 'day': 1, 'minute': 0}
  310. >>> start = datetime(2011,10,31, 12,00,00)
  311. >>> end = datetime(2011,11,01, 06,00,00)
  312. >>> compute_datetime_delta(start, end)
  313. {'hour': 18, 'second': 0, 'max_days': 0, 'year': 0, 'day': 0, 'minute': 0}
  314. >>> start = datetime(2011,11,01, 00,00,00)
  315. >>> end = datetime(2011,12,01, 01,00,00)
  316. >>> compute_datetime_delta(start, end)
  317. {'hour': 721, 'month': 1, 'second': 0, 'max_days': 30, 'year': 0, 'day': 0, 'minute': 0}
  318. >>> start = datetime(2011,11,01, 00,00,00)
  319. >>> end = datetime(2011,11,05, 00,00,00)
  320. >>> compute_datetime_delta(start, end)
  321. {'hour': 0, 'second': 0, 'max_days': 4, 'year': 0, 'day': 4, 'minute': 0}
  322. >>> start = datetime(2011,10,06, 00,00,00)
  323. >>> end = datetime(2011,11,05, 00,00,00)
  324. >>> compute_datetime_delta(start, end)
  325. {'hour': 0, 'second': 0, 'max_days': 30, 'year': 0, 'day': 30, 'minute': 0}
  326. >>> start = datetime(2011,12,02, 00,00,00)
  327. >>> end = datetime(2012,01,01, 00,00,00)
  328. >>> compute_datetime_delta(start, end)
  329. {'hour': 0, 'second': 0, 'max_days': 30, 'year': 1, 'day': 30, 'minute': 0}
  330. >>> start = datetime(2011,01,01, 00,00,00)
  331. >>> end = datetime(2011,02,01, 00,00,00)
  332. >>> compute_datetime_delta(start, end)
  333. {'hour': 0, 'month': 1, 'second': 0, 'max_days': 31, 'year': 0, 'day': 0, 'minute': 0}
  334. >>> start = datetime(2011,12,01, 00,00,00)
  335. >>> end = datetime(2012,01,01, 00,00,00)
  336. >>> compute_datetime_delta(start, end)
  337. {'hour': 0, 'month': 1, 'second': 0, 'max_days': 31, 'year': 1, 'day': 0, 'minute': 0}
  338. >>> start = datetime(2011,12,01, 00,00,00)
  339. >>> end = datetime(2012,06,01, 00,00,00)
  340. >>> compute_datetime_delta(start, end)
  341. {'hour': 0, 'month': 6, 'second': 0, 'max_days': 183, 'year': 1, 'day': 0, 'minute': 0}
  342. >>> start = datetime(2011,06,01, 00,00,00)
  343. >>> end = datetime(2021,06,01, 00,00,00)
  344. >>> compute_datetime_delta(start, end)
  345. {'hour': 0, 'month': 120, 'second': 0, 'max_days': 3653, 'year': 10, 'day': 0, 'minute': 0}
  346. >>> start = datetime(2011,06,01, 00,00,00)
  347. >>> end = datetime(2012,06,01, 12,00,00)
  348. >>> compute_datetime_delta(start, end)
  349. {'hour': 8796, 'month': 12, 'second': 0, 'max_days': 366, 'year': 1, 'day': 0, 'minute': 0}
  350. >>> start = datetime(2011,06,01, 00,00,00)
  351. >>> end = datetime(2012,06,01, 12,30,00)
  352. >>> compute_datetime_delta(start, end)
  353. {'hour': 8796, 'month': 12, 'second': 0, 'max_days': 366, 'year': 1, 'day': 0, 'minute': 527790}
  354. >>> start = datetime(2011,06,01, 00,00,00)
  355. >>> end = datetime(2012,06,01, 12,00,05)
  356. >>> compute_datetime_delta(start, end)
  357. {'hour': 8796, 'month': 12, 'second': 31665605, 'max_days': 366, 'year': 1, 'day': 0, 'minute': 0}
  358. >>> start = datetime(2011,06,01, 00,00,00)
  359. >>> end = datetime(2012,06,01, 00,30,00)
  360. >>> compute_datetime_delta(start, end)
  361. {'hour': 0, 'month': 12, 'second': 0, 'max_days': 366, 'year': 1, 'day': 0, 'minute': 527070}
  362. >>> start = datetime(2011,06,01, 00,00,00)
  363. >>> end = datetime(2012,06,01, 00,00,05)
  364. >>> compute_datetime_delta(start, end)
  365. {'hour': 0, 'month': 12, 'second': 31622405, 'max_days': 366, 'year': 1, 'day': 0, 'minute': 0}
  366. @endcode
  367. @return A dictionary with year, month, day, hour, minute and second as keys()
  368. """
  369. comp = {}
  370. day_diff = (end - start).days
  371. comp["max_days"] = day_diff
  372. # Date
  373. # Count full years
  374. d = end.year - start.year
  375. comp["year"] = d
  376. # Count full months
  377. if start.month == 1 and end.month == 1:
  378. comp["month"] = 0
  379. elif start.day == 1 and end.day == 1:
  380. d = end.month - start.month
  381. if d < 0:
  382. d = d + 12 * comp["year"]
  383. elif d == 0:
  384. d = 12 * comp["year"]
  385. comp["month"] = d
  386. # Count full days
  387. if start.day == 1 and end.day == 1:
  388. comp["day"] = 0
  389. else:
  390. comp["day"] = day_diff
  391. # Time
  392. # Hours
  393. if start.hour == 0 and end.hour == 0:
  394. comp["hour"] = 0
  395. else:
  396. d = end.hour - start.hour
  397. if d < 0:
  398. d = d + 24 + 24 * day_diff
  399. else:
  400. d = d + 24 * day_diff
  401. comp["hour"] = d
  402. # Minutes
  403. if start.minute == 0 and end.minute == 0:
  404. comp["minute"] = 0
  405. else:
  406. d = end.minute - start.minute
  407. if d != 0:
  408. if comp["hour"]:
  409. d = d + 60 * comp["hour"]
  410. else:
  411. d = d + 24 * 60 * day_diff
  412. elif d == 0:
  413. if comp["hour"]:
  414. d = 60 * comp["hour"]
  415. else:
  416. d = 24 * 60 * day_diff
  417. comp["minute"] = d
  418. # Seconds
  419. if start.second == 0 and end.second == 0:
  420. comp["second"] = 0
  421. else:
  422. d = end.second - start.second
  423. if d != 0:
  424. if comp["minute"]:
  425. d = d + 60 * comp["minute"]
  426. elif comp["hour"]:
  427. d = d + 3600 * comp["hour"]
  428. else:
  429. d = d + 24 * 60 * 60 * day_diff
  430. elif d == 0:
  431. if comp["minute"]:
  432. d = 60 * comp["minute"]
  433. elif comp["hour"]:
  434. d = 3600 * comp["hour"]
  435. else:
  436. d = 24 * 60 * 60 * day_diff
  437. comp["second"] = d
  438. return comp
  439. ###############################################################################
  440. def string_to_datetime(time_string):
  441. """!Convert a string into a datetime object
  442. Supported ISO string formats are:
  443. - YYYY-mm-dd
  444. - YYYY-mm-dd HH:MM:SS
  445. Time zones are not supported
  446. @param time_string The time string to convert
  447. @return datetime object or None in case of an error
  448. """
  449. # BC is not supported
  450. if time_string.find("bc") > 0:
  451. core.error("Dates Before Christ are not supported "
  452. "in the temporal database")
  453. return None
  454. # BC is not supported
  455. if time_string.find("+") > 0:
  456. core.error("Time zones are not supported "
  457. "in the temporal database")
  458. return None
  459. if time_string.find(":") > 0:
  460. time_format = "%Y-%m-%d %H:%M:%S"
  461. else:
  462. time_format = "%Y-%m-%d"
  463. try:
  464. return datetime.strptime(time_string, time_format)
  465. except:
  466. core.error("Unable to parse time string: %s"%time_string)
  467. return None
  468. ###############################################################################
  469. def datetime_to_grass_datetime_string(dt):
  470. """!Convert a python datetime object into a GRASS datetime string"""
  471. # GRASS datetime month names
  472. month_names = ["", "jan", "feb", "mar", "apr", "may", "jun",
  473. "jul", "aug", "sep", "oct", "nov", "dec"]
  474. # Check for time zone info in the datetime object
  475. if dt.tzinfo is not None:
  476. string = "%.2i %s %.2i %.2i:%.2i:%.2i %+.4i" % (dt.day,
  477. month_names[dt.month], dt.year,
  478. dt.hour, dt.minute, dt.second, dt.tzinfo._offset.seconds / 60)
  479. else:
  480. string = "%.2i %s %.4i %.2i:%.2i:%.2i" % (dt.day, month_names[
  481. dt.month], dt.year, dt.hour, dt.minute, dt.second)
  482. return string
  483. ###############################################################################
  484. if __name__ == "__main__":
  485. import doctest
  486. doctest.testmod()