package reflectext import ( "errors" "fmt" "gogs.mikescher.com/BlackForestBytes/goext/langext" "reflect" ) // PrimitiveStringSerializer is used to serialize primitive types (and a few more) from and to string // This is not really intended to be user facing, and more as a simple building block for other mechanisms // supports: // - golang primitives (ints, uints, floats, bool, string) // - type aliases // - time.Time // - primitive.ObjectID type PrimitiveStringSerializer struct{} func (pss PrimitiveStringSerializer) ValueToString(v any) (string, error) { inType := reflect.TypeOf(v) if inType.Kind() == reflect.Ptr && langext.IsNil(v) { return "", nil } if inType.Kind() == reflect.Ptr { rval1 := reflect.ValueOf(v) rval2 := rval1.Elem() rval3 := rval2.Interface() return pss.ValueToString(rval3) } if conv, ok := primitiveSerializer[inType]; ok { return conv.ToString(v) } for convType, conv := range primitiveSerializer { if castV, ok := TryCastType(v, convType); ok { return conv.ToString(castV) } } return "", errors.New(fmt.Sprintf("failed to find a matching generic <toString> conversion fo type %T", v)) } func (pss PrimitiveStringSerializer) ValueFromString(str string, outType reflect.Type) (any, error) { if outType.Kind() == reflect.Ptr && str == "" { return reflect.Zero(outType).Interface(), nil // = nil.(outType), nil } if str == "" { return reflect.Zero(outType).Interface(), nil // = <default>(outType), nil } if outType.Kind() == reflect.Ptr { innerValue, err := pss.ValueFromString(str, outType.Elem()) if err != nil { return nil, err } // this weird piece of shit converts innerValue to &innerValue (while keeping types) rval1 := reflect.ValueOf(innerValue) rval2 := rval1.Convert(outType.Elem()) rval3 := reflect.New(outType.Elem()) rval3.Elem().Set(rval2) rval4 := rval3.Interface() return rval4, nil } if conv, ok := primitiveSerializer[outType]; ok { return conv.FromString(str) } emptyResultVal := reflect.Zero(outType).Interface() for convType, conv := range primitiveSerializer { if _, ok := TryCastType(emptyResultVal, convType); ok { if convVal, err := conv.FromString(str); err == nil { if resVal, ok := TryCastType(convVal, outType); ok { return resVal, nil } } } } return "", errors.New(fmt.Sprintf("failed to find a matching generic <toString> conversion fo type %s", outType.String())) }