quo 1.0.0.alpha1 → 1.0.0.beta1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.standard.yml +1 -1
- data/Appraisals +4 -0
- data/Gemfile +1 -1
- data/LICENSE.txt +1 -1
- data/README.md +222 -233
- data/gemfiles/rails_8.0.gemfile +15 -0
- data/lib/quo/composed_query.rb +154 -44
- data/lib/quo/query.rb +0 -1
- data/lib/quo/relation_backed_query.rb +51 -78
- data/lib/quo/relation_backed_query_specification.rb +154 -0
- data/lib/quo/version.rb +1 -1
- data/lib/quo.rb +1 -0
- data/sig/generated/quo/composed_query.rbs +45 -16
- data/sig/generated/quo/relation_backed_query.rbs +21 -44
- data/sig/generated/quo/relation_backed_query_specification.rbs +94 -0
- data/sig/generated/quo/testing/collection_backed_fake.rbs +13 -0
- data/sig/generated/quo/testing/relation_backed_fake.rbs +23 -0
- data/sig/literal.rbs +7 -0
- metadata +15 -12
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: f72639830288de3776524bb1003fc88aeeee4d1adf83abeb44dd42e48caa0bfb
|
4
|
+
data.tar.gz: f72435e5e5ef3832bcf993362f0b56252bc17edaaedb197455308195c71f6516
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 64401d860bd1682d6b3d46d06f76eb644c9b3a3b1ad4d945b6d52b778d5deafe386f4588f9a7f381a8c484790711a3e0727f3d84f45469e8d53a787410741741
|
7
|
+
data.tar.gz: 5c148a23cc7ce4006fcac7c5cb9f6c118cbf08b35ea7b623743f48b3d36ae457d5902932a90669c5763e98db5df42b7ec31a470453752473821bc3e65ef7815d
|
data/.standard.yml
CHANGED
data/Appraisals
CHANGED
data/Gemfile
CHANGED
data/LICENSE.txt
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
The MIT License (MIT)
|
2
2
|
|
3
|
-
Copyright (c) 2022-
|
3
|
+
Copyright (c) 2022-2025 Stephen Ierodiaconou
|
4
4
|
|
5
5
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
6
|
of this software and associated documentation files (the "Software"), to deal
|
data/README.md
CHANGED
@@ -1,335 +1,324 @@
|
|
1
|
-
#
|
1
|
+
# Quo: Query Objects for ActiveRecord
|
2
2
|
|
3
|
-
|
3
|
+
Quo helps you organize database queries into reusable, composable, and testable objects.
|
4
4
|
|
5
|
-
|
6
|
-
interface.
|
5
|
+
## Core Features
|
7
6
|
|
8
|
-
|
9
|
-
|
7
|
+
* Wrap around an underlying ActiveRecord relation or array-like collection
|
8
|
+
* Supports pagination for ActiveRecord-based queries and collections that respond to `[]`
|
9
|
+
* Support composition with the `+` (`compose`) method to merge multiple query objects
|
10
|
+
* Allow transforming results with the `transform` method
|
11
|
+
* Offer utility methods that operate on the underlying collection (eg `exists?`)
|
12
|
+
* Act as a callable with chainable methods like ActiveRecord
|
13
|
+
* Provide a clear separation between query definition and execution with enumerable `Results` objects
|
14
|
+
* Type-safe properties with optional default values
|
10
15
|
|
11
|
-
|
16
|
+
## Core Concepts
|
12
17
|
|
13
|
-
|
14
|
-
* optionally provides paging behaviour to ActiveRecord based queries
|
15
|
-
* provides a number of utility methods that operate on the underlying collection (eg `exists?`)
|
16
|
-
* provides a `+` (`compose`) method which merges two query object instances (see section below for details!)
|
17
|
-
* can specify a mapping or transform method to `transform` to perform on results
|
18
|
-
* acts as a callable which executes the underlying query with `.first`
|
19
|
-
* can return an `Enumerable` of results
|
18
|
+
Query objects encapsulate query logic in dedicated classes, making complex queries more manageable and reusable.
|
20
19
|
|
20
|
+
Quo provides two main components:
|
21
|
+
1. **Query Objects** - Define and configure queries
|
22
|
+
2. **Results Objects** - Execute queries and provide access to the results
|
21
23
|
|
22
|
-
|
24
|
+
## Creating Query Objects
|
23
25
|
|
24
|
-
|
26
|
+
### Relation-Backed Queries
|
25
27
|
|
26
|
-
|
28
|
+
For queries based on ActiveRecord relations:
|
27
29
|
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
30
|
+
```ruby
|
31
|
+
class RecentActiveUsers < Quo::RelationBackedQuery
|
32
|
+
# Define typed properties
|
33
|
+
prop :days_ago, Integer, default: -> { 30 }
|
34
|
+
|
35
|
+
def query
|
36
|
+
User
|
37
|
+
.where(active: true)
|
38
|
+
.where("created_at > ?", days_ago.days.ago)
|
39
|
+
end
|
40
|
+
end
|
39
41
|
|
40
|
-
|
42
|
+
# Create and use the query
|
43
|
+
query = RecentActiveUsers.new(days_ago: 7)
|
44
|
+
results = query.results
|
41
45
|
|
42
|
-
|
46
|
+
# Work with results
|
47
|
+
results.each { |user| puts user.email }
|
48
|
+
puts "Found #{results.count} users"
|
49
|
+
```
|
43
50
|
|
44
|
-
|
51
|
+
### Collection-Backed Queries
|
45
52
|
|
46
|
-
|
53
|
+
For queries based on any Enumerable collection:
|
47
54
|
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
55
|
+
```ruby
|
56
|
+
class CachedUsers < Quo::CollectionBackedQuery
|
57
|
+
prop :role, String
|
58
|
+
|
59
|
+
def collection
|
60
|
+
@cached_users ||= Rails.cache.fetch("all_users", expires_in: 1.hour) do
|
61
|
+
User.all.to_a
|
62
|
+
end.select { |user| user.role == role }
|
63
|
+
end
|
64
|
+
end
|
53
65
|
|
54
|
-
|
55
|
-
|
66
|
+
# Use the query
|
67
|
+
admins = CachedUsers.new(role: "admin").results
|
68
|
+
```
|
56
69
|
|
57
|
-
##
|
70
|
+
## Quick Queries with Wrap
|
58
71
|
|
59
|
-
|
72
|
+
Create query objects without subclassing:
|
60
73
|
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
* left_outer_joins
|
66
|
-
* preload
|
67
|
-
* joins
|
74
|
+
```ruby
|
75
|
+
# Relation-backed
|
76
|
+
users_query = Quo::RelationBackedQuery.wrap(User.active).new
|
77
|
+
active_users = users_query.results
|
68
78
|
|
69
|
-
|
79
|
+
# Collection-backed
|
80
|
+
items_query = Quo::CollectionBackedQuery.wrap([1, 2, 3]).new
|
81
|
+
items = items_query.results
|
82
|
+
```
|
70
83
|
|
71
|
-
##
|
84
|
+
## Type-Safe Properties
|
72
85
|
|
73
|
-
Quo
|
74
|
-
and so under the hood `Quo::Query` uses that when composing relations. However since Queries can also abstract over
|
75
|
-
array-like collections (ie enumerable and define a `+` method) compose also handles concating them together.
|
86
|
+
Quo uses the `Literal` gem for typed properties:
|
76
87
|
|
77
|
-
|
88
|
+
```ruby
|
89
|
+
class UsersByState < Quo::RelationBackedQuery
|
90
|
+
prop :state, String
|
91
|
+
prop :minimum_age, Integer, default: -> { 18 }
|
92
|
+
prop :active_only, Boolean, default: -> { true }
|
78
93
|
|
79
|
-
|
80
|
-
|
81
|
-
|
94
|
+
def query
|
95
|
+
scope = User.where(state: state)
|
96
|
+
scope = scope.where("age >= ?", minimum_age) if minimum_age.present?
|
97
|
+
scope = scope.where(active: true) if active_only
|
98
|
+
scope
|
99
|
+
end
|
100
|
+
end
|
82
101
|
|
83
|
-
|
84
|
-
|
85
|
-
them correctly with the appropriate joins. Note with the alias you cant neatly specify optional parameters for joins
|
86
|
-
on relations.
|
102
|
+
query = UsersByState.new(state: "California", minimum_age: 21)
|
103
|
+
```
|
87
104
|
|
88
|
-
|
105
|
+
## Fluent API for Building Queries
|
89
106
|
|
90
|
-
|
107
|
+
```ruby
|
108
|
+
query = UsersByState.new(state: "California")
|
109
|
+
.order(created_at: :desc)
|
110
|
+
.includes(:profile)
|
111
|
+
.limit(10)
|
112
|
+
.where(verified: true)
|
91
113
|
|
92
|
-
|
93
|
-
|
94
|
-
3. compose two query objects which return array-likes
|
114
|
+
users = query.results
|
115
|
+
```
|
95
116
|
|
96
|
-
|
97
|
-
|
117
|
+
Available methods include:
|
118
|
+
* `where`
|
119
|
+
* `order`
|
120
|
+
* `limit`
|
121
|
+
* `includes`
|
122
|
+
* `preload`
|
123
|
+
* `left_outer_joins`
|
124
|
+
* `joins`
|
125
|
+
* `group`
|
98
126
|
|
99
|
-
|
100
|
-
to the array-like with `+`
|
127
|
+
Each method returns a new query instance without modifying the original.
|
101
128
|
|
102
|
-
|
129
|
+
## Pagination
|
103
130
|
|
104
|
-
|
131
|
+
```ruby
|
132
|
+
query = UsersByState.new(
|
133
|
+
state: "California",
|
134
|
+
page: 2,
|
135
|
+
page_size: 20
|
136
|
+
)
|
137
|
+
|
138
|
+
# Get paginated results
|
139
|
+
users = query.results
|
140
|
+
|
141
|
+
# Navigation
|
142
|
+
next_page = query.next_page_query
|
143
|
+
prev_page = query.previous_page_query
|
144
|
+
```
|
105
145
|
|
106
|
-
|
107
|
-
query object or and `ActiveRecord::Relation`. However `Quo::Query.compose(left, right)` also accepts
|
108
|
-
`ActiveRecord::Relation`s for left.
|
146
|
+
## Composing Queries
|
109
147
|
|
110
|
-
|
148
|
+
Combine multiple queries:
|
111
149
|
|
112
150
|
```ruby
|
113
|
-
class
|
151
|
+
class ActiveUsers < Quo::RelationBackedQuery
|
114
152
|
def query
|
115
|
-
|
116
|
-
.left_joins(:approval)
|
117
|
-
.where(approvals: {completed_at: nil})
|
153
|
+
User.where(active: true)
|
118
154
|
end
|
119
155
|
end
|
120
156
|
|
121
|
-
class
|
157
|
+
class PremiumUsers < Quo::RelationBackedQuery
|
122
158
|
def query
|
123
|
-
|
124
|
-
.joins(company: :address)
|
125
|
-
.where(addresses: {state: options[:state]})
|
159
|
+
User.where(subscription_tier: "premium")
|
126
160
|
end
|
127
161
|
end
|
128
162
|
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
# Compose
|
133
|
-
composed = query1 + query2 # or Quo::Query.compose(query1, query2) or query1.compose(query2)
|
134
|
-
composed.first
|
135
|
-
```
|
136
|
-
|
137
|
-
This effectively executes:
|
138
|
-
|
139
|
-
```ruby
|
140
|
-
Registration
|
141
|
-
.left_joins(:approval)
|
142
|
-
.joins(company: :address)
|
143
|
-
.where(approvals: {completed_at: nil})
|
144
|
-
.where(addresses: {state: options[:state]})
|
163
|
+
# Compose queries
|
164
|
+
active_premium = ActiveUsers.new + PremiumUsers.new
|
165
|
+
users = active_premium.results
|
145
166
|
```
|
146
167
|
|
147
|
-
|
148
|
-
|
168
|
+
You can compose queries using:
|
169
|
+
* `Quo::Query.compose(left, right)`
|
170
|
+
* `left.compose(right)`
|
171
|
+
* `left + right`
|
149
172
|
|
150
|
-
|
151
|
-
class RegistrationToBeApproved < Quo::RelationBackedQuery
|
152
|
-
def query
|
153
|
-
done = Registration.where(step: "complete")
|
154
|
-
approved = CompanyToBeApproved.new
|
155
|
-
# Here we use `.compose` utility method to wrap our Relation in a Query and
|
156
|
-
# then compose with the other Query
|
157
|
-
Quo::Query.compose(done, approved)
|
158
|
-
end
|
159
|
-
end
|
160
|
-
|
161
|
-
# A Relation can be composed directly to a Quo::Query
|
162
|
-
query = RegistrationToBeApproved.new + Registration.where(blocked: false)
|
163
|
-
```
|
164
|
-
|
165
|
-
Also you can use joins:
|
173
|
+
### Composing with Joins
|
166
174
|
|
167
175
|
```ruby
|
168
|
-
class
|
176
|
+
class ProductsQuery < Quo::RelationBackedQuery
|
169
177
|
def query
|
170
|
-
|
178
|
+
Product.where(active: true)
|
171
179
|
end
|
172
180
|
end
|
173
181
|
|
174
|
-
class
|
182
|
+
class CategoriesQuery < Quo::RelationBackedQuery
|
175
183
|
def query
|
176
|
-
Category.where(
|
184
|
+
Category.where(featured: true)
|
177
185
|
end
|
178
186
|
end
|
179
187
|
|
180
|
-
|
181
|
-
|
182
|
-
tags.compose(for_category, :category) # perform join on tag association `category`
|
188
|
+
# Compose with a join
|
189
|
+
products = ProductsQuery.new.compose(CategoriesQuery.new, joins: :category)
|
183
190
|
|
184
|
-
#
|
191
|
+
# Equivalent to:
|
192
|
+
# Product.joins(:category)
|
193
|
+
# .where(products: { active: true })
|
194
|
+
# .where(categories: { featured: true })
|
185
195
|
```
|
186
196
|
|
187
|
-
|
188
|
-
|
189
|
-
### Quo::ComposedQuery
|
190
|
-
|
191
|
-
The new instance of `Quo::ComposedQuery` from a compose process, retains references to the original entities that were
|
192
|
-
composed. These are then used to create a more useful output from `to_s`, so that it is easier to understand what the
|
193
|
-
merged query is actually made up of:
|
197
|
+
## Transforming Results
|
194
198
|
|
195
199
|
```ruby
|
196
|
-
|
197
|
-
|
198
|
-
|
200
|
+
query = UsersByState.new(state: "California")
|
201
|
+
.transform { |user| UserPresenter.new(user) }
|
202
|
+
|
203
|
+
# Results are automatically transformed
|
204
|
+
presenters = query.results.to_a # Array of UserPresenter objects
|
199
205
|
```
|
200
206
|
|
201
|
-
##
|
207
|
+
## Custom Association Preloading
|
202
208
|
|
203
|
-
|
209
|
+
```ruby
|
210
|
+
class UsersWithOrders < Quo::RelationBackedQuery
|
211
|
+
include Quo::Preloadable
|
212
|
+
|
213
|
+
def query
|
214
|
+
User.all
|
215
|
+
end
|
204
216
|
|
205
|
-
|
206
|
-
|
217
|
+
def preload_associations(collection)
|
218
|
+
# Custom preloading logic
|
219
|
+
ActiveRecord::Associations::Preloader.new(
|
220
|
+
records: collection,
|
221
|
+
associations: [:profile, :orders]
|
222
|
+
).call
|
223
|
+
|
224
|
+
collection
|
225
|
+
end
|
226
|
+
end
|
227
|
+
```
|
207
228
|
|
208
|
-
|
229
|
+
## Testing Helpers
|
209
230
|
|
210
|
-
|
211
|
-
by a collection (ie an enumerable such as an Array). This is useful for encapsulating data that doesn't come from an
|
212
|
-
ActiveRecord query or queries that execute immediately. Subclass this and override `collection` to return the data you
|
213
|
-
want to encapsulate.
|
231
|
+
### Minitest
|
214
232
|
|
215
233
|
```ruby
|
216
|
-
class
|
217
|
-
|
218
|
-
|
234
|
+
class UserQueryTest < ActiveSupport::TestCase
|
235
|
+
include Quo::Minitest::Helpers
|
236
|
+
|
237
|
+
test "filters users by state" do
|
238
|
+
users = [User.new(name: "Alice"), User.new(name: "Bob")]
|
239
|
+
|
240
|
+
fake_query(UsersByState, results: users) do
|
241
|
+
result = UsersByState.new(state: "California").results.to_a
|
242
|
+
assert_equal users, result
|
243
|
+
end
|
219
244
|
end
|
220
245
|
end
|
221
|
-
q = MyCollectionBackedQuery.new
|
222
|
-
q.collection? # is it a collection under the hood? Yes it is!
|
223
|
-
q.count # '3'
|
224
246
|
```
|
225
247
|
|
226
|
-
|
227
|
-
use `Quo::CollectionBackedQuery`:
|
248
|
+
### RSpec
|
228
249
|
|
229
250
|
```ruby
|
230
|
-
|
231
|
-
|
232
|
-
|
251
|
+
RSpec.describe UsersByState do
|
252
|
+
include Quo::RSpec::Helpers
|
253
|
+
|
254
|
+
it "filters users by state" do
|
255
|
+
users = [User.new(name: "Alice"), User.new(name: "Bob")]
|
256
|
+
|
257
|
+
fake_query(UsersByState, results: users) do
|
258
|
+
result = UsersByState.new(state: "California").results.to_a
|
259
|
+
expect(result).to eq(users)
|
260
|
+
end
|
261
|
+
end
|
262
|
+
end
|
233
263
|
```
|
234
264
|
|
235
|
-
|
236
|
-
actually just a page of the data and not the total count.
|
265
|
+
## Project Organization
|
237
266
|
|
238
|
-
|
267
|
+
Suggested directory structure:
|
239
268
|
|
240
|
-
```
|
241
|
-
|
269
|
+
```
|
270
|
+
app/
|
271
|
+
queries/
|
272
|
+
application_query.rb
|
273
|
+
users/
|
274
|
+
active_users_query.rb
|
275
|
+
by_state_query.rb
|
276
|
+
products/
|
277
|
+
featured_products_query.rb
|
242
278
|
```
|
243
279
|
|
244
|
-
|
245
|
-
results are returned from the other queries. An loaded or eager query will force all other queries to be eager loaded.
|
246
|
-
|
247
|
-
### Composition
|
248
|
-
|
249
|
-
Examples of composition of eager loaded queries
|
280
|
+
Base classes:
|
250
281
|
|
251
282
|
```ruby
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
end
|
283
|
+
# app/queries/application_query.rb
|
284
|
+
class ApplicationQuery < Quo::RelationBackedQuery
|
285
|
+
# Common functionality
|
256
286
|
end
|
257
287
|
|
258
|
-
|
259
|
-
|
260
|
-
#
|
261
|
-
|
262
|
-
# => #<Tag id: ...>
|
263
|
-
|
264
|
-
Quo::CollectionBackedQuery.new([3, 4]).compose(Quo::CollectionBackedQuery.new([1, 2])).last
|
265
|
-
# => 2
|
266
|
-
Quo::Query.compose([1, 2], [3, 4]).last
|
267
|
-
# => 4
|
288
|
+
# app/queries/application_collection_query.rb
|
289
|
+
class ApplicationCollectionQuery < Quo::CollectionBackedQuery
|
290
|
+
# Common functionality
|
291
|
+
end
|
268
292
|
```
|
269
293
|
|
270
|
-
##
|
271
|
-
|
272
|
-
Sometimes you want to specify a block to execute on each result for any method that returns results, such as `first`,
|
273
|
-
`last` and `each`.
|
294
|
+
## Installation
|
274
295
|
|
275
|
-
|
296
|
+
Add to your Gemfile:
|
276
297
|
|
277
298
|
```ruby
|
278
|
-
|
279
|
-
active: [true, false],
|
280
|
-
page: 1,
|
281
|
-
page_size: 30,
|
282
|
-
).transform { |tag| TagPresenter.new(tag) }
|
283
|
-
.first
|
284
|
-
# => #<TagPresenter ...>
|
299
|
+
gem "quo"
|
285
300
|
```
|
286
301
|
|
287
|
-
|
288
|
-
|
289
|
-
Tests for Query objects themselves should exercise the actual underlying query. But in other code stubbing the query
|
290
|
-
maybe desirable.
|
302
|
+
Then execute:
|
291
303
|
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
The `with` option is passed to the Query object on initialisation and used when setting up the method stub on the
|
296
|
-
query class.
|
304
|
+
```
|
305
|
+
$ bundle install
|
306
|
+
```
|
297
307
|
|
298
|
-
|
308
|
+
## Configuration
|
299
309
|
|
300
310
|
```ruby
|
301
|
-
|
302
|
-
|
311
|
+
# config/initializers/quo.rb
|
312
|
+
Quo.default_page_size = 25
|
313
|
+
Quo.max_page_size = 100
|
314
|
+
Quo.relation_backed_query_base_class = "ApplicationQuery"
|
315
|
+
Quo.collection_backed_query_base_class = "ApplicationCollectionQuery"
|
303
316
|
```
|
304
317
|
|
305
|
-
|
306
|
-
|
307
|
-
This returns an instance of CollectionBackedQuery, so will not work for cases were the actual type of the query instance is
|
308
|
-
important or where you are doing a composition of queries backed by relations!
|
309
|
-
|
310
|
-
If `compose` will be used then `Quo::Query.compose` needs to be stubbed. Something might be possible to make this
|
311
|
-
nicer in future.
|
312
|
-
|
313
|
-
## Other reading
|
314
|
-
|
315
|
-
See:
|
316
|
-
* [Includes vs preload vs eager_load](http://blog.scoutapp.com/articles/2017/01/24/activerecord-includes-vs-joins-vs-preload-vs-eager_load-when-and-where)
|
317
|
-
* [Objects on Rails](http://objectsonrails.com/#sec-14)
|
318
|
-
|
319
|
-
|
320
|
-
## Installation
|
321
|
-
|
322
|
-
Install the gem and add to the application's Gemfile by executing:
|
323
|
-
|
324
|
-
$ bundle add quo
|
325
|
-
|
326
|
-
If bundler is not being used to manage dependencies, install the gem by executing:
|
327
|
-
|
328
|
-
$ gem install quo
|
329
|
-
|
330
|
-
## Usage
|
318
|
+
## Requirements
|
331
319
|
|
332
|
-
|
320
|
+
- Ruby 3.1+
|
321
|
+
- Rails 7.0+
|
333
322
|
|
334
323
|
## Development
|
335
324
|
|
@@ -343,7 +332,7 @@ Bug reports and pull requests are welcome on GitHub at https://github.com/steveg
|
|
343
332
|
|
344
333
|
## Inspired by `rectify`
|
345
334
|
|
346
|
-
|
335
|
+
This implementation is inspired by the `Rectify` gem: https://github.com/andypike/rectify. Thanks to Andy Pike for the inspiration.
|
347
336
|
|
348
337
|
## License
|
349
338
|
|