// Copyright GoFrame Author(https://github.com/gogf/gf). All Rights Reserved. // // This Source Code Form is subject to the terms of the MIT License. // If a copy of the MIT was not distributed with this file, // You can obtain one at https://github.com/gogf/gf. package gdb import ( "fmt" "github.com/gogf/gf/container/gset" "github.com/gogf/gf/internal/empty" "github.com/gogf/gf/os/gtime" "github.com/gogf/gf/text/gregex" "github.com/gogf/gf/text/gstr" "github.com/gogf/gf/util/gconv" "github.com/gogf/gf/util/gutil" "time" ) // getModel creates and returns a cloned model of current model if is true, or else it returns // the current model. func (m *Model) getModel() *Model { if !m.safe { return m } else { return m.Clone() } } // mappingAndFilterToTableFields mappings and changes given field name to really table field name. func (m *Model) mappingAndFilterToTableFields(fields []string) []string { fieldsMap, err := m.db.TableFields(m.tables) if err != nil || len(fieldsMap) == 0 { return fields } var ( inputFieldsArray = gstr.SplitAndTrim(gstr.Join(fields, ","), ",") outputFieldsArray = make([]string, 0, len(inputFieldsArray)) ) fieldsKeyMap := make(map[string]interface{}, len(fieldsMap)) for k, _ := range fieldsMap { fieldsKeyMap[k] = nil } for _, field := range inputFieldsArray { if _, ok := fieldsKeyMap[field]; !ok { if !gregex.IsMatchString(regularFieldNameRegPattern, field) { outputFieldsArray = append(outputFieldsArray, field) continue } else { if foundKey, _ := gutil.MapPossibleItemByKey(fieldsKeyMap, field); foundKey != "" { outputFieldsArray = append(outputFieldsArray, foundKey) } } } else { outputFieldsArray = append(outputFieldsArray, field) } } return outputFieldsArray } // filterDataForInsertOrUpdate does filter feature with data for inserting/updating operations. // Note that, it does not filter list item, which is also type of map, for "omit empty" feature. func (m *Model) filterDataForInsertOrUpdate(data interface{}) (interface{}, error) { var err error switch value := data.(type) { case List: for k, item := range value { value[k], err = m.doMappingAndFilterForInsertOrUpdateDataMap(item, false) if err != nil { return nil, err } } return value, nil case Map: return m.doMappingAndFilterForInsertOrUpdateDataMap(value, true) default: return data, nil } } // doMappingAndFilterForInsertOrUpdateDataMap does the filter features for map. // Note that, it does not filter list item, which is also type of map, for "omit empty" feature. func (m *Model) doMappingAndFilterForInsertOrUpdateDataMap(data Map, allowOmitEmpty bool) (Map, error) { var err error data, err = m.db.mappingAndFilterData(m.schema, m.tables, data, m.filter) if err != nil { return nil, err } // Remove key-value pairs of which the value is empty. if allowOmitEmpty && m.option&OPTION_OMITEMPTY > 0 { tempMap := make(Map, len(data)) for k, v := range data { if empty.IsEmpty(v) { continue } // Special type filtering. switch r := v.(type) { case time.Time: if r.IsZero() { continue } case *time.Time: if r.IsZero() { continue } case gtime.Time: if r.IsZero() { continue } case *gtime.Time: if r.IsZero() { continue } } tempMap[k] = v } data = tempMap } if len(m.fields) > 0 && m.fields != "*" { // Keep specified fields. var ( set = gset.NewStrSetFrom(gstr.SplitAndTrim(m.fields, ",")) charL, charR = m.db.GetChars() chars = charL + charR ) set.Walk(func(item string) string { return gstr.Trim(item, chars) }) for k := range data { k = gstr.Trim(k, chars) if !set.Contains(k) { delete(data, k) } } } else if len(m.fieldsEx) > 0 { // Filter specified fields. for _, v := range gstr.SplitAndTrim(m.fieldsEx, ",") { delete(data, v) } } return data, nil } // getLink returns the underlying database link object with configured attribute. // The parameter specifies whether using the master node if master-slave configured. func (m *Model) getLink(master bool) Link { if m.tx != nil { return m.tx.tx } linkType := m.linkType if linkType == 0 { if master { linkType = linkTypeMaster } else { linkType = linkTypeSlave } } switch linkType { case linkTypeMaster: link, err := m.db.GetMaster(m.schema) if err != nil { panic(err) } return link case linkTypeSlave: link, err := m.db.GetSlave(m.schema) if err != nil { panic(err) } return link } return nil } // getPrimaryKey retrieves and returns the primary key name of the model table. // It parses m.tables to retrieve the primary table name, supporting m.tables like: // "user", "user u", "user as u, user_detail as ud". func (m *Model) getPrimaryKey() string { table := gstr.SplitAndTrim(m.tables, " ")[0] tableFields, err := m.db.TableFields(table) if err != nil { return "" } for name, field := range tableFields { if gstr.ContainsI(field.Key, "pri") { return name } } return "" } // formatCondition formats where arguments of the model and returns a new condition sql and its arguments. // Note that this function does not change any attribute value of the . // // The parameter specifies whether limits querying only one record if m.limit is not set. func (m *Model) formatCondition(limit1 bool, isCountStatement bool) (conditionWhere string, conditionExtra string, conditionArgs []interface{}) { if len(m.whereHolder) > 0 { for _, v := range m.whereHolder { switch v.operator { case whereHolderWhere: if conditionWhere == "" { newWhere, newArgs := formatWhere( m.db, v.where, v.args, m.option&OPTION_OMITEMPTY > 0, ) if len(newWhere) > 0 { conditionWhere = newWhere conditionArgs = newArgs } continue } fallthrough case whereHolderAnd: newWhere, newArgs := formatWhere( m.db, v.where, v.args, m.option&OPTION_OMITEMPTY > 0, ) if len(newWhere) > 0 { if len(conditionWhere) == 0 { conditionWhere = newWhere } else if conditionWhere[0] == '(' { conditionWhere = fmt.Sprintf(`%s AND (%s)`, conditionWhere, newWhere) } else { conditionWhere = fmt.Sprintf(`(%s) AND (%s)`, conditionWhere, newWhere) } conditionArgs = append(conditionArgs, newArgs...) } case whereHolderOr: newWhere, newArgs := formatWhere( m.db, v.where, v.args, m.option&OPTION_OMITEMPTY > 0, ) if len(newWhere) > 0 { if len(conditionWhere) == 0 { conditionWhere = newWhere } else if conditionWhere[0] == '(' { conditionWhere = fmt.Sprintf(`%s OR (%s)`, conditionWhere, newWhere) } else { conditionWhere = fmt.Sprintf(`(%s) OR (%s)`, conditionWhere, newWhere) } conditionArgs = append(conditionArgs, newArgs...) } } } } if conditionWhere != "" { conditionWhere = " WHERE " + conditionWhere } if m.groupBy != "" { conditionExtra += " GROUP BY " + m.groupBy } if len(m.having) > 0 { havingStr, havingArgs := formatWhere( m.db, m.having[0], gconv.Interfaces(m.having[1]), m.option&OPTION_OMITEMPTY > 0, ) if len(havingStr) > 0 { conditionExtra += " HAVING " + havingStr conditionArgs = append(conditionArgs, havingArgs...) } } if m.orderBy != "" { conditionExtra += " ORDER BY " + m.orderBy } if !isCountStatement { if m.limit != 0 { if m.start >= 0 { conditionExtra += fmt.Sprintf(" LIMIT %d,%d", m.start, m.limit) } else { conditionExtra += fmt.Sprintf(" LIMIT %d", m.limit) } } else if limit1 { conditionExtra += " LIMIT 1" } if m.offset >= 0 { conditionExtra += fmt.Sprintf(" OFFSET %d", m.offset) } } if m.lockInfo != "" { conditionExtra += " " + m.lockInfo } return } // mergeArguments creates and returns new arguments by merging and given . func (m *Model) mergeArguments(args []interface{}) []interface{} { if len(m.extraArgs) > 0 { newArgs := make([]interface{}, len(m.extraArgs)+len(args)) copy(newArgs, m.extraArgs) copy(newArgs[len(m.extraArgs):], args) return newArgs } return args }