This repository was archived by the owner on Apr 2, 2024. It is now read-only.
generated from mrz1836/go-template
-
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathmodels_internal.go
221 lines (189 loc) · 5.89 KB
/
models_internal.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
package bux
import (
"context"
"time"
"github.com/BuxOrg/bux/notifications"
)
// AfterDeleted will fire after a successful delete in the Datastore
func (m *Model) AfterDeleted(_ context.Context) error {
m.Client().Logger().Debug().Msgf("starting: %s AfterDeleted hook...", m.Name())
m.Client().Logger().Debug().Msgf("end: %s AfterDeleted hook", m.Name())
return nil
}
// BeforeUpdating will fire before updating a model in the Datastore
func (m *Model) BeforeUpdating(_ context.Context) error {
m.Client().Logger().Debug().Msgf("starting: %s BeforeUpdate hook...", m.Name())
m.Client().Logger().Debug().Msgf("end: %s BeforeUpdate hook", m.Name())
return nil
}
// Client will return the current client
func (m *Model) Client() ClientInterface {
return m.client
}
// ChildModels will return any child models
func (m *Model) ChildModels() []ModelInterface {
return nil
}
// enrich is run after getting a record from the database
func (m *Model) enrich(name ModelName, opts ...ModelOps) {
// Set the name
m.name = name
// Overwrite defaults
m.SetOptions(opts...)
}
// GetOptions will get the options that are set on that model
func (m *Model) GetOptions(isNewRecord bool) (opts []ModelOps) {
// Client was set on the model
if m.client != nil {
opts = append(opts, WithClient(m.client))
}
// New record flag
if isNewRecord {
opts = append(opts, New())
}
return
}
// IsNew returns true if the model is (or was) a new record
func (m *Model) IsNew() bool {
return m.newRecord
}
// GetID will get the model id, if overwritten in the actual model
func (m *Model) GetID() string {
return ""
}
// Name will get the collection name (model)
func (m *Model) Name() string {
return m.name.String()
}
// New will set the record to new
func (m *Model) New() {
m.newRecord = true
}
// NotNew sets newRecord to false
func (m *Model) NotNew() {
m.newRecord = false
}
// RawXpub returns the rawXpubKey
func (m *Model) RawXpub() string {
return m.rawXpubKey
}
// SetRecordTime will set the record timestamps (created is true for a new record)
func (m *Model) SetRecordTime(created bool) {
if created {
m.CreatedAt = time.Now().UTC()
} else {
m.UpdatedAt = time.Now().UTC()
}
}
// UpdateMetadata will update the metadata on the model
// any key set to nil will be removed, other keys updated or added
func (m *Model) UpdateMetadata(metadata Metadata) {
if m.Metadata == nil {
m.Metadata = make(Metadata)
}
for key, value := range metadata {
if value == nil {
delete(m.Metadata, key)
} else {
m.Metadata[key] = value
}
}
}
// SetOptions will set the options on the model
func (m *Model) SetOptions(opts ...ModelOps) {
for _, opt := range opts {
opt(m)
}
}
// Display filter the model for display
func (m *Model) Display() interface{} {
return m
}
// AfterUpdated will fire after a successful update into the Datastore
func (m *Model) AfterUpdated(_ context.Context) error {
m.Client().Logger().Debug().Msgf("starting: %s AfterUpdated hook...", m.Name())
m.Client().Logger().Debug().Msgf("end: %s AfterUpdated hook", m.Name())
return nil
}
// AfterCreated will fire after the model is created in the Datastore
func (m *Model) AfterCreated(_ context.Context) error {
m.Client().Logger().Debug().Msgf("starting: %s AfterCreated hook...", m.Name())
m.Client().Logger().Debug().Msgf("end: %s AfterCreated hook", m.Name())
return nil
}
// incrementField will increment the given field atomically in the datastore
func incrementField(ctx context.Context, model ModelInterface, fieldName string,
increment int64,
) (int64, error) {
// Check for client
c := model.Client()
if c == nil {
return 0, ErrMissingClient
}
// Increment
newValue, err := c.Datastore().IncrementModel(ctx, model, fieldName, increment)
if err != nil {
return 0, err
}
// AfterUpdate event should be called by parent function
return newValue, nil
}
// notify about an event on the model
func notify(eventType notifications.EventType, model interface{}) {
// run the notifications in a separate goroutine since there could be significant network delay
// communicating with a notification provider
go func() {
m := model.(ModelInterface)
if client := m.Client(); client != nil {
if n := client.Notifications(); n != nil {
if err := n.Notify(
context.Background(), m.GetModelName(), eventType, model, m.GetID(),
); err != nil {
client.Logger().Error().Msgf("failed notifying about %s on %s: %s", string(eventType), m.GetID(), err.Error())
}
}
}
}()
}
/*
// setFieldValueByJSONTag will parse the struct looking for the field (json tag) and updating the value if found
//
// todo: this was created because the increment field was not updating the model's value
func setFieldValueByJSONTag(item interface{}, fieldName string, value interface{}) error {
v := reflect.ValueOf(item).Elem()
if !v.CanAddr() {
return fmt.Errorf("cannot assign to the item passed, item must be a pointer in order to assign")
}
// It's possible we can cache this, which is why precompute all these ahead of time.
findJSONName := func(t reflect.StructTag) (string, error) {
if jt, ok := t.Lookup("json"); ok {
return strings.Split(jt, ",")[0], nil
}
return "", fmt.Errorf("tag provided does not define a json tag: %s", fieldName)
}
fieldNames := map[string]int{}
for i := 0; i < v.NumField(); i++ {
structTypeField := v.Type().Field(i)
jName, _ := findJSONName(structTypeField.Tag)
if jName != "" && jName != "-" {
fieldNames[jName] = i
}
}
fieldNum, ok := fieldNames[fieldName]
if !ok {
return fmt.Errorf("field %s does not exist within the provided item", fieldName)
}
fieldVal := v.Field(fieldNum)
switch fieldVal.Interface().(type) {
case uint8:
fieldVal.Set(reflect.ValueOf(uint8(value.(int64))))
case uint16:
fieldVal.Set(reflect.ValueOf(uint16(value.(int64))))
case uint32:
fieldVal.Set(reflect.ValueOf(uint32(value.(int64))))
case uint64:
fieldVal.Set(reflect.ValueOf(uint64(value.(int64))))
}
return nil
}
*/