Thread: Alternate interval output function

Alternate interval output function

From
"Ian E. Morgan"
Date:
After being frustrated with the inflexible output of intervals, I've written
a pl/pgsql function to do what I want, and hopefully some other people might
find it useful.

Output is a string that matches the output format of an interval as closely
as possible, but rather than counting days as a fixed 24-hours, it
recalculates days based on the number of hours in the desired 'day'.

For example, this is very useful for summing up work time and outputting a
number of 'work days' rather than 24 hour periods. The number of hours in
the new 'day' can be any double precision value (in hours), such as to
account for my 7.5-hour billable days:

# select dur as interval, workdays(dur,7.5) from worklog;

     interval    |     workdays
----------------+-------------------
  -00:10:00      | -00:10:00
  -00:10:00      | -00:10:00
  -00:15:00      | -00:15:00
  -04:00:00      | -04:00:00
  -04:00:00      | -04:00:00
  -04:00:00      | -04:00:00
  -13:00:00      | -1 days -05:30:00
  -02:00:00      | -02:00:00
  -00:30:00      | -00:30:00
  -01:00:00      | -01:00:00
  -00:15:00      | -00:15:00
  -02:00:00      | -02:00:00
  -03:25:00      | -03:25:00
  -00:30:00      | -00:30:00
  -00:30:00      | -00:30:00
  1 day 08:30:00 | 4 days 02:30:00
  -00:05:00      | -00:05:00
  -00:10:00      | -00:10:00
(18 rows)

# select sum(dur) as "sum of interval", workdays(sum(dur),7.5) as "workdays of
   sum" from worklog where dur < 0;

   sum of interval  |  workdays of sum
-------------------+-------------------
  -1 days -12:00:00 | -4 days -06:00:00
(1 row)

Formatting consistency:

# select foo as interval, workdays(foo,7.5) from intest order by foo;

      interval      |     workdays
-------------------+-------------------
  -1 days -04:00:00 | -3 days -05:30:00
  -1 days           | -3 days -01:30:00
  -20:00:00         | -2 days -05:00:00
  -07:30:00         | -1 days
  00:00:00.00123    | 00:00:00.00123
  00:15:10          | 00:15:10
  07:15:10          | 07:15:10
  07:30:00          | 1 day
  07:45:10          | 1 day 00:15:10
  14:59:59          | 1 day 07:29:59
  15:00:00          | 2 days
  1 day             | 3 days 01:30:00
  4 days 04:00:00   | 13 days 02:30:00
  100 days          | 320 days
  365 days          | 1168 days
  20 years          | 23376 days
(16 rows)


The only caveat, currently, is that the output of workdays() is a string, so
you can't effectively sort it, or perform arithmetic or comparisons on it.
You must perform these operations on the real interval _before_ running it
though workdays().


CREATE OR REPLACE FUNCTION workdays (interval, double precision) RETURNS text AS '
DECLARE
     in_epoch double precision;
     out_days double precision;
     out_hours double precision;
     out_minutes double precision;
     out_seconds double precision;
     temp double precision;
     out character varying;
     negative boolean;
BEGIN
     --- Copyright 2004 Webcon, Inc. Written by Ian Morgan.
     --- Distributed under the GNU Public License.
     in_epoch := extract(EPOCH FROM $1);
     negative := ''f'';
     if in_epoch < 0 then
         negative := ''t'';
         in_epoch := in_epoch * -1;
     end if;

     out_days := floor(in_epoch / ($2 * 60 * 60));
     temp := in_epoch - (out_days * ($2 * 60 * 60));
     out_hours := floor(temp / 3600);
     temp := temp - (out_hours * 3600);
     out_minutes := floor(temp / 60);
     out_seconds := temp - (out_minutes * 60);

     out := '''';
     if negative = ''t'' then
         out_days := out_days * -1;
         out_hours := out_hours * -1;
     end if;
     if out_days != 0 then
         out := out || out_days;
         if (out_days = 1) then
             out := out || '' day'';
         else
             out := out || '' days'';
         end if;
     end if;
     if (out_hours != 0) or (out_minutes != 0) or (out_seconds != 0) then
         if out_days != 0 then
             out := out || '' '';
         end if;
         out := out || to_char(out_hours, ''FM09'') || '':'' || to_char(out_minutes, ''FM09'') || '':'';
         if (out_seconds = floor(out_seconds)) then
             out := out || to_char(out_seconds, ''FM09'');
         else
             out := out || to_char(out_seconds, ''FM09.999999'');
         end if;
     end if;
     return out;
END;
' LANGUAGE plpgsql;


If anyone has improvements or optimizations, I'd be glad to see them.

Regards,
Ian Morgan

--
-------------------------------------------------------------------
  Ian E. Morgan          Vice President & C.O.O.       Webcon, Inc.
  imorgan at webcon dot ca       PGP: #2DA40D07       www.webcon.ca
     *  Customized Linux Network Solutions for your Business  *
-------------------------------------------------------------------