This source file includes following definitions.
- crm_get_utc_time
- crm_time_new
- crm_time_new_undefined
- crm_time_is_defined
- crm_time_free
- year_days
- crm_time_january1_weekday
- crm_time_weeks_in_year
- crm_time_days_in_month
- crm_time_leapyear
- get_ordinal_days
- crm_time_log_alias
- crm_time_get_sec
- crm_time_get_timeofday
- crm_time_get_timezone
- crm_time_get_seconds
- crm_time_get_seconds_since_epoch
- crm_time_get_gregorian
- crm_time_get_ordinal
- crm_time_get_isoweek
- crm_duration_as_string
- crm_time_as_string
- crm_time_parse_sec
- crm_time_parse_offset
- crm_time_parse
- parse_date
- parse_int
- crm_time_parse_duration
- crm_time_parse_period
- crm_time_free_period
- crm_time_set
- ha_set_tm_time
- crm_time_set_timet
- pcmk_copy_time
- crm_time_add
- crm_time_calculate_duration
- crm_time_subtract
- crm_time_check
- crm_time_compare
- crm_time_add_seconds
- crm_time_add_days
- crm_time_add_months
- crm_time_add_minutes
- crm_time_add_hours
- crm_time_add_weeks
- crm_time_add_years
- ha_get_tm_time
- pcmk__time_hr_convert
- pcmk__time_set_hr_dt
- pcmk__time_timeval_hr_convert
- pcmk__time_hr_new
- pcmk__time_hr_free
- pcmk__time_format_hr
- pcmk__epoch2str
- pcmk__readable_interval
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 #include <crm_internal.h>
17 #include <crm/crm.h>
18 #include <time.h>
19 #include <ctype.h>
20 #include <string.h>
21 #include <stdbool.h>
22 #include <crm/common/iso8601.h>
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39 #if defined(HAVE_STRUCT_TM_TM_GMTOFF)
40 # define GMTOFF(tm) ((tm)->tm_gmtoff)
41 #else
42
43 # define GMTOFF(tm) (-timezone+daylight)
44 #endif
45
46 #define HOUR_SECONDS (60 * 60)
47 #define DAY_SECONDS (HOUR_SECONDS * 24)
48
49
50 struct crm_time_s {
51 int years;
52 int months;
53 int days;
54 int seconds;
55 int offset;
56 bool duration;
57 };
58
59 static crm_time_t *parse_date(const char *date_str);
60
61 static crm_time_t *
62 crm_get_utc_time(crm_time_t *dt)
63 {
64 crm_time_t *utc = NULL;
65
66 if (dt == NULL) {
67 errno = EINVAL;
68 return NULL;
69 }
70
71 utc = crm_time_new_undefined();
72 utc->years = dt->years;
73 utc->days = dt->days;
74 utc->seconds = dt->seconds;
75 utc->offset = 0;
76
77 if (dt->offset) {
78 crm_time_add_seconds(utc, -dt->offset);
79 } else {
80
81 utc->months = dt->months;
82 }
83
84 crm_time_log(LOG_TRACE, "utc-source", dt,
85 crm_time_log_date | crm_time_log_timeofday | crm_time_log_with_timezone);
86 crm_time_log(LOG_TRACE, "utc-target", utc,
87 crm_time_log_date | crm_time_log_timeofday | crm_time_log_with_timezone);
88 return utc;
89 }
90
91 crm_time_t *
92 crm_time_new(const char *date_time)
93 {
94 time_t tm_now;
95 crm_time_t *dt = NULL;
96
97 tzset();
98 if (date_time == NULL) {
99 tm_now = time(NULL);
100 dt = crm_time_new_undefined();
101 crm_time_set_timet(dt, &tm_now);
102 } else {
103 dt = parse_date(date_time);
104 }
105 return dt;
106 }
107
108
109
110
111
112
113
114
115 crm_time_t *
116 crm_time_new_undefined()
117 {
118 crm_time_t *result = calloc(1, sizeof(crm_time_t));
119
120 CRM_ASSERT(result != NULL);
121 return result;
122 }
123
124
125
126
127
128
129
130
131 bool
132 crm_time_is_defined(const crm_time_t *t)
133 {
134
135 return (t != NULL) && (t->years || t->months || t->days || t->seconds
136 || t->offset || t->duration);
137 }
138
139 void
140 crm_time_free(crm_time_t * dt)
141 {
142 if (dt == NULL) {
143 return;
144 }
145 free(dt);
146 }
147
148 static int
149 year_days(int year)
150 {
151 int d = 365;
152
153 if (crm_time_leapyear(year)) {
154 d++;
155 }
156 return d;
157 }
158
159
160
161
162
163
164
165
166
167 int
168 crm_time_january1_weekday(int year)
169 {
170 int YY = (year - 1) % 100;
171 int C = (year - 1) - YY;
172 int G = YY + YY / 4;
173 int jan1 = 1 + (((((C / 100) % 4) * 5) + G) % 7);
174
175 crm_trace("YY=%d, C=%d, G=%d", YY, C, G);
176 crm_trace("January 1 %.4d: %d", year, jan1);
177 return jan1;
178 }
179
180 int
181 crm_time_weeks_in_year(int year)
182 {
183 int weeks = 52;
184 int jan1 = crm_time_january1_weekday(year);
185
186
187 if (jan1 == 4) {
188 weeks++;
189 } else {
190 jan1 = crm_time_january1_weekday(year + 1);
191
192 if (jan1 == 5) {
193 weeks++;
194 }
195
196 }
197 return weeks;
198 }
199
200
201 static int month_days[13] = {
202 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31, 29
203 };
204
205
206
207
208
209
210
211
212
213 int
214 crm_time_days_in_month(int month, int year)
215 {
216 if ((month < 1) || (month > 12)) {
217 return 0;
218 }
219 if ((month == 2) && crm_time_leapyear(year)) {
220 month = 13;
221 }
222 return month_days[month - 1];
223 }
224
225 bool
226 crm_time_leapyear(int year)
227 {
228 gboolean is_leap = FALSE;
229
230 if (year % 4 == 0) {
231 is_leap = TRUE;
232 }
233 if (year % 100 == 0 && year % 400 != 0) {
234 is_leap = FALSE;
235 }
236 return is_leap;
237 }
238
239 static uint32_t
240 get_ordinal_days(uint32_t y, uint32_t m, uint32_t d)
241 {
242 int lpc;
243
244 for (lpc = 1; lpc < m; lpc++) {
245 d += crm_time_days_in_month(lpc, y);
246 }
247 return d;
248 }
249
250 void
251 crm_time_log_alias(int log_level, const char *file, const char *function, int line,
252 const char *prefix, crm_time_t * date_time, int flags)
253 {
254 char *date_s = crm_time_as_string(date_time, flags);
255
256 if (log_level == LOG_STDOUT) {
257 printf("%s%s%s\n",
258 (prefix? prefix : ""), (prefix? ": " : ""), date_s);
259 } else {
260 do_crm_log_alias(log_level, file, function, line, "%s%s%s",
261 (prefix? prefix : ""), (prefix? ": " : ""), date_s);
262 }
263 free(date_s);
264 }
265
266 static void
267 crm_time_get_sec(int sec, uint * h, uint * m, uint * s)
268 {
269 uint hours, minutes, seconds;
270
271 if (sec < 0) {
272 seconds = 0 - sec;
273 } else {
274 seconds = sec;
275 }
276
277 hours = seconds / HOUR_SECONDS;
278 seconds -= HOUR_SECONDS * hours;
279
280 minutes = seconds / 60;
281 seconds -= 60 * minutes;
282
283 crm_trace("%d == %.2d:%.2d:%.2d", sec, hours, minutes, seconds);
284
285 *h = hours;
286 *m = minutes;
287 *s = seconds;
288 }
289
290 int
291 crm_time_get_timeofday(crm_time_t * dt, uint * h, uint * m, uint * s)
292 {
293 crm_time_get_sec(dt->seconds, h, m, s);
294 return TRUE;
295 }
296
297 int
298 crm_time_get_timezone(crm_time_t * dt, uint * h, uint * m)
299 {
300 uint s;
301
302 crm_time_get_sec(dt->seconds, h, m, &s);
303 return TRUE;
304 }
305
306 long long
307 crm_time_get_seconds(crm_time_t * dt)
308 {
309 int lpc;
310 crm_time_t *utc = NULL;
311 long long in_seconds = 0;
312
313 if (dt == NULL) {
314 return 0;
315 }
316
317 utc = crm_get_utc_time(dt);
318 if (utc == NULL) {
319 return 0;
320 }
321
322 for (lpc = 1; lpc < utc->years; lpc++) {
323 long long dmax = year_days(lpc);
324
325 in_seconds += DAY_SECONDS * dmax;
326 }
327
328
329
330
331
332
333
334
335 if (utc->months > 0) {
336 in_seconds += DAY_SECONDS * 30 * (long long) (utc->months);
337 }
338
339 if (utc->days > 0) {
340 in_seconds += DAY_SECONDS * (long long) (utc->days - 1);
341 }
342 in_seconds += utc->seconds;
343
344 crm_time_free(utc);
345 return in_seconds;
346 }
347
348 #define EPOCH_SECONDS 62135596800ULL
349 long long
350 crm_time_get_seconds_since_epoch(crm_time_t * dt)
351 {
352 return (dt == NULL)? 0 : (crm_time_get_seconds(dt) - EPOCH_SECONDS);
353 }
354
355 int
356 crm_time_get_gregorian(crm_time_t * dt, uint * y, uint * m, uint * d)
357 {
358 int months = 0;
359 int days = dt->days;
360
361 if(dt->years != 0) {
362 for (months = 1; months <= 12 && days > 0; months++) {
363 int mdays = crm_time_days_in_month(months, dt->years);
364
365 if (mdays >= days) {
366 break;
367 } else {
368 days -= mdays;
369 }
370 }
371
372 } else if (dt->months) {
373
374 months = dt->months;
375
376 } else {
377
378 }
379
380 *y = dt->years;
381 *m = months;
382 *d = days;
383 crm_trace("%.4d-%.3d -> %.4d-%.2d-%.2d", dt->years, dt->days, dt->years, months, days);
384 return TRUE;
385 }
386
387 int
388 crm_time_get_ordinal(crm_time_t * dt, uint * y, uint * d)
389 {
390 *y = dt->years;
391 *d = dt->days;
392 return TRUE;
393 }
394
395 int
396 crm_time_get_isoweek(crm_time_t * dt, uint * y, uint * w, uint * d)
397 {
398
399
400
401
402 int year_num = 0;
403 int jan1 = crm_time_january1_weekday(dt->years);
404 int h = -1;
405
406 CRM_CHECK(dt->days > 0, return FALSE);
407
408
409 h = dt->days + jan1 - 1;
410 *d = 1 + ((h - 1) % 7);
411
412
413 if (dt->days <= (8 - jan1) && jan1 > 4) {
414 crm_trace("year--, jan1=%d", jan1);
415 year_num = dt->years - 1;
416 *w = crm_time_weeks_in_year(year_num);
417
418 } else {
419 year_num = dt->years;
420 }
421
422
423 if (year_num == dt->years) {
424 int dmax = year_days(year_num);
425 int correction = 4 - *d;
426
427 if ((dmax - dt->days) < correction) {
428 crm_trace("year++, jan1=%d, i=%d vs. %d", jan1, dmax - dt->days, correction);
429 year_num = dt->years + 1;
430 *w = 1;
431 }
432 }
433
434
435 if (year_num == dt->years) {
436 int j = dt->days + (7 - *d) + (jan1 - 1);
437
438 *w = j / 7;
439 if (jan1 > 4) {
440 *w -= 1;
441 }
442 }
443
444 *y = year_num;
445 crm_trace("Converted %.4d-%.3d to %.4d-W%.2d-%d", dt->years, dt->days, *y, *w, *d);
446 return TRUE;
447 }
448
449 #define DATE_MAX 128
450
451 static void
452 crm_duration_as_string(crm_time_t *dt, char *result)
453 {
454 size_t offset = 0;
455
456 if (dt->years) {
457 offset += snprintf(result + offset, DATE_MAX - offset, "%4d year%s ",
458 dt->years, pcmk__plural_s(dt->years));
459 }
460 if (dt->months) {
461 offset += snprintf(result + offset, DATE_MAX - offset, "%2d month%s ",
462 dt->months, pcmk__plural_s(dt->months));
463 }
464 if (dt->days) {
465 offset += snprintf(result + offset, DATE_MAX - offset, "%2d day%s ",
466 dt->days, pcmk__plural_s(dt->days));
467 }
468
469 if (((offset == 0) || (dt->seconds != 0))
470 && (dt->seconds > -60) && (dt->seconds < 60)) {
471 offset += snprintf(result + offset, DATE_MAX - offset, "%d second%s",
472 dt->seconds, pcmk__plural_s(dt->seconds));
473 } else if (dt->seconds) {
474 uint h = 0, m = 0, s = 0;
475
476 offset += snprintf(result + offset, DATE_MAX - offset, "%d seconds (",
477 dt->seconds);
478 crm_time_get_sec(dt->seconds, &h, &m, &s);
479 if (h) {
480 offset += snprintf(result + offset, DATE_MAX - offset, "%u hour%s%s",
481 h, pcmk__plural_s(h), ((m || s)? " " : ""));
482 }
483 if (m) {
484 offset += snprintf(result + offset, DATE_MAX - offset, "%u minute%s%s",
485 m, pcmk__plural_s(m), (s? " " : ""));
486 }
487 if (s) {
488 offset += snprintf(result + offset, DATE_MAX - offset, "%u second%s",
489 s, pcmk__plural_s(s));
490 }
491 offset += snprintf(result + offset, DATE_MAX - offset, ")");
492 }
493 }
494
495 char *
496 crm_time_as_string(crm_time_t * date_time, int flags)
497 {
498 crm_time_t *dt = NULL;
499 crm_time_t *utc = NULL;
500 char result[DATE_MAX] = { '\0', };
501 char *result_copy = NULL;
502 size_t offset = 0;
503
504
505 if (date_time && date_time->offset
506 && !pcmk_is_set(flags, crm_time_log_with_timezone)) {
507 crm_trace("UTC conversion");
508 utc = crm_get_utc_time(date_time);
509 dt = utc;
510 } else {
511 dt = date_time;
512 }
513
514 if (!crm_time_is_defined(dt)) {
515 strcpy(result, "<undefined time>");
516 goto done;
517 }
518
519
520
521 if (flags & crm_time_log_duration) {
522 crm_duration_as_string(date_time, result);
523 goto done;
524 }
525
526 if (flags & crm_time_seconds) {
527 snprintf(result, DATE_MAX, "%lld", crm_time_get_seconds(date_time));
528 goto done;
529 }
530
531 if (flags & crm_time_epoch) {
532 snprintf(result, DATE_MAX, "%lld",
533 crm_time_get_seconds_since_epoch(date_time));
534 goto done;
535 }
536
537
538
539 if (flags & crm_time_log_date) {
540 if (flags & crm_time_weeks) {
541 uint y, w, d;
542
543 if (crm_time_get_isoweek(dt, &y, &w, &d)) {
544 offset += snprintf(result + offset, DATE_MAX - offset,
545 "%u-W%.2u-%u", y, w, d);
546 }
547
548 } else if (flags & crm_time_ordinal) {
549 uint y, d;
550
551 if (crm_time_get_ordinal(dt, &y, &d)) {
552 offset += snprintf(result + offset, DATE_MAX - offset,
553 "%u-%.3u", y, d);
554 }
555
556 } else {
557 uint y, m, d;
558
559 if (crm_time_get_gregorian(dt, &y, &m, &d)) {
560 offset += snprintf(result + offset, DATE_MAX - offset,
561 "%.4u-%.2u-%.2u", y, m, d);
562 }
563 }
564 }
565
566 if (flags & crm_time_log_timeofday) {
567 uint h = 0, m = 0, s = 0;
568
569 if (offset > 0) {
570 offset += snprintf(result + offset, DATE_MAX - offset, " ");
571 }
572
573 if (crm_time_get_timeofday(dt, &h, &m, &s)) {
574 offset += snprintf(result + offset, DATE_MAX - offset,
575 "%.2u:%.2u:%.2u", h, m, s);
576 }
577
578 if ((flags & crm_time_log_with_timezone) && (dt->offset != 0)) {
579 crm_time_get_sec(dt->offset, &h, &m, &s);
580 offset += snprintf(result + offset, DATE_MAX - offset,
581 " %c%.2u:%.2u",
582 ((dt->offset < 0)? '-' : '+'), h, m);
583 } else {
584 offset += snprintf(result + offset, DATE_MAX - offset, "Z");
585 }
586 }
587
588 done:
589 crm_time_free(utc);
590
591 result_copy = strdup(result);
592 CRM_ASSERT(result_copy != NULL);
593 return result_copy;
594 }
595
596
597
598
599
600
601
602
603
604
605
606
607 static bool
608 crm_time_parse_sec(const char *time_str, int *result)
609 {
610 int rc;
611 uint hour = 0;
612 uint minute = 0;
613 uint second = 0;
614
615 *result = 0;
616
617
618 rc = sscanf(time_str, "%d:%d:%d", &hour, &minute, &second);
619 if (rc == 1) {
620 rc = sscanf(time_str, "%2d%2d%2d", &hour, &minute, &second);
621 }
622 if (rc == 0) {
623 crm_err("%s is not a valid ISO 8601 time specification", time_str);
624 errno = EINVAL;
625 return FALSE;
626 }
627
628 crm_trace("Got valid time: %.2d:%.2d:%.2d", hour, minute, second);
629
630 if ((hour == 24) && (minute == 0) && (second == 0)) {
631
632 } else if (hour >= 24) {
633 crm_err("%s is not a valid ISO 8601 time specification "
634 "because %d is not a valid hour", time_str, hour);
635 errno = EINVAL;
636 return FALSE;
637 }
638 if (minute >= 60) {
639 crm_err("%s is not a valid ISO 8601 time specification "
640 "because %d is not a valid minute", time_str, minute);
641 errno = EINVAL;
642 return FALSE;
643 }
644 if (second >= 60) {
645 crm_err("%s is not a valid ISO 8601 time specification "
646 "because %d is not a valid second", time_str, second);
647 errno = EINVAL;
648 return FALSE;
649 }
650
651 *result = (hour * HOUR_SECONDS) + (minute * 60) + second;
652 return TRUE;
653 }
654
655 static bool
656 crm_time_parse_offset(const char *offset_str, int *offset)
657 {
658 tzset();
659
660 if (offset_str == NULL) {
661
662 #if defined(HAVE_STRUCT_TM_TM_GMTOFF)
663 time_t now = time(NULL);
664 struct tm *now_tm = localtime(&now);
665 #endif
666 int h_offset = GMTOFF(now_tm) / HOUR_SECONDS;
667 int m_offset = (GMTOFF(now_tm) - (HOUR_SECONDS * h_offset)) / 60;
668
669 if (h_offset < 0 && m_offset < 0) {
670 m_offset = 0 - m_offset;
671 }
672 *offset = (HOUR_SECONDS * h_offset) + (60 * m_offset);
673 return TRUE;
674 }
675
676 if (offset_str[0] == 'Z') {
677 *offset = 0;
678 return TRUE;
679 }
680
681 *offset = 0;
682 if ((offset_str[0] == '+') || (offset_str[0] == '-')
683 || isdigit((int)offset_str[0])) {
684
685 gboolean negate = FALSE;
686
687 if (offset_str[0] == '+') {
688 offset_str++;
689 } else if (offset_str[0] == '-') {
690 negate = TRUE;
691 offset_str++;
692 }
693 if (crm_time_parse_sec(offset_str, offset) == FALSE) {
694 return FALSE;
695 }
696 if (negate) {
697 *offset = 0 - *offset;
698 }
699 }
700 return TRUE;
701 }
702
703
704
705
706
707
708
709
710
711
712
713 static bool
714 crm_time_parse(const char *time_str, crm_time_t *a_time)
715 {
716 uint h, m, s;
717 char *offset_s = NULL;
718
719 tzset();
720
721 if (time_str) {
722 if (crm_time_parse_sec(time_str, &(a_time->seconds)) == FALSE) {
723 return FALSE;
724 }
725 offset_s = strstr(time_str, "Z");
726 if (offset_s == NULL) {
727 offset_s = strstr(time_str, " ");
728 if (offset_s) {
729 while (isspace(offset_s[0])) {
730 offset_s++;
731 }
732 }
733 }
734 }
735
736 if (crm_time_parse_offset(offset_s, &(a_time->offset)) == FALSE) {
737 return FALSE;
738 }
739 crm_time_get_sec(a_time->offset, &h, &m, &s);
740 crm_trace("Got tz: %c%2.d:%.2d", ((a_time->offset < 0)? '-' : '+'), h, m);
741
742 if (a_time->seconds == DAY_SECONDS) {
743
744 a_time->seconds = 0;
745 crm_time_add_days(a_time, 1);
746 }
747 return TRUE;
748 }
749
750
751
752
753
754
755
756
757
758 static crm_time_t *
759 parse_date(const char *date_str)
760 {
761 const char *time_s = NULL;
762 crm_time_t *dt = NULL;
763
764 int year = 0;
765 int month = 0;
766 int week = 0;
767 int day = 0;
768 int rc = 0;
769
770 if (pcmk__str_empty(date_str)) {
771 crm_err("No ISO 8601 date/time specification given");
772 goto invalid;
773 }
774
775 if ((date_str[0] == 'T') || (date_str[2] == ':')) {
776
777 dt = crm_time_new(NULL);
778 if (date_str[0] == 'T') {
779 time_s = date_str + 1;
780 } else {
781 time_s = date_str;
782 }
783 goto parse_time;
784 }
785
786 dt = crm_time_new_undefined();
787
788 if (!strncasecmp("epoch", date_str, 5)
789 && ((date_str[5] == '\0') || (date_str[5] == '/') || isspace(date_str[5]))) {
790 dt->days = 1;
791 dt->years = 1970;
792 crm_time_log(LOG_TRACE, "Unpacked", dt, crm_time_log_date | crm_time_log_timeofday);
793 return dt;
794 }
795
796
797 rc = sscanf(date_str, "%d-%d-%d", &year, &month, &day);
798 if (rc == 1) {
799
800 rc = sscanf(date_str, "%4d%2d%2d", &year, &month, &day);
801 }
802 if (rc == 3) {
803 if (month > 12) {
804 crm_err("'%s' is not a valid ISO 8601 date/time specification "
805 "because '%d' is not a valid month", date_str, month);
806 goto invalid;
807 } else if (day > crm_time_days_in_month(month, year)) {
808 crm_err("'%s' is not a valid ISO 8601 date/time specification "
809 "because '%d' is not a valid day of the month",
810 date_str, day);
811 goto invalid;
812 } else {
813 dt->years = year;
814 dt->days = get_ordinal_days(year, month, day);
815 crm_trace("Parsed Gregorian date '%.4d-%.3d' from date string '%s'",
816 year, dt->days, date_str);
817 }
818 goto parse_time;
819 }
820
821
822 rc = sscanf(date_str, "%d-%d", &year, &day);
823 if (rc == 2) {
824 if (day > year_days(year)) {
825 crm_err("'%s' is not a valid ISO 8601 date/time specification "
826 "because '%d' is not a valid day of the year (max %d)",
827 date_str, day, year_days(year));
828 goto invalid;
829 }
830 crm_trace("Parsed ordinal year %d and days %d from date string '%s'",
831 year, day, date_str);
832 dt->days = day;
833 dt->years = year;
834 goto parse_time;
835 }
836
837
838 rc = sscanf(date_str, "%d-W%d-%d", &year, &week, &day);
839 if (rc == 3) {
840 if (week > crm_time_weeks_in_year(year)) {
841 crm_err("'%s' is not a valid ISO 8601 date/time specification "
842 "because '%d' is not a valid week of the year (max %d)",
843 date_str, week, crm_time_weeks_in_year(year));
844 goto invalid;
845 } else if (day < 1 || day > 7) {
846 crm_err("'%s' is not a valid ISO 8601 date/time specification "
847 "because '%d' is not a valid day of the week",
848 date_str, day);
849 goto invalid;
850 } else {
851
852
853
854
855
856
857
858
859
860
861 int jan1 = crm_time_january1_weekday(year);
862
863 crm_trace("Got year %d (Jan 1 = %d), week %d, and day %d from date string '%s'",
864 year, jan1, week, day, date_str);
865
866 dt->years = year;
867 crm_time_add_days(dt, (week - 1) * 7);
868
869 if (jan1 <= 4) {
870 crm_time_add_days(dt, 1 - jan1);
871 } else {
872 crm_time_add_days(dt, 8 - jan1);
873 }
874
875 crm_time_add_days(dt, day);
876 }
877 goto parse_time;
878 }
879
880 crm_err("'%s' is not a valid ISO 8601 date/time specification", date_str);
881 goto invalid;
882
883 parse_time:
884
885 if (time_s == NULL) {
886 time_s = date_str + strspn(date_str, "0123456789-W");
887 if ((time_s[0] == ' ') || (time_s[0] == 'T')) {
888 ++time_s;
889 } else {
890 time_s = NULL;
891 }
892 }
893 if ((time_s != NULL) && (crm_time_parse(time_s, dt) == FALSE)) {
894 goto invalid;
895 }
896
897 crm_time_log(LOG_TRACE, "Unpacked", dt, crm_time_log_date | crm_time_log_timeofday);
898 if (crm_time_check(dt) == FALSE) {
899 crm_err("'%s' is not a valid ISO 8601 date/time specification",
900 date_str);
901 goto invalid;
902 }
903 return dt;
904
905 invalid:
906 crm_time_free(dt);
907 errno = EINVAL;
908 return NULL;
909 }
910
911
912
913
914
915 static int
916 parse_int(const char *str, int field_width, int upper_bound, int *result)
917 {
918 int lpc = 0;
919 int offset = 0;
920 int intermediate = 0;
921 gboolean fraction = FALSE;
922 gboolean negate = FALSE;
923
924 *result = 0;
925 if (*str == '\0') {
926 return 0;
927 }
928
929 if (str[offset] == 'T') {
930 offset++;
931 }
932
933 if (str[offset] == '.' || str[offset] == ',') {
934 fraction = TRUE;
935 field_width = -1;
936 offset++;
937 } else if (str[offset] == '-') {
938 negate = TRUE;
939 offset++;
940 } else if (str[offset] == '+' || str[offset] == ':') {
941 offset++;
942 }
943
944 for (; (fraction || lpc < field_width) && isdigit((int)str[offset]); lpc++) {
945 if (fraction) {
946 intermediate = (str[offset] - '0') / (10 ^ lpc);
947 } else {
948 *result *= 10;
949 intermediate = str[offset] - '0';
950 }
951 *result += intermediate;
952 offset++;
953 }
954 if (fraction) {
955 *result = (int)(*result * upper_bound);
956
957 } else if (upper_bound > 0 && *result > upper_bound) {
958 *result = upper_bound;
959 }
960 if (negate) {
961 *result = 0 - *result;
962 }
963 if (lpc > 0) {
964 crm_trace("Found int: %d. Stopped at str[%d]='%c'", *result, lpc, str[lpc]);
965 return offset;
966 }
967 return 0;
968 }
969
970
971
972
973
974
975
976
977
978
979
980
981 crm_time_t *
982 crm_time_parse_duration(const char *period_s)
983 {
984 gboolean is_time = FALSE;
985 crm_time_t *diff = NULL;
986
987 if (pcmk__str_empty(period_s)) {
988 crm_err("No ISO 8601 time duration given");
989 goto invalid;
990 }
991 if (period_s[0] != 'P') {
992 crm_err("'%s' is not a valid ISO 8601 time duration "
993 "because it does not start with a 'P'", period_s);
994 goto invalid;
995 }
996 if ((period_s[1] == '\0') || isspace(period_s[1])) {
997 crm_err("'%s' is not a valid ISO 8601 time duration "
998 "because nothing follows 'P'", period_s);
999 goto invalid;
1000 }
1001
1002 diff = crm_time_new_undefined();
1003 diff->duration = TRUE;
1004
1005 for (const char *current = period_s + 1;
1006 current[0] && (current[0] != '/') && !isspace(current[0]);
1007 ++current) {
1008
1009 int an_int = 0, rc;
1010
1011 if (current[0] == 'T') {
1012
1013
1014
1015
1016 is_time = TRUE;
1017 continue;
1018 }
1019
1020
1021 rc = parse_int(current, 10, 0, &an_int);
1022 if (rc == 0) {
1023 crm_err("'%s' is not a valid ISO 8601 time duration "
1024 "because no integer at '%s'", period_s, current);
1025 goto invalid;
1026 }
1027 current += rc;
1028
1029
1030 switch (current[0]) {
1031 case 'Y':
1032 diff->years = an_int;
1033 break;
1034 case 'M':
1035 if (is_time) {
1036
1037 diff->seconds += an_int * 60;
1038 } else {
1039 diff->months = an_int;
1040 }
1041 break;
1042 case 'W':
1043 diff->days += an_int * 7;
1044 break;
1045 case 'D':
1046 diff->days += an_int;
1047 break;
1048 case 'H':
1049 diff->seconds += an_int * HOUR_SECONDS;
1050 break;
1051 case 'S':
1052 diff->seconds += an_int;
1053 break;
1054 case '\0':
1055 crm_err("'%s' is not a valid ISO 8601 time duration "
1056 "because no units after %d", period_s, an_int);
1057 goto invalid;
1058 default:
1059 crm_err("'%s' is not a valid ISO 8601 time duration "
1060 "because '%c' is not a valid time unit",
1061 period_s, current[0]);
1062 goto invalid;
1063 }
1064 }
1065
1066 if (!crm_time_is_defined(diff)) {
1067 crm_err("'%s' is not a valid ISO 8601 time duration "
1068 "because no amounts and units given", period_s);
1069 goto invalid;
1070 }
1071 return diff;
1072
1073 invalid:
1074 crm_time_free(diff);
1075 errno = EINVAL;
1076 return NULL;
1077 }
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089 crm_time_period_t *
1090 crm_time_parse_period(const char *period_str)
1091 {
1092 const char *original = period_str;
1093 crm_time_period_t *period = NULL;
1094
1095 if (pcmk__str_empty(period_str)) {
1096 crm_err("No ISO 8601 time period given");
1097 goto invalid;
1098 }
1099
1100 tzset();
1101 period = calloc(1, sizeof(crm_time_period_t));
1102 CRM_ASSERT(period != NULL);
1103
1104 if (period_str[0] == 'P') {
1105 period->diff = crm_time_parse_duration(period_str);
1106 if (period->diff == NULL) {
1107 goto error;
1108 }
1109 } else {
1110 period->start = parse_date(period_str);
1111 if (period->start == NULL) {
1112 goto error;
1113 }
1114 }
1115
1116 period_str = strstr(original, "/");
1117 if (period_str) {
1118 ++period_str;
1119 if (period_str[0] == 'P') {
1120 if (period->diff != NULL) {
1121 crm_err("'%s' is not a valid ISO 8601 time period "
1122 "because it has two durations",
1123 original);
1124 goto invalid;
1125 }
1126 period->diff = crm_time_parse_duration(period_str);
1127 if (period->diff == NULL) {
1128 goto error;
1129 }
1130 } else {
1131 period->end = parse_date(period_str);
1132 if (period->end == NULL) {
1133 goto error;
1134 }
1135 }
1136
1137 } else if (period->diff != NULL) {
1138
1139 period->start = crm_time_new(NULL);
1140
1141 } else {
1142
1143 crm_err("'%s' is not a valid ISO 8601 time period "
1144 "because it has no duration or ending time",
1145 original);
1146 goto invalid;
1147 }
1148
1149 if (period->start == NULL) {
1150 period->start = crm_time_subtract(period->end, period->diff);
1151
1152 } else if (period->end == NULL) {
1153 period->end = crm_time_add(period->start, period->diff);
1154 }
1155
1156 if (crm_time_check(period->start) == FALSE) {
1157 crm_err("'%s' is not a valid ISO 8601 time period "
1158 "because the start is invalid", period_str);
1159 goto invalid;
1160 }
1161 if (crm_time_check(period->end) == FALSE) {
1162 crm_err("'%s' is not a valid ISO 8601 time period "
1163 "because the end is invalid", period_str);
1164 goto invalid;
1165 }
1166 return period;
1167
1168 invalid:
1169 errno = EINVAL;
1170 error:
1171 crm_time_free_period(period);
1172 return NULL;
1173 }
1174
1175
1176
1177
1178
1179
1180 void
1181 crm_time_free_period(crm_time_period_t *period)
1182 {
1183 if (period) {
1184 crm_time_free(period->start);
1185 crm_time_free(period->end);
1186 crm_time_free(period->diff);
1187 free(period);
1188 }
1189 }
1190
1191 void
1192 crm_time_set(crm_time_t * target, crm_time_t * source)
1193 {
1194 crm_trace("target=%p, source=%p", target, source);
1195
1196 CRM_CHECK(target != NULL && source != NULL, return);
1197
1198 target->years = source->years;
1199 target->days = source->days;
1200 target->months = source->months;
1201 target->seconds = source->seconds;
1202 target->offset = source->offset;
1203
1204 crm_time_log(LOG_TRACE, "source", source,
1205 crm_time_log_date | crm_time_log_timeofday | crm_time_log_with_timezone);
1206 crm_time_log(LOG_TRACE, "target", target,
1207 crm_time_log_date | crm_time_log_timeofday | crm_time_log_with_timezone);
1208 }
1209
1210 static void
1211 ha_set_tm_time(crm_time_t * target, struct tm *source)
1212 {
1213 int h_offset = 0;
1214 int m_offset = 0;
1215
1216
1217 target->years = 0;
1218 target->months = 0;
1219 target->days = 0;
1220 target->seconds = 0;
1221 target->offset = 0;
1222 target->duration = FALSE;
1223
1224 if (source->tm_year > 0) {
1225
1226 target->years = 1900 + source->tm_year;
1227 }
1228
1229 if (source->tm_yday >= 0) {
1230
1231 target->days = 1 + source->tm_yday;
1232 }
1233
1234 if (source->tm_hour >= 0) {
1235 target->seconds += HOUR_SECONDS * source->tm_hour;
1236 }
1237 if (source->tm_min >= 0) {
1238 target->seconds += 60 * source->tm_min;
1239 }
1240 if (source->tm_sec >= 0) {
1241 target->seconds += source->tm_sec;
1242 }
1243
1244
1245 h_offset = GMTOFF(source) / HOUR_SECONDS;
1246 m_offset = (GMTOFF(source) - (HOUR_SECONDS * h_offset)) / 60;
1247 crm_trace("Time offset is %lds (%.2d:%.2d)",
1248 GMTOFF(source), h_offset, m_offset);
1249
1250 target->offset += HOUR_SECONDS * h_offset;
1251 target->offset += 60 * m_offset;
1252 }
1253
1254 void
1255 crm_time_set_timet(crm_time_t * target, time_t * source)
1256 {
1257 ha_set_tm_time(target, localtime(source));
1258 }
1259
1260 crm_time_t *
1261 pcmk_copy_time(crm_time_t *source)
1262 {
1263 crm_time_t *target = crm_time_new_undefined();
1264
1265 crm_time_set(target, source);
1266 return target;
1267 }
1268
1269 crm_time_t *
1270 crm_time_add(crm_time_t * dt, crm_time_t * value)
1271 {
1272 crm_time_t *utc = NULL;
1273 crm_time_t *answer = NULL;
1274
1275 if ((dt == NULL) || (value == NULL)) {
1276 errno = EINVAL;
1277 return NULL;
1278 }
1279
1280 answer = pcmk_copy_time(dt);
1281
1282 utc = crm_get_utc_time(value);
1283 if (utc == NULL) {
1284 crm_time_free(answer);
1285 return NULL;
1286 }
1287
1288 answer->years += utc->years;
1289 crm_time_add_months(answer, utc->months);
1290 crm_time_add_days(answer, utc->days);
1291 crm_time_add_seconds(answer, utc->seconds);
1292
1293 crm_time_free(utc);
1294 return answer;
1295 }
1296
1297 crm_time_t *
1298 crm_time_calculate_duration(crm_time_t * dt, crm_time_t * value)
1299 {
1300 crm_time_t *utc = NULL;
1301 crm_time_t *answer = NULL;
1302
1303 if ((dt == NULL) || (value == NULL)) {
1304 errno = EINVAL;
1305 return NULL;
1306 }
1307
1308 utc = crm_get_utc_time(value);
1309 if (utc == NULL) {
1310 return NULL;
1311 }
1312
1313 answer = crm_get_utc_time(dt);
1314 if (answer == NULL) {
1315 crm_time_free(utc);
1316 return NULL;
1317 }
1318 answer->duration = TRUE;
1319
1320 answer->years -= utc->years;
1321 if(utc->months != 0) {
1322 crm_time_add_months(answer, -utc->months);
1323 }
1324 crm_time_add_days(answer, -utc->days);
1325 crm_time_add_seconds(answer, -utc->seconds);
1326
1327 crm_time_free(utc);
1328 return answer;
1329 }
1330
1331 crm_time_t *
1332 crm_time_subtract(crm_time_t * dt, crm_time_t * value)
1333 {
1334 crm_time_t *utc = NULL;
1335 crm_time_t *answer = NULL;
1336
1337 if ((dt == NULL) || (value == NULL)) {
1338 errno = EINVAL;
1339 return NULL;
1340 }
1341
1342 utc = crm_get_utc_time(value);
1343 if (utc == NULL) {
1344 return NULL;
1345 }
1346
1347 answer = pcmk_copy_time(dt);
1348 answer->years -= utc->years;
1349 if(utc->months != 0) {
1350 crm_time_add_months(answer, -utc->months);
1351 }
1352 crm_time_add_days(answer, -utc->days);
1353 crm_time_add_seconds(answer, -utc->seconds);
1354
1355 return answer;
1356 }
1357
1358
1359
1360
1361
1362
1363
1364
1365 bool
1366 crm_time_check(crm_time_t * dt)
1367 {
1368 return (dt != NULL)
1369 && (dt->days > 0) && (dt->days <= year_days(dt->years))
1370 && (dt->seconds >= 0) && (dt->seconds < DAY_SECONDS);
1371 }
1372
1373 #define do_cmp_field(l, r, field) \
1374 if(rc == 0) { \
1375 if(l->field > r->field) { \
1376 crm_trace("%s: %d > %d", \
1377 #field, l->field, r->field); \
1378 rc = 1; \
1379 } else if(l->field < r->field) { \
1380 crm_trace("%s: %d < %d", \
1381 #field, l->field, r->field); \
1382 rc = -1; \
1383 } \
1384 }
1385
1386 int
1387 crm_time_compare(crm_time_t *a, crm_time_t *b)
1388 {
1389 int rc = 0;
1390 crm_time_t *t1 = crm_get_utc_time(a);
1391 crm_time_t *t2 = crm_get_utc_time(b);
1392
1393 if ((t1 == NULL) && (t2 == NULL)) {
1394 rc = 0;
1395 } else if (t1 == NULL) {
1396 rc = -1;
1397 } else if (t2 == NULL) {
1398 rc = 1;
1399 } else {
1400 do_cmp_field(t1, t2, years);
1401 do_cmp_field(t1, t2, days);
1402 do_cmp_field(t1, t2, seconds);
1403 }
1404
1405 crm_time_free(t1);
1406 crm_time_free(t2);
1407 return rc;
1408 }
1409
1410
1411
1412
1413
1414
1415
1416 void
1417 crm_time_add_seconds(crm_time_t *a_time, int extra)
1418 {
1419 int days = 0;
1420
1421 crm_trace("Adding %d seconds to %d (max=%d)",
1422 extra, a_time->seconds, DAY_SECONDS);
1423 a_time->seconds += extra;
1424 days = a_time->seconds / DAY_SECONDS;
1425 a_time->seconds %= DAY_SECONDS;
1426
1427
1428 if (a_time->seconds < 0) {
1429 a_time->seconds += DAY_SECONDS;
1430 --days;
1431 }
1432
1433 crm_time_add_days(a_time, days);
1434 }
1435
1436 void
1437 crm_time_add_days(crm_time_t * a_time, int extra)
1438 {
1439 int lower_bound = 1;
1440 int ydays = crm_time_leapyear(a_time->years) ? 366 : 365;
1441
1442 crm_trace("Adding %d days to %.4d-%.3d", extra, a_time->years, a_time->days);
1443
1444 a_time->days += extra;
1445 while (a_time->days > ydays) {
1446 a_time->years++;
1447 a_time->days -= ydays;
1448 ydays = crm_time_leapyear(a_time->years) ? 366 : 365;
1449 }
1450
1451 if(a_time->duration) {
1452 lower_bound = 0;
1453 }
1454
1455 while (a_time->days < lower_bound) {
1456 a_time->years--;
1457 a_time->days += crm_time_leapyear(a_time->years) ? 366 : 365;
1458 }
1459 }
1460
1461 void
1462 crm_time_add_months(crm_time_t * a_time, int extra)
1463 {
1464 int lpc;
1465 uint32_t y, m, d, dmax;
1466
1467 crm_time_get_gregorian(a_time, &y, &m, &d);
1468 crm_trace("Adding %d months to %.4d-%.2d-%.2d", extra, y, m, d);
1469
1470 if (extra > 0) {
1471 for (lpc = extra; lpc > 0; lpc--) {
1472 m++;
1473 if (m == 13) {
1474 m = 1;
1475 y++;
1476 }
1477 }
1478 } else {
1479 for (lpc = -extra; lpc > 0; lpc--) {
1480 m--;
1481 if (m == 0) {
1482 m = 12;
1483 y--;
1484 }
1485 }
1486 }
1487
1488 dmax = crm_time_days_in_month(m, y);
1489 if (dmax < d) {
1490
1491 d = dmax;
1492 }
1493
1494 crm_trace("Calculated %.4d-%.2d-%.2d", y, m, d);
1495
1496 a_time->years = y;
1497 a_time->days = get_ordinal_days(y, m, d);
1498
1499 crm_time_get_gregorian(a_time, &y, &m, &d);
1500 crm_trace("Got %.4d-%.2d-%.2d", y, m, d);
1501 }
1502
1503 void
1504 crm_time_add_minutes(crm_time_t * a_time, int extra)
1505 {
1506 crm_time_add_seconds(a_time, extra * 60);
1507 }
1508
1509 void
1510 crm_time_add_hours(crm_time_t * a_time, int extra)
1511 {
1512 crm_time_add_seconds(a_time, extra * HOUR_SECONDS);
1513 }
1514
1515 void
1516 crm_time_add_weeks(crm_time_t * a_time, int extra)
1517 {
1518 crm_time_add_days(a_time, extra * 7);
1519 }
1520
1521 void
1522 crm_time_add_years(crm_time_t * a_time, int extra)
1523 {
1524 a_time->years += extra;
1525 }
1526
1527 static void
1528 ha_get_tm_time( struct tm *target, crm_time_t *source)
1529 {
1530 *target = (struct tm) {
1531 .tm_year = source->years - 1900,
1532 .tm_mday = source->days,
1533 .tm_sec = source->seconds % 60,
1534 .tm_min = ( source->seconds / 60 ) % 60,
1535 .tm_hour = source->seconds / HOUR_SECONDS,
1536 .tm_isdst = -1,
1537
1538 #if defined(HAVE_STRUCT_TM_TM_GMTOFF)
1539 .tm_gmtoff = source->offset
1540 #endif
1541 };
1542 mktime(target);
1543 }
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555 pcmk__time_hr_t *
1556 pcmk__time_hr_convert(pcmk__time_hr_t *target, crm_time_t *dt)
1557 {
1558 pcmk__time_hr_t *hr_dt = NULL;
1559
1560 if (dt) {
1561 hr_dt = target?target:calloc(1, sizeof(pcmk__time_hr_t));
1562 CRM_ASSERT(hr_dt != NULL);
1563 *hr_dt = (pcmk__time_hr_t) {
1564 .years = dt->years,
1565 .months = dt->months,
1566 .days = dt->days,
1567 .seconds = dt->seconds,
1568 .offset = dt->offset,
1569 .duration = dt->duration
1570 };
1571 }
1572
1573 return hr_dt;
1574 }
1575
1576 void
1577 pcmk__time_set_hr_dt(crm_time_t *target, pcmk__time_hr_t *hr_dt)
1578 {
1579 CRM_ASSERT((hr_dt) && (target));
1580 *target = (crm_time_t) {
1581 .years = hr_dt->years,
1582 .months = hr_dt->months,
1583 .days = hr_dt->days,
1584 .seconds = hr_dt->seconds,
1585 .offset = hr_dt->offset,
1586 .duration = hr_dt->duration
1587 };
1588 }
1589
1590 pcmk__time_hr_t *
1591 pcmk__time_timeval_hr_convert(pcmk__time_hr_t *target, struct timeval *tv)
1592 {
1593 crm_time_t dt;
1594 pcmk__time_hr_t *ret;
1595
1596 crm_time_set_timet(&dt, &tv->tv_sec);
1597 ret = pcmk__time_hr_convert(target, &dt);
1598 if (ret) {
1599 ret->useconds = tv->tv_usec;
1600 }
1601 return ret;
1602 }
1603
1604 pcmk__time_hr_t *
1605 pcmk__time_hr_new(const char *date_time)
1606 {
1607 pcmk__time_hr_t *hr_dt = NULL;
1608 struct timeval tv_now;
1609
1610 if (!date_time) {
1611 if (gettimeofday(&tv_now, NULL) == 0) {
1612 hr_dt = pcmk__time_timeval_hr_convert(NULL, &tv_now);
1613 }
1614 } else {
1615 crm_time_t *dt;
1616
1617 dt = parse_date(date_time);
1618 hr_dt = pcmk__time_hr_convert(NULL, dt);
1619 crm_time_free(dt);
1620 }
1621 return hr_dt;
1622 }
1623
1624 void
1625 pcmk__time_hr_free(pcmk__time_hr_t * hr_dt)
1626 {
1627 free(hr_dt);
1628 }
1629
1630 char *
1631 pcmk__time_format_hr(const char *format, pcmk__time_hr_t * hr_dt)
1632 {
1633 const char *mark_s;
1634 int max = 128, scanned_pos = 0, printed_pos = 0, fmt_pos = 0,
1635 date_len = 0, nano_digits = 0;
1636 char nano_s[10], date_s[max+1], nanofmt_s[5] = "%", *tmp_fmt_s;
1637 struct tm tm;
1638 crm_time_t dt;
1639
1640 if (!format) {
1641 return NULL;
1642 }
1643 pcmk__time_set_hr_dt(&dt, hr_dt);
1644 ha_get_tm_time(&tm, &dt);
1645 sprintf(nano_s, "%06d000", hr_dt->useconds);
1646
1647 while ((format[scanned_pos]) != '\0') {
1648 mark_s = strchr(&format[scanned_pos], '%');
1649 if (mark_s) {
1650 int fmt_len = 1;
1651
1652 fmt_pos = mark_s - format;
1653 while ((format[fmt_pos+fmt_len] != '\0') &&
1654 (format[fmt_pos+fmt_len] >= '0') &&
1655 (format[fmt_pos+fmt_len] <= '9')) {
1656 fmt_len++;
1657 }
1658 scanned_pos = fmt_pos + fmt_len + 1;
1659 if (format[fmt_pos+fmt_len] == 'N') {
1660 nano_digits = atoi(&format[fmt_pos+1]);
1661 nano_digits = (nano_digits > 6)?6:nano_digits;
1662 nano_digits = (nano_digits < 0)?0:nano_digits;
1663 sprintf(&nanofmt_s[1], ".%ds", nano_digits);
1664 } else {
1665 if (format[scanned_pos] != '\0') {
1666 continue;
1667 }
1668 fmt_pos = scanned_pos;
1669 }
1670 } else {
1671 scanned_pos = strlen(format);
1672 fmt_pos = scanned_pos;
1673 }
1674 tmp_fmt_s = strndup(&format[printed_pos], fmt_pos - printed_pos);
1675 #ifdef GCC_FORMAT_NONLITERAL_CHECKING_ENABLED
1676 #pragma GCC diagnostic push
1677 #pragma GCC diagnostic ignored "-Wformat-nonliteral"
1678 #endif
1679 date_len += strftime(&date_s[date_len], max-date_len, tmp_fmt_s, &tm);
1680 #ifdef GCC_FORMAT_NONLITERAL_CHECKING_ENABLED
1681 #pragma GCC diagnostic pop
1682 #endif
1683 printed_pos = scanned_pos;
1684 free(tmp_fmt_s);
1685 if (nano_digits) {
1686 #ifdef GCC_FORMAT_NONLITERAL_CHECKING_ENABLED
1687 #pragma GCC diagnostic push
1688 #pragma GCC diagnostic ignored "-Wformat-nonliteral"
1689 #endif
1690 date_len += snprintf(&date_s[date_len], max-date_len,
1691 nanofmt_s, nano_s);
1692 #ifdef GCC_FORMAT_NONLITERAL_CHECKING_ENABLED
1693 #pragma GCC diagnostic pop
1694 #endif
1695 nano_digits = 0;
1696 }
1697 }
1698
1699 return (date_len == 0)?NULL:strdup(date_s);
1700 }
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714 const char *
1715 pcmk__epoch2str(time_t *when)
1716 {
1717 char *since_epoch = NULL;
1718
1719 if (when == NULL) {
1720 time_t a_time = time(NULL);
1721
1722 if (a_time == (time_t) -1) {
1723 return NULL;
1724 } else {
1725 since_epoch = ctime(&a_time);
1726 }
1727 } else {
1728 since_epoch = ctime(when);
1729 }
1730
1731 if (since_epoch == NULL) {
1732 return NULL;
1733 } else {
1734 return pcmk__trim(since_epoch);
1735 }
1736 }
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749 const char *
1750 pcmk__readable_interval(guint interval_ms)
1751 {
1752 #define MS_IN_S (1000)
1753 #define MS_IN_M (MS_IN_S * 60)
1754 #define MS_IN_H (MS_IN_M * 60)
1755 #define MS_IN_D (MS_IN_H * 24)
1756 #define MAXSTR sizeof("..d..h..m..s...ms")
1757 static char str[MAXSTR] = { '\0', };
1758 int offset = 0;
1759
1760 if (interval_ms > MS_IN_D) {
1761 offset += snprintf(str + offset, MAXSTR - offset, "%ud",
1762 interval_ms / MS_IN_D);
1763 interval_ms -= (interval_ms / MS_IN_D) * MS_IN_D;
1764 }
1765 if (interval_ms > MS_IN_H) {
1766 offset += snprintf(str + offset, MAXSTR - offset, "%uh",
1767 interval_ms / MS_IN_H);
1768 interval_ms -= (interval_ms / MS_IN_H) * MS_IN_H;
1769 }
1770 if (interval_ms > MS_IN_M) {
1771 offset += snprintf(str + offset, MAXSTR - offset, "%um",
1772 interval_ms / MS_IN_M);
1773 interval_ms -= (interval_ms / MS_IN_M) * MS_IN_M;
1774 }
1775
1776
1777 if (interval_ms > MS_IN_S) {
1778 offset += snprintf(str + offset, MAXSTR - offset, "%u",
1779 interval_ms / MS_IN_S);
1780 interval_ms -= (interval_ms / MS_IN_S) * MS_IN_S;
1781 if (interval_ms > 0) {
1782 offset += snprintf(str + offset, MAXSTR - offset, ".%03u",
1783 interval_ms);
1784 }
1785 (void) snprintf(str + offset, MAXSTR - offset, "s");
1786
1787 } else if (interval_ms > 0) {
1788 (void) snprintf(str + offset, MAXSTR - offset, "%ums", interval_ms);
1789
1790 } else if (str[0] == '\0') {
1791 strcpy(str, "0s");
1792 }
1793 return str;
1794 }