Pembagian/normalisasi tanggal waktu SQL/Postgres

Saya memiliki tabel aktivitas ini

+--------------+------------------+
| Field        | Type             |
+--------------+------------------+
| id           | int(11) unsigned |
| start_date   | timestamp        |
| end_date     | timestamp        |
| ...          |                  |
+--------------+------------------+

Saya memerlukan tampilan yang mengelompokkan aktivitas ini berdasarkan tanggal_mulai berdasarkan HARI, tetapi sedemikian rupa sehingga, jika tanggal_akhir tidak pada hari yang sama dengan tanggal_mulai, tampilan tersebut berisi entri lagi tetapi dengan tanggal_mulai disetel ke 00:00 berikutnya hari.. (dan seterusnya, diulangi sebanyak yang diperlukan hingga tanggal_mulai berada di hari yang sama dengan tanggal_akhir)

Sebagai contoh:

jika tabel aktivitas berisi:

+--------------+----------------------------+----------------------------+
| id           | start_date                 | end_date                   |
+--------------+----------------------------+----------------------------+
|  1           | 2014-12-02 14:12:00+00     | 2014-12-03 06:45:00+00     |   
|  2           | 2014-12-05 15:25:00+00     | 2014-12-05 07:29:00+00     |                                
+--------------+----------------------------+----------------------------+

Tampilan harus berisi:

+--------------+----------------------------+----------------------------+
| activity_id  | start_date                 | end_date                   |
+--------------+----------------------------+----------------------------+
|  1           | 2014-12-02 14:12:00+00     | 2014-12-02 23:59:59+00     |   
|  1           | 2014-12-03 00:00:00+00     | 2014-12-03 06:45:00+00     |
|  2           | 2014-12-05 15:25:00+00     | 2014-12-05 07:29:00+00     |                                  
+--------------+----------------------------+----------------------------+

Bantuan apa pun akan sangat dihargai!

PS: Saya menggunakan postgresql


person Alexandr    schedule 10.12.2014    source sumber


Jawaban (1)


Untuk mendapatkan baris yang diperlukan, mulailah dengan menggunakan mengatur fungsi pengembalian bersama dengan gabungan lateral. Dari sana, gunakan pernyataan CASE dan aritmatika tanggal untuk mengeluarkan nilai-nilai yang relevan.

Berikut ini contoh untuk Anda mulai:

with data as (
  select id, start_date, end_date
  from (values
    (1, '2014-12-02 14:12:00+00'::timestamptz, '2014-12-03 06:45:00+00'::timestamptz),
    (2, '2014-12-05 15:25:00+00'::timestamptz, '2014-12-05 07:29:00+00'::timestamptz)
  ) as rows (id, start_date, end_date)
)
select data.id,
      case days.d = date_trunc('day', data.start_date)
        when true then data.start_date
        else days.d
      end as start_date,
      case days.d = date_trunc('day', data.end_date)
        when true then data.end_date
        else days.d + interval '1 day' - interval '1 sec'
      end as end_date
from data
join generate_series(
      date_trunc('day', data.start_date),
      date_trunc('day', data.end_date),
      '1 day'
      ) as days (d)
      on days.d >= date_trunc('day', data.start_date)
      and days.d <= date_trunc('day', data.end_date)

 id |       start_date       |        end_date        
----+------------------------+------------------------
  1 | 2014-12-02 15:12:00+01 | 2014-12-02 23:59:59+01
  1 | 2014-12-03 00:00:00+01 | 2014-12-03 07:45:00+01
  2 | 2014-12-05 16:25:00+01 | 2014-12-05 08:29:00+01
(3 rows)

Selain itu, bergantung pada apa yang Anda lakukan, mungkin lebih masuk akal jika Anda menggunakan rentang tanggal:

with data as (
  select id, start_date, end_date
  from (values
    (1, '2014-12-02 14:12:00+00'::timestamptz, '2014-12-03 06:45:00+00'::timestamptz),
    (2, '2014-12-05 07:25:00+00'::timestamptz, '2014-12-05 15:29:00+00'::timestamptz)
  ) as rows (id, start_date, end_date)
)
select data.id,
      tstzrange(data.start_date, data.end_date)
from data;

 id |                      tstzrange                      
----+-----------------------------------------------------
  1 | ["2014-12-02 15:12:00+01","2014-12-03 07:45:00+01")
  2 | ["2014-12-05 08:25:00+01","2014-12-05 16:29:00+01")
(2 rows)
person Denis de Bernardy    schedule 10.12.2014