forked from gobuffalo/pop
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathquery.go
230 lines (205 loc) · 6.61 KB
/
query.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
package pop
import (
"fmt"
"strings"
"github.com/friesencr/pop/v6/logging"
)
type operation string
const (
Select operation = "SELECT"
Delete operation = "DELETE"
)
// Query is the main value that is used to build up a query
// to be executed against the `Connection`.
type Query struct {
RawSQL *clause
limitResults int
addColumns []string
eagerMode EagerMode
eager bool
eagerFields []string
whereClauses clauses
orderClauses clauses
fromClauses fromClauses
belongsToThroughClauses belongsToThroughClauses
joinClauses joinClauses
groupClauses groupClauses
havingClauses havingClauses
Paginator *Paginator
Connection *Connection
Operation operation
}
// Clone will fill targetQ query with the connection used in q, if
// targetQ is not empty, Clone will override all the fields.
func (q *Query) Clone(targetQ *Query) {
rawSQL := *q.RawSQL
targetQ.RawSQL = &rawSQL
targetQ.limitResults = q.limitResults
targetQ.whereClauses = q.whereClauses
targetQ.orderClauses = q.orderClauses
targetQ.fromClauses = q.fromClauses
targetQ.belongsToThroughClauses = q.belongsToThroughClauses
targetQ.joinClauses = q.joinClauses
targetQ.groupClauses = q.groupClauses
targetQ.havingClauses = q.havingClauses
targetQ.addColumns = q.addColumns
targetQ.Operation = q.Operation
if q.Paginator != nil {
paginator := *q.Paginator
targetQ.Paginator = &paginator
}
if q.Connection != nil {
connection := *q.Connection
targetQ.Connection = &connection
}
}
// RawQuery will override the query building feature of Pop and will use
// whatever query you want to execute against the `Connection`. You can continue
// to use the `?` argument syntax.
//
// c.RawQuery("select * from foo where id = ?", 1)
func (c *Connection) RawQuery(stmt string, args ...interface{}) *Query {
return Q(c).RawQuery(stmt, args...)
}
// RawQuery will override the query building feature of Pop and will use
// whatever query you want to execute against the `Connection`. You can continue
// to use the `?` argument syntax.
//
// q.RawQuery("select * from foo where id = ?", 1)
func (q *Query) RawQuery(stmt string, args ...interface{}) *Query {
q.RawSQL = &clause{stmt, args}
return q
}
// Eager will enable associations loading of the model.
// by defaults loads all the associations on the model,
// but can take a variadic list of associations to load.
//
// c.Eager().Find(model, 1) // will load all associations for model.
// c.Eager("Books").Find(model, 1) // will load only Book association for model.
//
// Eager also enable nested models creation:
//
// model := Parent{Child: Child{}, Parent: &Parent{}}
// c.Eager().Create(&model) // will create all associations for model.
// c.Eager("Child").Create(&model) // will only create the Child association for model.
func (c *Connection) Eager(fields ...string) *Connection {
con := c.copy()
con.eager = true
con.eagerFields = append(c.eagerFields, fields...)
return con
}
// Eager will enable load associations of the model.
// by defaults loads all the associations on the model,
// but can take a variadic list of associations to load.
//
// q.Eager().Find(model, 1) // will load all associations for model.
// q.Eager("Books").Find(model, 1) // will load only Book association for model.
func (q *Query) Eager(fields ...string) *Query {
q.eager = true
q.eagerFields = append(q.eagerFields, fields...)
return q
}
// disableEager disables eager mode for current query and Connection.
func (q *Query) disableEager() {
q.Connection.eager, q.eager = false, false
q.Connection.eagerFields, q.eagerFields = []string{}, []string{}
}
// Where will append a where clause to the query. You may use `?` in place of
// arguments.
//
// c.Where("id = ?", 1)
// q.Where("id in (?)", 1, 2, 3)
func (c *Connection) Where(stmt string, args ...interface{}) *Query {
q := Q(c)
return q.Where(stmt, args...)
}
// Where will append a where clause to the query. You may use `?` in place of
// arguments.
//
// q.Where("id = ?", 1)
// q.Where("id in (?)", 1, 2, 3)
func (q *Query) Where(stmt string, args ...interface{}) *Query {
if q.RawSQL.Fragment != "" {
log(logging.Warn, "Query is setup to use raw SQL")
return q
}
if inRegex.MatchString(stmt) {
var inq []string
for i := 0; i < len(args); i++ {
inq = append(inq, "?")
}
qs := fmt.Sprintf("(%s)", strings.Join(inq, ","))
stmt = inRegex.ReplaceAllString(stmt, " IN "+qs)
}
q.whereClauses = append(q.whereClauses, clause{stmt, args})
return q
}
// Order will append an order clause to the query.
//
// c.Order("name desc")
func (c *Connection) Order(stmt string, args ...interface{}) *Query {
return Q(c).Order(stmt, args...)
}
// Order will append an order clause to the query.
//
// q.Order("name desc")
func (q *Query) Order(stmt string, args ...interface{}) *Query {
if q.RawSQL.Fragment != "" {
log(logging.Warn, "Query is setup to use raw SQL")
return q
}
q.orderClauses = append(q.orderClauses, clause{stmt, args})
return q
}
// Limit will create a query and add a limit clause to it.
//
// c.Limit(10)
func (c *Connection) Limit(limit int) *Query {
return Q(c).Limit(limit)
}
// Limit will add a limit clause to the query.
//
// q.Limit(10)
func (q *Query) Limit(limit int) *Query {
q.limitResults = limit
return q
}
// Preload activates preload eager Mode automatically.
func (c *Connection) EagerPreload(fields ...string) *Query {
return Q(c).EagerPreload(fields...)
}
// Preload activates preload eager Mode automatically.
func (q *Query) EagerPreload(fields ...string) *Query {
q.Eager(fields...)
q.eagerMode = EagerPreload
return q
}
// Q will create a new "empty" query from the current connection.
func Q(c *Connection) *Query {
return &Query{
RawSQL: &clause{},
Connection: c,
eager: c.eager,
eagerFields: c.eagerFields,
eagerMode: eagerModeNil,
Operation: Select,
}
}
// ToSQL will generate SQL and the appropriate arguments for that SQL
// from the `Model` passed in.
func (q Query) ToSQL(model *Model, addColumns ...string) (string, []interface{}) {
sb := q.toSQLBuilder(model, addColumns...)
// nil model is allowed only when if RawSQL is provided.
if model == nil && (q.RawSQL == nil || q.RawSQL.Fragment == "") {
return "", nil
}
return sb.String(), sb.Args()
}
// ToSQLBuilder returns a new `SQLBuilder` that can be used to generate SQL,
// get arguments, and more.
func (q Query) toSQLBuilder(model *Model, addColumns ...string) *sqlBuilder {
if len(q.addColumns) != 0 {
addColumns = q.addColumns
}
return newSQLBuilder(q, model, addColumns...)
}