-
Notifications
You must be signed in to change notification settings - Fork 14
/
dialect.go
137 lines (120 loc) · 3.05 KB
/
dialect.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
package sqlf
import (
"strconv"
"sync/atomic"
"github.com/valyala/bytebufferpool"
)
// Dialect defines the method SQL statement is to be built.
//
// NoDialect is a default statement builder mode.
// No SQL fragments will be altered.
// PostgreSQL mode can be set for a statement:
//
// q := sqlf.PostgreSQL.From("table").Select("field")
// ...
// q.Close()
//
// or as default mode:
//
// sqlf.SetDialect(sqlf.PostgreSQL)
// ...
// q := sqlf.From("table").Select("field")
// q.Close()
//
// Wher PostgreSQL mode is activated, ? placeholders are
// replaced with numbered positional arguments like $1, $2...
type Dialect uint32
const (
// NoDialect is a default statement builder mode.
NoDialect Dialect = iota
// PostgreSQL mode is to be used to automatically replace ? placeholders with $1, $2...
PostgreSQL
)
var defaultDialect = NoDialect
/*
SetDialect selects a Dialect to be used by default.
Dialect can be one of sqlf.NoDialect or sqlf.PostgreSQL
sqlf.SetDialect(sqlf.PostgreSQL)
*/
func SetDialect(dialect Dialect) {
atomic.StoreUint32((*uint32)(&defaultDialect), uint32(dialect))
}
/*
New starts an SQL statement with an arbitrary verb.
Use From, Select, InsertInto or DeleteFrom methods to create
an instance of an SQL statement builder for common statements.
*/
func (b Dialect) New(verb string, args ...interface{}) *Stmt {
q := getStmt(b)
q.addChunk(posSelect, verb, "", args, ", ")
return q
}
/*
From starts a SELECT statement.
*/
func (b Dialect) From(expr string, args ...interface{}) *Stmt {
q := getStmt(b)
return q.From(expr, args...)
}
/*
Select starts a SELECT statement.
Consider using From method to start a SELECT statement - you may find
it easier to read and maintain.
*/
func (b Dialect) Select(expr string, args ...interface{}) *Stmt {
q := getStmt(b)
return q.Select(expr, args...)
}
// Update starts an UPDATE statement.
func (b Dialect) Update(tableName string) *Stmt {
q := getStmt(b)
return q.Update(tableName)
}
// InsertInto starts an INSERT statement.
func (b Dialect) InsertInto(tableName string) *Stmt {
q := getStmt(b)
return q.InsertInto(tableName)
}
// DeleteFrom starts a DELETE statement.
func (b Dialect) DeleteFrom(tableName string) *Stmt {
q := getStmt(b)
return q.DeleteFrom(tableName)
}
// writePg function copies s into buf and replaces ? placeholders with $1, $2...
func writePg(argNo int64, s []byte, buf *bytebufferpool.ByteBuffer) (int64, error) {
var err error
start := 0
// Iterate by runes
for pos, r := range bufToString(&s) {
if start > pos {
continue
}
switch r {
case '\\':
if pos < len(s)-1 && s[pos+1] == '?' {
_, err = buf.Write(s[start:pos])
if err == nil {
err = buf.WriteByte('?')
}
start = pos + 2
}
case '?':
_, err = buf.Write(s[start:pos])
start = pos + 1
if err == nil {
err = buf.WriteByte('$')
if err == nil {
buf.B = strconv.AppendInt(buf.B, int64(argNo), 10)
argNo++
}
}
}
if err != nil {
break
}
}
if err == nil && start < len(s) {
_, err = buf.Write(s[start:])
}
return argNo, err
}