-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathprettycron.js
197 lines (170 loc) · 6.42 KB
/
prettycron.js
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
////////////////////////////////////////////////////////////////////////////////////
//
// prettycron.js
// Generates human-readable sentences from a schedule string in cron format
//
// Based on an earlier version by Pehr Johansson
// http://dsysadm.blogspot.com.au/2012/09/human-readable-cron-expressions-using.html
//
////////////////////////////////////////////////////////////////////////////////////
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published
// by the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
////////////////////////////////////////////////////////////////////////////////////
if ((!moment || !later) && (typeof require !== 'undefined')) {
var moment = require('moment');
var later = require('later');
}
(function() {
/*
* For an array of numbers, e.g. a list of hours in a schedule,
* return a string listing out all of the values (complete with
* "and" plus ordinal text on the last item).
*/
var numberList = function(numbers) {
if (numbers.length < 2) {
return moment()._locale.ordinal(numbers);
}
var last_val = numbers.pop();
return numbers.join(', ') + ' and ' + moment()._locale.ordinal(last_val);
};
/*
* Parse a number into day of week, or a month name;
* used in dateList below.
*/
var numberToDateName = function(value, type) {
if (type == 'dow') {
return moment().day(value - 1).format('ddd');
} else if (type == 'mon') {
return moment().month(value - 1).format('MMM');
}
};
/*
* From an array of numbers corresponding to dates (given in type: either
* days of the week, or months), return a string listing all the values.
*/
var dateList = function(numbers, type) {
if (numbers.length < 2) {
return numberToDateName(''+numbers[0], type);
}
var last_val = '' + numbers.pop();
var output_text = '';
for (var i=0, value; value=numbers[i]; i++) {
if (output_text.length > 0) {
output_text += ', ';
}
output_text += numberToDateName(value, type);
}
return output_text + ' and ' + numberToDateName(last_val, type);
};
/*
* Pad to equivalent of sprintf('%02d'). Both moment.js and later.js
* have zero-fill functions, but alas, they're private.
*/
var zeroPad = function(x) {
return (x < 10) ? '0' + x : x;
};
//----------------
/*
* Given a schedule from later.js (i.e. after parsing the cronspec),
* generate a friendly sentence description.
*/
var scheduleToSentence = function(schedule) {
var output_text = 'Every ';
if (schedule['h'] && schedule['m'] && schedule['h'].length <= 2 && schedule['m'].length <= 2) {
// If there are only one or two specified values for
// hour or minute, print them in HH:MM format
var hm = [];
for (var i=0; i < schedule['h'].length; i++) {
for (var j=0; j < schedule['m'].length; j++) {
hm.push(zeroPad(schedule['h'][i]) + ':' + zeroPad(schedule['m'][j]));
}
}
if (hm.length < 2) {
output_text = hm[0];
} else {
var last_val = hm.pop();
output_text = hm.join(', ') + ' and ' + last_val;
}
if (!schedule['d'] && !schedule['D']) {
output_text += ' every day';
}
} else {
// Otherwise, list out every specified hour/minute value.
if(schedule['h']) { // runs only at specific hours
if (schedule['m']) { // and only at specific minutes
output_text += numberList(schedule['m']) + ' minute past the ' + numberList(schedule['h']) + ' hour';
} else { // specific hours, but every minute
output_text += 'minute of ' + numberList(schedule['h']) + ' hour';
}
} else if(schedule['m']) { // every hour, but specific minutes
if (schedule['m'].length == 1 && schedule['m'][0] == 0) {
output_text += 'hour, on the hour';
} else {
output_text += numberList(schedule['m']) + ' minute past every hour';
}
} else { // cronspec has "*" for both hour and minute
output_text += 'minute';
}
}
if (schedule['D']) { // runs only on specific day(s) of month
output_text += ' on the ' + numberList(schedule['D']);
if (!schedule['M']) {
output_text += ' of every month';
}
}
if (schedule['d']) { // runs only on specific day(s) of week
if (schedule['D']) {
// if both day fields are specified, cron uses both; superuser.com/a/348372
output_text += ' and every ';
} else {
output_text += ' on ';
}
output_text += dateList(schedule['d'], 'dow');
}
if (schedule['M']) {
// runs only in specific months; put this output last
output_text += ' in ' + dateList(schedule['M'], 'mon');
}
return output_text;
};
//----------------
/*
* Given a cronspec, return the human-readable string.
*/
var toString = function(cronspec, sixth) {
var schedule = later.parse.cron(cronspec, sixth);
return scheduleToSentence(schedule['schedules'][0]);
};
/*
* Given a cronspec, return the next date for when it will next run.
* (This is just a wrapper for later.js)
*/
var getNextDate = function(cronspec, sixth) {
var schedule = later.parse.cron(cronspec, sixth);
return later.schedule(schedule).next();
};
/*
* Given a cronspec, return a friendly string for when it will next run.
* (This is just a wrapper for later.js and moment.js)
*/
var getNext = function(cronspec, sixth) {
return moment( getNextDate( cronspec, sixth ) ).calendar();
};
//----------------
// attach ourselves to window in the browser, and to exports in Node,
// so our functions can always be called as prettyCron.toString()
var global_obj = (typeof exports !== "undefined" && exports !== null) ? exports : window.prettyCron = {};
global_obj.toString = toString;
global_obj.getNext = getNext;
global_obj.getNextDate = getNextDate;
}).call(this);