1996-08-19 06:14:33 +08:00
|
|
|
/*
|
|
|
|
* datetime_functions.c --
|
|
|
|
*
|
|
|
|
* This file defines new functions for the time and date data types.
|
|
|
|
*
|
|
|
|
* Copyright (c) 1996, Massimo Dal Zotto <dz@cs.unitn.it>
|
|
|
|
*/
|
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
#include <stdio.h> /* for sprintf() */
|
1997-11-06 05:38:25 +08:00
|
|
|
#include <string.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#ifdef HAVE_FLOAT_H
|
|
|
|
#include <float.h>
|
|
|
|
#endif
|
1996-08-19 06:14:33 +08:00
|
|
|
|
|
|
|
#include "postgres.h"
|
1997-11-06 05:38:25 +08:00
|
|
|
#include "miscadmin.h"
|
|
|
|
#include "utils/builtins.h"
|
|
|
|
#include "utils/nabstime.h"
|
1997-08-26 03:41:52 +08:00
|
|
|
#include "utils/datetime.h"
|
1997-11-06 05:38:25 +08:00
|
|
|
#include "access/xact.h"
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1997-11-06 05:38:25 +08:00
|
|
|
#include "datetime_functions.h"
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1997-11-06 05:38:25 +08:00
|
|
|
/* Constant to replace calls to date2j(2000,1,1) */
|
|
|
|
#define JDATE_2000 2451545
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A modified version of time_in which allows the value 24:00:00 for
|
|
|
|
* time and converts it to TimeADT data type forcing seconds to 0.
|
|
|
|
* This can be Useful if you need to handle TimeADT values limited
|
|
|
|
* to hh:mm like in timetables.
|
|
|
|
*/
|
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *
|
1997-11-06 05:38:25 +08:00
|
|
|
hhmm_in(char *str)
|
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *time;
|
|
|
|
|
|
|
|
double fsec;
|
|
|
|
struct tm tt,
|
|
|
|
*tm = &tt;
|
|
|
|
|
|
|
|
int nf;
|
|
|
|
char lowstr[MAXDATELEN + 1];
|
|
|
|
char *field[MAXDATEFIELDS];
|
|
|
|
int dtype;
|
|
|
|
int ftype[MAXDATEFIELDS];
|
|
|
|
|
|
|
|
if (!PointerIsValid(str))
|
|
|
|
elog(ERROR, "Bad (null) time external representation", NULL);
|
|
|
|
|
|
|
|
if ((ParseDateTime(str, lowstr, field, ftype, MAXDATEFIELDS, &nf) != 0)
|
|
|
|
|| (DecodeTimeOnly(field, ftype, nf, &dtype, tm, &fsec) != 0))
|
|
|
|
elog(ERROR, "Bad time external representation '%s'", str);
|
|
|
|
|
|
|
|
if (tm->tm_hour < 0 || tm->tm_hour > 24 ||
|
|
|
|
(tm->tm_hour == 24 && (tm->tm_min != 0 || tm->tm_sec != 0 || fsec != 0)))
|
|
|
|
{
|
|
|
|
elog(ERROR,
|
|
|
|
"time_in: hour must be limited to values 0 through 24:00 "
|
|
|
|
"in \"%s\"",
|
|
|
|
str);
|
|
|
|
}
|
|
|
|
if ((tm->tm_min < 0) || (tm->tm_min > 59))
|
|
|
|
elog(ERROR, "Minute must be limited to values 0 through 59 in '%s'", str);
|
|
|
|
if ((tm->tm_sec < 0) || ((tm->tm_sec + fsec) >= 60))
|
|
|
|
elog(ERROR, "Second must be limited to values 0 through < 60 in '%s'",
|
|
|
|
str);
|
|
|
|
|
|
|
|
time = palloc(sizeof(TimeADT));
|
|
|
|
|
|
|
|
*time = ((((tm->tm_hour * 60) + tm->tm_min) * 60));
|
|
|
|
|
|
|
|
return (time);
|
1997-11-06 05:38:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A modified version of time_out which converts from TimeADT data type
|
|
|
|
* omitting the seconds field when it is 0.
|
|
|
|
* Useful if you need to handle TimeADT values limited to hh:mm.
|
|
|
|
*/
|
|
|
|
|
|
|
|
char *
|
|
|
|
hhmm_out(TimeADT *time)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
char *result;
|
|
|
|
struct tm tt,
|
|
|
|
*tm = &tt;
|
|
|
|
char buf[MAXDATELEN + 1];
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
if (!PointerIsValid(time))
|
|
|
|
return NULL;
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
tm->tm_hour = (*time / (60 * 60));
|
|
|
|
tm->tm_min = (((int) (*time / 60)) % 60);
|
|
|
|
tm->tm_sec = (((int) *time) % 60);
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
if (tm->tm_sec == 0)
|
|
|
|
sprintf(buf, "%02d:%02d", tm->tm_hour, tm->tm_min);
|
|
|
|
else
|
|
|
|
sprintf(buf, "%02d:%02d:%02d", tm->tm_hour, tm->tm_min, tm->tm_sec);
|
1997-09-07 13:04:48 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
result = palloc(strlen(buf) + 1);
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
strcpy(result, buf);
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (result);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *
|
1997-11-06 05:38:25 +08:00
|
|
|
hhmm(TimeADT *time)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *result = palloc(sizeof(TimeADT));
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
*result = (((int) *time) / 60 * 60);
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (result);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *
|
1997-11-06 05:38:25 +08:00
|
|
|
time_difference(TimeADT *time1, TimeADT *time2)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *time = palloc(sizeof(TimeADT));
|
1997-09-07 13:04:48 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
*time = (*time1 - *time2);
|
|
|
|
return (time);
|
1997-11-06 05:38:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int4
|
|
|
|
time_hours(TimeADT *time)
|
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
return (((int) *time) / 3600);
|
1997-11-06 05:38:25 +08:00
|
|
|
}
|
1997-09-07 13:04:48 +08:00
|
|
|
|
1997-11-06 05:38:25 +08:00
|
|
|
int4
|
|
|
|
time_minutes(TimeADT *time)
|
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
return ((((int) *time) / 60) % 60);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
time_seconds(TimeADT *time)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
return (((int) *time) % 60);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
as_minutes(TimeADT *time)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
return (((int) *time) / 60);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
as_seconds(TimeADT *time)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
return ((int) *time);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
date_day(DateADT val)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
int year,
|
|
|
|
month,
|
|
|
|
day;
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
j2date(val + JDATE_2000, &year, &month, &day);
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (day);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
date_month(DateADT val)
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
int year,
|
|
|
|
month,
|
|
|
|
day;
|
1997-08-26 03:41:52 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
j2date(val + JDATE_2000, &year, &month, &day);
|
1997-08-26 03:41:52 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (month);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1997-09-07 13:04:48 +08:00
|
|
|
int4
|
1997-11-06 05:38:25 +08:00
|
|
|
date_year(DateADT val)
|
1997-08-26 03:41:52 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
int year,
|
|
|
|
month,
|
|
|
|
day;
|
1997-08-26 03:41:52 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
j2date(val + JDATE_2000, &year, &month, &day);
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (year);
|
1997-08-26 03:41:52 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *
|
1997-11-06 05:38:25 +08:00
|
|
|
currenttime()
|
1996-08-19 06:14:33 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
TimeADT *result = palloc(sizeof(TimeADT));
|
|
|
|
struct tm *tm;
|
|
|
|
time_t current_time;
|
1997-11-06 05:38:25 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
current_time = time(NULL);
|
|
|
|
tm = localtime(¤t_time);
|
|
|
|
*result = ((((tm->tm_hour * 60) + tm->tm_min) * 60) + tm->tm_sec);
|
1997-08-26 03:41:52 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
return (result);
|
1996-08-19 06:14:33 +08:00
|
|
|
}
|
1997-09-08 10:41:22 +08:00
|
|
|
|
1997-11-06 05:38:25 +08:00
|
|
|
DateADT
|
|
|
|
currentdate()
|
1997-08-26 03:41:52 +08:00
|
|
|
{
|
1998-02-26 12:46:47 +08:00
|
|
|
DateADT date;
|
|
|
|
struct tm tt,
|
|
|
|
*tm = &tt;
|
1996-08-19 06:14:33 +08:00
|
|
|
|
1998-02-26 12:46:47 +08:00
|
|
|
GetCurrentTime(tm);
|
|
|
|
date = (date2j(tm->tm_year, tm->tm_mon, tm->tm_mday) - JDATE_2000);
|
|
|
|
return (date);
|
1997-08-26 03:41:52 +08:00
|
|
|
}
|
1997-11-06 05:38:25 +08:00
|
|
|
|
|
|
|
/* end of file */
|