123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190 |
- // 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.
- package schemas
- import (
- "fmt"
- "reflect"
- "strconv"
- "strings"
- )
- // Table represents a database table
- type Table struct {
- Name string
- Type reflect.Type
- columnsSeq []string
- columnsMap map[string][]*Column
- columns []*Column
- Indexes map[string]*Index
- PrimaryKeys []string
- AutoIncrement string
- Created map[string]bool
- Updated string
- Deleted string
- Version string
- StoreEngine string
- Charset string
- Comment string
- }
- // NewEmptyTable creates an empty table
- func NewEmptyTable() *Table {
- return NewTable("", nil)
- }
- // NewTable creates a new Table object
- func NewTable(name string, t reflect.Type) *Table {
- return &Table{Name: name, Type: t,
- columnsSeq: make([]string, 0),
- columns: make([]*Column, 0),
- columnsMap: make(map[string][]*Column),
- Indexes: make(map[string]*Index),
- Created: make(map[string]bool),
- PrimaryKeys: make([]string, 0),
- }
- }
- // Columns returns table's columns
- func (table *Table) Columns() []*Column {
- return table.columns
- }
- // ColumnsSeq returns table's column names according sequence
- func (table *Table) ColumnsSeq() []string {
- return table.columnsSeq
- }
- func (table *Table) columnsByName(name string) []*Column {
- return table.columnsMap[strings.ToLower(name)]
- }
- // GetColumn returns column according column name, if column not found, return nil
- func (table *Table) GetColumn(name string) *Column {
- cols := table.columnsByName(name)
- if cols != nil {
- return cols[0]
- }
- return nil
- }
- // GetColumnIdx returns column according name and idx
- func (table *Table) GetColumnIdx(name string, idx int) *Column {
- cols := table.columnsByName(name)
- if cols != nil && idx < len(cols) {
- return cols[idx]
- }
- return nil
- }
- // PKColumns reprents all primary key columns
- func (table *Table) PKColumns() []*Column {
- columns := make([]*Column, len(table.PrimaryKeys))
- for i, name := range table.PrimaryKeys {
- columns[i] = table.GetColumn(name)
- }
- return columns
- }
- func (table *Table) ColumnType(name string) reflect.Type {
- t, _ := table.Type.FieldByName(name)
- return t.Type
- }
- func (table *Table) AutoIncrColumn() *Column {
- return table.GetColumn(table.AutoIncrement)
- }
- func (table *Table) VersionColumn() *Column {
- return table.GetColumn(table.Version)
- }
- func (table *Table) UpdatedColumn() *Column {
- return table.GetColumn(table.Updated)
- }
- func (table *Table) DeletedColumn() *Column {
- return table.GetColumn(table.Deleted)
- }
- // AddColumn adds a column to table
- func (table *Table) AddColumn(col *Column) {
- table.columnsSeq = append(table.columnsSeq, col.Name)
- table.columns = append(table.columns, col)
- colName := strings.ToLower(col.Name)
- if c, ok := table.columnsMap[colName]; ok {
- table.columnsMap[colName] = append(c, col)
- } else {
- table.columnsMap[colName] = []*Column{col}
- }
- if col.IsPrimaryKey {
- table.PrimaryKeys = append(table.PrimaryKeys, col.Name)
- }
- if col.IsAutoIncrement {
- table.AutoIncrement = col.Name
- }
- if col.IsCreated {
- table.Created[col.Name] = true
- }
- if col.IsUpdated {
- table.Updated = col.Name
- }
- if col.IsDeleted {
- table.Deleted = col.Name
- }
- if col.IsVersion {
- table.Version = col.Name
- }
- }
- // AddIndex adds an index or an unique to table
- func (table *Table) AddIndex(index *Index) {
- table.Indexes[index.Name] = index
- }
- // IDOfV get id from one value of struct
- func (table *Table) IDOfV(rv reflect.Value) (PK, error) {
- v := reflect.Indirect(rv)
- pk := make([]interface{}, len(table.PrimaryKeys))
- for i, col := range table.PKColumns() {
- var err error
- fieldName := col.FieldName
- for {
- parts := strings.SplitN(fieldName, ".", 2)
- if len(parts) == 1 {
- break
- }
- v = v.FieldByName(parts[0])
- if v.Kind() == reflect.Ptr {
- v = v.Elem()
- }
- if v.Kind() != reflect.Struct {
- return nil, fmt.Errorf("Unsupported read value of column %s from field %s", col.Name, col.FieldName)
- }
- fieldName = parts[1]
- }
- pkField := v.FieldByName(fieldName)
- switch pkField.Kind() {
- case reflect.String:
- pk[i], err = col.ConvertID(pkField.String())
- case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
- pk[i], err = col.ConvertID(strconv.FormatInt(pkField.Int(), 10))
- case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
- // id of uint will be converted to int64
- pk[i], err = col.ConvertID(strconv.FormatUint(pkField.Uint(), 10))
- }
- if err != nil {
- return nil, err
- }
- }
- return PK(pk), nil
- }
|