package persistence import ( "fmt" "regexp" "strconv" "time" ) var yearRE = regexp.MustCompile(`^(\d{4})$`) var monthRE = regexp.MustCompile(`^(\d{4})-(\d{2})$`) var weekRE = regexp.MustCompile(`^(\d{4})-W(\d{2})$`) var dayRE = regexp.MustCompile(`^(\d{4})-(\d{2})-(\d{2})$`) var hourRE = regexp.MustCompile(`^(\d{4})-(\d{2})-(\d{2})T(\d{2})$`) type Granularity int const ( Year Granularity = iota Month Week Day Hour ) // ParseISO8601 is **not** a function to parse all and every kind of valid ISO 8601 // date, nor it's intended to be, since we don't need that. func ParseISO8601(s string) (*time.Time, Granularity, error) { if matches := yearRE.FindStringSubmatch(s); len(matches) != 0 { year, err := parseYear(matches[1]) if err != nil { return nil, -1, err } t := time.Date(year, time.December, daysOfMonth(time.December, year), 23, 59, 59, 0, time.UTC) return &t, Year, nil } if matches := monthRE.FindStringSubmatch(s); len(matches) != 0 { month, err := parseMonth(matches[2]) if err != nil { return nil, -1, err } year, err := parseYear(matches[1]) if err != nil { return nil, -1, err } t := time.Date(year, month, 31, 23, 59, 59, 0, time.UTC) return &t, Month, nil } if matches := weekRE.FindStringSubmatch(s); len(matches) != 0 { week, err := parseWeek(matches[2]) if err != nil { return nil, -1, err } year, err := parseYear(matches[1]) if err != nil { return nil, -1, err } t := time.Date(year, time.January, week*7, 23, 59, 59, 0, time.UTC) return &t, Week, nil } if matches := dayRE.FindStringSubmatch(s); len(matches) != 0 { month, err := parseMonth(matches[2]) if err != nil { return nil, -1, err } year, err := parseYear(matches[1]) if err != nil { return nil, -1, err } day, err := parseDay(matches[3], daysOfMonth(month, year)) if err != nil { return nil, -1, err } t := time.Date(year, month, day, 23, 59, 59, 0, time.UTC) return &t, Day, nil } if matches := hourRE.FindStringSubmatch(s); len(matches) != 0 { month, err := parseMonth(matches[2]) if err != nil { return nil, -1, err } year, err := parseYear(matches[1]) if err != nil { return nil, -1, err } hour, err := parseHour(matches[4]) if err != nil { return nil, -1, err } day, err := parseDay(matches[3], daysOfMonth(month, year)) if err != nil { return nil, -1, err } t := time.Date(year, month, day, hour, 59, 59, 0, time.UTC) return &t, Hour, nil } return nil, -1, fmt.Errorf("string does not match any formats") } func daysOfMonth(month time.Month, year int) int { switch month { case time.January: return 31 case time.February: if isLeap(year) { return 29 } else { return 28 } case time.March: return 31 case time.April: return 30 case time.May: return 31 case time.June: return 30 case time.July: return 31 case time.August: return 31 case time.September: return 30 case time.October: return 31 case time.November: return 30 case time.December: return 31 default: panic("invalid month!") } } func isLeap(year int) bool { if year%4 != 0 { return false } else if year%100 != 0 { return true } else if year%400 != 0 { return false } else { return true } } func atoi(s string) int { i, e := strconv.Atoi(s) if e != nil { // panic on error since atoi() will be called only after we parse it with regex // (hopefully `\d`!) panic(e.Error()) } return i } func parseYear(s string) (int, error) { year := atoi(s) if year <= 1583 { return 0, fmt.Errorf("years before 1583 are not allowed") } return year, nil } func parseMonth(s string) (time.Month, error) { month := atoi(s) if month <= 0 || month >= 13 { return time.Month(-1), fmt.Errorf("month is not in range [01, 12]") } return time.Month(month), nil } func parseWeek(s string) (int, error) { week := atoi(s) if week <= 0 || week >= 54 { return -1, fmt.Errorf("week is not in range [01, 53]") } return week, nil } func parseDay(s string, max int) (int, error) { day := atoi(s) if day <= 0 || day > max { return -1, fmt.Errorf("day is not in range [01, %d]", max) } return day, nil } func parseHour(s string) (int, error) { hour := atoi(s) if hour <= -1 || hour >= 25 { return -1, fmt.Errorf("hour is not in range [00, 24]") } return hour, nil }