date_parser.go 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860
  1. package goja
  2. // This is a slightly modified version of the standard Go parser to make it more compatible with ECMAScript 5.1
  3. // Changes:
  4. // - 6-digit extended years are supported in place of long year (2006) in the form of +123456
  5. // - Timezone formats tolerate colons, e.g. -0700 will parse -07:00
  6. // - Short week day will also parse long week day
  7. // - Timezone in brackets, "(MST)", will match any string in brackets (e.g. "(GMT Standard Time)")
  8. // - If offset is not set and timezone name is unknown, an error is returned
  9. // - If offset and timezone name are both set the offset takes precedence and the resulting Location will be FixedZone("", offset)
  10. // Original copyright message:
  11. // Copyright 2010 The Go Authors. All rights reserved.
  12. // Use of this source code is governed by a BSD-style
  13. // license that can be found in the LICENSE file.
  14. import (
  15. "errors"
  16. "time"
  17. )
  18. const (
  19. _ = iota
  20. stdLongMonth = iota + stdNeedDate // "January"
  21. stdMonth // "Jan"
  22. stdNumMonth // "1"
  23. stdZeroMonth // "01"
  24. stdLongWeekDay // "Monday"
  25. stdWeekDay // "Mon"
  26. stdDay // "2"
  27. stdUnderDay // "_2"
  28. stdZeroDay // "02"
  29. stdHour = iota + stdNeedClock // "15"
  30. stdHour12 // "3"
  31. stdZeroHour12 // "03"
  32. stdMinute // "4"
  33. stdZeroMinute // "04"
  34. stdSecond // "5"
  35. stdZeroSecond // "05"
  36. stdLongYear = iota + stdNeedDate // "2006"
  37. stdYear // "06"
  38. stdPM = iota + stdNeedClock // "PM"
  39. stdpm // "pm"
  40. stdTZ = iota // "MST"
  41. stdBracketTZ // "(MST)"
  42. stdISO8601TZ // "Z0700" // prints Z for UTC
  43. stdISO8601SecondsTZ // "Z070000"
  44. stdISO8601ShortTZ // "Z07"
  45. stdISO8601ColonTZ // "Z07:00" // prints Z for UTC
  46. stdISO8601ColonSecondsTZ // "Z07:00:00"
  47. stdNumTZ // "-0700" // always numeric
  48. stdNumSecondsTz // "-070000"
  49. stdNumShortTZ // "-07" // always numeric
  50. stdNumColonTZ // "-07:00" // always numeric
  51. stdNumColonSecondsTZ // "-07:00:00"
  52. stdFracSecond0 // ".0", ".00", ... , trailing zeros included
  53. stdFracSecond9 // ".9", ".99", ..., trailing zeros omitted
  54. stdNeedDate = 1 << 8 // need month, day, year
  55. stdNeedClock = 2 << 8 // need hour, minute, second
  56. stdArgShift = 16 // extra argument in high bits, above low stdArgShift
  57. stdMask = 1<<stdArgShift - 1 // mask out argument
  58. )
  59. var errBad = errors.New("bad value for field") // placeholder not passed to user
  60. func parseDate(layout, value string, defaultLocation *time.Location) (time.Time, error) {
  61. alayout, avalue := layout, value
  62. rangeErrString := "" // set if a value is out of range
  63. amSet := false // do we need to subtract 12 from the hour for midnight?
  64. pmSet := false // do we need to add 12 to the hour?
  65. // Time being constructed.
  66. var (
  67. year int
  68. month int = 1 // January
  69. day int = 1
  70. hour int
  71. min int
  72. sec int
  73. nsec int
  74. z *time.Location
  75. zoneOffset int = -1
  76. zoneName string
  77. )
  78. // Each iteration processes one std value.
  79. for {
  80. var err error
  81. prefix, std, suffix := nextStdChunk(layout)
  82. stdstr := layout[len(prefix) : len(layout)-len(suffix)]
  83. value, err = skip(value, prefix)
  84. if err != nil {
  85. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, LayoutElem: prefix, ValueElem: value}
  86. }
  87. if std == 0 {
  88. if len(value) != 0 {
  89. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, ValueElem: value, Message: ": extra text: " + value}
  90. }
  91. break
  92. }
  93. layout = suffix
  94. var p string
  95. switch std & stdMask {
  96. case stdYear:
  97. if len(value) < 2 {
  98. err = errBad
  99. break
  100. }
  101. p, value = value[0:2], value[2:]
  102. year, err = atoi(p)
  103. if year >= 69 { // Unix time starts Dec 31 1969 in some time zones
  104. year += 1900
  105. } else {
  106. year += 2000
  107. }
  108. case stdLongYear:
  109. if len(value) >= 7 && (value[0] == '-' || value[0] == '+') { // extended year
  110. neg := value[0] == '-'
  111. p, value = value[1:7], value[7:]
  112. year, err = atoi(p)
  113. if neg {
  114. year = -year
  115. }
  116. } else {
  117. if len(value) < 4 || !isDigit(value, 0) {
  118. err = errBad
  119. break
  120. }
  121. p, value = value[0:4], value[4:]
  122. year, err = atoi(p)
  123. }
  124. case stdMonth:
  125. month, value, err = lookup(shortMonthNames, value)
  126. month++
  127. case stdLongMonth:
  128. month, value, err = lookup(longMonthNames, value)
  129. month++
  130. case stdNumMonth, stdZeroMonth:
  131. month, value, err = getnum(value, std == stdZeroMonth)
  132. if month <= 0 || 12 < month {
  133. rangeErrString = "month"
  134. }
  135. case stdWeekDay:
  136. // Ignore weekday except for error checking.
  137. _, value, err = lookup(longDayNames, value)
  138. if err != nil {
  139. _, value, err = lookup(shortDayNames, value)
  140. }
  141. case stdLongWeekDay:
  142. _, value, err = lookup(longDayNames, value)
  143. case stdDay, stdUnderDay, stdZeroDay:
  144. if std == stdUnderDay && len(value) > 0 && value[0] == ' ' {
  145. value = value[1:]
  146. }
  147. day, value, err = getnum(value, std == stdZeroDay)
  148. if day < 0 {
  149. // Note that we allow any one- or two-digit day here.
  150. rangeErrString = "day"
  151. }
  152. case stdHour:
  153. hour, value, err = getnum(value, false)
  154. if hour < 0 || 24 <= hour {
  155. rangeErrString = "hour"
  156. }
  157. case stdHour12, stdZeroHour12:
  158. hour, value, err = getnum(value, std == stdZeroHour12)
  159. if hour < 0 || 12 < hour {
  160. rangeErrString = "hour"
  161. }
  162. case stdMinute, stdZeroMinute:
  163. min, value, err = getnum(value, std == stdZeroMinute)
  164. if min < 0 || 60 <= min {
  165. rangeErrString = "minute"
  166. }
  167. case stdSecond, stdZeroSecond:
  168. sec, value, err = getnum(value, std == stdZeroSecond)
  169. if sec < 0 || 60 <= sec {
  170. rangeErrString = "second"
  171. break
  172. }
  173. // Special case: do we have a fractional second but no
  174. // fractional second in the format?
  175. if len(value) >= 2 && value[0] == '.' && isDigit(value, 1) {
  176. _, std, _ = nextStdChunk(layout)
  177. std &= stdMask
  178. if std == stdFracSecond0 || std == stdFracSecond9 {
  179. // Fractional second in the layout; proceed normally
  180. break
  181. }
  182. // No fractional second in the layout but we have one in the input.
  183. n := 2
  184. for ; n < len(value) && isDigit(value, n); n++ {
  185. }
  186. nsec, rangeErrString, err = parseNanoseconds(value, n)
  187. value = value[n:]
  188. }
  189. case stdPM:
  190. if len(value) < 2 {
  191. err = errBad
  192. break
  193. }
  194. p, value = value[0:2], value[2:]
  195. switch p {
  196. case "PM":
  197. pmSet = true
  198. case "AM":
  199. amSet = true
  200. default:
  201. err = errBad
  202. }
  203. case stdpm:
  204. if len(value) < 2 {
  205. err = errBad
  206. break
  207. }
  208. p, value = value[0:2], value[2:]
  209. switch p {
  210. case "pm":
  211. pmSet = true
  212. case "am":
  213. amSet = true
  214. default:
  215. err = errBad
  216. }
  217. case stdISO8601TZ, stdISO8601ColonTZ, stdISO8601SecondsTZ, stdISO8601ShortTZ, stdISO8601ColonSecondsTZ, stdNumTZ, stdNumShortTZ, stdNumColonTZ, stdNumSecondsTz, stdNumColonSecondsTZ:
  218. if (std == stdISO8601TZ || std == stdISO8601ShortTZ || std == stdISO8601ColonTZ ||
  219. std == stdISO8601SecondsTZ || std == stdISO8601ColonSecondsTZ) && len(value) >= 1 && value[0] == 'Z' {
  220. value = value[1:]
  221. z = time.UTC
  222. break
  223. }
  224. var sign, hour, min, seconds string
  225. if std == stdISO8601ColonTZ || std == stdNumColonTZ || std == stdNumTZ || std == stdISO8601TZ {
  226. if len(value) < 4 {
  227. err = errBad
  228. break
  229. }
  230. if value[3] != ':' {
  231. if std == stdNumColonTZ || std == stdISO8601ColonTZ || len(value) < 5 {
  232. err = errBad
  233. break
  234. }
  235. sign, hour, min, seconds, value = value[0:1], value[1:3], value[3:5], "00", value[5:]
  236. } else {
  237. if len(value) < 6 {
  238. err = errBad
  239. break
  240. }
  241. sign, hour, min, seconds, value = value[0:1], value[1:3], value[4:6], "00", value[6:]
  242. }
  243. } else if std == stdNumShortTZ || std == stdISO8601ShortTZ {
  244. if len(value) < 3 {
  245. err = errBad
  246. break
  247. }
  248. sign, hour, min, seconds, value = value[0:1], value[1:3], "00", "00", value[3:]
  249. } else if std == stdISO8601ColonSecondsTZ || std == stdNumColonSecondsTZ || std == stdISO8601SecondsTZ || std == stdNumSecondsTz {
  250. if len(value) < 7 {
  251. err = errBad
  252. break
  253. }
  254. if value[3] != ':' || value[6] != ':' {
  255. if std == stdISO8601ColonSecondsTZ || std == stdNumColonSecondsTZ || len(value) < 7 {
  256. err = errBad
  257. break
  258. }
  259. sign, hour, min, seconds, value = value[0:1], value[1:3], value[3:5], value[5:7], value[7:]
  260. } else {
  261. if len(value) < 9 {
  262. err = errBad
  263. break
  264. }
  265. sign, hour, min, seconds, value = value[0:1], value[1:3], value[4:6], value[7:9], value[9:]
  266. }
  267. }
  268. var hr, mm, ss int
  269. hr, err = atoi(hour)
  270. if err == nil {
  271. mm, err = atoi(min)
  272. }
  273. if err == nil {
  274. ss, err = atoi(seconds)
  275. }
  276. zoneOffset = (hr*60+mm)*60 + ss // offset is in seconds
  277. switch sign[0] {
  278. case '+':
  279. case '-':
  280. zoneOffset = -zoneOffset
  281. default:
  282. err = errBad
  283. }
  284. case stdTZ:
  285. // Does it look like a time zone?
  286. if len(value) >= 3 && value[0:3] == "UTC" {
  287. z = time.UTC
  288. value = value[3:]
  289. break
  290. }
  291. n, ok := parseTimeZone(value)
  292. if !ok {
  293. err = errBad
  294. break
  295. }
  296. zoneName, value = value[:n], value[n:]
  297. case stdBracketTZ:
  298. if len(value) < 3 || value[0] != '(' {
  299. err = errBad
  300. break
  301. }
  302. i := 1
  303. for ; ; i++ {
  304. if i >= len(value) {
  305. err = errBad
  306. break
  307. }
  308. if value[i] == ')' {
  309. zoneName, value = value[1:i], value[i+1:]
  310. break
  311. }
  312. }
  313. case stdFracSecond0:
  314. // stdFracSecond0 requires the exact number of digits as specified in
  315. // the layout.
  316. ndigit := 1 + (std >> stdArgShift)
  317. if len(value) < ndigit {
  318. err = errBad
  319. break
  320. }
  321. nsec, rangeErrString, err = parseNanoseconds(value, ndigit)
  322. value = value[ndigit:]
  323. case stdFracSecond9:
  324. if len(value) < 2 || value[0] != '.' || value[1] < '0' || '9' < value[1] {
  325. // Fractional second omitted.
  326. break
  327. }
  328. // Take any number of digits, even more than asked for,
  329. // because it is what the stdSecond case would do.
  330. i := 0
  331. for i < 9 && i+1 < len(value) && '0' <= value[i+1] && value[i+1] <= '9' {
  332. i++
  333. }
  334. nsec, rangeErrString, err = parseNanoseconds(value, 1+i)
  335. value = value[1+i:]
  336. }
  337. if rangeErrString != "" {
  338. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, LayoutElem: stdstr, ValueElem: value, Message: ": " + rangeErrString + " out of range"}
  339. }
  340. if err != nil {
  341. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, LayoutElem: stdstr, ValueElem: value}
  342. }
  343. }
  344. if pmSet && hour < 12 {
  345. hour += 12
  346. } else if amSet && hour == 12 {
  347. hour = 0
  348. }
  349. // Validate the day of the month.
  350. if day < 1 || day > daysIn(time.Month(month), year) {
  351. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, ValueElem: value, Message: ": day out of range"}
  352. }
  353. if z == nil {
  354. if zoneOffset == -1 {
  355. if zoneName != "" {
  356. if z1, err := time.LoadLocation(zoneName); err == nil {
  357. z = z1
  358. } else {
  359. return time.Time{}, &time.ParseError{Layout: alayout, Value: avalue, ValueElem: value, Message: ": unknown timezone"}
  360. }
  361. } else {
  362. z = defaultLocation
  363. }
  364. } else if zoneOffset == 0 {
  365. z = time.UTC
  366. } else {
  367. z = time.FixedZone("", zoneOffset)
  368. }
  369. }
  370. return time.Date(year, time.Month(month), day, hour, min, sec, nsec, z), nil
  371. }
  372. var errLeadingInt = errors.New("time: bad [0-9]*") // never printed
  373. func signedLeadingInt(s string) (x int64, rem string, err error) {
  374. neg := false
  375. if s != "" && (s[0] == '-' || s[0] == '+') {
  376. neg = s[0] == '-'
  377. s = s[1:]
  378. }
  379. x, rem, err = leadingInt(s)
  380. if err != nil {
  381. return
  382. }
  383. if neg {
  384. x = -x
  385. }
  386. return
  387. }
  388. // leadingInt consumes the leading [0-9]* from s.
  389. func leadingInt(s string) (x int64, rem string, err error) {
  390. i := 0
  391. for ; i < len(s); i++ {
  392. c := s[i]
  393. if c < '0' || c > '9' {
  394. break
  395. }
  396. if x > (1<<63-1)/10 {
  397. // overflow
  398. return 0, "", errLeadingInt
  399. }
  400. x = x*10 + int64(c) - '0'
  401. if x < 0 {
  402. // overflow
  403. return 0, "", errLeadingInt
  404. }
  405. }
  406. return x, s[i:], nil
  407. }
  408. // nextStdChunk finds the first occurrence of a std string in
  409. // layout and returns the text before, the std string, and the text after.
  410. func nextStdChunk(layout string) (prefix string, std int, suffix string) {
  411. for i := 0; i < len(layout); i++ {
  412. switch c := int(layout[i]); c {
  413. case 'J': // January, Jan
  414. if len(layout) >= i+3 && layout[i:i+3] == "Jan" {
  415. if len(layout) >= i+7 && layout[i:i+7] == "January" {
  416. return layout[0:i], stdLongMonth, layout[i+7:]
  417. }
  418. if !startsWithLowerCase(layout[i+3:]) {
  419. return layout[0:i], stdMonth, layout[i+3:]
  420. }
  421. }
  422. case 'M': // Monday, Mon, MST
  423. if len(layout) >= i+3 {
  424. if layout[i:i+3] == "Mon" {
  425. if len(layout) >= i+6 && layout[i:i+6] == "Monday" {
  426. return layout[0:i], stdLongWeekDay, layout[i+6:]
  427. }
  428. if !startsWithLowerCase(layout[i+3:]) {
  429. return layout[0:i], stdWeekDay, layout[i+3:]
  430. }
  431. }
  432. if layout[i:i+3] == "MST" {
  433. return layout[0:i], stdTZ, layout[i+3:]
  434. }
  435. }
  436. case '0': // 01, 02, 03, 04, 05, 06
  437. if len(layout) >= i+2 && '1' <= layout[i+1] && layout[i+1] <= '6' {
  438. return layout[0:i], std0x[layout[i+1]-'1'], layout[i+2:]
  439. }
  440. case '1': // 15, 1
  441. if len(layout) >= i+2 && layout[i+1] == '5' {
  442. return layout[0:i], stdHour, layout[i+2:]
  443. }
  444. return layout[0:i], stdNumMonth, layout[i+1:]
  445. case '2': // 2006, 2
  446. if len(layout) >= i+4 && layout[i:i+4] == "2006" {
  447. return layout[0:i], stdLongYear, layout[i+4:]
  448. }
  449. return layout[0:i], stdDay, layout[i+1:]
  450. case '_': // _2, _2006
  451. if len(layout) >= i+2 && layout[i+1] == '2' {
  452. //_2006 is really a literal _, followed by stdLongYear
  453. if len(layout) >= i+5 && layout[i+1:i+5] == "2006" {
  454. return layout[0 : i+1], stdLongYear, layout[i+5:]
  455. }
  456. return layout[0:i], stdUnderDay, layout[i+2:]
  457. }
  458. case '3':
  459. return layout[0:i], stdHour12, layout[i+1:]
  460. case '4':
  461. return layout[0:i], stdMinute, layout[i+1:]
  462. case '5':
  463. return layout[0:i], stdSecond, layout[i+1:]
  464. case 'P': // PM
  465. if len(layout) >= i+2 && layout[i+1] == 'M' {
  466. return layout[0:i], stdPM, layout[i+2:]
  467. }
  468. case 'p': // pm
  469. if len(layout) >= i+2 && layout[i+1] == 'm' {
  470. return layout[0:i], stdpm, layout[i+2:]
  471. }
  472. case '-': // -070000, -07:00:00, -0700, -07:00, -07
  473. if len(layout) >= i+7 && layout[i:i+7] == "-070000" {
  474. return layout[0:i], stdNumSecondsTz, layout[i+7:]
  475. }
  476. if len(layout) >= i+9 && layout[i:i+9] == "-07:00:00" {
  477. return layout[0:i], stdNumColonSecondsTZ, layout[i+9:]
  478. }
  479. if len(layout) >= i+5 && layout[i:i+5] == "-0700" {
  480. return layout[0:i], stdNumTZ, layout[i+5:]
  481. }
  482. if len(layout) >= i+6 && layout[i:i+6] == "-07:00" {
  483. return layout[0:i], stdNumColonTZ, layout[i+6:]
  484. }
  485. if len(layout) >= i+3 && layout[i:i+3] == "-07" {
  486. return layout[0:i], stdNumShortTZ, layout[i+3:]
  487. }
  488. case 'Z': // Z070000, Z07:00:00, Z0700, Z07:00,
  489. if len(layout) >= i+7 && layout[i:i+7] == "Z070000" {
  490. return layout[0:i], stdISO8601SecondsTZ, layout[i+7:]
  491. }
  492. if len(layout) >= i+9 && layout[i:i+9] == "Z07:00:00" {
  493. return layout[0:i], stdISO8601ColonSecondsTZ, layout[i+9:]
  494. }
  495. if len(layout) >= i+5 && layout[i:i+5] == "Z0700" {
  496. return layout[0:i], stdISO8601TZ, layout[i+5:]
  497. }
  498. if len(layout) >= i+6 && layout[i:i+6] == "Z07:00" {
  499. return layout[0:i], stdISO8601ColonTZ, layout[i+6:]
  500. }
  501. if len(layout) >= i+3 && layout[i:i+3] == "Z07" {
  502. return layout[0:i], stdISO8601ShortTZ, layout[i+3:]
  503. }
  504. case '.': // .000 or .999 - repeated digits for fractional seconds.
  505. if i+1 < len(layout) && (layout[i+1] == '0' || layout[i+1] == '9') {
  506. ch := layout[i+1]
  507. j := i + 1
  508. for j < len(layout) && layout[j] == ch {
  509. j++
  510. }
  511. // String of digits must end here - only fractional second is all digits.
  512. if !isDigit(layout, j) {
  513. std := stdFracSecond0
  514. if layout[i+1] == '9' {
  515. std = stdFracSecond9
  516. }
  517. std |= (j - (i + 1)) << stdArgShift
  518. return layout[0:i], std, layout[j:]
  519. }
  520. }
  521. case '(':
  522. if len(layout) >= i+5 && layout[i:i+5] == "(MST)" {
  523. return layout[0:i], stdBracketTZ, layout[i+5:]
  524. }
  525. }
  526. }
  527. return layout, 0, ""
  528. }
  529. var longDayNames = []string{
  530. "Sunday",
  531. "Monday",
  532. "Tuesday",
  533. "Wednesday",
  534. "Thursday",
  535. "Friday",
  536. "Saturday",
  537. }
  538. var shortDayNames = []string{
  539. "Sun",
  540. "Mon",
  541. "Tue",
  542. "Wed",
  543. "Thu",
  544. "Fri",
  545. "Sat",
  546. }
  547. var shortMonthNames = []string{
  548. "Jan",
  549. "Feb",
  550. "Mar",
  551. "Apr",
  552. "May",
  553. "Jun",
  554. "Jul",
  555. "Aug",
  556. "Sep",
  557. "Oct",
  558. "Nov",
  559. "Dec",
  560. }
  561. var longMonthNames = []string{
  562. "January",
  563. "February",
  564. "March",
  565. "April",
  566. "May",
  567. "June",
  568. "July",
  569. "August",
  570. "September",
  571. "October",
  572. "November",
  573. "December",
  574. }
  575. // isDigit reports whether s[i] is in range and is a decimal digit.
  576. func isDigit(s string, i int) bool {
  577. if len(s) <= i {
  578. return false
  579. }
  580. c := s[i]
  581. return '0' <= c && c <= '9'
  582. }
  583. // getnum parses s[0:1] or s[0:2] (fixed forces the latter)
  584. // as a decimal integer and returns the integer and the
  585. // remainder of the string.
  586. func getnum(s string, fixed bool) (int, string, error) {
  587. if !isDigit(s, 0) {
  588. return 0, s, errBad
  589. }
  590. if !isDigit(s, 1) {
  591. if fixed {
  592. return 0, s, errBad
  593. }
  594. return int(s[0] - '0'), s[1:], nil
  595. }
  596. return int(s[0]-'0')*10 + int(s[1]-'0'), s[2:], nil
  597. }
  598. func cutspace(s string) string {
  599. for len(s) > 0 && s[0] == ' ' {
  600. s = s[1:]
  601. }
  602. return s
  603. }
  604. // skip removes the given prefix from value,
  605. // treating runs of space characters as equivalent.
  606. func skip(value, prefix string) (string, error) {
  607. for len(prefix) > 0 {
  608. if prefix[0] == ' ' {
  609. if len(value) > 0 && value[0] != ' ' {
  610. return value, errBad
  611. }
  612. prefix = cutspace(prefix)
  613. value = cutspace(value)
  614. continue
  615. }
  616. if len(value) == 0 || value[0] != prefix[0] {
  617. return value, errBad
  618. }
  619. prefix = prefix[1:]
  620. value = value[1:]
  621. }
  622. return value, nil
  623. }
  624. // Never printed, just needs to be non-nil for return by atoi.
  625. var atoiError = errors.New("time: invalid number")
  626. // Duplicates functionality in strconv, but avoids dependency.
  627. func atoi(s string) (x int, err error) {
  628. q, rem, err := signedLeadingInt(s)
  629. x = int(q)
  630. if err != nil || rem != "" {
  631. return 0, atoiError
  632. }
  633. return x, nil
  634. }
  635. // match reports whether s1 and s2 match ignoring case.
  636. // It is assumed s1 and s2 are the same length.
  637. func match(s1, s2 string) bool {
  638. for i := 0; i < len(s1); i++ {
  639. c1 := s1[i]
  640. c2 := s2[i]
  641. if c1 != c2 {
  642. // Switch to lower-case; 'a'-'A' is known to be a single bit.
  643. c1 |= 'a' - 'A'
  644. c2 |= 'a' - 'A'
  645. if c1 != c2 || c1 < 'a' || c1 > 'z' {
  646. return false
  647. }
  648. }
  649. }
  650. return true
  651. }
  652. func lookup(tab []string, val string) (int, string, error) {
  653. for i, v := range tab {
  654. if len(val) >= len(v) && match(val[0:len(v)], v) {
  655. return i, val[len(v):], nil
  656. }
  657. }
  658. return -1, val, errBad
  659. }
  660. // daysBefore[m] counts the number of days in a non-leap year
  661. // before month m begins. There is an entry for m=12, counting
  662. // the number of days before January of next year (365).
  663. var daysBefore = [...]int32{
  664. 0,
  665. 31,
  666. 31 + 28,
  667. 31 + 28 + 31,
  668. 31 + 28 + 31 + 30,
  669. 31 + 28 + 31 + 30 + 31,
  670. 31 + 28 + 31 + 30 + 31 + 30,
  671. 31 + 28 + 31 + 30 + 31 + 30 + 31,
  672. 31 + 28 + 31 + 30 + 31 + 30 + 31 + 31,
  673. 31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30,
  674. 31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31,
  675. 31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30,
  676. 31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30 + 31,
  677. }
  678. func isLeap(year int) bool {
  679. return year%4 == 0 && (year%100 != 0 || year%400 == 0)
  680. }
  681. func daysIn(m time.Month, year int) int {
  682. if m == time.February && isLeap(year) {
  683. return 29
  684. }
  685. return int(daysBefore[m] - daysBefore[m-1])
  686. }
  687. // parseTimeZone parses a time zone string and returns its length. Time zones
  688. // are human-generated and unpredictable. We can't do precise error checking.
  689. // On the other hand, for a correct parse there must be a time zone at the
  690. // beginning of the string, so it's almost always true that there's one
  691. // there. We look at the beginning of the string for a run of upper-case letters.
  692. // If there are more than 5, it's an error.
  693. // If there are 4 or 5 and the last is a T, it's a time zone.
  694. // If there are 3, it's a time zone.
  695. // Otherwise, other than special cases, it's not a time zone.
  696. // GMT is special because it can have an hour offset.
  697. func parseTimeZone(value string) (length int, ok bool) {
  698. if len(value) < 3 {
  699. return 0, false
  700. }
  701. // Special case 1: ChST and MeST are the only zones with a lower-case letter.
  702. if len(value) >= 4 && (value[:4] == "ChST" || value[:4] == "MeST") {
  703. return 4, true
  704. }
  705. // Special case 2: GMT may have an hour offset; treat it specially.
  706. if value[:3] == "GMT" {
  707. length = parseGMT(value)
  708. return length, true
  709. }
  710. // Special Case 3: Some time zones are not named, but have +/-00 format
  711. if value[0] == '+' || value[0] == '-' {
  712. length = parseSignedOffset(value)
  713. return length, true
  714. }
  715. // How many upper-case letters are there? Need at least three, at most five.
  716. var nUpper int
  717. for nUpper = 0; nUpper < 6; nUpper++ {
  718. if nUpper >= len(value) {
  719. break
  720. }
  721. if c := value[nUpper]; c < 'A' || 'Z' < c {
  722. break
  723. }
  724. }
  725. switch nUpper {
  726. case 0, 1, 2, 6:
  727. return 0, false
  728. case 5: // Must end in T to match.
  729. if value[4] == 'T' {
  730. return 5, true
  731. }
  732. case 4:
  733. // Must end in T, except one special case.
  734. if value[3] == 'T' || value[:4] == "WITA" {
  735. return 4, true
  736. }
  737. case 3:
  738. return 3, true
  739. }
  740. return 0, false
  741. }
  742. // parseGMT parses a GMT time zone. The input string is known to start "GMT".
  743. // The function checks whether that is followed by a sign and a number in the
  744. // range -14 through 12 excluding zero.
  745. func parseGMT(value string) int {
  746. value = value[3:]
  747. if len(value) == 0 {
  748. return 3
  749. }
  750. return 3 + parseSignedOffset(value)
  751. }
  752. // parseSignedOffset parses a signed timezone offset (e.g. "+03" or "-04").
  753. // The function checks for a signed number in the range -14 through +12 excluding zero.
  754. // Returns length of the found offset string or 0 otherwise
  755. func parseSignedOffset(value string) int {
  756. sign := value[0]
  757. if sign != '-' && sign != '+' {
  758. return 0
  759. }
  760. x, rem, err := leadingInt(value[1:])
  761. if err != nil {
  762. return 0
  763. }
  764. if sign == '-' {
  765. x = -x
  766. }
  767. if x == 0 || x < -14 || 12 < x {
  768. return 0
  769. }
  770. return len(value) - len(rem)
  771. }
  772. func parseNanoseconds(value string, nbytes int) (ns int, rangeErrString string, err error) {
  773. if value[0] != '.' {
  774. err = errBad
  775. return
  776. }
  777. if ns, err = atoi(value[1:nbytes]); err != nil {
  778. return
  779. }
  780. if ns < 0 || 1e9 <= ns {
  781. rangeErrString = "fractional second"
  782. return
  783. }
  784. // We need nanoseconds, which means scaling by the number
  785. // of missing digits in the format, maximum length 10. If it's
  786. // longer than 10, we won't scale.
  787. scaleDigits := 10 - nbytes
  788. for i := 0; i < scaleDigits; i++ {
  789. ns *= 10
  790. }
  791. return
  792. }
  793. // std0x records the std values for "01", "02", ..., "06".
  794. var std0x = [...]int{stdZeroMonth, stdZeroDay, stdZeroHour12, stdZeroMinute, stdZeroSecond, stdYear}
  795. // startsWithLowerCase reports whether the string has a lower-case letter at the beginning.
  796. // Its purpose is to prevent matching strings like "Month" when looking for "Mon".
  797. func startsWithLowerCase(str string) bool {
  798. if len(str) == 0 {
  799. return false
  800. }
  801. c := str[0]
  802. return 'a' <= c && c <= 'z'
  803. }