forked from tee8z/nullable
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathuint32.go
145 lines (126 loc) · 2.84 KB
/
uint32.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
package nullable
import (
"context"
"database/sql/driver"
"encoding/json"
"strconv"
"gorm.io/gorm"
"gorm.io/gorm/clause"
"gorm.io/gorm/schema"
)
// Uint32 SQL type that can retrieve NULL value
type Uint32 struct {
realValue uint32
isValid bool
}
// NewUint32 creates a new nullable 32-bit unsigned integer
func NewUint32(value *uint32) Uint32 {
if value == nil {
return Uint32{
realValue: 0,
isValid: false,
}
}
return Uint32{
realValue: *value,
isValid: true,
}
}
// Get either nil or 32-bit unsigned integer
func (n Uint32) Get() *uint32 {
if !n.isValid {
return nil
}
return &n.realValue
}
// Set either nil or 32-bit unsigned integer
func (n *Uint32) Set(value *uint32) {
n.isValid = (value != nil)
if n.isValid {
n.realValue = *value
} else {
n.realValue = 0
}
}
// MarshalJSON converts current value to JSON
func (n Uint32) MarshalJSON() ([]byte, error) {
return json.Marshal(n.Get())
}
// UnmarshalJSON writes JSON to this type
func (n *Uint32) UnmarshalJSON(data []byte) error {
dataString := string(data)
if len(dataString) == 0 || dataString == "null" {
n.isValid = false
n.realValue = 0
return nil
}
var parsed uint32
if err := json.Unmarshal(data, &parsed); err != nil {
return err
}
n.isValid = true
n.realValue = parsed
return nil
}
// Scan implements scanner interface
func (n *Uint32) Scan(value interface{}) error {
if value == nil {
n.realValue, n.isValid = 0, false
return nil
}
var scanned string
if err := convertAssign(&scanned, value); err != nil {
return err
}
radix := 10
if len(scanned) == 32 {
radix = 2
}
parsed, err := strconv.ParseUint(scanned, radix, 32)
if err != nil {
return err
}
n.realValue = uint32(parsed)
n.isValid = true
return nil
}
// Value implements the driver Valuer interface.
func (n Uint32) Value() (driver.Value, error) {
if !n.isValid {
return nil, nil
}
return strconv.FormatUint(uint64(n.realValue), 10), nil
}
// GormValue implements the driver Valuer interface via GORM.
func (n Uint32) GormValue(ctx context.Context, db *gorm.DB) clause.Expr {
switch db.Dialector.Name() {
case "sqlite", "mysql":
// MySQL and SQLite are using Value() instead of GormValue()
value, err := n.Value()
if err != nil {
db.AddError(err)
return clause.Expr{}
}
return clause.Expr{SQL: "?", Vars: []interface{}{value}}
case "postgres":
if !n.isValid {
return clause.Expr{SQL: "?", Vars: []interface{}{nil}}
}
return clause.Expr{SQL: "?", Vars: []interface{}{n.realValue}}
}
return clause.Expr{}
}
// GormDataType gorm common data type
func (Uint32) GormDataType() string {
return "uint32_null"
}
// GormDBDataType gorm db data type
func (Uint32) GormDBDataType(db *gorm.DB, field *schema.Field) string {
switch db.Dialector.Name() {
case "sqlite", "mysql":
return "INT UNSIGNED"
case "postgres":
return "numeric"
}
return ""
}