package toml import ( "errors" "fmt" "math" "strconv" "strings" "time" ) func parseInteger(b []byte) (int64, error) { if len(b) > 2 && b[0] == '0' { switch b[1] { case 'x': return parseIntHex(b) case 'b': return parseIntBin(b) case 'o': return parseIntOct(b) default: return 0, fmt.Errorf("invalid base: '%c'", b[1]) } } return parseIntDec(b) } func parseLocalDate(b []byte) (LocalDate, error) { // full-date = date-fullyear "-" date-month "-" date-mday // date-fullyear = 4DIGIT // date-month = 2DIGIT ; 01-12 // date-mday = 2DIGIT ; 01-28, 01-29, 01-30, 01-31 based on month/year date := LocalDate{} if len(b) != 10 || b[4] != '-' || b[7] != '-' { return date, fmt.Errorf("dates are expected to have the format YYYY-MM-DD") } var err error date.Year, err = parseDecimalDigits(b[0:4]) if err != nil { return date, err } v, err := parseDecimalDigits(b[5:7]) if err != nil { return date, err } date.Month = time.Month(v) date.Day, err = parseDecimalDigits(b[8:10]) return date, nil } func parseDecimalDigits(b []byte) (int, error) { v := 0 for _, c := range b { if !isDigit(c) { return 0, fmt.Errorf("expected digit") } v *= 10 v += int(c - '0') } return v, nil } func parseDateTime(b []byte) (time.Time, error) { // offset-date-time = full-date time-delim full-time // full-time = partial-time time-offset // time-offset = "Z" / time-numoffset // time-numoffset = ( "+" / "-" ) time-hour ":" time-minute dt, b, err := parseLocalDateTime(b) if err != nil { return time.Time{}, err } var zone *time.Location if len(b) == 0 { return time.Time{}, fmt.Errorf("date-time missing timezone information") } if b[0] == 'Z' { b = b[1:] zone = time.UTC } else { if len(b) != 6 { return time.Time{}, fmt.Errorf("invalid date-time timezone") } direction := 1 switch b[0] { case '+': case '-': direction = -1 default: return time.Time{}, fmt.Errorf("invalid timezone offset character") } hours := digitsToInt(b[1:3]) minutes := digitsToInt(b[4:6]) seconds := direction * (hours*3600 + minutes*60) zone = time.FixedZone("", seconds) } if len(b) > 0 { return time.Time{}, fmt.Errorf("extra bytes at the end of the timezone") } t := time.Date( dt.Date.Year, dt.Date.Month, dt.Date.Day, dt.Time.Hour, dt.Time.Minute, dt.Time.Second, dt.Time.Nanosecond, zone) return t, nil } func parseLocalDateTime(b []byte) (LocalDateTime, []byte, error) { dt := LocalDateTime{} if len(b) < 11 { return dt, nil, fmt.Errorf("local datetimes are expected to have the format YYYY-MM-DDTHH:MM:SS[.NNNNNN]") } date, err := parseLocalDate(b[:10]) if err != nil { return dt, nil, err } dt.Date = date sep := b[10] if sep != 'T' && sep != ' ' { return dt, nil, fmt.Errorf("datetime separator is expected to be T or a space") } t, rest, err := parseLocalTime(b[11:]) if err != nil { return dt, nil, err } dt.Time = t return dt, rest, nil } // parseLocalTime is a bit different because it also returns the remaining // []byte that is didn't need. This is to allow parseDateTime to parse those // remaining bytes as a timezone. func parseLocalTime(b []byte) (LocalTime, []byte, error) { t := LocalTime{} if len(b) < 8 { return t, nil, fmt.Errorf("times are expected to have the format HH:MM:SS[.NNNNNN]") } var err error t.Hour, err = parseDecimalDigits(b[0:2]) if err != nil { return t, nil, err } if b[2] != ':' { return t, nil, fmt.Errorf("expecting colon between hours and minutes") } t.Minute, err = parseDecimalDigits(b[3:5]) if err != nil { return t, nil, err } if b[5] != ':' { return t, nil, fmt.Errorf("expecting colon between minutes and seconds") } t.Second, err = parseDecimalDigits(b[6:8]) if err != nil { return t, nil, err } if len(b) >= 15 && b[8] == '.' { t.Nanosecond, err = parseDecimalDigits(b[9:15]) return t, b[15:], nil } return t, b[8:], nil } func parseFloat(b []byte) (float64, error) { // TODO: inefficient if len(b) == 4 && (b[0] == '+' || b[0] == '-') && b[1] == 'n' && b[2] == 'a' && b[3] == 'n' { return math.NaN(), nil } tok := string(b) err := numberContainsInvalidUnderscore(tok) if err != nil { return 0, err } cleanedVal := cleanupNumberToken(tok) return strconv.ParseFloat(cleanedVal, 64) } func parseIntHex(b []byte) (int64, error) { tok := string(b) cleanedVal := cleanupNumberToken(tok) err := hexNumberContainsInvalidUnderscore(cleanedVal) if err != nil { return 0, nil } return strconv.ParseInt(cleanedVal[2:], 16, 64) } func parseIntOct(b []byte) (int64, error) { tok := string(b) cleanedVal := cleanupNumberToken(tok) err := numberContainsInvalidUnderscore(cleanedVal) if err != nil { return 0, err } return strconv.ParseInt(cleanedVal[2:], 8, 64) } func parseIntBin(b []byte) (int64, error) { tok := string(b) cleanedVal := cleanupNumberToken(tok) err := numberContainsInvalidUnderscore(cleanedVal) if err != nil { return 0, err } return strconv.ParseInt(cleanedVal[2:], 2, 64) } func parseIntDec(b []byte) (int64, error) { tok := string(b) cleanedVal := cleanupNumberToken(tok) err := numberContainsInvalidUnderscore(cleanedVal) if err != nil { return 0, err } return strconv.ParseInt(cleanedVal, 10, 64) } func numberContainsInvalidUnderscore(value string) error { // For large numbers, you may use underscores between digits to enhance // readability. Each underscore must be surrounded by at least one digit on // each side. hasBefore := false for idx, r := range value { if r == '_' { if !hasBefore || idx+1 >= len(value) { // can't end with an underscore return errInvalidUnderscore } } hasBefore = isDigitRune(r) } return nil } func hexNumberContainsInvalidUnderscore(value string) error { hasBefore := false for idx, r := range value { if r == '_' { if !hasBefore || idx+1 >= len(value) { // can't end with an underscore return errInvalidUnderscoreHex } } hasBefore = isHexDigit(r) } return nil } func cleanupNumberToken(value string) string { cleanedVal := strings.Replace(value, "_", "", -1) return cleanedVal } func isHexDigit(r rune) bool { return isDigitRune(r) || (r >= 'a' && r <= 'f') || (r >= 'A' && r <= 'F') } func isDigitRune(r rune) bool { return r >= '0' && r <= '9' } var errInvalidUnderscore = errors.New("invalid use of _ in number") var errInvalidUnderscoreHex = errors.New("invalid use of _ in hex number")