-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #128 from os2display/feature/rrule-fixes
Fixed rrule evaluation to handle local time correctly
- Loading branch information
Showing
5 changed files
with
90 additions
and
45 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
import { RRule } from 'rrule'; | ||
|
||
class ScheduleUtils { | ||
static occursNow(rruleString, durationSeconds) { | ||
const rrule = RRule.fromString(rruleString.replace("\\n", "\n")); | ||
const duration = durationSeconds * 1000; | ||
|
||
const now = new Date(); | ||
|
||
// From the RRULE docs:_ "Returned "UTC" dates are always meant to be | ||
// interpreted as dates in your local timezone. This may mean you have to | ||
// do additional conversion to get the "correct" local time with offset | ||
// applied." | ||
// | ||
// We do the opposite to ensure that datetime comparisons works as expected. | ||
// For evaluation with the RRule library we pretend that "now" is in UTC instead of the local timezone. | ||
// That is 9:00 in Europe/Copenhagen time will be evaluated as if it was 9:00 in UTC. | ||
// @see https://github.com/jkbrzt/rrule#important-use-utc-dates | ||
const nowWithoutTimezone = new Date(Date.UTC(now.getFullYear(), now.getMonth(), now.getDate(), now.getHours(), now.getMinutes(), now.getSeconds())); | ||
|
||
// Subtract duration from now to make sure all relevant occurrences are considered. | ||
const from = new Date(nowWithoutTimezone.getTime() - duration); | ||
|
||
let occurs = false; | ||
|
||
// RRule.prototype.between(after, before, inc=false [, iterator]) | ||
// The between() function expects "after" and "before" to be in pretend UTC as | ||
// described above. | ||
rrule.between( | ||
from, | ||
nowWithoutTimezone, | ||
true, | ||
function iterator(occurrenceDate) { | ||
// The "ccurrenceDate" we are iterating over contains a "pretend UTC" datetime | ||
// object. As above, if the time for "occurrenceDate" is 09:00 UTC it should be | ||
// treated as 09:00 local time regardsless of the actual local timezone | ||
const end = new Date(occurrenceDate.getTime() + duration); | ||
|
||
if (nowWithoutTimezone >= occurrenceDate && nowWithoutTimezone <= end) { | ||
occurs = true; | ||
// break iteration. | ||
return false; | ||
} | ||
|
||
// continue iteration. | ||
return true; | ||
} | ||
); | ||
|
||
return occurs; | ||
} | ||
} | ||
|
||
export default ScheduleUtils; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters