Rails с postgresql: извлечение поля из времени не работает - PullRequest
0 голосов
/ 05 июня 2018

У меня проблема с заказом коллекции по часам.Но обо всем по порядку.

Детали проекта:

Rails version             5.1.3
Ruby version              2.4.1-p111 (x86_64-linux)
Database adapter          postgresql

Вот так выглядит моя коллекция:

 #<DeliveryTimeslot:0x00562dd1ad2690
  id: 1,
  start: Sun, 02 Jan 2000 01:00:00 +03 +03:00,
  stop: Sun, 02 Jan 2000 02:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:39 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:25 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abcf98
  id: 2,
  start: Sun, 02 Jan 2000 02:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 03:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:39 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:25 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ad23c0
  id: 3,
  start: Sat, 01 Jan 2000 03:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 04:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:39 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ad1e70
  id: 4,
  start: Sat, 01 Jan 2000 04:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 05:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ad1470
  id: 5,
  start: Sat, 01 Jan 2000 05:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 06:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ad0bd8
  id: 6,
  start: Sat, 01 Jan 2000 06:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 07:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acfdc8
id: 7,
  start: Sat, 01 Jan 2000 07:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 08:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acf7b0
  id: 8,
  start: Sat, 01 Jan 2000 08:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 09:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acf198
  id: 9,
  start: Sat, 01 Jan 2000 09:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 10:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acee00
  id: 10,
  start: Sat, 01 Jan 2000 10:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 11:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1aceb80
  id: 11,
  start: Sat, 01 Jan 2000 11:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 12:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ace838
  id: 12,
  start: Sat, 01 Jan 2000 12:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 13:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1ace4c8
  id: 13,
start: Sat, 01 Jan 2000 13:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 14:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acdde8
  id: 14,
  start: Sat, 01 Jan 2000 14:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 15:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acd758
  id: 15,
  start: Sat, 01 Jan 2000 15:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 16:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acd168
  id: 16,
  start: Sat, 01 Jan 2000 16:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 17:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:26 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acce20
  id: 17,
  start: Sat, 01 Jan 2000 17:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 18:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1acc678
  id: 18,
  start: Sat, 01 Jan 2000 18:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 19:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abfb30
  id: 19,
  start: Sat, 01 Jan 2000 19:00:00 +03 +03:00,
stop: Sat, 01 Jan 2000 20:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abf220
  id: 20,
  start: Sat, 01 Jan 2000 20:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 21:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abea78
  id: 21,
  start: Sat, 01 Jan 2000 21:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 22:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abe500
  id: 22,
  start: Sat, 01 Jan 2000 22:00:00 +03 +03:00,
  stop: Sat, 01 Jan 2000 23:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abdf10
  id: 23,
  start: Sat, 01 Jan 2000 23:00:00 +03 +03:00,
  stop: Sun, 02 Jan 2000 00:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>,
 #<DeliveryTimeslot:0x00562dd1abd3f8
  id: 24,
  start: Sun, 02 Jan 2000 00:00:00 +03 +03:00,
  stop: Sun, 02 Jan 2000 01:00:00 +03 +03:00,
  created_at: Wed, 05 Oct 2016 17:57:40 +03 +03:00,
  updated_at: Thu, 16 Mar 2017 12:40:27 +03 +03:00>]

Обе колонки start и stopt.time в схеме (поэтому записи ActiveSupport::TimeWithZone).

Я хочу упорядочить их по часам, и я должен сделать это с помощью операторов SQL.

Итак, у меня естьпопробовал это:

DeliveryTimeslot.all.order("EXTRACT (HOUR FROM start) DESC").map(&:id)

и я ожидал получить массив, как это:

[23, 22, 21, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 24]

но вместо этого я получил это:

[2, 1, 24, 23, 22, 21, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3]

Моя мысль былачто это потому, что три записи (с идентификаторами 1, 2 и 24) имеют разные даты начала (2 января, в то время как остальные из них 1 января).Разве запрос, который я написал выше, не должен извлекать час из записи типа времени?Это действительно из-за даты?IDK, почему это не работает, и в принципе я ничего не могу изменить в БД.Любые намеки, что я делаю не так?Любые советы для сделать это без каких-либо изменений в БД?

Ответы [ 3 ]

0 голосов
/ 06 июня 2018

Нашел решение.Он работал с моим существующим приложением

Попробуйте

DeliveryTimeslot.all.order('start DESC').sort_by {|item| item.start.to_date}.map{|item| item.id}

0 голосов
/ 06 июня 2018

Проблема решена.

Я полагаю, что ответ заключается в том, что эта проблема вызвана созданием самой базы данных.В postgresql doc: (https://www.postgresql.org/docs/9.2/static/datatype-datetime.html) они говорят:

Мы не рекомендуем использовать тип времени с часовым поясом

... и ActiveRecord работает с ошибкамикак-то с ActiveSupport :: TimeWithZone.

Коллега сказал мне отсортировать его по извлеченным часам. В psql это выглядит нормально:

kriseqsdb=# SELECT EXTRACT (HOUR FROM start) AS extracted_hour, * FROM delivery_timeslots ORDER BY EXTRACT (HOUR FROM start) DESC;
 extracted_hour | id |  start   |   stop   |         created_at         |         updated_at         
----------------+----+----------+----------+----------------------------+----------------------------
             23 |  2 | 23:00:00 | 00:00:00 | 2016-10-05 14:57:39.981764 | 2017-03-16 09:40:25.960571
             22 |  1 | 22:00:00 | 23:00:00 | 2016-10-05 14:57:39.960375 | 2017-03-16 09:40:25.738402
             21 | 24 | 21:00:00 | 22:00:00 | 2016-10-05 14:57:40.381005 | 2017-03-16 09:40:27.558225
             20 | 23 | 20:00:00 | 21:00:00 | 2016-10-05 14:57:40.362648 | 2017-03-16 09:40:27.485668
             19 | 22 | 19:00:00 | 20:00:00 | 2016-10-05 14:57:40.347286 | 2017-03-16 09:40:27.413084
             18 | 21 | 18:00:00 | 19:00:00 | 2016-10-05 14:57:40.329297 | 2017-03-16 09:40:27.340032
             17 | 20 | 17:00:00 | 18:00:00 | 2016-10-05 14:57:40.309307 | 2017-03-16 09:40:27.267091
             16 | 19 | 16:00:00 | 17:00:00 | 2016-10-05 14:57:40.291424 | 2017-03-16 09:40:27.194525
             15 | 18 | 15:00:00 | 16:00:00 | 2016-10-05 14:57:40.261229 | 2017-03-16 09:40:27.122137
             14 | 17 | 14:00:00 | 15:00:00 | 2016-10-05 14:57:40.244531 | 2017-03-16 09:40:27.049617
             13 | 16 | 13:00:00 | 14:00:00 | 2016-10-05 14:57:40.228901 | 2017-03-16 09:40:26.977144
             12 | 15 | 12:00:00 | 13:00:00 | 2016-10-05 14:57:40.2118   | 2017-03-16 09:40:26.904671
             11 | 14 | 11:00:00 | 12:00:00 | 2016-10-05 14:57:40.194678 | 2017-03-16 09:40:26.832177
             10 | 13 | 10:00:00 | 11:00:00 | 2016-10-05 14:57:40.175353 | 2017-03-16 09:40:26.759804
              9 | 12 | 09:00:00 | 10:00:00 | 2016-10-05 14:57:40.159382 | 2017-03-16 09:40:26.687357
              8 | 11 | 08:00:00 | 09:00:00 | 2016-10-05 14:57:40.144921 | 2017-03-16 09:40:26.614746
              7 | 10 | 07:00:00 | 08:00:00 | 2016-10-05 14:57:40.127898 | 2017-03-16 09:40:26.542091
              6 |  9 | 06:00:00 | 07:00:00 | 2016-10-05 14:57:40.106023 | 2017-03-16 09:40:26.469586
              5 |  8 | 05:00:00 | 06:00:00 | 2016-10-05 14:57:40.082284 | 2017-03-16 09:40:26.397126
              4 |  7 | 04:00:00 | 05:00:00 | 2016-10-05 14:57:40.06161  | 2017-03-16 09:40:26.324512
              3 |  6 | 03:00:00 | 04:00:00 | 2016-10-05 14:57:40.046009 | 2017-03-16 09:40:26.251731
              2 |  5 | 02:00:00 | 03:00:00 | 2016-10-05 14:57:40.03066  | 2017-03-16 09:40:26.178915
              1 |  4 | 01:00:00 | 02:00:00 | 2016-10-05 14:57:40.015266 | 2017-03-16 09:40:26.106172
              0 |  3 | 00:00:00 | 01:00:00 | 2016-10-05 14:57:39.996529 | 2017-03-16 09:40:26.033393
(24 rows)

и так происходит через консоль rails:

DeliveryTimeslot.connection.select_all("SELECT EXTRACT (HOUR FROM start) AS extracted_hour, * FROM delivery_timeslots ORDER BY EXTRACT (HOUR FROM start) DESC")
=>
[23.0, 2, "23:00:00", "00:00:00", "2016-10-05 14:57:39.981764", "2017-03-16 09:40:25.960571"],
   [22.0, 1, "22:00:00", "23:00:00", "2016-10-05 14:57:39.960375", "2017-03-16 09:40:25.738402"],
   [21.0, 24, "21:00:00", "22:00:00", "2016-10-05 14:57:40.381005", "2017-03-16 09:40:27.558225"],
   [20.0, 23, "20:00:00", "21:00:00", "2016-10-05 14:57:40.362648", "2017-03-16 09:40:27.485668"],
   [19.0, 22, "19:00:00", "20:00:00", "2016-10-05 14:57:40.347286", "2017-03-16 09:40:27.413084"],
   [18.0, 21, "18:00:00", "19:00:00", "2016-10-05 14:57:40.329297", "2017-03-16 09:40:27.340032"],
   [17.0, 20, "17:00:00", "18:00:00", "2016-10-05 14:57:40.309307", "2017-03-16 09:40:27.267091"],
   [16.0, 19, "16:00:00", "17:00:00", "2016-10-05 14:57:40.291424", "2017-03-16 09:40:27.194525"],
   [15.0, 18, "15:00:00", "16:00:00", "2016-10-05 14:57:40.261229", "2017-03-16 09:40:27.122137"],
   [14.0, 17, "14:00:00", "15:00:00", "2016-10-05 14:57:40.244531", "2017-03-16 09:40:27.049617"],
   [13.0, 16, "13:00:00", "14:00:00", "2016-10-05 14:57:40.228901", "2017-03-16 09:40:26.977144"],
   [12.0, 15, "12:00:00", "13:00:00", "2016-10-05 14:57:40.2118", "2017-03-16 09:40:26.904671"],
   [11.0, 14, "11:00:00", "12:00:00", "2016-10-05 14:57:40.194678", "2017-03-16 09:40:26.832177"],
   [10.0, 13, "10:00:00", "11:00:00", "2016-10-05 14:57:40.175353", "2017-03-16 09:40:26.759804"],
   [9.0, 12, "09:00:00", "10:00:00", "2016-10-05 14:57:40.159382", "2017-03-16 09:40:26.687357"],
   [8.0, 11, "08:00:00", "09:00:00", "2016-10-05 14:57:40.144921", "2017-03-16 09:40:26.614746"],
   [7.0, 10, "07:00:00", "08:00:00", "2016-10-05 14:57:40.127898", "2017-03-16 09:40:26.542091"],
   [6.0, 9, "06:00:00", "07:00:00", "2016-10-05 14:57:40.106023", "2017-03-16 09:40:26.469586"],
   [5.0, 8, "05:00:00", "06:00:00", "2016-10-05 14:57:40.082284", "2017-03-16 09:40:26.397126"],
   [4.0, 7, "04:00:00", "05:00:00", "2016-10-05 14:57:40.06161", "2017-03-16 09:40:26.324512"],
   [3.0, 6, "03:00:00", "04:00:00", "2016-10-05 14:57:40.046009", "2017-03-16 09:40:26.251731"],
   [2.0, 5, "02:00:00", "03:00:00", "2016-10-05 14:57:40.03066", "2017-03-16 09:40:26.178915"],
   [1.0, 4, "01:00:00", "02:00:00", "2016-10-05 14:57:40.015266", "2017-03-16 09:40:26.106172"],
   [0.0, 3, "00:00:00", "01:00:00", "2016-10-05 14:57:39.996529", "2017-03-16 09:40:26.033393"]

В этом случае это нормально для меня, так как это было моей главной заботой. Но чтобы быть ясным в моем вопросе и получить правильный массив, я просто сопоставил его с помощью extract_hour:

DeliveryTimeslot.connection.select_all("SELECT EXTRACT (HOUR FROM start) AS extracted_hour, * FROM delivery_timeslots ORDER BY EXTRACT (HOUR FROM start) DESC").rows.map { |row| row[0].to_i }

и в основномполучил то, о чем я спросил.

    => [23, 22, 21, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
0 голосов
/ 05 июня 2018

Просто используйте DeliveryTimeslot.all.order("start DESC").map(&:id)

...