Skip to content

3268. Find Overlapping Shifts II

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
WITH
  EmployeeTimes AS (
    SELECT DISTINCT employee_id, start_time AS `time`
    FROM EmployeeShifts
    UNION DISTINCT
    SELECT DISTINCT employee_id, end_time AS `time`
    FROM EmployeeShifts
  ),
  Segments AS (
    SELECT
      employee_id,
      `time` AS start_time,
      LEAD(`time`) OVER(PARTITION BY employee_id ORDER BY `time`) AS end_time
    FROM EmployeeTimes
  ),
  SegmentsCount AS (
    SELECT
      Segments.*,
      COUNT(*) AS concurrent_count
    FROM Segments
    INNER JOIN EmployeeShifts
      USING (employee_id)
    WHERE
      Segments.start_time >= EmployeeShifts.start_time
      AND Segments.end_time <= EmployeeShifts.end_time
    GROUP BY 1, 2, 3
  )
SELECT
  employee_id,
  MAX(concurrent_count) AS max_overlapping_shifts,
  SUM(
    concurrent_count * (concurrent_count - 1) / 2 * TIMESTAMPDIFF(
      MINUTE,
      start_time,
      end_time
    )
  ) AS total_overlap_duration -- C(concurrent_count, 2)
FROM SegmentsCount
GROUP BY 1
ORDER BY 1;