Path: blob/main/vendor/github.com/spf13/cast/number.go
2875 views
// Copyright © 2014 Steve Francia <[email protected]>.1//2// Use of this source code is governed by an MIT-style3// license that can be found in the LICENSE file.45package cast67import (8"encoding/json"9"errors"10"fmt"11"regexp"12"strconv"13"strings"14"time"15)1617var errNegativeNotAllowed = errors.New("unable to cast negative value")1819type float64EProvider interface {20Float64() (float64, error)21}2223type float64Provider interface {24Float64() float6425}2627// Number is a type parameter constraint for functions accepting number types.28//29// It represents the supported number types this package can cast to.30type Number interface {31int | int8 | int16 | int32 | int64 | uint | uint8 | uint16 | uint32 | uint64 | float32 | float6432}3334type integer interface {35int | int8 | int16 | int32 | int6436}3738type unsigned interface {39uint | uint8 | uint16 | uint32 | uint6440}4142type float interface {43float32 | float6444}4546// ToNumberE casts any value to a [Number] type.47func ToNumberE[T Number](i any) (T, error) {48var t T4950switch any(t).(type) {51case int:52return toNumberE[T](i, parseNumber[T])53case int8:54return toNumberE[T](i, parseNumber[T])55case int16:56return toNumberE[T](i, parseNumber[T])57case int32:58return toNumberE[T](i, parseNumber[T])59case int64:60return toNumberE[T](i, parseNumber[T])61case uint:62return toUnsignedNumberE[T](i, parseNumber[T])63case uint8:64return toUnsignedNumberE[T](i, parseNumber[T])65case uint16:66return toUnsignedNumberE[T](i, parseNumber[T])67case uint32:68return toUnsignedNumberE[T](i, parseNumber[T])69case uint64:70return toUnsignedNumberE[T](i, parseNumber[T])71case float32:72return toNumberE[T](i, parseNumber[T])73case float64:74return toNumberE[T](i, parseNumber[T])75default:76return 0, fmt.Errorf("unknown number type: %T", t)77}78}7980// ToNumber casts any value to a [Number] type.81func ToNumber[T Number](i any) T {82v, _ := ToNumberE[T](i)8384return v85}8687// toNumber's semantics differ from other "to" functions.88// It returns false as the second parameter if the conversion fails.89// This is to signal other callers that they should proceed with their own conversions.90func toNumber[T Number](i any) (T, bool) {91i, _ = indirect(i)9293switch s := i.(type) {94case T:95return s, true96case int:97return T(s), true98case int8:99return T(s), true100case int16:101return T(s), true102case int32:103return T(s), true104case int64:105return T(s), true106case uint:107return T(s), true108case uint8:109return T(s), true110case uint16:111return T(s), true112case uint32:113return T(s), true114case uint64:115return T(s), true116case float32:117return T(s), true118case float64:119return T(s), true120case bool:121if s {122return 1, true123}124125return 0, true126case nil:127return 0, true128case time.Weekday:129return T(s), true130case time.Month:131return T(s), true132}133134return 0, false135}136137func toNumberE[T Number](i any, parseFn func(string) (T, error)) (T, error) {138n, ok := toNumber[T](i)139if ok {140return n, nil141}142143i, _ = indirect(i)144145switch s := i.(type) {146case string:147if s == "" {148return 0, nil149}150151v, err := parseFn(s)152if err != nil {153return 0, fmt.Errorf(errorMsgWith, i, i, n, err)154}155156return v, nil157case json.Number:158if s == "" {159return 0, nil160}161162v, err := parseFn(string(s))163if err != nil {164return 0, fmt.Errorf(errorMsgWith, i, i, n, err)165}166167return v, nil168case float64EProvider:169if _, ok := any(n).(float64); !ok {170return 0, fmt.Errorf(errorMsg, i, i, n)171}172173v, err := s.Float64()174if err != nil {175return 0, fmt.Errorf(errorMsg, i, i, n)176}177178return T(v), nil179case float64Provider:180if _, ok := any(n).(float64); !ok {181return 0, fmt.Errorf(errorMsg, i, i, n)182}183184return T(s.Float64()), nil185default:186if i, ok := resolveAlias(i); ok {187return toNumberE(i, parseFn)188}189190return 0, fmt.Errorf(errorMsg, i, i, n)191}192}193194func toUnsignedNumber[T Number](i any) (T, bool, bool) {195i, _ = indirect(i)196197switch s := i.(type) {198case T:199return s, true, true200case int:201if s < 0 {202return 0, false, false203}204205return T(s), true, true206case int8:207if s < 0 {208return 0, false, false209}210211return T(s), true, true212case int16:213if s < 0 {214return 0, false, false215}216217return T(s), true, true218case int32:219if s < 0 {220return 0, false, false221}222223return T(s), true, true224case int64:225if s < 0 {226return 0, false, false227}228229return T(s), true, true230case uint:231return T(s), true, true232case uint8:233return T(s), true, true234case uint16:235return T(s), true, true236case uint32:237return T(s), true, true238case uint64:239return T(s), true, true240case float32:241if s < 0 {242return 0, false, false243}244245return T(s), true, true246case float64:247if s < 0 {248return 0, false, false249}250251return T(s), true, true252case bool:253if s {254return 1, true, true255}256257return 0, true, true258case nil:259return 0, true, true260case time.Weekday:261if s < 0 {262return 0, false, false263}264265return T(s), true, true266case time.Month:267if s < 0 {268return 0, false, false269}270271return T(s), true, true272}273274return 0, true, false275}276277func toUnsignedNumberE[T Number](i any, parseFn func(string) (T, error)) (T, error) {278n, valid, ok := toUnsignedNumber[T](i)279if ok {280return n, nil281}282283i, _ = indirect(i)284285if !valid {286return 0, errNegativeNotAllowed287}288289switch s := i.(type) {290case string:291if s == "" {292return 0, nil293}294295v, err := parseFn(s)296if err != nil {297return 0, fmt.Errorf(errorMsgWith, i, i, n, err)298}299300return v, nil301case json.Number:302if s == "" {303return 0, nil304}305306v, err := parseFn(string(s))307if err != nil {308return 0, fmt.Errorf(errorMsgWith, i, i, n, err)309}310311return v, nil312case float64EProvider:313if _, ok := any(n).(float64); !ok {314return 0, fmt.Errorf(errorMsg, i, i, n)315}316317v, err := s.Float64()318if err != nil {319return 0, fmt.Errorf(errorMsg, i, i, n)320}321322if v < 0 {323return 0, errNegativeNotAllowed324}325326return T(v), nil327case float64Provider:328if _, ok := any(n).(float64); !ok {329return 0, fmt.Errorf(errorMsg, i, i, n)330}331332v := s.Float64()333334if v < 0 {335return 0, errNegativeNotAllowed336}337338return T(v), nil339default:340if i, ok := resolveAlias(i); ok {341return toUnsignedNumberE(i, parseFn)342}343344return 0, fmt.Errorf(errorMsg, i, i, n)345}346}347348func parseNumber[T Number](s string) (T, error) {349var t T350351switch any(t).(type) {352case int:353v, err := parseInt[int](s)354355return T(v), err356case int8:357v, err := parseInt[int8](s)358359return T(v), err360case int16:361v, err := parseInt[int16](s)362363return T(v), err364case int32:365v, err := parseInt[int32](s)366367return T(v), err368case int64:369v, err := parseInt[int64](s)370371return T(v), err372case uint:373v, err := parseUint[uint](s)374375return T(v), err376case uint8:377v, err := parseUint[uint8](s)378379return T(v), err380case uint16:381v, err := parseUint[uint16](s)382383return T(v), err384case uint32:385v, err := parseUint[uint32](s)386387return T(v), err388case uint64:389v, err := parseUint[uint64](s)390391return T(v), err392case float32:393v, err := strconv.ParseFloat(s, 32)394395return T(v), err396case float64:397v, err := strconv.ParseFloat(s, 64)398399return T(v), err400401default:402return 0, fmt.Errorf("unknown number type: %T", t)403}404}405406func parseInt[T integer](s string) (T, error) {407v, err := strconv.ParseInt(trimDecimal(s), 0, 0)408if err != nil {409return 0, err410}411412return T(v), nil413}414415func parseUint[T unsigned](s string) (T, error) {416v, err := strconv.ParseUint(strings.TrimLeft(trimDecimal(s), "+"), 0, 0)417if err != nil {418return 0, err419}420421return T(v), nil422}423424func parseFloat[T float](s string) (T, error) {425var t T426427var v any428var err error429430switch any(t).(type) {431case float32:432n, e := strconv.ParseFloat(s, 32)433434v = float32(n)435err = e436case float64:437n, e := strconv.ParseFloat(s, 64)438439v = float64(n)440err = e441}442443return v.(T), err444}445446// ToFloat64E casts an interface to a float64 type.447func ToFloat64E(i any) (float64, error) {448return toNumberE[float64](i, parseFloat[float64])449}450451// ToFloat32E casts an interface to a float32 type.452func ToFloat32E(i any) (float32, error) {453return toNumberE[float32](i, parseFloat[float32])454}455456// ToInt64E casts an interface to an int64 type.457func ToInt64E(i any) (int64, error) {458return toNumberE[int64](i, parseInt[int64])459}460461// ToInt32E casts an interface to an int32 type.462func ToInt32E(i any) (int32, error) {463return toNumberE[int32](i, parseInt[int32])464}465466// ToInt16E casts an interface to an int16 type.467func ToInt16E(i any) (int16, error) {468return toNumberE[int16](i, parseInt[int16])469}470471// ToInt8E casts an interface to an int8 type.472func ToInt8E(i any) (int8, error) {473return toNumberE[int8](i, parseInt[int8])474}475476// ToIntE casts an interface to an int type.477func ToIntE(i any) (int, error) {478return toNumberE[int](i, parseInt[int])479}480481// ToUintE casts an interface to a uint type.482func ToUintE(i any) (uint, error) {483return toUnsignedNumberE[uint](i, parseUint[uint])484}485486// ToUint64E casts an interface to a uint64 type.487func ToUint64E(i any) (uint64, error) {488return toUnsignedNumberE[uint64](i, parseUint[uint64])489}490491// ToUint32E casts an interface to a uint32 type.492func ToUint32E(i any) (uint32, error) {493return toUnsignedNumberE[uint32](i, parseUint[uint32])494}495496// ToUint16E casts an interface to a uint16 type.497func ToUint16E(i any) (uint16, error) {498return toUnsignedNumberE[uint16](i, parseUint[uint16])499}500501// ToUint8E casts an interface to a uint type.502func ToUint8E(i any) (uint8, error) {503return toUnsignedNumberE[uint8](i, parseUint[uint8])504}505506func trimZeroDecimal(s string) string {507var foundZero bool508for i := len(s); i > 0; i-- {509switch s[i-1] {510case '.':511if foundZero {512return s[:i-1]513}514case '0':515foundZero = true516default:517return s518}519}520return s521}522523var stringNumberRe = regexp.MustCompile(`^([-+]?\d*)(\.\d*)?$`)524525// see [BenchmarkDecimal] for details about the implementation526func trimDecimal(s string) string {527if !strings.Contains(s, ".") {528return s529}530531matches := stringNumberRe.FindStringSubmatch(s)532if matches != nil {533// matches[1] is the captured integer part with sign534s = matches[1]535536// handle special cases537switch s {538case "-", "+":539s += "0"540case "":541s = "0"542}543544return s545}546547return s548}549550551