-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstructfieldmapping.go
150 lines (126 loc) · 4.5 KB
/
structfieldmapping.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
package sqldb
import (
"fmt"
"reflect"
"strings"
"github.com/domonda/go-types/strutil"
)
// FieldFlag is a bitmask for special properties
// of how struct fields relate to database columns.
type FieldFlag uint
// PrimaryKey indicates if FieldFlagPrimaryKey is set
func (f FieldFlag) PrimaryKey() bool { return f&FieldFlagPrimaryKey != 0 }
// ReadOnly indicates if FieldFlagReadOnly is set
func (f FieldFlag) ReadOnly() bool { return f&FieldFlagReadOnly != 0 }
// Default indicates if FieldFlagDefault is set
func (f FieldFlag) Default() bool { return f&FieldFlagDefault != 0 }
const (
// FieldFlagPrimaryKey marks a field as primary key
FieldFlagPrimaryKey FieldFlag = 1 << iota
// FieldFlagReadOnly marks a field as read-only
FieldFlagReadOnly
// FieldFlagDefault marks a field as having a column default value
FieldFlagDefault
)
// StructFieldMapper is used to map struct type fields to column names
// and indicate special column properies via flags.
type StructFieldMapper interface {
// MapStructField returns the column name for a reflected struct field
// and flags for special column properies.
// If false is returned for use then the field is not mapped.
// An empty name and true for use indicates an embedded struct
// field whose fields should be recursively mapped.
MapStructField(field reflect.StructField) (table, column string, flags FieldFlag, use bool)
}
// NewTaggedStructFieldMapping returns a default mapping.
func NewTaggedStructFieldMapping() *TaggedStructFieldMapping {
return &TaggedStructFieldMapping{
NameTag: "db",
Ignore: "-",
PrimaryKey: "pk",
ReadOnly: "readonly",
Default: "default",
UntaggedNameFunc: IgnoreStructField,
}
}
// DefaultStructFieldMapping provides the default StructFieldTagNaming
// using "db" as NameTag and IgnoreStructField as UntaggedNameFunc.
// Implements StructFieldMapper.
var DefaultStructFieldMapping = NewTaggedStructFieldMapping()
// TaggedStructFieldMapping implements StructFieldMapper
var _ StructFieldMapper = new(TaggedStructFieldMapping)
// TaggedStructFieldMapping implements StructFieldMapper with a struct field NameTag
// to be used for naming and a UntaggedNameFunc in case the NameTag is not set.
type TaggedStructFieldMapping struct {
_Named_Fields_Required struct{}
// NameTag is the struct field tag to be used as column name
NameTag string
// Ignore will cause a struct field to be ignored if it has that name
Ignore string
PrimaryKey string
ReadOnly string
Default string
// UntaggedNameFunc will be called with the struct field name to
// return a column name in case the struct field has no tag named NameTag.
UntaggedNameFunc func(fieldName string) string
}
func (m *TaggedStructFieldMapping) MapStructField(field reflect.StructField) (table, column string, flags FieldFlag, use bool) {
if field.Anonymous {
column, hasTag := field.Tag.Lookup(m.NameTag)
if !hasTag {
// Embedded struct fields are ok if not tagged with IgnoreName
return "", "", 0, true
}
if i := strings.IndexByte(column, ','); i != -1 {
column = column[:i]
}
// Embedded struct fields are ok if not tagged with IgnoreName
return "", "", 0, column != m.Ignore
}
if !field.IsExported() {
// Not exported struct fields that are not
// anonymously embedded structs are not ok
return "", "", 0, false
}
tag, hasTag := field.Tag.Lookup(m.NameTag)
if hasTag {
for i, part := range strings.Split(tag, ",") {
// First part is the name
if i == 0 {
column = part
continue
}
// Follow on parts are flags
flag, value, _ := strings.Cut(part, "=")
switch flag {
case "":
// Ignore empty flags
case m.PrimaryKey:
flags |= FieldFlagPrimaryKey
table = value
case m.ReadOnly:
flags |= FieldFlagReadOnly
case m.Default:
flags |= FieldFlagDefault
}
}
} else {
column = m.UntaggedNameFunc(field.Name)
}
if column == "" || column == m.Ignore {
return "", "", 0, false
}
return table, column, flags, true
}
func (n TaggedStructFieldMapping) String() string {
return fmt.Sprintf("NameTag: %q", n.NameTag)
}
// IgnoreStructField can be used as TaggedStructFieldMapping.UntaggedNameFunc
// to ignore fields that don't have TaggedStructFieldMapping.NameTag.
func IgnoreStructField(string) string { return "" }
// ToSnakeCase converts s to snake case
// by lower casing everything and inserting '_'
// before every new upper case character in s.
// Whitespace, symbol, and punctuation characters
// will be replace by '_'.
var ToSnakeCase = strutil.ToSnakeCase