-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy patherror.go
More file actions
235 lines (210 loc) · 5.18 KB
/
error.go
File metadata and controls
235 lines (210 loc) · 5.18 KB
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
package pg
import (
"errors"
"fmt"
// Packages
pgx "github.com/jackc/pgx/v5"
pgconn "github.com/jackc/pgx/v5/pgconn"
)
/////////////////////////////////////////////////////////////////////
// TYPES
type Err int
type DatabaseError struct {
code string
message string
err error
kinds []Err
}
/////////////////////////////////////////////////////////////////////
// GLOBALS
const (
ErrSuccess Err = iota
ErrNotFound
ErrNotImplemented
ErrBadParameter
ErrNotAvailable
ErrConflict
ErrDatabase
ErrUniqueViolation
ErrForeignKeyViolation
ErrNotNullViolation
ErrCheckViolation
ErrInvalidTextRepresentation
ErrInvalidDatetimeFormat
ErrDatetimeFieldOverflow
)
const (
sqlStateUniqueViolation = "23505"
sqlStateForeignKeyViolation = "23503"
sqlStateNotNullViolation = "23502"
sqlStateCheckViolation = "23514"
sqlStateInvalidTextRepresentation = "22P02"
sqlStateInvalidDatetimeFormat = "22007"
sqlStateDatetimeFieldOverflow = "22008"
)
// Error returns the string representation of the error.
func (e Err) Error() string {
switch e {
case ErrSuccess:
return "success"
case ErrNotFound:
return "not found"
case ErrNotImplemented:
return "not implemented"
case ErrBadParameter:
return "bad parameter"
case ErrNotAvailable:
return "not available"
case ErrConflict:
return "conflict"
case ErrDatabase:
return "database error"
case ErrUniqueViolation:
return "unique violation"
case ErrForeignKeyViolation:
return "foreign key violation"
case ErrNotNullViolation:
return "not null violation"
case ErrCheckViolation:
return "check violation"
case ErrInvalidTextRepresentation:
return "invalid text representation"
case ErrInvalidDatetimeFormat:
return "invalid datetime format"
case ErrDatetimeFieldOverflow:
return "datetime field overflow"
default:
return fmt.Sprint("Unknown error ", int(e))
}
}
// With returns the error with additional context appended.
func (e Err) With(a ...any) error {
return fmt.Errorf("%w: %s", e, fmt.Sprint(a...))
}
// Withf returns the error with formatted context appended.
func (e Err) Withf(format string, a ...any) error {
return fmt.Errorf("%w: %s", e, fmt.Sprintf(format, a...))
}
// Error returns the wrapped database error string.
func (e *DatabaseError) Error() string {
if e == nil {
return "<nil>"
}
if e.err != nil {
return e.err.Error()
}
if e.message != "" {
return e.message
}
return e.code
}
// Unwrap returns the underlying driver error.
func (e *DatabaseError) Unwrap() error {
if e == nil {
return nil
}
return e.err
}
// Is supports errors.Is checks against broad and specific package errors.
func (e *DatabaseError) Is(target error) bool {
if e == nil {
return false
}
if kind, ok := target.(Err); ok {
for _, candidate := range e.kinds {
if candidate == kind {
return true
}
}
}
return errors.Is(e.err, target)
}
// SQLState returns the PostgreSQL SQLSTATE code.
func (e *DatabaseError) SQLState() string {
if e == nil {
return ""
}
return e.code
}
// Message returns the PostgreSQL error message.
func (e *DatabaseError) Message() string {
if e == nil {
return ""
}
return e.message
}
/////////////////////////////////////////////////////////////////////
// PUBLIC METHODS
// NormalizeError maps driver-specific PostgreSQL errors to package errors.
func NormalizeError(err error) error {
if err == nil {
return nil
}
if errors.Is(err, ErrNotFound) {
return err
}
var dbErr *DatabaseError
if errors.As(err, &dbErr) {
return err
}
if errors.Is(err, pgx.ErrNoRows) {
return ErrNotFound
}
var pgErr *pgconn.PgError
if errors.As(err, &pgErr) {
return newDatabaseError(pgErr)
}
return err
}
// IsDatabaseError reports whether err is a PostgreSQL error with a SQLSTATE code.
func IsDatabaseError(err error) bool {
return SQLState(err) != ""
}
// SQLState returns the PostgreSQL SQLSTATE code for err, if one is available.
func SQLState(err error) string {
if err == nil {
return ""
}
var dbErr *DatabaseError
if errors.As(err, &dbErr) {
return dbErr.SQLState()
}
var pgErr *pgconn.PgError
if errors.As(err, &pgErr) {
return pgErr.Code
}
return ""
}
func pgerror(err error) error {
return NormalizeError(err)
}
/////////////////////////////////////////////////////////////////////
// PRIVATE METHODS
func newDatabaseError(err *pgconn.PgError) error {
if err == nil {
return nil
}
kinds := []Err{ErrDatabase}
switch err.Code {
case sqlStateUniqueViolation:
kinds = append(kinds, ErrConflict, ErrUniqueViolation)
case sqlStateForeignKeyViolation:
kinds = append(kinds, ErrBadParameter, ErrForeignKeyViolation)
case sqlStateNotNullViolation:
kinds = append(kinds, ErrBadParameter, ErrNotNullViolation)
case sqlStateCheckViolation:
kinds = append(kinds, ErrBadParameter, ErrCheckViolation)
case sqlStateInvalidTextRepresentation:
kinds = append(kinds, ErrBadParameter, ErrInvalidTextRepresentation)
case sqlStateInvalidDatetimeFormat:
kinds = append(kinds, ErrBadParameter, ErrInvalidDatetimeFormat)
case sqlStateDatetimeFieldOverflow:
kinds = append(kinds, ErrBadParameter, ErrDatetimeFieldOverflow)
}
return &DatabaseError{
code: err.Code,
message: err.Message,
err: err,
kinds: kinds,
}
}