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 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235
|
/* copyright (c) 2022 - 2026 grunfink et al. / MIT license */
#ifndef _XS_TIME_H
#define _XS_TIME_H
#include <time.h>
xs_str *xs_str_time(time_t t, const char *fmt, int local);
#define xs_str_localtime(t, fmt) xs_str_time(t, fmt, 1)
#define xs_str_utctime(t, fmt) xs_str_time(t, fmt, 0)
#define xs_str_iso_date(t) xs_str_time(t, "%Y-%m-%dT%H:%M:%SZ", 0)
time_t xs_parse_iso_date(const char *iso_date, int local);
time_t xs_parse_time(const char *str, const char *fmt, int local);
#define xs_parse_localtime(str, fmt) xs_parse_time(str, fmt, 1)
#define xs_parse_utctime(str, fmt) xs_parse_time(str, fmt, 0)
xs_str *xs_str_time_diff(time_t time_diff);
xs_list *xs_tz_list(void);
int xs_tz_offset(const char *tz);
#ifdef XS_IMPLEMENTATION
xs_str *xs_str_time(time_t t, const char *fmt, int local)
/* returns a string with a formated time */
{
struct tm tm;
char tmp[64];
if (t == 0)
t = time(NULL);
if (local)
localtime_r(&t, &tm);
else
gmtime_r(&t, &tm);
strftime(tmp, sizeof(tmp), fmt, &tm);
return xs_str_new(tmp);
}
xs_str *xs_str_time_diff(time_t time_diff)
/* returns time_diff in seconds to 'human' units (d:hh:mm:ss) */
{
int secs = time_diff % 60;
int mins = (time_diff /= 60) % 60;
int hours = (time_diff /= 60) % 24;
int days = (time_diff /= 24);
return xs_fmt("%d:%02d:%02d:%02d", days, hours, mins, secs);
}
char *strptime(const char *s, const char *format, struct tm *tm);
time_t xs_parse_time(const char *str, const char *fmt, int local)
{
time_t t = 0;
#ifndef WITHOUT_STRPTIME
struct tm tm = {0};
strptime(str, fmt, &tm);
/* try to guess the Daylight Saving Time */
if (local)
tm.tm_isdst = -1;
t = local ? mktime(&tm) : timegm(&tm);
#endif /* WITHOUT_STRPTIME */
return t;
}
time_t xs_parse_iso_date(const char *iso_date, int local)
/* parses a YYYY-MM-DDTHH:MM:SS date string */
{
time_t t = 0;
#ifndef WITHOUT_STRPTIME
t = xs_parse_time(iso_date, "%Y-%m-%dT%H:%M:%S", local);
#else /* WITHOUT_STRPTIME */
struct tm tm = {0};
if (sscanf(iso_date, "%d-%d-%dT%d:%d:%d",
&tm.tm_year, &tm.tm_mon, &tm.tm_mday,
&tm.tm_hour, &tm.tm_min, &tm.tm_sec) == 6) {
tm.tm_year -= 1900;
tm.tm_mon -= 1;
if (local)
tm.tm_isdst = -1;
t = local ? mktime(&tm) : timegm(&tm);
}
#endif /* WITHOUT_STRPTIME */
return t;
}
/** timezones **/
/* intentionally dead simple */
struct {
const char *tz; /* timezone name */
float h_offset; /* hour offset */
} xs_tz[] = {
{ "UTC", 0 },
{ "WET (Western European Time)", 0 },
{ "WEST (Western European Summer Time)", 1 },
{ "CET (Central European Time)", 1 },
{ "CEST (Central European Summer Time)", 2 },
{ "EET (Eastern European Time)", 2 },
{ "EEST (Eastern European Summer Time)", 3 },
{ "MSK (Moskow Time Zone)", 3 },
{ "EST (Eastern Time Zone)", -5 },
{ "AST (Atlantic Time Zone)", -4 },
{ "ADT (Atlantic Daylight Time Zone)", -3 },
{ "CST (Central Time Zone)", -6 },
{ "CDT (Central Daylight Time Zone)", -5 },
{ "MST (Mountain Time Zone)", -7 },
{ "MDT (Mountain Daylight Time Zone)", -6 },
{ "PST (Pacific Time Zone)", -8 },
{ "PDT (Pacific Daylight Time Zone)", -7 },
{ "AKST (Alaska Time Zone)", -9 },
{ "AKDT (Alaska Daylight Time Zone)", -8 },
{ "China Time Zone", 8 },
{ "IST (Indian Standard Time)", 5.5 },
{ "IDT (Israel Daylight Standard Time)", 3 },
{ "WIB (Western Indonesia Time)", 7 },
{ "WITA (Central Indonesia Time)", 8 },
{ "WIT (Eastern Indonesia Time)", 9 },
{ "AWST (Australian Western Time)", 8 },
{ "ACST (Australian Eastern Time)", 9.5 },
{ "ACDT (Australian Daylight Eastern Time)", 10.5 },
{ "AEST (Australian Eastern Time)", 10 },
{ "AEDT (Australian Daylight Eastern Time)", 11 },
{ "NZST (New Zealand Time)", 12 },
{ "NZDT (New Zealand Daylight Time)", 13 },
{ "UTC", 0 },
{ "UTC+1", 1 },
{ "UTC+2", 2 },
{ "UTC+3", 3 },
{ "UTC+4", 4 },
{ "UTC+5", 5 },
{ "UTC+6", 6 },
{ "UTC+7", 7 },
{ "UTC+8", 8 },
{ "UTC+9", 9 },
{ "UTC+10", 10 },
{ "UTC+11", 11 },
{ "UTC+12", 12 },
{ "UTC-1", -1 },
{ "UTC-2", -2 },
{ "UTC-3", -3 },
{ "UTC-4", -4 },
{ "UTC-5", -5 },
{ "UTC-6", -6 },
{ "UTC-7", -7 },
{ "UTC-8", -8 },
{ "UTC-9", -9 },
{ "UTC-10", -10 },
{ "UTC-11", -11 },
{ "UTC-12", -12 },
{ "UTC-13", -13 },
{ "UTC-14", -14 },
{ "GMT", 0 },
{ "GMT+1", -1 },
{ "GMT+2", -2 },
{ "GMT+3", -3 },
{ "GMT+4", -4 },
{ "GMT+5", -5 },
{ "GMT+6", -6 },
{ "GMT+7", -7 },
{ "GMT+8", -8 },
{ "GMT+9", -9 },
{ "GMT+10", -10 },
{ "GMT+11", -11 },
{ "GMT+12", -12 },
{ "GMT-1", 1 },
{ "GMT-2", 2 },
{ "GMT-3", 3 },
{ "GMT-4", 4 },
{ "GMT-5", 5 },
{ "GMT-6", 6 },
{ "GMT-7", 7 },
{ "GMT-8", 8 },
{ "GMT-9", 9 },
{ "GMT-10", 10 },
{ "GMT-11", 11 },
{ "GMT-12", 12 },
{ "GMT-13", 13 },
{ "GMT-14", 14 },
{ NULL, 0 }
};
xs_list *xs_tz_list(void)
/* returns the list of supported timezones */
{
xs_list *l = xs_list_new();
for (int n = 0; xs_tz[n].tz != NULL; n++)
l = xs_list_append(l, xs_tz[n].tz);
return l;
}
int xs_tz_offset(const char *tz)
/* returns the offset in seconds from the specified Time Zone to UTC */
{
for (int n = 0; xs_tz[n].tz != NULL; n++) {
if (strcmp(xs_tz[n].tz, tz) == 0)
return xs_tz[n].h_offset * 3600;
}
return 0;
}
#endif /* XS_IMPLEMENTATION */
#endif /* _XS_TIME_H */
|