OLD | NEW |
| (Empty) |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | |
2 // for details. All rights reserved. Use of this source code is governed by a | |
3 // BSD-style license that can be found in the LICENSE file. | |
4 | |
5 part of dart.io; | |
6 | |
7 /** | |
8 * Utility functions for working with dates with HTTP specific date | |
9 * formats. | |
10 */ | |
11 class HttpDate { | |
12 // From RFC-2616 section "3.3.1 Full Date", | |
13 // http://tools.ietf.org/html/rfc2616#section-3.3.1 | |
14 // | |
15 // HTTP-date = rfc1123-date | rfc850-date | asctime-date | |
16 // rfc1123-date = wkday "," SP date1 SP time SP "GMT" | |
17 // rfc850-date = weekday "," SP date2 SP time SP "GMT" | |
18 // asctime-date = wkday SP date3 SP time SP 4DIGIT | |
19 // date1 = 2DIGIT SP month SP 4DIGIT | |
20 // ; day month year (e.g., 02 Jun 1982) | |
21 // date2 = 2DIGIT "-" month "-" 2DIGIT | |
22 // ; day-month-year (e.g., 02-Jun-82) | |
23 // date3 = month SP ( 2DIGIT | ( SP 1DIGIT )) | |
24 // ; month day (e.g., Jun 2) | |
25 // time = 2DIGIT ":" 2DIGIT ":" 2DIGIT | |
26 // ; 00:00:00 - 23:59:59 | |
27 // wkday = "Mon" | "Tue" | "Wed" | |
28 // | "Thu" | "Fri" | "Sat" | "Sun" | |
29 // weekday = "Monday" | "Tuesday" | "Wednesday" | |
30 // | "Thursday" | "Friday" | "Saturday" | "Sunday" | |
31 // month = "Jan" | "Feb" | "Mar" | "Apr" | |
32 // | "May" | "Jun" | "Jul" | "Aug" | |
33 // | "Sep" | "Oct" | "Nov" | "Dec" | |
34 | |
35 /** | |
36 * Format a date according to | |
37 * [RFC-1123](http://tools.ietf.org/html/rfc1123 "RFC-1123"), | |
38 * e.g. `Thu, 1 Jan 1970 00:00:00 GMT`. | |
39 */ | |
40 static String format(DateTime date) { | |
41 const List wkday = const ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]; | |
42 const List month = const ["Jan", "Feb", "Mar", "Apr", "May", "Jun", | |
43 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]; | |
44 | |
45 DateTime d = date.toUtc(); | |
46 StringBuffer sb = new StringBuffer() | |
47 ..write(wkday[d.weekday - 1]) | |
48 ..write(", ") | |
49 ..write(d.day <= 9 ? "0" : "") | |
50 ..write(d.day.toString()) | |
51 ..write(" ") | |
52 ..write(month[d.month - 1]) | |
53 ..write(" ") | |
54 ..write(d.year.toString()) | |
55 ..write(d.hour <= 9 ? " 0" : " ") | |
56 ..write(d.hour.toString()) | |
57 ..write(d.minute <= 9 ? ":0" : ":") | |
58 ..write(d.minute.toString()) | |
59 ..write(d.second <= 9 ? ":0" : ":") | |
60 ..write(d.second.toString()) | |
61 ..write(" GMT"); | |
62 return sb.toString(); | |
63 } | |
64 | |
65 /** | |
66 * Parse a date string in either of the formats | |
67 * [RFC-1123](http://tools.ietf.org/html/rfc1123 "RFC-1123"), | |
68 * [RFC-850](http://tools.ietf.org/html/rfc850 "RFC-850") or | |
69 * ANSI C's asctime() format. These formats are listed here. | |
70 * | |
71 * Thu, 1 Jan 1970 00:00:00 GMT | |
72 * Thursday, 1-Jan-1970 00:00:00 GMT | |
73 * Thu Jan 1 00:00:00 1970 | |
74 * | |
75 * For more information see [RFC-2616 section | |
76 * 3.1.1](http://tools.ietf.org/html/rfc2616#section-3.3.1 | |
77 * "RFC-2616 section 3.1.1"). | |
78 */ | |
79 static DateTime parse(String date) { | |
80 final int SP = 32; | |
81 const List wkdays = const ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]; | |
82 const List weekdays = const ["Monday", "Tuesday", "Wednesday", "Thursday", | |
83 "Friday", "Saturday", "Sunday"]; | |
84 const List months = const ["Jan", "Feb", "Mar", "Apr", "May", "Jun", | |
85 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]; | |
86 const List wkdaysLowerCase = | |
87 const ["mon", "tue", "wed", "thu", "fri", "sat", "sun"]; | |
88 const List weekdaysLowerCase = const ["monday", "tuesday", "wednesday", | |
89 "thursday", "friday", "saturday", | |
90 "sunday"]; | |
91 const List monthsLowerCase = const ["jan", "feb", "mar", "apr", "may", | |
92 "jun", "jul", "aug", "sep", "oct", | |
93 "nov", "dec"]; | |
94 | |
95 final int formatRfc1123 = 0; | |
96 final int formatRfc850 = 1; | |
97 final int formatAsctime = 2; | |
98 | |
99 int index = 0; | |
100 String tmp; | |
101 int format; | |
102 | |
103 void expect(String s) { | |
104 if (date.length - index < s.length) { | |
105 throw new HttpException("Invalid HTTP date $date"); | |
106 } | |
107 String tmp = date.substring(index, index + s.length); | |
108 if (tmp != s) { | |
109 throw new HttpException("Invalid HTTP date $date"); | |
110 } | |
111 index += s.length; | |
112 } | |
113 | |
114 int expectWeekday() { | |
115 int weekday; | |
116 // The formatting of the weekday signals the format of the date string. | |
117 int pos = date.indexOf(",", index); | |
118 if (pos == -1) { | |
119 int pos = date.indexOf(" ", index); | |
120 if (pos == -1) throw new HttpException("Invalid HTTP date $date"); | |
121 tmp = date.substring(index, pos); | |
122 index = pos + 1; | |
123 weekday = wkdays.indexOf(tmp); | |
124 if (weekday != -1) { | |
125 format = formatAsctime; | |
126 return weekday; | |
127 } | |
128 } else { | |
129 tmp = date.substring(index, pos); | |
130 index = pos + 1; | |
131 weekday = wkdays.indexOf(tmp); | |
132 if (weekday != -1) { | |
133 format = formatRfc1123; | |
134 return weekday; | |
135 } | |
136 weekday = weekdays.indexOf(tmp); | |
137 if (weekday != -1) { | |
138 format = formatRfc850; | |
139 return weekday; | |
140 } | |
141 } | |
142 throw new HttpException("Invalid HTTP date $date"); | |
143 } | |
144 | |
145 int expectMonth(String separator) { | |
146 int pos = date.indexOf(separator, index); | |
147 if (pos - index != 3) throw new HttpException("Invalid HTTP date $date"); | |
148 tmp = date.substring(index, pos); | |
149 index = pos + 1; | |
150 int month = months.indexOf(tmp); | |
151 if (month != -1) return month; | |
152 throw new HttpException("Invalid HTTP date $date"); | |
153 } | |
154 | |
155 int expectNum(String separator) { | |
156 int pos; | |
157 if (separator.length > 0) { | |
158 pos = date.indexOf(separator, index); | |
159 } else { | |
160 pos = date.length; | |
161 } | |
162 String tmp = date.substring(index, pos); | |
163 index = pos + separator.length; | |
164 try { | |
165 int value = int.parse(tmp); | |
166 return value; | |
167 } on FormatException catch (e) { | |
168 throw new HttpException("Invalid HTTP date $date"); | |
169 } | |
170 } | |
171 | |
172 void expectEnd() { | |
173 if (index != date.length) { | |
174 throw new HttpException("Invalid HTTP date $date"); | |
175 } | |
176 } | |
177 | |
178 int weekday = expectWeekday(); | |
179 int day; | |
180 int month; | |
181 int year; | |
182 int hours; | |
183 int minutes; | |
184 int seconds; | |
185 if (format == formatAsctime) { | |
186 month = expectMonth(" "); | |
187 if (date.codeUnitAt(index) == SP) index++; | |
188 day = expectNum(" "); | |
189 hours = expectNum(":"); | |
190 minutes = expectNum(":"); | |
191 seconds = expectNum(" "); | |
192 year = expectNum(""); | |
193 } else { | |
194 expect(" "); | |
195 day = expectNum(format == formatRfc1123 ? " " : "-"); | |
196 month = expectMonth(format == formatRfc1123 ? " " : "-"); | |
197 year = expectNum(" "); | |
198 hours = expectNum(":"); | |
199 minutes = expectNum(":"); | |
200 seconds = expectNum(" "); | |
201 expect("GMT"); | |
202 } | |
203 expectEnd(); | |
204 return new DateTime.utc(year, month + 1, day, hours, minutes, seconds, 0); | |
205 } | |
206 | |
207 // Parse a cookie date string. | |
208 static DateTime _parseCookieDate(String date) { | |
209 const List monthsLowerCase = const ["jan", "feb", "mar", "apr", "may", | |
210 "jun", "jul", "aug", "sep", "oct", | |
211 "nov", "dec"]; | |
212 | |
213 int position = 0; | |
214 | |
215 void error() { | |
216 throw new HttpException("Invalid cookie date $date"); | |
217 } | |
218 | |
219 bool isEnd() => position == date.length; | |
220 | |
221 bool isDelimiter(String s) { | |
222 int char = s.codeUnitAt(0); | |
223 if (char == 0x09) return true; | |
224 if (char >= 0x20 && char <= 0x2F) return true; | |
225 if (char >= 0x3B && char <= 0x40) return true; | |
226 if (char >= 0x5B && char <= 0x60) return true; | |
227 if (char >= 0x7B && char <= 0x7E) return true; | |
228 return false; | |
229 } | |
230 | |
231 bool isNonDelimiter(String s) { | |
232 int char = s.codeUnitAt(0); | |
233 if (char >= 0x00 && char <= 0x08) return true; | |
234 if (char >= 0x0A && char <= 0x1F) return true; | |
235 if (char >= 0x30 && char <= 0x39) return true; // Digit | |
236 if (char == 0x3A) return true; // ':' | |
237 if (char >= 0x41 && char <= 0x5A) return true; // Alpha | |
238 if (char >= 0x61 && char <= 0x7A) return true; // Alpha | |
239 if (char >= 0x7F && char <= 0xFF) return true; // Alpha | |
240 return false; | |
241 } | |
242 | |
243 bool isDigit(String s) { | |
244 int char = s.codeUnitAt(0); | |
245 if (char > 0x2F && char < 0x3A) return true; | |
246 return false; | |
247 } | |
248 | |
249 int getMonth(String month) { | |
250 if (month.length < 3) return -1; | |
251 return monthsLowerCase.indexOf(month.substring(0, 3)); | |
252 } | |
253 | |
254 int toInt(String s) { | |
255 int index = 0; | |
256 for (; index < s.length && isDigit(s[index]); index++); | |
257 return int.parse(s.substring(0, index)); | |
258 } | |
259 | |
260 var tokens = []; | |
261 while (!isEnd()) { | |
262 while (!isEnd() && isDelimiter(date[position])) position++; | |
263 int start = position; | |
264 while (!isEnd() && isNonDelimiter(date[position])) position++; | |
265 tokens.add(date.substring(start, position).toLowerCase()); | |
266 while (!isEnd() && isDelimiter(date[position])) position++; | |
267 } | |
268 | |
269 String timeStr; | |
270 String dayOfMonthStr; | |
271 String monthStr; | |
272 String yearStr; | |
273 | |
274 for (var token in tokens) { | |
275 if (token.length < 1) continue; | |
276 if (timeStr == null && token.length >= 5 && isDigit(token[0]) && | |
277 (token[1] == ":" || (isDigit(token[1]) && token[2] == ":"))) { | |
278 timeStr = token; | |
279 } else if (dayOfMonthStr == null && isDigit(token[0])) { | |
280 dayOfMonthStr = token; | |
281 } else if (monthStr == null && getMonth(token) >= 0) { | |
282 monthStr = token; | |
283 } else if (yearStr == null && token.length >= 2 && | |
284 isDigit(token[0]) && isDigit(token[1])) { | |
285 yearStr = token; | |
286 } | |
287 } | |
288 | |
289 if (timeStr == null || dayOfMonthStr == null || | |
290 monthStr == null || yearStr == null) { | |
291 error(); | |
292 } | |
293 | |
294 int year = toInt(yearStr); | |
295 if (year >= 70 && year <= 99) year += 1900; | |
296 else if (year >= 0 && year <= 69) year += 2000; | |
297 if (year < 1601) error(); | |
298 | |
299 int dayOfMonth = toInt(dayOfMonthStr); | |
300 if (dayOfMonth < 1 || dayOfMonth > 31) error(); | |
301 | |
302 int month = getMonth(monthStr) + 1; | |
303 | |
304 var timeList = timeStr.split(":"); | |
305 if (timeList.length != 3) error(); | |
306 int hour = toInt(timeList[0]); | |
307 int minute = toInt(timeList[1]); | |
308 int second = toInt(timeList[2]); | |
309 if (hour > 23) error(); | |
310 if (minute > 59) error(); | |
311 if (second > 59) error(); | |
312 | |
313 return new DateTime.utc(year, month, dayOfMonth, hour, minute, second, 0); | |
314 } | |
315 } | |
OLD | NEW |