2019-06-23 18:22:43 +03:00
|
|
|
// Copyright 2019 The Xorm Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2020-03-22 18:12:55 +03:00
|
|
|
package schemas
|
2016-11-04 01:16:01 +03:00
|
|
|
|
|
|
|
import (
|
2020-06-15 23:46:01 +03:00
|
|
|
"errors"
|
2016-11-04 01:16:01 +03:00
|
|
|
"fmt"
|
|
|
|
"reflect"
|
2020-06-15 23:46:01 +03:00
|
|
|
"strconv"
|
2016-11-04 01:16:01 +03:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2021-05-14 22:17:06 +03:00
|
|
|
// enumerates all database mapping way
|
2016-11-04 01:16:01 +03:00
|
|
|
const (
|
|
|
|
TWOSIDES = iota + 1
|
|
|
|
ONLYTODB
|
|
|
|
ONLYFROMDB
|
|
|
|
)
|
|
|
|
|
2017-08-22 14:39:52 +03:00
|
|
|
// Column defines database column
|
2016-11-04 01:16:01 +03:00
|
|
|
type Column struct {
|
|
|
|
Name string
|
|
|
|
TableName string
|
2020-03-22 18:12:55 +03:00
|
|
|
FieldName string // Avaiable only when parsed from a struct
|
2016-11-04 01:16:01 +03:00
|
|
|
SQLType SQLType
|
2017-08-22 14:39:52 +03:00
|
|
|
IsJSON bool
|
2016-11-04 01:16:01 +03:00
|
|
|
Length int
|
|
|
|
Length2 int
|
|
|
|
Nullable bool
|
|
|
|
Default string
|
|
|
|
Indexes map[string]int
|
|
|
|
IsPrimaryKey bool
|
|
|
|
IsAutoIncrement bool
|
|
|
|
MapType int
|
|
|
|
IsCreated bool
|
|
|
|
IsUpdated bool
|
|
|
|
IsDeleted bool
|
|
|
|
IsCascade bool
|
|
|
|
IsVersion bool
|
2020-01-20 18:45:14 +03:00
|
|
|
DefaultIsEmpty bool // false means column has no default set, but not default value is empty
|
2016-11-04 01:16:01 +03:00
|
|
|
EnumOptions map[string]int
|
|
|
|
SetOptions map[string]int
|
|
|
|
DisableTimeZone bool
|
|
|
|
TimeZone *time.Location // column specified time zone
|
2017-08-22 14:39:52 +03:00
|
|
|
Comment string
|
2016-11-04 01:16:01 +03:00
|
|
|
}
|
|
|
|
|
2019-06-23 18:22:43 +03:00
|
|
|
// NewColumn creates a new column
|
2016-11-04 01:16:01 +03:00
|
|
|
func NewColumn(name, fieldName string, sqlType SQLType, len1, len2 int, nullable bool) *Column {
|
|
|
|
return &Column{
|
|
|
|
Name: name,
|
2021-01-05 09:28:51 +03:00
|
|
|
IsJSON: sqlType.IsJson(),
|
2016-11-04 01:16:01 +03:00
|
|
|
TableName: "",
|
|
|
|
FieldName: fieldName,
|
|
|
|
SQLType: sqlType,
|
|
|
|
Length: len1,
|
|
|
|
Length2: len2,
|
|
|
|
Nullable: nullable,
|
|
|
|
Default: "",
|
|
|
|
Indexes: make(map[string]int),
|
|
|
|
IsPrimaryKey: false,
|
|
|
|
IsAutoIncrement: false,
|
|
|
|
MapType: TWOSIDES,
|
|
|
|
IsCreated: false,
|
|
|
|
IsUpdated: false,
|
|
|
|
IsDeleted: false,
|
|
|
|
IsCascade: false,
|
|
|
|
IsVersion: false,
|
2020-01-20 18:45:14 +03:00
|
|
|
DefaultIsEmpty: true, // default should be no default
|
2016-11-04 01:16:01 +03:00
|
|
|
EnumOptions: make(map[string]int),
|
2017-08-22 14:39:52 +03:00
|
|
|
Comment: "",
|
2016-11-04 01:16:01 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-23 18:22:43 +03:00
|
|
|
// ValueOf returns column's filed of struct's value
|
2016-11-04 01:16:01 +03:00
|
|
|
func (col *Column) ValueOf(bean interface{}) (*reflect.Value, error) {
|
|
|
|
dataStruct := reflect.Indirect(reflect.ValueOf(bean))
|
|
|
|
return col.ValueOfV(&dataStruct)
|
|
|
|
}
|
|
|
|
|
2019-06-23 18:22:43 +03:00
|
|
|
// ValueOfV returns column's filed of struct's value accept reflevt value
|
2016-11-04 01:16:01 +03:00
|
|
|
func (col *Column) ValueOfV(dataStruct *reflect.Value) (*reflect.Value, error) {
|
|
|
|
var fieldValue reflect.Value
|
2017-02-10 18:02:26 +03:00
|
|
|
fieldPath := strings.Split(col.FieldName, ".")
|
2016-11-04 01:16:01 +03:00
|
|
|
|
|
|
|
if dataStruct.Type().Kind() == reflect.Map {
|
2017-02-10 18:02:26 +03:00
|
|
|
keyValue := reflect.ValueOf(fieldPath[len(fieldPath)-1])
|
2016-11-04 01:16:01 +03:00
|
|
|
fieldValue = dataStruct.MapIndex(keyValue)
|
|
|
|
return &fieldValue, nil
|
|
|
|
} else if dataStruct.Type().Kind() == reflect.Interface {
|
|
|
|
structValue := reflect.ValueOf(dataStruct.Interface())
|
|
|
|
dataStruct = &structValue
|
|
|
|
}
|
|
|
|
|
2017-02-10 18:02:26 +03:00
|
|
|
level := len(fieldPath)
|
|
|
|
fieldValue = dataStruct.FieldByName(fieldPath[0])
|
2016-11-04 01:16:01 +03:00
|
|
|
for i := 0; i < level-1; i++ {
|
|
|
|
if !fieldValue.IsValid() {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if fieldValue.Kind() == reflect.Struct {
|
2017-02-10 18:02:26 +03:00
|
|
|
fieldValue = fieldValue.FieldByName(fieldPath[i+1])
|
2016-11-04 01:16:01 +03:00
|
|
|
} else if fieldValue.Kind() == reflect.Ptr {
|
|
|
|
if fieldValue.IsNil() {
|
|
|
|
fieldValue.Set(reflect.New(fieldValue.Type().Elem()))
|
|
|
|
}
|
2017-02-10 18:02:26 +03:00
|
|
|
fieldValue = fieldValue.Elem().FieldByName(fieldPath[i+1])
|
2016-11-04 01:16:01 +03:00
|
|
|
} else {
|
2018-07-20 05:10:17 +03:00
|
|
|
return nil, fmt.Errorf("field %v is not valid", col.FieldName)
|
2016-11-04 01:16:01 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !fieldValue.IsValid() {
|
2018-07-20 05:10:17 +03:00
|
|
|
return nil, fmt.Errorf("field %v is not valid", col.FieldName)
|
2016-11-04 01:16:01 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return &fieldValue, nil
|
|
|
|
}
|
2020-06-15 23:46:01 +03:00
|
|
|
|
|
|
|
// ConvertID converts id content to suitable type according column type
|
|
|
|
func (col *Column) ConvertID(sid string) (interface{}, error) {
|
|
|
|
if col.SQLType.IsNumeric() {
|
|
|
|
n, err := strconv.ParseInt(sid, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return n, nil
|
|
|
|
} else if col.SQLType.IsText() {
|
|
|
|
return sid, nil
|
|
|
|
}
|
|
|
|
return nil, errors.New("not supported")
|
|
|
|
}
|