2014-04-05 13:21:52 +08:00
|
|
|
// Copyright © 2014 Steve Francia <spf@spf13.com>.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by an MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package cast
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2014-04-25 00:21:33 +08:00
|
|
|
"reflect"
|
2014-04-05 13:21:52 +08:00
|
|
|
"strconv"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
jww "github.com/spf13/jwalterweatherman"
|
|
|
|
)
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToTimeE(i interface{}) (tim time.Time, err error) {
|
|
|
|
jww.DEBUG.Println("ToTimeE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
switch s := i.(type) {
|
|
|
|
case time.Time:
|
2014-04-25 00:21:33 +08:00
|
|
|
return s, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case string:
|
|
|
|
d, e := StringToDate(s)
|
|
|
|
if e == nil {
|
2014-04-25 00:21:33 +08:00
|
|
|
return d, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return time.Time{}, fmt.Errorf("Could not parse Date/Time format: %v\n", e)
|
2014-04-05 13:21:52 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return time.Time{}, fmt.Errorf("Unable to Cast %#v to Time\n", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToBoolE(i interface{}) (bool, error) {
|
|
|
|
jww.DEBUG.Println("ToBoolE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
switch b := i.(type) {
|
|
|
|
case bool:
|
2014-04-25 00:21:33 +08:00
|
|
|
return b, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case nil:
|
2014-04-25 00:21:33 +08:00
|
|
|
return false, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int:
|
|
|
|
if i.(int) > 0 {
|
2014-04-25 00:21:33 +08:00
|
|
|
return true, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return false, nil
|
2014-08-31 09:16:09 +08:00
|
|
|
case string:
|
|
|
|
return strconv.ParseBool(i.(string))
|
2014-04-05 13:21:52 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return false, fmt.Errorf("Unable to Cast %#v to bool", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToFloat64E(i interface{}) (float64, error) {
|
|
|
|
jww.DEBUG.Println("ToFloat64E called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
switch s := i.(type) {
|
|
|
|
case float64:
|
2014-04-25 00:21:33 +08:00
|
|
|
return s, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case float32:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-07 23:43:25 +08:00
|
|
|
case int64:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-07 23:43:25 +08:00
|
|
|
case int32:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-07 23:43:25 +08:00
|
|
|
case int16:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-07 23:43:25 +08:00
|
|
|
case int8:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-07 23:43:25 +08:00
|
|
|
case int:
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case string:
|
|
|
|
v, err := strconv.ParseFloat(s, 64)
|
|
|
|
if err == nil {
|
2014-04-25 00:21:33 +08:00
|
|
|
return float64(v), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
} else {
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0.0, fmt.Errorf("Unable to Cast %#v to float", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0.0, fmt.Errorf("Unable to Cast %#v to float", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToIntE(i interface{}) (int, error) {
|
|
|
|
jww.DEBUG.Println("ToIntE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
switch s := i.(type) {
|
|
|
|
case int:
|
2014-04-25 00:21:33 +08:00
|
|
|
return s, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int64:
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int32:
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int16:
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int8:
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case string:
|
|
|
|
v, err := strconv.ParseInt(s, 0, 0)
|
|
|
|
if err == nil {
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(v), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
} else {
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0, fmt.Errorf("Unable to Cast %#v to int", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
2014-04-06 03:18:23 +08:00
|
|
|
case float64:
|
2014-04-25 00:21:33 +08:00
|
|
|
return int(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case bool:
|
|
|
|
if bool(s) {
|
2014-04-25 00:21:33 +08:00
|
|
|
return 1, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
} else {
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
2014-04-06 03:18:23 +08:00
|
|
|
case nil:
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return 0, fmt.Errorf("Unable to Cast %#v to int", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToStringE(i interface{}) (string, error) {
|
|
|
|
jww.DEBUG.Println("ToStringE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
switch s := i.(type) {
|
|
|
|
case string:
|
2014-04-25 00:21:33 +08:00
|
|
|
return s, nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case float64:
|
2014-04-25 00:21:33 +08:00
|
|
|
return strconv.FormatFloat(i.(float64), 'f', -1, 64), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case int:
|
2014-04-25 00:21:33 +08:00
|
|
|
return strconv.FormatInt(int64(i.(int)), 10), nil
|
2014-04-06 03:18:23 +08:00
|
|
|
case []byte:
|
2014-04-25 00:21:33 +08:00
|
|
|
return string(s), nil
|
2014-04-05 13:21:52 +08:00
|
|
|
case nil:
|
2014-04-25 00:21:33 +08:00
|
|
|
return "", nil
|
2014-04-05 13:21:52 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return "", fmt.Errorf("Unable to Cast %#v to string", i)
|
2014-04-05 13:21:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToStringMapStringE(i interface{}) (map[string]string, error) {
|
|
|
|
jww.DEBUG.Println("ToStringMapStringE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-06 03:18:23 +08:00
|
|
|
var m = map[string]string{}
|
|
|
|
|
|
|
|
switch v := i.(type) {
|
|
|
|
case map[interface{}]interface{}:
|
|
|
|
for k, val := range v {
|
|
|
|
m[ToString(k)] = ToString(val)
|
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, nil
|
2014-04-08 10:40:49 +08:00
|
|
|
case map[string]interface{}:
|
|
|
|
for k, val := range v {
|
|
|
|
m[ToString(k)] = ToString(val)
|
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, nil
|
2014-04-07 23:42:50 +08:00
|
|
|
case map[string]string:
|
2014-04-25 00:21:33 +08:00
|
|
|
return v, nil
|
2014-04-06 03:18:23 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, fmt.Errorf("Unable to Cast %#v to map[string]string", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, fmt.Errorf("Unable to Cast %#v to map[string]string", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToStringMapE(i interface{}) (map[string]interface{}, error) {
|
|
|
|
jww.DEBUG.Println("ToStringMapE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-06 03:18:23 +08:00
|
|
|
var m = map[string]interface{}{}
|
|
|
|
|
|
|
|
switch v := i.(type) {
|
|
|
|
case map[interface{}]interface{}:
|
|
|
|
for k, val := range v {
|
|
|
|
m[ToString(k)] = val
|
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, nil
|
2014-04-07 23:42:50 +08:00
|
|
|
case map[string]interface{}:
|
2014-04-25 00:21:33 +08:00
|
|
|
return v, nil
|
2014-04-06 03:18:23 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, fmt.Errorf("Unable to Cast %#v to map[string]interface{}", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
return m, fmt.Errorf("Unable to Cast %#v to map[string]interface{}", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
|
|
|
|
2014-04-27 06:56:25 +08:00
|
|
|
func ToSliceE(i interface{}) ([]interface{}, error) {
|
|
|
|
jww.DEBUG.Println("ToSliceE called on type:", reflect.TypeOf(i))
|
|
|
|
|
|
|
|
var s []interface{}
|
|
|
|
|
|
|
|
switch v := i.(type) {
|
|
|
|
case []interface{}:
|
|
|
|
fmt.Println("here")
|
|
|
|
for _, u := range v {
|
|
|
|
s = append(s, u)
|
|
|
|
}
|
|
|
|
return s, nil
|
|
|
|
case []map[string]interface{}:
|
|
|
|
for _, u := range v {
|
|
|
|
s = append(s, u)
|
|
|
|
}
|
|
|
|
return s, nil
|
|
|
|
default:
|
|
|
|
return s, fmt.Errorf("Unable to Cast %#v of type %v to []interface{}", i, reflect.TypeOf(i))
|
|
|
|
}
|
|
|
|
|
|
|
|
return s, fmt.Errorf("Unable to Cast %#v to []interface{}", i)
|
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
func ToStringSliceE(i interface{}) ([]string, error) {
|
|
|
|
jww.DEBUG.Println("ToStringSliceE called on type:", reflect.TypeOf(i))
|
|
|
|
|
2014-04-06 03:18:23 +08:00
|
|
|
var a []string
|
|
|
|
|
|
|
|
switch v := i.(type) {
|
|
|
|
case []interface{}:
|
|
|
|
for _, u := range v {
|
|
|
|
a = append(a, ToString(u))
|
|
|
|
}
|
2014-04-25 00:21:33 +08:00
|
|
|
return a, nil
|
2014-04-07 23:42:50 +08:00
|
|
|
case []string:
|
2014-04-25 00:21:33 +08:00
|
|
|
return v, nil
|
2014-04-06 03:18:23 +08:00
|
|
|
default:
|
2014-04-25 00:21:33 +08:00
|
|
|
return a, fmt.Errorf("Unable to Cast %#v to []string", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
|
|
|
|
2014-04-25 00:21:33 +08:00
|
|
|
return a, fmt.Errorf("Unable to Cast %#v to []string", i)
|
2014-04-06 03:18:23 +08:00
|
|
|
}
|
|
|
|
|
2014-04-05 13:21:52 +08:00
|
|
|
func StringToDate(s string) (time.Time, error) {
|
|
|
|
return parseDateWith(s, []string{
|
|
|
|
time.RFC3339,
|
|
|
|
"2006-01-02T15:04:05", // iso8601 without timezone
|
|
|
|
time.RFC1123Z,
|
|
|
|
time.RFC1123,
|
|
|
|
time.RFC822Z,
|
|
|
|
time.RFC822,
|
|
|
|
time.ANSIC,
|
|
|
|
time.UnixDate,
|
|
|
|
time.RubyDate,
|
|
|
|
"2006-01-02 15:04:05Z07:00",
|
|
|
|
"02 Jan 06 15:04 MST",
|
|
|
|
"2006-01-02",
|
|
|
|
"02 Jan 2006",
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseDateWith(s string, dates []string) (d time.Time, e error) {
|
|
|
|
for _, dateType := range dates {
|
|
|
|
if d, e = time.Parse(dateType, s); e == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return d, errors.New(fmt.Sprintf("Unable to parse date: %s", s))
|
|
|
|
}
|