5 #ifndef V8_DATEPARSER_INL_H_ 6 #define V8_DATEPARSER_INL_H_ 8 #include "src/char-predicates-inl.h" 9 #include "src/dateparser.h" 10 #include "src/isolate.h" 15 template <
typename Char>
16 bool DateParser::Parse(Isolate* isolate, Vector<Char> str, FixedArray out) {
17 DCHECK(out->length() >= OUTPUT_SIZE);
18 InputReader<Char> in(str);
19 DateStringTokenizer<Char> scanner(&in);
73 DateToken next_unhandled_token = ParseES5DateTime(&scanner, &day, &time, &tz);
74 if (next_unhandled_token.IsInvalid())
return false;
75 bool has_read_number = !day.IsEmpty();
77 bool legacy_parser =
false;
78 for (DateToken token = next_unhandled_token;
79 !token.IsEndOfInput();
80 token = scanner.Next()) {
81 if (token.IsNumber()) {
83 has_read_number =
true;
84 int n = token.number();
85 if (scanner.SkipSymbol(
':')) {
86 if (scanner.SkipSymbol(
':')) {
88 if (!time.IsEmpty())
return false;
93 if (!time.Add(n))
return false;
94 if (scanner.Peek().IsSymbol(
'.')) scanner.Next();
96 }
else if (scanner.SkipSymbol(
'.') && time.IsExpecting(n)) {
98 if (!scanner.Peek().IsNumber())
return false;
99 int n = ReadMilliseconds(scanner.Next());
100 if (n < 0)
return false;
102 }
else if (tz.IsExpecting(n)) {
103 tz.SetAbsoluteMinute(n);
104 }
else if (time.IsExpecting(n)) {
108 DateToken peek = scanner.Peek();
109 if (!peek.IsEndOfInput() &&
110 !peek.IsWhiteSpace() &&
111 !peek.IsKeywordZ() &&
112 !peek.IsAsciiSign())
return false;
114 if (!day.Add(n))
return false;
115 scanner.SkipSymbol(
'-');
117 }
else if (token.IsKeyword()) {
118 legacy_parser =
true;
120 KeywordType type = token.keyword_type();
121 int value = token.keyword_value();
122 if (type == AM_PM && !time.IsEmpty()) {
123 time.SetHourOffset(value);
124 }
else if (type == MONTH_NAME) {
125 day.SetNamedMonth(value);
126 scanner.SkipSymbol(
'-');
127 }
else if (type == TIME_ZONE_NAME && has_read_number) {
131 if (has_read_number)
return false;
134 if (scanner.Peek().IsNumber())
return false;
136 }
else if (token.IsAsciiSign() && (tz.IsUTC() || !time.IsEmpty())) {
137 legacy_parser =
true;
139 tz.SetSign(token.ascii_sign());
143 if (scanner.Peek().IsNumber()) {
144 DateToken token = scanner.Next();
145 length = token.length();
148 has_read_number =
true;
150 if (scanner.Peek().IsSymbol(
':')) {
151 tz.SetAbsoluteHour(n);
153 tz.SetAbsoluteMinute(kNone);
154 }
else if (length == 2 || length == 1) {
156 tz.SetAbsoluteHour(n);
157 tz.SetAbsoluteMinute(0);
158 }
else if (length == 4 || length == 3) {
160 tz.SetAbsoluteHour(n / 100);
161 tz.SetAbsoluteMinute(n % 100);
166 }
else if ((token.IsAsciiSign() || token.IsSymbol(
')')) &&
175 bool success = day.Write(out) && time.Write(out) && tz.Write(out);
177 if (legacy_parser && success) {
178 isolate->CountUsage(v8::Isolate::kLegacyDateParser);
184 template<
typename CharType>
185 DateParser::DateToken DateParser::DateStringTokenizer<CharType>::Scan() {
186 int pre_pos = in_->position();
187 if (in_->IsEnd())
return DateToken::EndOfInput();
188 if (in_->IsAsciiDigit()) {
189 int n = in_->ReadUnsignedNumeral();
190 int length = in_->position() - pre_pos;
191 return DateToken::Number(n, length);
193 if (in_->Skip(
':'))
return DateToken::Symbol(
':');
194 if (in_->Skip(
'-'))
return DateToken::Symbol(
'-');
195 if (in_->Skip(
'+'))
return DateToken::Symbol(
'+');
196 if (in_->Skip(
'.'))
return DateToken::Symbol(
'.');
197 if (in_->Skip(
')'))
return DateToken::Symbol(
')');
198 if (in_->IsAsciiAlphaOrAbove()) {
199 DCHECK_EQ(KeywordTable::kPrefixLength, 3);
201 int length = in_->ReadWord(buffer, 3);
202 int index = KeywordTable::Lookup(buffer, length);
203 return DateToken::Keyword(KeywordTable::GetType(index),
204 KeywordTable::GetValue(index),
207 if (in_->SkipWhiteSpace()) {
208 return DateToken::WhiteSpace(in_->position() - pre_pos);
210 if (in_->SkipParentheses()) {
211 return DateToken::Unknown();
214 return DateToken::Unknown();
218 template <
typename Char>
219 bool DateParser::InputReader<Char>::SkipWhiteSpace() {
220 if (IsWhiteSpaceOrLineTerminator(ch_)) {
228 template <
typename Char>
229 bool DateParser::InputReader<Char>::SkipParentheses() {
230 if (ch_ !=
'(')
return false;
233 if (ch_ ==
')') --balance;
234 else if (ch_ ==
'(') ++balance;
236 }
while (balance > 0 && ch_);
241 template <
typename Char>
242 DateParser::DateToken DateParser::ParseES5DateTime(
243 DateStringTokenizer<Char>* scanner, DayComposer* day, TimeComposer* time,
244 TimeZoneComposer* tz) {
245 DCHECK(day->IsEmpty());
246 DCHECK(time->IsEmpty());
247 DCHECK(tz->IsEmpty());
250 if (scanner->Peek().IsAsciiSign()) {
253 DateToken sign_token = scanner->Next();
254 if (!scanner->Peek().IsFixedLengthNumber(6))
return sign_token;
255 int sign = sign_token.ascii_sign();
256 int year = scanner->Next().number();
257 if (sign < 0 && year == 0)
return sign_token;
258 day->Add(sign * year);
259 }
else if (scanner->Peek().IsFixedLengthNumber(4)) {
260 day->Add(scanner->Next().number());
262 return scanner->Next();
264 if (scanner->SkipSymbol(
'-')) {
265 if (!scanner->Peek().IsFixedLengthNumber(2) ||
266 !DayComposer::IsMonth(scanner->Peek().number()))
return scanner->Next();
267 day->Add(scanner->Next().number());
268 if (scanner->SkipSymbol(
'-')) {
269 if (!scanner->Peek().IsFixedLengthNumber(2) ||
270 !DayComposer::IsDay(scanner->Peek().number()))
return scanner->Next();
271 day->Add(scanner->Next().number());
275 if (!scanner->Peek().IsKeywordType(TIME_SEPARATOR)) {
276 if (!scanner->Peek().IsEndOfInput())
return scanner->Next();
280 if (!scanner->Peek().IsFixedLengthNumber(2) ||
281 !Between(scanner->Peek().number(), 0, 24)) {
282 return DateToken::Invalid();
285 bool hour_is_24 = (scanner->Peek().number() == 24);
286 time->Add(scanner->Next().number());
287 if (!scanner->SkipSymbol(
':'))
return DateToken::Invalid();
288 if (!scanner->Peek().IsFixedLengthNumber(2) ||
289 !TimeComposer::IsMinute(scanner->Peek().number()) ||
290 (hour_is_24 && scanner->Peek().number() > 0)) {
291 return DateToken::Invalid();
293 time->Add(scanner->Next().number());
294 if (scanner->SkipSymbol(
':')) {
295 if (!scanner->Peek().IsFixedLengthNumber(2) ||
296 !TimeComposer::IsSecond(scanner->Peek().number()) ||
297 (hour_is_24 && scanner->Peek().number() > 0)) {
298 return DateToken::Invalid();
300 time->Add(scanner->Next().number());
301 if (scanner->SkipSymbol(
'.')) {
302 if (!scanner->Peek().IsNumber() ||
303 (hour_is_24 && scanner->Peek().number() > 0)) {
304 return DateToken::Invalid();
307 time->Add(ReadMilliseconds(scanner->Next()));
311 if (scanner->Peek().IsKeywordZ()) {
314 }
else if (scanner->Peek().IsSymbol(
'+') ||
315 scanner->Peek().IsSymbol(
'-')) {
316 tz->SetSign(scanner->Next().symbol() ==
'+' ? 1 : -1);
317 if (scanner->Peek().IsFixedLengthNumber(4)) {
319 int hourmin = scanner->Next().number();
320 int hour = hourmin / 100;
321 int min = hourmin % 100;
322 if (!TimeComposer::IsHour(hour) || !TimeComposer::IsMinute(min)) {
323 return DateToken::Invalid();
325 tz->SetAbsoluteHour(hour);
326 tz->SetAbsoluteMinute(min);
329 if (!scanner->Peek().IsFixedLengthNumber(2) ||
330 !TimeComposer::IsHour(scanner->Peek().number())) {
331 return DateToken::Invalid();
333 tz->SetAbsoluteHour(scanner->Next().number());
334 if (!scanner->SkipSymbol(
':'))
return DateToken::Invalid();
335 if (!scanner->Peek().IsFixedLengthNumber(2) ||
336 !TimeComposer::IsMinute(scanner->Peek().number())) {
337 return DateToken::Invalid();
339 tz->SetAbsoluteMinute(scanner->Next().number());
342 if (!scanner->Peek().IsEndOfInput())
return DateToken::Invalid();
348 if (tz->IsEmpty() && time->IsEmpty()) {
352 return DateToken::EndOfInput();
359 #endif // V8_DATEPARSER_INL_H_