metaclass: (Default)
metaclass ([personal profile] metaclass) wrote2012-02-16 12:54 am

Ох, какой ад

в Java c датой и временем.
Конкретно, чтобы на Clojure засунуть в jdbc параметр дату, пришлось перерыть весь гугл, исходники jaybird (firebird jdbc драйвера), внутренности joda-time и ее кложурной обертки clj-time

http://www.paullegato.com/blog/clojure-joda-sql-date-time/

(defn to-sql-date [date]
"Convert any Joda-readable date object (including a string) to a java.sql.Date"
(java.sql.Date. (.. (LocalDate. date) toDateMidnight toInstant getMillis)))

Основной затык - только в org.joda.time.LocalDate нормально сделана дата, без времени и временных зон. А jdbc драйвер понимает только java.sql.Date и вроде еще пару вариаций на тему "эпоха в long"

[identity profile] slonopotamus.livejournal.com 2012-02-16 05:06 pm (UTC)(link)
Всë в sql есть, timestamp with timezone.

[identity profile] sergiej.livejournal.com 2012-02-16 05:56 pm (UTC)(link)
Нихрена там нет таймзоны, Дата это универсальное время, оно не содержит информации о временном поясе, совсем. Проблема в том что Дата всегда создаётся как "локальная", и если её надо использовать "глобально" надо заниматься извращением - тянуть в дополнение к Дате ещё и временную зону в которой эту дату создавали.
Проблема даже глобальнее, она в том что для людей есть такое понятие как "дата" а для Джавы - нет. Тупо есть только время. То есть если контракт заключён в Минске с датой 1.1.2012 (на контракте так написано) и клиентское приложение в соответствии со всеми законами и правилами создало java.util.Date как полночь локального времени, итак для нормального человека это просто 1.1.2012, а для системы
это 31.12.2011 GMT + 3, и тут начинается ад, для клиентского приложения в Лондоне это 31.12.2011, для клиентского приложения в Москве это 1.1.2012. Что есть бред, даже если контракт заключён ночью у него одна "дата" - 1.1.2012.
Волевое решение писать все "даты" в одном временном поясе - например сервера помогает частично, танцы с переводам этой даты в человеческую дату клиента всё равно никто не отменял.
Edited 2012-02-16 17:57 (UTC)

[identity profile] slonopotamus.livejournal.com 2012-02-16 06:21 pm (UTC)(link)
Я хер знает какой там у вас sql (видимо тоже fb, в котором, как выяснилось, даже лога запросов нет), в моëм sql всë есть: http://www.postgresql.org/docs/9.1/static/datatype-datetime.html#DATATYPE-DATETIME-TABLE
И с таймзоной и без таймзоны и отдельно дата без времени и отдельно время без даты.

[identity profile] sergiej.livejournal.com 2012-02-16 06:24 pm (UTC)(link)
Где вы там увидели таймзон у "даты"???

[identity profile] slonopotamus.livejournal.com 2012-02-16 06:33 pm (UTC)(link)
Само название ТАЙМзоны совершенно недвусмысленно намекает что она имеет отношение ко ВРЕМЕНИ. Если у нас нет ВРЕМЕНИ, то и нет смысла говорить о таймзоне.

И этим вашим Минским контрактам глубоко похер на таймзону. Не пишут в контракте в какой таймзоне он был заключен. Более того, можно сначала продать а потом в конце месяца задним числом подписать бумаги. И никого не колебет какая там была таймзона.

Ваш пример надуман.

[identity profile] sergiej.livejournal.com 2012-02-16 06:38 pm (UTC)(link)
Таймзона не имела бы отношения к дате, если бы лядский Date не был ВРЕМЕНЕМ - полночью ЛОКАЛЬНОГО времени. Каким боком я могу узнать в базе, в каком лядском локальном времени был тот, кто коздавал Дату если я этого специально где-то не запишу?

[identity profile] slonopotamus.livejournal.com 2012-02-16 06:49 pm (UTC)(link)
Когда в лядском sql хранится лядская дата без времени, утверждается, что у неë нет понятия таймзоны. Вааще нет. И не нужно думать о таймзоне когда делается распечатка контракта. Надо просто печатать "2012-01-01".

[identity profile] sergiej.livejournal.com 2012-02-16 07:19 pm (UTC)(link)
Вы что, издеваетесь? В этом и проблема что сколько бы правильно база дату не записывала, в жабе Дата это будет полночь ЛОКАЛЬНОГО времени. То есть получая вашу дату из базы жабовскому приложению нужно гадать для какого же пояса дату создавать.

[identity profile] slonopotamus.livejournal.com 2012-02-16 07:26 pm (UTC)(link)
Это кривизна жабской Даты, а вовсе не sql'я.

[identity profile] sergiej.livejournal.com 2012-02-16 07:54 pm (UTC)(link)
У жабы проблема это факт, но у жабы и отмазка простая - в отсутствиии единого стандарта получается кривой java.sql.Date, который не знает что "с той стороны" за дата имеется в виду.
Кроме того со стороны джавы проблема легко решается расширением типа даты.

[identity profile] metaclass.livejournal.com 2012-02-16 06:42 pm (UTC)(link)
Ну, про это речь и идет. О том, что нужен тип "дата" :)

[identity profile] slonopotamus.livejournal.com 2012-02-16 06:50 pm (UTC)(link)
Да есть в базе тип "дата". И называется он, как это ни странно, date.

[identity profile] metaclass.livejournal.com 2012-02-16 07:01 pm (UTC)(link)
А в жабе java.util.Date - внезапно, содержит время, равное полночи, и таймзону.

[identity profile] slonopotamus.livejournal.com 2012-02-16 07:04 pm (UTC)(link)
java.util.Date (как и java.util.Calendar) - феерический пиздец, который починен только в joda-time. Однако посыл тред-стартера (не путать с топик-стартером) был в том, что в sql нет типа "дата".

[identity profile] sergiej.livejournal.com 2012-02-16 07:22 pm (UTC)(link)
Посыл в том что нет стандарта, позволяющая эту дату переделывать однозначно в жавовскую дату, и нужно тягать ещё дополнительную информацию, или забить, и читать её в стринг.

[identity profile] antontsau.livejournal.com 2012-02-17 05:28 am (UTC)(link)
нинада. В контрактах очень часто пишут. В страховках так вообще обязательно - "вступает в действие с 00.00 надцатого мартобря", а привязка по месту заключения контракта (что тоже обязательный аттрибут!) или прямо указывается временная зона (AEDT какой-нибудь). Ибо оно реально работает, расколотил застрахованный мошын на 15 минут раньше или позже границы - все, контракт не катит.