forked from src-d/go-kallax
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmodel.go
487 lines (413 loc) · 12.6 KB
/
model.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
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
package kallax
import (
"bytes"
"database/sql"
"database/sql/driver"
"encoding/hex"
"fmt"
"math/rand"
"sync"
"time"
"github.com/oklog/ulid"
uuid "github.com/satori/go.uuid"
)
// Model contains all the basic fields that make something a model, that is,
// the ID and some internal data used by kallax.
// To make a struct a model, it only needs to have Model embedded.
//
// type MyModel struct {
// kallax.Model
// Foo string
// }
//
// Custom name for the table can be specified using the struct tag `table` when
// embedding the Model.
//
// type MyModel struct {
// kallax.Model `table:"custom_name"`
// }
//
// Otherwise, the default name of the table is the name of the model converted
// to lower snake case. E.g: MyModel => my_model.
// No pluralization is done right now, but might be done in the future, so
// please, set the name of the tables yourself.
type Model struct {
virtualColumns map[string]Identifier
persisted bool
writable bool
saving bool
}
// NewModel creates a new Model that is writable and not persisted.
func NewModel() Model {
return Model{
persisted: false,
writable: true,
virtualColumns: make(map[string]Identifier),
saving: false,
}
}
// IsPersisted returns whether the Model has already been persisted to the
// database or not.
func (m *Model) IsPersisted() bool {
return m.persisted
}
func (m *Model) setPersisted() {
m.persisted = true
}
// IsWritable returns whether this Model can be saved into the database.
// For example, a model with partially retrieved data is not writable, so
// it is not saved by accident and the data is corrupted. For example, if
// you select only 2 columns out of all the ones the table has, it will not
// be writable.
func (m *Model) IsWritable() bool {
return m.writable
}
func (m *Model) setWritable(w bool) {
m.writable = w
}
// IsSaving reports whether the model is in the process of being saved or not.
func (m *Model) IsSaving() bool {
return m.saving
}
// SetSaving sets the saving status of the model.
// This is an internal function, even though it is exposed for the concrete
// kallax generated code to use. Please, don't use it, and if you do,
// bad things may happen.
func (m *Model) SetSaving(saving bool) {
m.saving = saving
}
// ClearVirtualColumns clears all the previous virtual columns.
// This method is only intended for internal use. It is only exposed for
// technical reasons.
func (m *Model) ClearVirtualColumns() {
m.virtualColumns = make(map[string]Identifier)
}
func (m *Model) getVirtualColumns() map[string]Identifier {
return m.virtualColumns
}
// AddVirtualColumn adds a new virtual column with the given name and value.
// This method is only intended for internal use. It is only exposed for
// technical reasons.
func (m *Model) AddVirtualColumn(name string, v Identifier) {
if m.virtualColumns == nil {
m.ClearVirtualColumns()
}
m.virtualColumns[name] = v
}
// VirtualColumn returns the value of the virtual column with the given column name.
// This method is only intended for internal use. It is only exposed for
// technical reasons.
func (m *Model) VirtualColumn(name string) Identifier {
if m.virtualColumns == nil {
m.ClearVirtualColumns()
}
return m.virtualColumns[name]
}
// Identifier is a type used to identify a model.
type Identifier interface {
sql.Scanner
driver.Valuer
// Equals reports whether the identifier and the given one are equal.
Equals(Identifier) bool
// IsEmpty returns whether the ID is empty or not.
IsEmpty() bool
// Raw returns the internal value of the identifier.
Raw() interface{}
}
// Identifiable must be implemented by those values that can be identified by an ID.
type Identifiable interface {
// GetID returns the ID.
GetID() Identifier
}
// Persistable must be implemented by those values that can be persisted.
type Persistable interface {
// IsPersisted returns whether this Model is new in the store or not.
IsPersisted() bool
setPersisted()
}
// Writable must be implemented by those values that defines internally
// if they can be sent back to the database to be stored with its changes.
type Writable interface {
// IsWritable returns whether this Model can be saved into the database.
IsWritable() bool
setWritable(bool)
}
// ColumnAddresser provides the pointer addresses of columns.
type ColumnAddresser interface {
// ColumnAddress returns the pointer to the column value of the given
// column name, or an error if it does not exist in the model.
ColumnAddress(string) (interface{}, error)
}
// Relationable can perform operations related to relationships of a record.
type Relationable interface {
// NewRelationshipRecord returns a new Record for the relationship at the
// given field.
NewRelationshipRecord(string) (Record, error)
// SetRelationship sets the relationship value at the given field.
SetRelationship(string, interface{}) error
}
// Valuer provides the values for columns.
type Valuer interface {
// Value returns the value of the given column, or an error if it does not
// exist in the model.
Value(string) (interface{}, error)
}
// VirtualColumnContainer contains a collection of virtual columns and
// manages them.
type VirtualColumnContainer interface {
// ClearVirtualColumns removes all virtual columns.
ClearVirtualColumns()
// AddVirtualColumn adds a new virtual column with the given name and value
AddVirtualColumn(string, Identifier)
// VirtualColumn returns the virtual column with the given column name.
VirtualColumn(string) Identifier
getVirtualColumns() map[string]Identifier
}
var ErrEmptyVirtualColumn = fmt.Errorf("empty virtual column")
// RecordValues returns the values of a record at the given columns in the same
// order as the columns.
// It also returns the columns with any empty virtual column removed.
func RecordValues(record Valuer, columns ...string) ([]interface{}, []string, error) {
var cols = make([]string, 0, len(columns))
var values = make([]interface{}, 0, len(columns))
for _, col := range columns {
v, err := record.Value(col)
if err == ErrEmptyVirtualColumn {
continue
}
if err != nil {
return nil, nil, err
}
values = append(values, v)
cols = append(cols, col)
}
return values, cols, nil
}
// Saveable can report whether it's being saved or change the saving status.
type Saveable interface {
IsSaving() bool
SetSaving(bool)
}
// Record is something that can be stored as a row in the database.
type Record interface {
Identifiable
Persistable
Writable
Relationable
ColumnAddresser
Valuer
VirtualColumnContainer
Saveable
}
var randPool = &sync.Pool{
New: func() interface{} {
return rand.NewSource(time.Now().UnixNano())
},
}
// ULID is an ID type provided by kallax that is a lexically sortable UUID.
// The internal representation is an ULID (https://github.com/oklog/ulid).
// It already implements sql.Scanner and driver.Valuer, so it's perfectly
// safe for database usage.
type ULID uuid.UUID
// NewULID returns a new ULID, which is a lexically sortable UUID.
func NewULID() ULID {
entropy := randPool.Get().(rand.Source)
id := ULID(ulid.MustNew(ulid.Timestamp(time.Now()), rand.New(entropy)))
randPool.Put(entropy)
return id
}
// NewULIDFromText creates a new ULID from its string representation. Will
// return an error if the text is not a valid ULID.
func NewULIDFromText(text string) (ULID, error) {
var id ULID
err := id.UnmarshalText([]byte(text))
return id, err
}
// Scan implements the Scanner interface.
func (id *ULID) Scan(src interface{}) error {
switch src := src.(type) {
case []byte:
if len(src) != 16 {
return id.UnmarshalText(src)
}
var ulid ulid.ULID
if err := ulid.UnmarshalBinary(src); err != nil {
return err
}
*id = ULID(ulid)
return nil
case string:
return id.Scan([]byte(src))
default:
return fmt.Errorf("kallax: cannot scan %T into ULID", src)
}
}
var (
urnPrefix = []byte("urn:uuid:")
byteGroups = []int{8, 4, 4, 4, 12}
)
// UnmarshalText implements the encoding.TextUnmarshaler interface.
// Following formats are supported:
// "6ba7b810-9dad-11d1-80b4-00c04fd430c8",
// "{6ba7b810-9dad-11d1-80b4-00c04fd430c8}",
// "urn:uuid:6ba7b810-9dad-11d1-80b4-00c04fd430c8"
// Implements the exact same code as the UUID UnmarshalText removing the
// version check.
func (u *ULID) UnmarshalText(text []byte) (err error) {
if len(text) < 32 {
err = fmt.Errorf("uuid: UUID string too short: %s", text)
return
}
t := text[:]
braced := false
if bytes.Equal(t[:9], urnPrefix) {
t = t[9:]
} else if t[0] == '{' {
braced = true
t = t[1:]
}
b := u[:]
for i, byteGroup := range byteGroups {
if i > 0 && t[0] == '-' {
t = t[1:]
} else if i > 0 && t[0] != '-' {
err = fmt.Errorf("kallax: invalid ulid string format")
return
}
if len(t) < byteGroup {
err = fmt.Errorf("kallax: ulid string too short: %s", text)
return
}
if i == 4 && len(t) > byteGroup &&
((braced && t[byteGroup] != '}') || len(t[byteGroup:]) > 1 || !braced) {
err = fmt.Errorf("kallax: ulid string too long: %s", t)
return
}
_, err = hex.Decode(b[:byteGroup/2], t[:byteGroup])
if err != nil {
return
}
t = t[byteGroup:]
b = b[byteGroup/2:]
}
return
}
func (id ULID) MarshalText() ([]byte, error) {
return []byte(id.String()), nil
}
// Value implements the Valuer interface.
func (id ULID) Value() (driver.Value, error) {
return uuid.UUID(id).Value()
}
// IsEmpty returns whether the ID is empty or not. An empty ID means it has not
// been set yet.
func (id ULID) IsEmpty() bool {
return uuid.Equal(uuid.UUID(id), uuid.Nil)
}
// String returns the string representation of the ID.
func (id ULID) String() string {
return uuid.UUID(id).String()
}
// Equals reports whether the ID and the given one are equals.
func (id ULID) Equals(other Identifier) bool {
v, ok := other.(*ULID)
if !ok {
return false
}
return uuid.Equal(uuid.UUID(id), uuid.UUID(*v))
}
// Raw returns the underlying raw value.
func (id ULID) Raw() interface{} {
return id
}
// NumericID is a wrapper for int64 that implements the Identifier interface.
// You don't need to actually use this as a type in your model. They will be
// automatically converted to and from in the generated code.
type NumericID int64
// Scan implements the Scanner interface.
func (id *NumericID) Scan(src interface{}) error {
switch src := src.(type) {
case int64:
*(*int64)(id) = src
default:
return fmt.Errorf("kallax: cannot scan value of type %T into a numeric ID", src)
}
return nil
}
// Value implements the Valuer interface.
func (id NumericID) Value() (driver.Value, error) {
return int64(id), nil
}
// IsEmpty returns whether the ID is empty or not. An empty ID means it has not
// been set yet.
func (id NumericID) IsEmpty() bool {
return int64(id) == 0
}
// String returns the string representation of the ID.
func (id NumericID) String() string {
return fmt.Sprint(int64(id))
}
// Equals reports whether the ID and the given one are equals.
func (id NumericID) Equals(other Identifier) bool {
v, ok := other.(*NumericID)
if !ok {
return false
}
return int64(id) == int64(*v)
}
// Raw returns the underlying raw value.
func (id NumericID) Raw() interface{} {
return id
}
// UUID is a wrapper type for uuid.UUID that implements the Identifier
// interface.
// You don't need to actually use this as a type in your model. They will be
// automatically converted to and from in the generated code.
type UUID uuid.UUID
// Scan implements the Scanner interface.
func (id *UUID) Scan(src interface{}) error {
return (*uuid.UUID)(id).Scan(src)
}
// Value implements the Valuer interface.
func (id UUID) Value() (driver.Value, error) {
return uuid.UUID(id).Value()
}
// IsEmpty returns whether the ID is empty or not. An empty ID means it has not
// been set yet.
func (id UUID) IsEmpty() bool {
return uuid.Equal(uuid.UUID(id), uuid.Nil)
}
// String returns the string representation of the ID.
func (id UUID) String() string {
return uuid.UUID(id).String()
}
// Equals reports whether the ID and the given one are equals.
func (id UUID) Equals(other Identifier) bool {
v, ok := other.(*UUID)
if !ok {
return false
}
return uuid.Equal(uuid.UUID(id), uuid.UUID(*v))
}
// Raw returns the underlying raw value.
func (id UUID) Raw() interface{} {
return id
}
type virtualColumn struct {
r Record
col string
id Identifier
}
// VirtualColumn returns a sql.Scanner that will scan the given column as a
// virtual column in the given record.
func VirtualColumn(col string, r Record, id Identifier) sql.Scanner {
return &virtualColumn{r, col, id}
}
// Scan implements the scanner interface.
func (c *virtualColumn) Scan(src interface{}) error {
if err := c.id.Scan(src); err != nil {
return err
}
c.r.AddVirtualColumn(c.col, c.id)
return nil
}