123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 |
- package types
- import (
- "database/sql"
- "database/sql/driver"
- "errors"
- "fmt"
- "reflect"
- "strconv"
- "time"
- )
- // TextNull is an implementation of a string for the MySQL type char/varchar/text ....
- type TextNull string
- // Value implements the driver.Valuer interface,
- // and turns the string into a bytes for MySQL storage.
- func (s TextNull) Value() (driver.Value, error) {
- return []byte(s), nil
- }
- // Scan implements the sql.Scanner interface,
- // and turns the bytes incoming from MySQL into a string
- func (s *TextNull) Scan(src interface{}) error {
- switch t := src.(type) {
- case string:
- *s = TextNull(src.(string))
- case []byte:
- if len(t) == 0 {
- *s = TextNull("")
- } else {
- *s = TextNull(src.([]byte))
- }
- case nil:
- *s = TextNull("")
- default:
- return errors.New("bad []byte type assertion")
- }
- return nil
- /*
- if src != nil {
- v, ok := src.([]byte)
- if !ok {
- return errors.New("bad []byte type assertion")
- }
- *s = TextNull(v)
- return nil
- }
- *s = TextNull("")
- return nil
- //*/
- }
- var errNilPtr = errors.New("destination pointer is nil") // embedded in descriptive error
- // convertAssign is the same as convertAssignRows, but without the optional
- // rows argument.
- func convertAssign(dest, src interface{}) error {
- return convertAssignRows(dest, src)
- }
- // convertAssignRows copies to dest the value in src, converting it if possible.
- // An error is returned if the copy would result in loss of information.
- // dest should be a pointer type. If rows is passed in, the rows will
- // be used as the parent for any cursor values converted from a
- // driver.Rows to a *Rows.
- func convertAssignRows(dest, src interface{}) error {
- // Common cases, without reflect.
- switch s := src.(type) {
- case string:
- switch d := dest.(type) {
- case *string:
- if d == nil {
- return errNilPtr
- }
- *d = s
- return nil
- case *[]byte:
- if d == nil {
- return errNilPtr
- }
- *d = []byte(s)
- return nil
- case *sql.RawBytes:
- if d == nil {
- return errNilPtr
- }
- *d = append((*d)[:0], s...)
- return nil
- }
- case []byte:
- switch d := dest.(type) {
- case *string:
- if d == nil {
- return errNilPtr
- }
- *d = string(s)
- return nil
- case *interface{}:
- if d == nil {
- return errNilPtr
- }
- *d = cloneBytes(s)
- return nil
- case *[]byte:
- if d == nil {
- return errNilPtr
- }
- *d = cloneBytes(s)
- return nil
- case *sql.RawBytes:
- if d == nil {
- return errNilPtr
- }
- *d = s
- return nil
- }
- case time.Time:
- switch d := dest.(type) {
- case *time.Time:
- *d = s
- return nil
- case *string:
- *d = s.Format(time.RFC3339Nano)
- return nil
- case *[]byte:
- if d == nil {
- return errNilPtr
- }
- *d = []byte(s.Format(time.RFC3339Nano))
- return nil
- case *sql.RawBytes:
- if d == nil {
- return errNilPtr
- }
- *d = s.AppendFormat((*d)[:0], time.RFC3339Nano)
- return nil
- }
- case nil:
- switch d := dest.(type) {
- case *interface{}:
- if d == nil {
- return errNilPtr
- }
- *d = nil
- return nil
- case *[]byte:
- if d == nil {
- return errNilPtr
- }
- *d = nil
- return nil
- case *sql.RawBytes:
- if d == nil {
- return errNilPtr
- }
- *d = nil
- return nil
- }
- // The driver is returning a cursor the client may iterate over.
- case driver.Rows:
- switch d := dest.(type) {
- case *sql.Rows:
- if d == nil {
- return errNilPtr
- }
- return nil
- }
- }
- var sv reflect.Value
- switch d := dest.(type) {
- case *string:
- sv = reflect.ValueOf(src)
- switch sv.Kind() {
- case reflect.Bool,
- reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
- reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64,
- reflect.Float32, reflect.Float64:
- *d = asString(src)
- return nil
- }
- case *[]byte:
- sv = reflect.ValueOf(src)
- if b, ok := asBytes(nil, sv); ok {
- *d = b
- return nil
- }
- case *sql.RawBytes:
- sv = reflect.ValueOf(src)
- if b, ok := asBytes([]byte(*d)[:0], sv); ok {
- *d = sql.RawBytes(b)
- return nil
- }
- case *bool:
- bv, err := driver.Bool.ConvertValue(src)
- if err == nil {
- *d = bv.(bool)
- }
- return err
- case *interface{}:
- *d = src
- return nil
- }
- return fmt.Errorf("unsupported Scan, storing driver.Value type %T into type %T", src, dest)
- }
- func cloneBytes(b []byte) []byte {
- if b == nil {
- return nil
- }
- c := make([]byte, len(b))
- copy(c, b)
- return c
- }
- func asString(src interface{}) string {
- switch v := src.(type) {
- case string:
- return v
- case []byte:
- return string(v)
- }
- rv := reflect.ValueOf(src)
- switch rv.Kind() {
- case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
- return strconv.FormatInt(rv.Int(), 10)
- case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
- return strconv.FormatUint(rv.Uint(), 10)
- case reflect.Float64:
- return strconv.FormatFloat(rv.Float(), 'g', -1, 64)
- case reflect.Float32:
- return strconv.FormatFloat(rv.Float(), 'g', -1, 32)
- case reflect.Bool:
- return strconv.FormatBool(rv.Bool())
- }
- return fmt.Sprintf("%v", src)
- }
- func asBytes(buf []byte, rv reflect.Value) (b []byte, ok bool) {
- switch rv.Kind() {
- case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
- return strconv.AppendInt(buf, rv.Int(), 10), true
- case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
- return strconv.AppendUint(buf, rv.Uint(), 10), true
- case reflect.Float32:
- return strconv.AppendFloat(buf, rv.Float(), 'g', -1, 32), true
- case reflect.Float64:
- return strconv.AppendFloat(buf, rv.Float(), 'g', -1, 64), true
- case reflect.Bool:
- return strconv.AppendBool(buf, rv.Bool()), true
- case reflect.String:
- s := rv.String()
- return append(buf, s...), true
- }
- return
- }
|