0917d32013-03-04Anders Johansson // $Id$
1494372003-07-31Anders Johansson 
4eed962022-06-28Henrik Grubbström (Grubba) #if constant(roxenp)
fc10292022-06-27Henrik Grubbström (Grubba) inherit /*Variable*/.Variable;
0a29482001-08-22Per Hedbor 
5004d12001-08-22Martin Nilsson //! This class implements a scheduler widget with three main states, //! never index, index every n:th hour or index every n:th x-day at y //! o'clock. In the "index every n:th hour" case the range is 1 to 23. //! In the "index every n:th x-day at y o'clock" the n-range is //! 1 to 9, the units are day and all the weekdays. The time range for //! y is all hours in a day.
0a29482001-08-22Per Hedbor // Locale macros //<locale-token project="roxen_config"> LOCALE </locale-token> #define LOCALE(X,Y) \ ([string](mixed)Locale.translate("roxen_config",roxenp()->locale->get(),X,Y))
b525652022-06-27Henrik Grubbström (Grubba) #else // Test mode. #define LOCALE(X, Y) (Y) array(int) val = ({ }); array(int) query() { return val; } // Deterministic timezone... #define localtime(X) gmtime(X) int main(int argc, array(string) argv) { int successes; int failures; foreach(({
8fa1b62022-06-27Henrik Grubbström (Grubba)  ({ ({ 0, 2, 1, 6, 3, 0, }),
b525652022-06-27Henrik Grubbström (Grubba)  // Disabled. ({ 0, -1 }), }),
8fa1b62022-06-27Henrik Grubbström (Grubba)  ({ ({ 1, 2, 1, 6, 3, 0, }),
b525652022-06-27Henrik Grubbström (Grubba)  // Every other hour. // 2022-06-22T14:11:43 (Wed) ==> 2022-06-22T16:11:43 (Wed) ({ 1655907103, 1655914303 }), }),
8fa1b62022-06-27Henrik Grubbström (Grubba)  ({ ({ 2, 2, 1, 6, 3, 0, }),
b525652022-06-27Henrik Grubbström (Grubba)  // Every Friday at 03:00. // 2022-06-22T14:11:43 (Wed) ==> 2022-06-24T03:00:00 (Fri) ({ 1655907103, 1656039600 }), // 2022-06-23T14:11:43 (Thu) ==> 2022-06-24T03:00:00 (Fri) ({ 1655993503, 1656039600 }), // 2022-06-24T02:11:43 (Fri) ==> 2022-06-24T03:00:00 (Fri) ({ 1656036703, 1656039600 }), // 2022-06-24T03:11:43 (Fri) ==> 2022-07-01T03:00:00 (Fri) ({ 1656040303, 1656644400 }), // 2022-06-24T03:21:43 (Fri) ==> 2022-07-01T03:00:00 (Fri) ({ 1656040903, 1656644400 }), // 2022-06-24T03:51:43 (Fri) ==> 2022-07-01T03:00:00 (Fri) ({ 1656042703, 1656644400 }), // 2022-06-24T04:11:43 (Fri) ==> 2022-07-01T03:00:00 (Fri) ({ 1656043903, 1656644400 }), // 2022-06-24T14:11:43 (Fri) ==> 2022-07-01T03:00:00 (Fri) ({ 1656079903, 1656644400 }), }),
8fa1b62022-06-27Henrik Grubbström (Grubba)  ({ ({ 2, 2, 1, 6, 3, 45, }), // Every Friday at 03:45. // 2022-06-22T14:11:43 (Wed) ==> 2022-06-24T03:45:00 (Fri) ({ 1655907103, 1656042300 }), // 2022-06-23T14:11:43 (Thu) ==> 2022-06-24T03:45:00 (Fri) ({ 1655993503, 1656042300 }), // 2022-06-24T02:11:43 (Fri) ==> 2022-06-24T03:45:00 (Fri) ({ 1656036703, 1656042300 }), // 2022-06-24T03:11:43 (Fri) ==> 2022-06-24T03:45:00 (Fri) ({ 1656040303, 1656042300 }), // 2022-06-24T03:21:43 (Fri) ==> 2022-06-24T03:45:00 (Fri) ({ 1656040903, 1656042300 }), // 2022-06-24T03:51:43 (Fri) ==> 2022-07-01T03:45:00 (Fri) ({ 1656042703, 1656647100 }), // 2022-06-24T04:11:43 (Fri) ==> 2022-07-01T03:45:00 (Fri) ({ 1656043903, 1656647100 }), // 2022-06-24T14:11:43 (Fri) ==> 2022-07-01T03:45:00 (Fri) ({ 1656079903, 1656647100 }), }),
55f3422022-08-24Henrik Grubbström (Grubba)  ({ ({ 2, 1, 1, 0, 3, 0 }), // PI-172 adjusted from 02:00 to 03:00 // Every day at 03:00. // 2022-06-22T14:11:43 (Wed) ==> 2022-06-23T03:00:00 (Thu) ({ 1655907103, 1655953200 }), // 2022-06-23T14:11:43 (Thu) ==> 2022-06-24T03:00:00 (Fri) ({ 1655993503, 1656039600 }), // 2022-06-24T02:11:43 (Fri) ==> 2022-06-24T03:00:00 (Fri) ({ 1656036703, 1656039600 }), // 2022-06-24T03:11:43 (Fri) ==> 2022-06-25T03:00:00 (Sat) ({ 1656040303, 1656126000 }), // 2022-06-24T03:21:43 (Fri) ==> 2022-06-25T03:00:00 (Sat) ({ 1656040903, 1656126000 }), // Borken 2022-06-25T03:15:00 // 2022-06-24T03:51:43 (Fri) ==> 2022-06-25T03:00:00 (Sat) ({ 1656042703, 1656126000 }), // Borken 2022-06-25T03:45:00 // 2022-06-24T04:11:43 (Fri) ==> 2022-06-25T03:00:00 (Sat) ({ 1656043903, 1656126000 }), // 2022-06-24T14:11:43 (Fri) ==> 2022-06-25T03:00:00 (Sat) ({ 1656079903, 1656126000 }), }),
b525652022-06-27Henrik Grubbström (Grubba)  }), array(array(int)) test) { val = test[0]; while(1) { foreach(test[1..], [int when, int expected]) { int got = get_next(when); if (got != expected) { failures++; werror("Test failed for %O\n" "When: %d\n" "%O\n" "Expected: %d\n" "%O\n" "Got: %d\n" "%O\n", test, when, localtime(when), expected, localtime(expected), got, localtime(got)); } else { successes++; } }
8fa1b62022-06-27Henrik Grubbström (Grubba)  if ((sizeof(val) > 5) && !val[5]) { // Redo in compat mode. val = val[..4]; continue; }
b525652022-06-27Henrik Grubbström (Grubba)  break; } } werror("Succeeded on %d, Failed on %d.\n", successes, failures); return !!failures; }
fc10292022-06-27Henrik Grubbström (Grubba) #endif #define VALS_SORT 0 #define VALS_REPEAT_HOURS 1 #define VALS_REPEAT_COUNT 2 #define VALS_DAY 3 #define VALS_HOUR 4
8fa1b62022-06-27Henrik Grubbström (Grubba) #define VALS_MINUTE 5
fc10292022-06-27Henrik Grubbström (Grubba) 
44128c2022-09-13Henrik Grubbström (Grubba) protected multiset(int(0..2)) valid_sorts = (< 0, 1, 2 >);
5004d12001-08-22Martin Nilsson //! Transforms the form variables given in the @[vl] attribute //! to the internal time representation as follows. //!
0a29482001-08-22Per Hedbor //! @array //! @elem int(0..2) sort //! @int //! @value 0 //! Never //! @value 1 //! Every x hour //! @value 2 //! Every x y at z //! @endint
44128c2022-09-13Henrik Grubbström (Grubba) //!
0a29482001-08-22Per Hedbor //! @elem int(1..23) hour
97c8322002-04-23Henrik Grubbström (Grubba) //! Number of hours between restarts.
5004d12001-08-22Martin Nilsson //!
0a29482001-08-22Per Hedbor //! @elem int(1..9) everynth
97c8322002-04-23Henrik Grubbström (Grubba) //! Number of days or weeks to skip between restarts.
5004d12001-08-22Martin Nilsson //!
0a29482001-08-22Per Hedbor //! @elem int(0..7) day //! @int //! @value 0 //! Day //! @value 1 //! Sunday //! @value 2..7 //! Rest of weekdays //! @endint
44128c2022-09-13Henrik Grubbström (Grubba) //!
8fa1b62022-06-27Henrik Grubbström (Grubba) //! @elem int(0..23) time_hour //! Time at which to restart (hour).
44128c2022-09-13Henrik Grubbström (Grubba) //!
8fa1b62022-06-27Henrik Grubbström (Grubba) //! @elem int(0..59)|void time_min //! Time at which to restart (minute). //! If not present at minute 0 (compat).
0a29482001-08-22Per Hedbor //! @endarray
e7c3402002-10-03Anders Johansson array transform_from_form( string what, mapping vl )
0a29482001-08-22Per Hedbor { array res = query() + ({});
8fa1b62022-06-27Henrik Grubbström (Grubba)  if(sizeof(res) <= VALS_HOUR) { res = ({ 0, 2, 1, 6, 3, -1 }); } else if (sizeof(res) <= VALS_MINUTE) { // Compat. res += ({ 0 }); }
0a29482001-08-22Per Hedbor 
fc10292022-06-27Henrik Grubbström (Grubba)  res[VALS_SORT] = (int)what;
8fa1b62022-06-27Henrik Grubbström (Grubba)  for(int i=1; i <= VALS_MINUTE; i++) {
0a29482001-08-22Per Hedbor  res[i] = (int)vl[(string)i];
8fa1b62022-06-27Henrik Grubbström (Grubba)  res[i] = max( ({ 0, 1, 1, 0, 0, 0 })[i], res[i] ); res[i] = min( ({ 2, 23, 9, 7, 23, 59 })[i], res[i] ); } if (!res[VALS_MINUTE]) { // Compat. res = res[..VALS_HOUR];
0a29482001-08-22Per Hedbor  } return res; }
44128c2022-09-13Henrik Grubbström (Grubba) array verify_set_from_form(array val) { if ((sizeof(val) >= VALS_SORT) && !valid_sorts[val[VALS_SORT]]) { throw("Invalid operation mode.\n"); } return ::verify_set_from_form(val); }
96718a2019-04-29Henrik Grubbström (Grubba) protected int mktime(mapping m) { int t = predef::mktime(m); if (m->timezone) { // Compensate for cases where predef::mktime() is broken. // Cf [WS-469]. t += t - predef::mktime(localtime(t)); } return t; }
8fa1b62022-06-27Henrik Grubbström (Grubba) private mapping next_or_same_day(mapping from, int day, int hour, int minute)
0a29482001-08-22Per Hedbor {
e7c3402002-10-03Anders Johansson  if(from->wday==day && from->hour<hour) return from;
8fa1b62022-06-27Henrik Grubbström (Grubba)  if(from->wday==day && from->hour == hour && from->min<minute) return from;
0a29482001-08-22Per Hedbor  return next_day(from, day); } private mapping next_day(mapping from, int day) {
795b952020-03-27Henrik Grubbström (Grubba)  int num_days = ((6 + day - from->wday) % 7) + 1; // NB: Use a time in the middle of the date to ensure that we // don't miss the next day due to DST or similar. // Adjust the hour back to 00 afterwards. from->hour = 12; mapping m = localtime(mktime(from) + num_days * 3600 * 24); m->hour = from->hour = 0;
8fa1b62022-06-27Henrik Grubbström (Grubba)  m->min = from->min = 0;
795b952020-03-27Henrik Grubbström (Grubba)  return m;
0a29482001-08-22Per Hedbor }
8fa1b62022-06-27Henrik Grubbström (Grubba) private mapping next_or_same_time(mapping from, int hour, int minute, void|int delta)
0a29482001-08-22Per Hedbor {
8fa1b62022-06-27Henrik Grubbström (Grubba)  if (from->hour == hour) { if (minute < 0) { return from; } if ((from->min - (from->min % 15)) == minute) { return from; } } return next_time(from, hour, minute, delta);
0a29482001-08-22Per Hedbor }
8fa1b62022-06-27Henrik Grubbström (Grubba) private mapping next_time(mapping from, int hour, int minute, void|int delta)
0a29482001-08-22Per Hedbor { if(from->hour<hour) { from->hour = hour;
8fa1b62022-06-27Henrik Grubbström (Grubba)  if (minute < 0) { from->min = 0; } else { from->min = minute; } return from; } else if ((from->hour == hour) && (from->min < minute)) { from->min = minute;
0a29482001-08-22Per Hedbor  return from; }
3a9f072022-08-24Henrik Grubbström (Grubba)  from->min = minute;
4dbfcf2002-04-25Anders Johansson  return localtime(mktime(from) + (24 - from->hour + hour)*3600 + delta);
0a29482001-08-22Per Hedbor } int get_next( int last )
5004d12001-08-22Martin Nilsson //! Get the next time that matches this schedule, starting from the //! posix time @[last]. If last is 0, time(1) will be used instead. //! //! @returns //! When the next scheduled event is, represented by a posix time integer.
fc10292022-06-27Henrik Grubbström (Grubba) //! Note that the returned time may already have occured, so all return
5004d12001-08-22Martin Nilsson //! values < time() essentially means go ahead and do it right away. //! Minutes and seconds are cleared in the return value, so if the scheduler //! is set to every day at 5 o'clock, and this method is called at 5:42 it //! will return the posix time representing 5:00, unless of course @[last] //! was set to a posix time >= 5:00.
97c8322002-04-23Henrik Grubbström (Grubba) //! Returns @tt{-1@} if the schedule is disabled (@tt{"Never"@}).
0a29482001-08-22Per Hedbor { array vals = query();
8fa1b62022-06-27Henrik Grubbström (Grubba)  if (sizeof(vals) == VALS_MINUTE) { vals += ({ 0 }); }
fc10292022-06-27Henrik Grubbström (Grubba)  if( !vals[VALS_SORT] )
0a29482001-08-22Per Hedbor  return -1;
e7c3402002-10-03Anders Johansson  // Every n:th hour.
fc10292022-06-27Henrik Grubbström (Grubba)  if( vals[VALS_SORT] == 1 )
0a29482001-08-22Per Hedbor  if( !last ) return time(1); else
fc10292022-06-27Henrik Grubbström (Grubba)  return last + 3600 * vals[VALS_REPEAT_HOURS];
0a29482001-08-22Per Hedbor  mapping m = localtime( last || time(1) );
8fa1b62022-06-27Henrik Grubbström (Grubba)  m->sec = 0; m->min -= (m->min % 15);
fc10292022-06-27Henrik Grubbström (Grubba)  if( !vals[VALS_DAY] ) {
0757312002-10-09Anders Johansson  // Every n:th day at x. if (!last) {
fc10292022-06-27Henrik Grubbström (Grubba)  for(int i; i<vals[VALS_REPEAT_COUNT]; i++)
8fa1b62022-06-27Henrik Grubbström (Grubba)  m = next_or_same_time( m, vals[VALS_HOUR], vals[VALS_MINUTE] );
0757312002-10-09Anders Johansson  return mktime(m); } else {
fc10292022-06-27Henrik Grubbström (Grubba)  for(int i; i<vals[VALS_REPEAT_COUNT]; i++)
8fa1b62022-06-27Henrik Grubbström (Grubba)  m = next_time( m, vals[VALS_HOUR], vals[VALS_MINUTE] );
0757312002-10-09Anders Johansson  return mktime(m); } } // Every x-day at y. if (!last) {
fc10292022-06-27Henrik Grubbström (Grubba)  for(int i; i<vals[VALS_REPEAT_COUNT]; i++)
0757312002-10-09Anders Johansson  {
fc10292022-06-27Henrik Grubbström (Grubba)  m = next_or_same_time( next_or_same_day( m, vals[VALS_DAY]-1,
8fa1b62022-06-27Henrik Grubbström (Grubba)  vals[VALS_HOUR]+1, vals[VALS_MINUTE] ), vals[VALS_HOUR], vals[VALS_MINUTE], 6*24*3600 );
0757312002-10-09Anders Johansson  }
0a29482001-08-22Per Hedbor  }
0757312002-10-09Anders Johansson  else
4dbfcf2002-04-25Anders Johansson  {
fc10292022-06-27Henrik Grubbström (Grubba)  for(int i; i<vals[VALS_REPEAT_COUNT]; i++)
0757312002-10-09Anders Johansson  {
fc10292022-06-27Henrik Grubbström (Grubba)  m = next_or_same_time( next_or_same_day( m, vals[VALS_DAY]-1,
8fa1b62022-06-27Henrik Grubbström (Grubba)  vals[VALS_HOUR], vals[VALS_MINUTE] ), vals[VALS_HOUR], vals[VALS_MINUTE], 6*24*3600 );
0757312002-10-09Anders Johansson  }
4dbfcf2002-04-25Anders Johansson  } return mktime(m);
0a29482001-08-22Per Hedbor }
4eed962022-06-28Henrik Grubbström (Grubba) #if constant(roxenp)
fc10292022-06-27Henrik Grubbström (Grubba)  private string checked( int pos, int alt ) { if(alt==query()[pos]) return " checked='checked'"; return ""; }
0a29482001-08-22Per Hedbor string render_form( RequestID id, void|mapping additional_args ) {
8fa1b62022-06-27Henrik Grubbström (Grubba)  string res, inp1, inp2, inp3, inp4;
fc10292022-06-27Henrik Grubbström (Grubba)  array vals = query();
8fa1b62022-06-27Henrik Grubbström (Grubba)  if (sizeof(vals) == VALS_MINUTE) { vals += ({ 0 }); }
0a29482001-08-22Per Hedbor 
44128c2022-09-13Henrik Grubbström (Grubba)  res = "<table>"; if (valid_sorts[0]) { res += "<tr valign='top'><td><input name='" + path() + "' value='0' type='radio' " + checked(0,0) + " /></td><td>" + LOCALE(482, "Never") + "</td></tr>\n"; } if (valid_sorts[1]) { inp1 = HTML.select(path()+"1", "123456789"/1 + "1011121314151617181920212223"/2, (string)vals[VALS_REPEAT_HOURS]); res += "<tr valign='top'><td><input name='" + path() + "' value='1' type='radio' " + checked(0,1) + " /></td><td>" + sprintf( LOCALE(483, "Every %s hour(s)."), inp1) + "</td></tr>\n"; } if (valid_sorts[2]) { inp1 = HTML.select(path()+"2", "123456789"/1, (string)vals[VALS_REPEAT_COUNT]); inp2 = HTML.select(path()+"3", ({ ({ "0", LOCALE(484, "Day") }), ({ "1", LOCALE(485, "Sunday") }), ({ "2", LOCALE(486, "Monday") }), ({ "3", LOCALE(487, "Tuesday") }), ({ "4", LOCALE(488, "Wednesday") }), ({ "5", LOCALE(489, "Thursday") }), ({ "6", LOCALE(490, "Friday") }), ({ "7", LOCALE(491, "Saturday") }) }), (string)vals[VALS_DAY]); inp3 = HTML.select(path()+"4", "000102030405060708091011121314151617181920212223"/2, sprintf("%02d", vals[VALS_HOUR])); inp4 = HTML.select(path()+"5", "00153045"/2, sprintf("%02d", vals[VALS_MINUTE])); res += "<tr valign='top'><td><input name='" + path() + "' value='2' type='radio' " + checked(0,2) + " /></td>\n<td>" + sprintf(LOCALE(492, "Every %s %s at %s:%s o'clock."), inp1, inp2, inp3, inp4) + "</td></tr>\n"; } res += "</table>";
0a29482001-08-22Per Hedbor  return res; } string render_view( RequestID id, void|mapping additional_args ) { array res = query();
8fa1b62022-06-27Henrik Grubbström (Grubba)  if (sizeof(res) == VALS_MINUTE) { res += ({ 0 }); }
fc10292022-06-27Henrik Grubbström (Grubba)  switch(res[VALS_SORT]) {
0a29482001-08-22Per Hedbor  case 0:
9fa11d2001-08-24Martin Nilsson  return LOCALE(482, "Never");
0a29482001-08-22Per Hedbor  case 1:
fc10292022-06-27Henrik Grubbström (Grubba)  return sprintf(LOCALE(493, "Every %d hour."), res[VALS_REPEAT_HOURS]);
0a29482001-08-22Per Hedbor  case 2: string period = ({
9fa11d2001-08-24Martin Nilsson  LOCALE(484, "Day"),
e7c3402002-10-03Anders Johansson  LOCALE(485, "Sunday"),
9fa11d2001-08-24Martin Nilsson  LOCALE(486, "Monday"), LOCALE(487, "Tuesday"), LOCALE(488, "Wednesday"), LOCALE(489, "Thursday"), LOCALE(490, "Friday"),
e7c3402002-10-03Anders Johansson  LOCALE(491, "Saturday")
fc10292022-06-27Henrik Grubbström (Grubba)  })[res[VALS_DAY]];
0a29482001-08-22Per Hedbor 
8fa1b62022-06-27Henrik Grubbström (Grubba)  return sprintf(LOCALE(494, "Every %d %s at %02d:%02d"), res[VALS_REPEAT_COUNT], period, res[VALS_HOUR], res[VALS_MINUTE]);
0a29482001-08-22Per Hedbor  default:
9fa11d2001-08-24Martin Nilsson  return LOCALE(495, "Error in stored value.");
0a29482001-08-22Per Hedbor  } }
fc10292022-06-27Henrik Grubbström (Grubba) 
44128c2022-09-13Henrik Grubbström (Grubba) protected void create(array(int) default_value, void|int flags, void|LocaleString std_name, void|LocaleString std_doc, multiset(int(0..2))|void valid_sorts) { if (valid_sorts) { this_program::valid_sorts &= valid_sorts; if (!sizeof(this_program::valid_sorts)) { error("Invalid set of operation modes for Schedule: %O\n", valid_sorts); } } if (sizeof(default_value||({})) && !this_program::valid_sorts[default_value[VALS_SORT]]) { error("Invalid default mode for Schedule: %O\n", default_value[VALS_SORT]); } ::create(default_value, flags, std_name, std_doc); }
fc10292022-06-27Henrik Grubbström (Grubba) #endif