github.com/xushiwei/go@v0.0.0-20130601165731-2b9d83f45bc9/src/pkg/time/format.go (about) 1 // Copyright 2010 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 package time 6 7 import "errors" 8 9 // These are predefined layouts for use in Time.Format and Time.Parse. 10 // The reference time used in the layouts is: 11 // Mon Jan 2 15:04:05 MST 2006 12 // which is Unix time 1136239445. Since MST is GMT-0700, 13 // the reference time can be thought of as 14 // 01/02 03:04:05PM '06 -0700 15 // To define your own format, write down what the reference time would look 16 // like formatted your way; see the values of constants like ANSIC, 17 // StampMicro or Kitchen for examples. The model is to demonstrate what the 18 // reference time looks like so that the Format and Parse methods can apply 19 // the same transformation to a general time value. 20 // 21 // Within the format string, an underscore _ represents a space that may be 22 // replaced by a digit if the following number (a day) has two digits; for 23 // compatibility with fixed-width Unix time formats. 24 // 25 // A decimal point followed by one or more zeros represents a fractional 26 // second, printed to the given number of decimal places. A decimal point 27 // followed by one or more nines represents a fractional second, printed to 28 // the given number of decimal places, with trailing zeros removed. 29 // When parsing (only), the input may contain a fractional second 30 // field immediately after the seconds field, even if the layout does not 31 // signify its presence. In that case a decimal point followed by a maximal 32 // series of digits is parsed as a fractional second. 33 // 34 // Numeric time zone offsets format as follows: 35 // -0700 ±hhmm 36 // -07:00 ±hh:mm 37 // Replacing the sign in the format with a Z triggers 38 // the ISO 8601 behavior of printing Z instead of an 39 // offset for the UTC zone. Thus: 40 // Z0700 Z or ±hhmm 41 // Z07:00 Z or ±hh:mm 42 const ( 43 ANSIC = "Mon Jan _2 15:04:05 2006" 44 UnixDate = "Mon Jan _2 15:04:05 MST 2006" 45 RubyDate = "Mon Jan 02 15:04:05 -0700 2006" 46 RFC822 = "02 Jan 06 15:04 MST" 47 RFC822Z = "02 Jan 06 15:04 -0700" // RFC822 with numeric zone 48 RFC850 = "Monday, 02-Jan-06 15:04:05 MST" 49 RFC1123 = "Mon, 02 Jan 2006 15:04:05 MST" 50 RFC1123Z = "Mon, 02 Jan 2006 15:04:05 -0700" // RFC1123 with numeric zone 51 RFC3339 = "2006-01-02T15:04:05Z07:00" 52 RFC3339Nano = "2006-01-02T15:04:05.999999999Z07:00" 53 Kitchen = "3:04PM" 54 // Handy time stamps. 55 Stamp = "Jan _2 15:04:05" 56 StampMilli = "Jan _2 15:04:05.000" 57 StampMicro = "Jan _2 15:04:05.000000" 58 StampNano = "Jan _2 15:04:05.000000000" 59 ) 60 61 const ( 62 _ = iota 63 stdLongMonth = iota + stdNeedDate // "January" 64 stdMonth // "Jan" 65 stdNumMonth // "1" 66 stdZeroMonth // "01" 67 stdLongWeekDay // "Monday" 68 stdWeekDay // "Mon" 69 stdDay // "2" 70 stdUnderDay // "_2" 71 stdZeroDay // "02" 72 stdHour = iota + stdNeedClock // "15" 73 stdHour12 // "3" 74 stdZeroHour12 // "03" 75 stdMinute // "4" 76 stdZeroMinute // "04" 77 stdSecond // "5" 78 stdZeroSecond // "05" 79 stdLongYear = iota + stdNeedDate // "2006" 80 stdYear // "06" 81 stdPM = iota + stdNeedClock // "PM" 82 stdpm // "pm" 83 stdTZ = iota // "MST" 84 stdISO8601TZ // "Z0700" // prints Z for UTC 85 stdISO8601ColonTZ // "Z07:00" // prints Z for UTC 86 stdNumTZ // "-0700" // always numeric 87 stdNumShortTZ // "-07" // always numeric 88 stdNumColonTZ // "-07:00" // always numeric 89 stdFracSecond0 // ".0", ".00", ... , trailing zeros included 90 stdFracSecond9 // ".9", ".99", ..., trailing zeros omitted 91 92 stdNeedDate = 1 << 8 // need month, day, year 93 stdNeedClock = 2 << 8 // need hour, minute, second 94 stdArgShift = 16 // extra argument in high bits, above low stdArgShift 95 stdMask = 1<<stdArgShift - 1 // mask out argument 96 ) 97 98 // std0x records the std values for "01", "02", ..., "06". 99 var std0x = [...]int{stdZeroMonth, stdZeroDay, stdZeroHour12, stdZeroMinute, stdZeroSecond, stdYear} 100 101 // nextStdChunk finds the first occurrence of a std string in 102 // layout and returns the text before, the std string, and the text after. 103 func nextStdChunk(layout string) (prefix string, std int, suffix string) { 104 for i := 0; i < len(layout); i++ { 105 switch c := int(layout[i]); c { 106 case 'J': // January, Jan 107 if len(layout) >= i+3 && layout[i:i+3] == "Jan" { 108 if len(layout) >= i+7 && layout[i:i+7] == "January" { 109 return layout[0:i], stdLongMonth, layout[i+7:] 110 } 111 return layout[0:i], stdMonth, layout[i+3:] 112 } 113 114 case 'M': // Monday, Mon, MST 115 if len(layout) >= i+3 { 116 if layout[i:i+3] == "Mon" { 117 if len(layout) >= i+6 && layout[i:i+6] == "Monday" { 118 return layout[0:i], stdLongWeekDay, layout[i+6:] 119 } 120 return layout[0:i], stdWeekDay, layout[i+3:] 121 } 122 if layout[i:i+3] == "MST" { 123 return layout[0:i], stdTZ, layout[i+3:] 124 } 125 } 126 127 case '0': // 01, 02, 03, 04, 05, 06 128 if len(layout) >= i+2 && '1' <= layout[i+1] && layout[i+1] <= '6' { 129 return layout[0:i], std0x[layout[i+1]-'1'], layout[i+2:] 130 } 131 132 case '1': // 15, 1 133 if len(layout) >= i+2 && layout[i+1] == '5' { 134 return layout[0:i], stdHour, layout[i+2:] 135 } 136 return layout[0:i], stdNumMonth, layout[i+1:] 137 138 case '2': // 2006, 2 139 if len(layout) >= i+4 && layout[i:i+4] == "2006" { 140 return layout[0:i], stdLongYear, layout[i+4:] 141 } 142 return layout[0:i], stdDay, layout[i+1:] 143 144 case '_': // _2 145 if len(layout) >= i+2 && layout[i+1] == '2' { 146 return layout[0:i], stdUnderDay, layout[i+2:] 147 } 148 149 case '3': 150 return layout[0:i], stdHour12, layout[i+1:] 151 152 case '4': 153 return layout[0:i], stdMinute, layout[i+1:] 154 155 case '5': 156 return layout[0:i], stdSecond, layout[i+1:] 157 158 case 'P': // PM 159 if len(layout) >= i+2 && layout[i+1] == 'M' { 160 return layout[0:i], stdPM, layout[i+2:] 161 } 162 163 case 'p': // pm 164 if len(layout) >= i+2 && layout[i+1] == 'm' { 165 return layout[0:i], stdpm, layout[i+2:] 166 } 167 168 case '-': // -0700, -07:00, -07 169 if len(layout) >= i+5 && layout[i:i+5] == "-0700" { 170 return layout[0:i], stdNumTZ, layout[i+5:] 171 } 172 if len(layout) >= i+6 && layout[i:i+6] == "-07:00" { 173 return layout[0:i], stdNumColonTZ, layout[i+6:] 174 } 175 if len(layout) >= i+3 && layout[i:i+3] == "-07" { 176 return layout[0:i], stdNumShortTZ, layout[i+3:] 177 } 178 case 'Z': // Z0700, Z07:00 179 if len(layout) >= i+5 && layout[i:i+5] == "Z0700" { 180 return layout[0:i], stdISO8601TZ, layout[i+5:] 181 } 182 if len(layout) >= i+6 && layout[i:i+6] == "Z07:00" { 183 return layout[0:i], stdISO8601ColonTZ, layout[i+6:] 184 } 185 case '.': // .000 or .999 - repeated digits for fractional seconds. 186 if i+1 < len(layout) && (layout[i+1] == '0' || layout[i+1] == '9') { 187 ch := layout[i+1] 188 j := i + 1 189 for j < len(layout) && layout[j] == ch { 190 j++ 191 } 192 // String of digits must end here - only fractional second is all digits. 193 if !isDigit(layout, j) { 194 std := stdFracSecond0 195 if layout[i+1] == '9' { 196 std = stdFracSecond9 197 } 198 std |= (j - (i + 1)) << stdArgShift 199 return layout[0:i], std, layout[j:] 200 } 201 } 202 } 203 } 204 return layout, 0, "" 205 } 206 207 var longDayNames = []string{ 208 "Sunday", 209 "Monday", 210 "Tuesday", 211 "Wednesday", 212 "Thursday", 213 "Friday", 214 "Saturday", 215 } 216 217 var shortDayNames = []string{ 218 "Sun", 219 "Mon", 220 "Tue", 221 "Wed", 222 "Thu", 223 "Fri", 224 "Sat", 225 } 226 227 var shortMonthNames = []string{ 228 "---", 229 "Jan", 230 "Feb", 231 "Mar", 232 "Apr", 233 "May", 234 "Jun", 235 "Jul", 236 "Aug", 237 "Sep", 238 "Oct", 239 "Nov", 240 "Dec", 241 } 242 243 var longMonthNames = []string{ 244 "---", 245 "January", 246 "February", 247 "March", 248 "April", 249 "May", 250 "June", 251 "July", 252 "August", 253 "September", 254 "October", 255 "November", 256 "December", 257 } 258 259 // match returns true if s1 and s2 match ignoring case. 260 // It is assumed s1 and s2 are the same length. 261 func match(s1, s2 string) bool { 262 for i := 0; i < len(s1); i++ { 263 c1 := s1[i] 264 c2 := s2[i] 265 if c1 != c2 { 266 // Switch to lower-case; 'a'-'A' is known to be a single bit. 267 c1 |= 'a' - 'A' 268 c2 |= 'a' - 'A' 269 if c1 != c2 || c1 < 'a' || c1 > 'z' { 270 return false 271 } 272 } 273 } 274 return true 275 } 276 277 func lookup(tab []string, val string) (int, string, error) { 278 for i, v := range tab { 279 if len(val) >= len(v) && match(val[0:len(v)], v) { 280 return i, val[len(v):], nil 281 } 282 } 283 return -1, val, errBad 284 } 285 286 // appendUint appends the decimal form of x to b and returns the result. 287 // If x is a single-digit number and pad != 0, appendUint inserts the pad byte 288 // before the digit. 289 // Duplicates functionality in strconv, but avoids dependency. 290 func appendUint(b []byte, x uint, pad byte) []byte { 291 if x < 10 { 292 if pad != 0 { 293 b = append(b, pad) 294 } 295 return append(b, byte('0'+x)) 296 } 297 if x < 100 { 298 b = append(b, byte('0'+x/10)) 299 b = append(b, byte('0'+x%10)) 300 return b 301 } 302 303 var buf [32]byte 304 n := len(buf) 305 if x == 0 { 306 return append(b, '0') 307 } 308 for x >= 10 { 309 n-- 310 buf[n] = byte(x%10 + '0') 311 x /= 10 312 } 313 n-- 314 buf[n] = byte(x + '0') 315 return append(b, buf[n:]...) 316 } 317 318 // Never printed, just needs to be non-nil for return by atoi. 319 var atoiError = errors.New("time: invalid number") 320 321 // Duplicates functionality in strconv, but avoids dependency. 322 func atoi(s string) (x int, err error) { 323 neg := false 324 if s != "" && s[0] == '-' { 325 neg = true 326 s = s[1:] 327 } 328 q, rem, err := leadingInt(s) 329 x = int(q) 330 if err != nil || rem != "" { 331 return 0, atoiError 332 } 333 if neg { 334 x = -x 335 } 336 return x, nil 337 } 338 339 // formatNano appends a fractional second, as nanoseconds, to b 340 // and returns the result. 341 func formatNano(b []byte, nanosec uint, n int, trim bool) []byte { 342 u := nanosec 343 var buf [9]byte 344 for start := len(buf); start > 0; { 345 start-- 346 buf[start] = byte(u%10 + '0') 347 u /= 10 348 } 349 350 if n > 9 { 351 n = 9 352 } 353 if trim { 354 for n > 0 && buf[n-1] == '0' { 355 n-- 356 } 357 if n == 0 { 358 return b 359 } 360 } 361 b = append(b, '.') 362 return append(b, buf[:n]...) 363 } 364 365 // String returns the time formatted using the format string 366 // "2006-01-02 15:04:05.999999999 -0700 MST" 367 func (t Time) String() string { 368 return t.Format("2006-01-02 15:04:05.999999999 -0700 MST") 369 } 370 371 // Format returns a textual representation of the time value formatted 372 // according to layout, which defines the format by showing how the reference 373 // time, 374 // Mon Jan 2 15:04:05 -0700 MST 2006 375 // would be displayed if it were the value; it serves as an example of the 376 // desired output. The same display rules will then be applied to the time 377 // value. 378 // Predefined layouts ANSIC, UnixDate, RFC3339 and others describe standard 379 // and convenient representations of the reference time. For more information 380 // about the formats and the definition of the reference time, see the 381 // documentation for ANSIC and the other constants defined by this package. 382 func (t Time) Format(layout string) string { 383 var ( 384 name, offset, abs = t.locabs() 385 386 year int = -1 387 month Month 388 day int 389 hour int = -1 390 min int 391 sec int 392 393 b []byte 394 buf [64]byte 395 ) 396 max := len(layout) + 10 397 if max <= len(buf) { 398 b = buf[:0] 399 } else { 400 b = make([]byte, 0, max) 401 } 402 // Each iteration generates one std value. 403 for layout != "" { 404 prefix, std, suffix := nextStdChunk(layout) 405 if prefix != "" { 406 b = append(b, prefix...) 407 } 408 if std == 0 { 409 break 410 } 411 layout = suffix 412 413 // Compute year, month, day if needed. 414 if year < 0 && std&stdNeedDate != 0 { 415 year, month, day, _ = absDate(abs, true) 416 } 417 418 // Compute hour, minute, second if needed. 419 if hour < 0 && std&stdNeedClock != 0 { 420 hour, min, sec = absClock(abs) 421 } 422 423 switch std & stdMask { 424 case stdYear: 425 y := year 426 if y < 0 { 427 y = -y 428 } 429 b = appendUint(b, uint(y%100), '0') 430 case stdLongYear: 431 // Pad year to at least 4 digits. 432 y := year 433 switch { 434 case year <= -1000: 435 b = append(b, '-') 436 y = -y 437 case year <= -100: 438 b = append(b, "-0"...) 439 y = -y 440 case year <= -10: 441 b = append(b, "-00"...) 442 y = -y 443 case year < 0: 444 b = append(b, "-000"...) 445 y = -y 446 case year < 10: 447 b = append(b, "000"...) 448 case year < 100: 449 b = append(b, "00"...) 450 case year < 1000: 451 b = append(b, '0') 452 } 453 b = appendUint(b, uint(y), 0) 454 case stdMonth: 455 b = append(b, month.String()[:3]...) 456 case stdLongMonth: 457 m := month.String() 458 b = append(b, m...) 459 case stdNumMonth: 460 b = appendUint(b, uint(month), 0) 461 case stdZeroMonth: 462 b = appendUint(b, uint(month), '0') 463 case stdWeekDay: 464 b = append(b, absWeekday(abs).String()[:3]...) 465 case stdLongWeekDay: 466 s := absWeekday(abs).String() 467 b = append(b, s...) 468 case stdDay: 469 b = appendUint(b, uint(day), 0) 470 case stdUnderDay: 471 b = appendUint(b, uint(day), ' ') 472 case stdZeroDay: 473 b = appendUint(b, uint(day), '0') 474 case stdHour: 475 b = appendUint(b, uint(hour), '0') 476 case stdHour12: 477 // Noon is 12PM, midnight is 12AM. 478 hr := hour % 12 479 if hr == 0 { 480 hr = 12 481 } 482 b = appendUint(b, uint(hr), 0) 483 case stdZeroHour12: 484 // Noon is 12PM, midnight is 12AM. 485 hr := hour % 12 486 if hr == 0 { 487 hr = 12 488 } 489 b = appendUint(b, uint(hr), '0') 490 case stdMinute: 491 b = appendUint(b, uint(min), 0) 492 case stdZeroMinute: 493 b = appendUint(b, uint(min), '0') 494 case stdSecond: 495 b = appendUint(b, uint(sec), 0) 496 case stdZeroSecond: 497 b = appendUint(b, uint(sec), '0') 498 case stdPM: 499 if hour >= 12 { 500 b = append(b, "PM"...) 501 } else { 502 b = append(b, "AM"...) 503 } 504 case stdpm: 505 if hour >= 12 { 506 b = append(b, "pm"...) 507 } else { 508 b = append(b, "am"...) 509 } 510 case stdISO8601TZ, stdISO8601ColonTZ, stdNumTZ, stdNumColonTZ: 511 // Ugly special case. We cheat and take the "Z" variants 512 // to mean "the time zone as formatted for ISO 8601". 513 if offset == 0 && (std == stdISO8601TZ || std == stdISO8601ColonTZ) { 514 b = append(b, 'Z') 515 break 516 } 517 zone := offset / 60 // convert to minutes 518 if zone < 0 { 519 b = append(b, '-') 520 zone = -zone 521 } else { 522 b = append(b, '+') 523 } 524 b = appendUint(b, uint(zone/60), '0') 525 if std == stdISO8601ColonTZ || std == stdNumColonTZ { 526 b = append(b, ':') 527 } 528 b = appendUint(b, uint(zone%60), '0') 529 case stdTZ: 530 if name != "" { 531 b = append(b, name...) 532 break 533 } 534 // No time zone known for this time, but we must print one. 535 // Use the -0700 format. 536 zone := offset / 60 // convert to minutes 537 if zone < 0 { 538 b = append(b, '-') 539 zone = -zone 540 } else { 541 b = append(b, '+') 542 } 543 b = appendUint(b, uint(zone/60), '0') 544 b = appendUint(b, uint(zone%60), '0') 545 case stdFracSecond0, stdFracSecond9: 546 b = formatNano(b, uint(t.Nanosecond()), std>>stdArgShift, std&stdMask == stdFracSecond9) 547 } 548 } 549 return string(b) 550 } 551 552 var errBad = errors.New("bad value for field") // placeholder not passed to user 553 554 // ParseError describes a problem parsing a time string. 555 type ParseError struct { 556 Layout string 557 Value string 558 LayoutElem string 559 ValueElem string 560 Message string 561 } 562 563 func quote(s string) string { 564 return "\"" + s + "\"" 565 } 566 567 // Error returns the string representation of a ParseError. 568 func (e *ParseError) Error() string { 569 if e.Message == "" { 570 return "parsing time " + 571 quote(e.Value) + " as " + 572 quote(e.Layout) + ": cannot parse " + 573 quote(e.ValueElem) + " as " + 574 quote(e.LayoutElem) 575 } 576 return "parsing time " + 577 quote(e.Value) + e.Message 578 } 579 580 // isDigit returns true if s[i] is a decimal digit, false if not or 581 // if s[i] is out of range. 582 func isDigit(s string, i int) bool { 583 if len(s) <= i { 584 return false 585 } 586 c := s[i] 587 return '0' <= c && c <= '9' 588 } 589 590 // getnum parses s[0:1] or s[0:2] (fixed forces the latter) 591 // as a decimal integer and returns the integer and the 592 // remainder of the string. 593 func getnum(s string, fixed bool) (int, string, error) { 594 if !isDigit(s, 0) { 595 return 0, s, errBad 596 } 597 if !isDigit(s, 1) { 598 if fixed { 599 return 0, s, errBad 600 } 601 return int(s[0] - '0'), s[1:], nil 602 } 603 return int(s[0]-'0')*10 + int(s[1]-'0'), s[2:], nil 604 } 605 606 func cutspace(s string) string { 607 for len(s) > 0 && s[0] == ' ' { 608 s = s[1:] 609 } 610 return s 611 } 612 613 // skip removes the given prefix from value, 614 // treating runs of space characters as equivalent. 615 func skip(value, prefix string) (string, error) { 616 for len(prefix) > 0 { 617 if prefix[0] == ' ' { 618 if len(value) > 0 && value[0] != ' ' { 619 return value, errBad 620 } 621 prefix = cutspace(prefix) 622 value = cutspace(value) 623 continue 624 } 625 if len(value) == 0 || value[0] != prefix[0] { 626 return value, errBad 627 } 628 prefix = prefix[1:] 629 value = value[1:] 630 } 631 return value, nil 632 } 633 634 // Parse parses a formatted string and returns the time value it represents. 635 // The layout defines the format by showing how the reference time, 636 // Mon Jan 2 15:04:05 -0700 MST 2006 637 // would be interpreted if it were the value; it serves as an example of 638 // the input format. The same interpretation will then be made to the 639 // input string. 640 // Predefined layouts ANSIC, UnixDate, RFC3339 and others describe standard 641 // and convenient representations of the reference time. For more information 642 // about the formats and the definition of the reference time, see the 643 // documentation for ANSIC and the other constants defined by this package. 644 // 645 // Elements omitted from the value are assumed to be zero or, when 646 // zero is impossible, one, so parsing "3:04pm" returns the time 647 // corresponding to Jan 1, year 0, 15:04:00 UTC (note that because the year is 648 // 0, this time is before the zero Time). 649 // Years must be in the range 0000..9999. The day of the week is checked 650 // for syntax but it is otherwise ignored. 651 // 652 // In the absence of a time zone indicator, Parse returns a time in UTC. 653 // 654 // When parsing a time with a zone offset like -0700, if the offset corresponds 655 // to a time zone used by the current location (Local), then Parse uses that 656 // location and zone in the returned time. Otherwise it records the time as 657 // being in a fabricated location with time fixed at the given zone offset. 658 // 659 // When parsing a time with a zone abbreviation like MST, if the zone abbreviation 660 // has a defined offset in the current location, then that offset is used. 661 // The zone abbreviation "UTC" is recognized as UTC regardless of location. 662 // If the zone abbreviation is unknown, Parse records the time as being 663 // in a fabricated location with the given zone abbreviation and a zero offset. 664 // This choice means that such a time can be parse and reformatted with the 665 // same layout losslessly, but the exact instant used in the representation will 666 // differ by the actual zone offset. To avoid such problems, prefer time layouts 667 // that use a numeric zone offset, or use ParseInLocation. 668 func Parse(layout, value string) (Time, error) { 669 return parse(layout, value, UTC, Local) 670 } 671 672 // ParseInLocation is like Parse but differs in two important ways. 673 // First, in the absence of time zone information, Parse interprets a time as UTC; 674 // ParseInLocation interprets the time as in the given location. 675 // Second, when given a zone offset or abbreviation, Parse tries to match it 676 // against the Local location; ParseInLocation uses the given location. 677 func ParseInLocation(layout, value string, loc *Location) (Time, error) { 678 return parse(layout, value, loc, loc) 679 } 680 681 func parse(layout, value string, defaultLocation, local *Location) (Time, error) { 682 alayout, avalue := layout, value 683 rangeErrString := "" // set if a value is out of range 684 amSet := false // do we need to subtract 12 from the hour for midnight? 685 pmSet := false // do we need to add 12 to the hour? 686 687 // Time being constructed. 688 var ( 689 year int 690 month int = 1 // January 691 day int = 1 692 hour int 693 min int 694 sec int 695 nsec int 696 z *Location 697 zoneOffset int = -1 698 zoneName string 699 ) 700 701 // Each iteration processes one std value. 702 for { 703 var err error 704 prefix, std, suffix := nextStdChunk(layout) 705 stdstr := layout[len(prefix) : len(layout)-len(suffix)] 706 value, err = skip(value, prefix) 707 if err != nil { 708 return Time{}, &ParseError{alayout, avalue, prefix, value, ""} 709 } 710 if std == 0 { 711 if len(value) != 0 { 712 return Time{}, &ParseError{alayout, avalue, "", value, ": extra text: " + value} 713 } 714 break 715 } 716 layout = suffix 717 var p string 718 switch std & stdMask { 719 case stdYear: 720 if len(value) < 2 { 721 err = errBad 722 break 723 } 724 p, value = value[0:2], value[2:] 725 year, err = atoi(p) 726 if year >= 69 { // Unix time starts Dec 31 1969 in some time zones 727 year += 1900 728 } else { 729 year += 2000 730 } 731 case stdLongYear: 732 if len(value) < 4 || !isDigit(value, 0) { 733 err = errBad 734 break 735 } 736 p, value = value[0:4], value[4:] 737 year, err = atoi(p) 738 case stdMonth: 739 month, value, err = lookup(shortMonthNames, value) 740 case stdLongMonth: 741 month, value, err = lookup(longMonthNames, value) 742 case stdNumMonth, stdZeroMonth: 743 month, value, err = getnum(value, std == stdZeroMonth) 744 if month <= 0 || 12 < month { 745 rangeErrString = "month" 746 } 747 case stdWeekDay: 748 // Ignore weekday except for error checking. 749 _, value, err = lookup(shortDayNames, value) 750 case stdLongWeekDay: 751 _, value, err = lookup(longDayNames, value) 752 case stdDay, stdUnderDay, stdZeroDay: 753 if std == stdUnderDay && len(value) > 0 && value[0] == ' ' { 754 value = value[1:] 755 } 756 day, value, err = getnum(value, std == stdZeroDay) 757 if day < 0 || 31 < day { 758 rangeErrString = "day" 759 } 760 case stdHour: 761 hour, value, err = getnum(value, false) 762 if hour < 0 || 24 <= hour { 763 rangeErrString = "hour" 764 } 765 case stdHour12, stdZeroHour12: 766 hour, value, err = getnum(value, std == stdZeroHour12) 767 if hour < 0 || 12 < hour { 768 rangeErrString = "hour" 769 } 770 case stdMinute, stdZeroMinute: 771 min, value, err = getnum(value, std == stdZeroMinute) 772 if min < 0 || 60 <= min { 773 rangeErrString = "minute" 774 } 775 case stdSecond, stdZeroSecond: 776 sec, value, err = getnum(value, std == stdZeroSecond) 777 if sec < 0 || 60 <= sec { 778 rangeErrString = "second" 779 } 780 // Special case: do we have a fractional second but no 781 // fractional second in the format? 782 if len(value) >= 2 && value[0] == '.' && isDigit(value, 1) { 783 _, std, _ := nextStdChunk(layout) 784 std &= stdMask 785 if std == stdFracSecond0 || std == stdFracSecond9 { 786 // Fractional second in the layout; proceed normally 787 break 788 } 789 // No fractional second in the layout but we have one in the input. 790 n := 2 791 for ; n < len(value) && isDigit(value, n); n++ { 792 } 793 nsec, rangeErrString, err = parseNanoseconds(value, n) 794 value = value[n:] 795 } 796 case stdPM: 797 if len(value) < 2 { 798 err = errBad 799 break 800 } 801 p, value = value[0:2], value[2:] 802 switch p { 803 case "PM": 804 pmSet = true 805 case "AM": 806 amSet = true 807 default: 808 err = errBad 809 } 810 case stdpm: 811 if len(value) < 2 { 812 err = errBad 813 break 814 } 815 p, value = value[0:2], value[2:] 816 switch p { 817 case "pm": 818 pmSet = true 819 case "am": 820 amSet = true 821 default: 822 err = errBad 823 } 824 case stdISO8601TZ, stdISO8601ColonTZ, stdNumTZ, stdNumShortTZ, stdNumColonTZ: 825 if (std == stdISO8601TZ || std == stdISO8601ColonTZ) && len(value) >= 1 && value[0] == 'Z' { 826 value = value[1:] 827 z = UTC 828 break 829 } 830 var sign, hour, min string 831 if std == stdISO8601ColonTZ || std == stdNumColonTZ { 832 if len(value) < 6 { 833 err = errBad 834 break 835 } 836 if value[3] != ':' { 837 err = errBad 838 break 839 } 840 sign, hour, min, value = value[0:1], value[1:3], value[4:6], value[6:] 841 } else if std == stdNumShortTZ { 842 if len(value) < 3 { 843 err = errBad 844 break 845 } 846 sign, hour, min, value = value[0:1], value[1:3], "00", value[3:] 847 } else { 848 if len(value) < 5 { 849 err = errBad 850 break 851 } 852 sign, hour, min, value = value[0:1], value[1:3], value[3:5], value[5:] 853 } 854 var hr, mm int 855 hr, err = atoi(hour) 856 if err == nil { 857 mm, err = atoi(min) 858 } 859 zoneOffset = (hr*60 + mm) * 60 // offset is in seconds 860 switch sign[0] { 861 case '+': 862 case '-': 863 zoneOffset = -zoneOffset 864 default: 865 err = errBad 866 } 867 case stdTZ: 868 // Does it look like a time zone? 869 if len(value) >= 3 && value[0:3] == "UTC" { 870 z = UTC 871 value = value[3:] 872 break 873 } 874 875 if len(value) >= 3 && value[2] == 'T' { 876 p, value = value[0:3], value[3:] 877 } else if len(value) >= 4 && value[3] == 'T' { 878 p, value = value[0:4], value[4:] 879 } else { 880 err = errBad 881 break 882 } 883 for i := 0; i < len(p); i++ { 884 if p[i] < 'A' || 'Z' < p[i] { 885 err = errBad 886 } 887 } 888 if err != nil { 889 break 890 } 891 // It's a valid format. 892 zoneName = p 893 894 case stdFracSecond0: 895 // stdFracSecond0 requires the exact number of digits as specified in 896 // the layout. 897 ndigit := 1 + (std >> stdArgShift) 898 if len(value) < ndigit { 899 err = errBad 900 break 901 } 902 nsec, rangeErrString, err = parseNanoseconds(value, ndigit) 903 value = value[ndigit:] 904 905 case stdFracSecond9: 906 if len(value) < 2 || value[0] != '.' || value[1] < '0' || '9' < value[1] { 907 // Fractional second omitted. 908 break 909 } 910 // Take any number of digits, even more than asked for, 911 // because it is what the stdSecond case would do. 912 i := 0 913 for i < 9 && i+1 < len(value) && '0' <= value[i+1] && value[i+1] <= '9' { 914 i++ 915 } 916 nsec, rangeErrString, err = parseNanoseconds(value, 1+i) 917 value = value[1+i:] 918 } 919 if rangeErrString != "" { 920 return Time{}, &ParseError{alayout, avalue, stdstr, value, ": " + rangeErrString + " out of range"} 921 } 922 if err != nil { 923 return Time{}, &ParseError{alayout, avalue, stdstr, value, ""} 924 } 925 } 926 if pmSet && hour < 12 { 927 hour += 12 928 } else if amSet && hour == 12 { 929 hour = 0 930 } 931 932 if z != nil { 933 return Date(year, Month(month), day, hour, min, sec, nsec, z), nil 934 } 935 936 if zoneOffset != -1 { 937 t := Date(year, Month(month), day, hour, min, sec, nsec, UTC) 938 t.sec -= int64(zoneOffset) 939 940 // Look for local zone with the given offset. 941 // If that zone was in effect at the given time, use it. 942 name, offset, _, _, _ := local.lookup(t.sec + internalToUnix) 943 if offset == zoneOffset && (zoneName == "" || name == zoneName) { 944 t.loc = local 945 return t, nil 946 } 947 948 // Otherwise create fake zone to record offset. 949 t.loc = FixedZone(zoneName, zoneOffset) 950 return t, nil 951 } 952 953 if zoneName != "" { 954 t := Date(year, Month(month), day, hour, min, sec, nsec, UTC) 955 // Look for local zone with the given offset. 956 // If that zone was in effect at the given time, use it. 957 offset, _, ok := local.lookupName(zoneName, t.sec+internalToUnix) 958 if ok { 959 t.sec -= int64(offset) 960 t.loc = local 961 return t, nil 962 } 963 964 // Otherwise, create fake zone with unknown offset. 965 t.loc = FixedZone(zoneName, 0) 966 return t, nil 967 } 968 969 // Otherwise, fall back to default. 970 return Date(year, Month(month), day, hour, min, sec, nsec, defaultLocation), nil 971 } 972 973 func parseNanoseconds(value string, nbytes int) (ns int, rangeErrString string, err error) { 974 if value[0] != '.' { 975 err = errBad 976 return 977 } 978 if ns, err = atoi(value[1:nbytes]); err != nil { 979 return 980 } 981 if ns < 0 || 1e9 <= ns { 982 rangeErrString = "fractional second" 983 return 984 } 985 // We need nanoseconds, which means scaling by the number 986 // of missing digits in the format, maximum length 10. If it's 987 // longer than 10, we won't scale. 988 scaleDigits := 10 - nbytes 989 for i := 0; i < scaleDigits; i++ { 990 ns *= 10 991 } 992 return 993 } 994 995 var errLeadingInt = errors.New("time: bad [0-9]*") // never printed 996 997 // leadingInt consumes the leading [0-9]* from s. 998 func leadingInt(s string) (x int64, rem string, err error) { 999 i := 0 1000 for ; i < len(s); i++ { 1001 c := s[i] 1002 if c < '0' || c > '9' { 1003 break 1004 } 1005 if x >= (1<<63-10)/10 { 1006 // overflow 1007 return 0, "", errLeadingInt 1008 } 1009 x = x*10 + int64(c) - '0' 1010 } 1011 return x, s[i:], nil 1012 } 1013 1014 var unitMap = map[string]float64{ 1015 "ns": float64(Nanosecond), 1016 "us": float64(Microsecond), 1017 "µs": float64(Microsecond), // U+00B5 = micro symbol 1018 "μs": float64(Microsecond), // U+03BC = Greek letter mu 1019 "ms": float64(Millisecond), 1020 "s": float64(Second), 1021 "m": float64(Minute), 1022 "h": float64(Hour), 1023 } 1024 1025 // ParseDuration parses a duration string. 1026 // A duration string is a possibly signed sequence of 1027 // decimal numbers, each with optional fraction and a unit suffix, 1028 // such as "300ms", "-1.5h" or "2h45m". 1029 // Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h". 1030 func ParseDuration(s string) (Duration, error) { 1031 // [-+]?([0-9]*(\.[0-9]*)?[a-z]+)+ 1032 orig := s 1033 f := float64(0) 1034 neg := false 1035 1036 // Consume [-+]? 1037 if s != "" { 1038 c := s[0] 1039 if c == '-' || c == '+' { 1040 neg = c == '-' 1041 s = s[1:] 1042 } 1043 } 1044 // Special case: if all that is left is "0", this is zero. 1045 if s == "0" { 1046 return 0, nil 1047 } 1048 if s == "" { 1049 return 0, errors.New("time: invalid duration " + orig) 1050 } 1051 for s != "" { 1052 g := float64(0) // this element of the sequence 1053 1054 var x int64 1055 var err error 1056 1057 // The next character must be [0-9.] 1058 if !(s[0] == '.' || ('0' <= s[0] && s[0] <= '9')) { 1059 return 0, errors.New("time: invalid duration " + orig) 1060 } 1061 // Consume [0-9]* 1062 pl := len(s) 1063 x, s, err = leadingInt(s) 1064 if err != nil { 1065 return 0, errors.New("time: invalid duration " + orig) 1066 } 1067 g = float64(x) 1068 pre := pl != len(s) // whether we consumed anything before a period 1069 1070 // Consume (\.[0-9]*)? 1071 post := false 1072 if s != "" && s[0] == '.' { 1073 s = s[1:] 1074 pl := len(s) 1075 x, s, err = leadingInt(s) 1076 if err != nil { 1077 return 0, errors.New("time: invalid duration " + orig) 1078 } 1079 scale := 1 1080 for n := pl - len(s); n > 0; n-- { 1081 scale *= 10 1082 } 1083 g += float64(x) / float64(scale) 1084 post = pl != len(s) 1085 } 1086 if !pre && !post { 1087 // no digits (e.g. ".s" or "-.s") 1088 return 0, errors.New("time: invalid duration " + orig) 1089 } 1090 1091 // Consume unit. 1092 i := 0 1093 for ; i < len(s); i++ { 1094 c := s[i] 1095 if c == '.' || ('0' <= c && c <= '9') { 1096 break 1097 } 1098 } 1099 if i == 0 { 1100 return 0, errors.New("time: missing unit in duration " + orig) 1101 } 1102 u := s[:i] 1103 s = s[i:] 1104 unit, ok := unitMap[u] 1105 if !ok { 1106 return 0, errors.New("time: unknown unit " + u + " in duration " + orig) 1107 } 1108 1109 f += g * unit 1110 } 1111 1112 if neg { 1113 f = -f 1114 } 1115 return Duration(f), nil 1116 }