rotulus 2.0.0 → 2.1.1
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/CHANGELOG.md +7 -1
- data/README.md +18 -17
- data/lib/rotulus/version.rb +1 -1
- metadata +7 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 1111980eccb3c7fd8e75206ad2f51f90035a0c82636772704a4c747ba0287b23
|
4
|
+
data.tar.gz: 4305e2d6522ea570707f6f100151f695265d8e09020d7609aded7bb149af4f9e
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 0d5138726de48dfe807c4d1da2f11072aabeecba892162cfd415d94e2844a481a4134f47d32632fe6e322b76b479615ee04b10d661332698efc28ec2ba02d38e
|
7
|
+
data.tar.gz: 31c5d0f13bfbe50bd93f618b1ebafc07a846ec4e09ccb85c9ee61e3a18d31dd2a4cd1c2dedaa63d78071363cd291764510521d5a9f02f4e4997bcaa0e32fef30
|
data/CHANGELOG.md
CHANGED
@@ -18,4 +18,10 @@
|
|
18
18
|
- Make error class names consistent.
|
19
19
|
|
20
20
|
## 2.0.0
|
21
|
-
- Use multi_json instead of locking clients into using Oj gem.
|
21
|
+
- Use multi_json instead of locking clients into using Oj gem.
|
22
|
+
|
23
|
+
## 2.1.0
|
24
|
+
- Rails 7.1 support.
|
25
|
+
|
26
|
+
## 2.1.1
|
27
|
+
- Rails 7.2 and 8.0 support.
|
data/README.md
CHANGED
@@ -8,9 +8,10 @@ Cursor-based pagination is an alternative to OFFSET-based pagination that provid
|
|
8
8
|
|
9
9
|
Some advantages of this approach are:
|
10
10
|
|
11
|
-
* Reduces inaccuracies such as duplicate
|
11
|
+
* Reduces inaccuracies such as duplicate or skipped records as records are being manipulated.
|
12
12
|
* Can significantly improve performance(with proper DB indexing on ordered columns) especially as you move forward on large datasets.
|
13
13
|
|
14
|
+
|
14
15
|
**TL;DR** See [ sample usage for Rails here ](#rails-usage).
|
15
16
|
|
16
17
|
## Features
|
@@ -98,7 +99,7 @@ page = Rotulus::Page.new(users)
|
|
98
99
|
|
99
100
|
page = Rotulus::Page.new(users, order: { first_name: :asc, last_name: :desc }, limit: 3)
|
100
101
|
```
|
101
|
-
|
102
|
+
The gem will automatically add the table's PK(`users.id`) in the `ORDER BY` as the tie-breaker column to ensure stable sorting and pagination.
|
102
103
|
|
103
104
|
|
104
105
|
#### Access the page records
|
@@ -204,7 +205,7 @@ puts page.as_table
|
|
204
205
|
|
205
206
|
### Advanced Usage
|
206
207
|
#### Expanded order definition
|
207
|
-
Instead of just specifying the column sorting such as ```{ first_name: :asc }``` in the :order param, one can use the expanded order config in `Hash` format for more sorting options:
|
208
|
+
Instead of just specifying the column sorting such as ```{ first_name: :asc }``` in the :order param, one can use the expanded order config in `Hash` format for more sorting options that would help the library to generate the optimal query:
|
208
209
|
|
209
210
|
| Column Configuration | Description |
|
210
211
|
| ----------- | ----------- |
|
@@ -339,14 +340,14 @@ private
|
|
339
340
|
|
340
341
|
# Allow clients to sort by first_name, last_name, and/or email.
|
341
342
|
# example sort values:
|
342
|
-
# a. params[:sort] = +
|
343
|
+
# a. params[:sort] = +last_name,-email
|
343
344
|
# b. params[:sort] = -first_name
|
344
345
|
def index_order
|
345
346
|
SortParam.define do
|
346
|
-
field '
|
347
|
-
field '
|
348
|
-
field '
|
349
|
-
end.load!(params[:sort].presence || '
|
347
|
+
field 'first_name'
|
348
|
+
field 'last_name', nulls: :last, nullable: true
|
349
|
+
field 'email', distinct: true
|
350
|
+
end.load!(params[:sort].presence || 'first_name')
|
350
351
|
end
|
351
352
|
```
|
352
353
|
|
@@ -360,19 +361,19 @@ end
|
|
360
361
|
| `Rotulus::CursorError` | Generic error for cursor related validations |
|
361
362
|
| `Rotulus::InvalidColumn` | Column provided in the :order param can't be found. |
|
362
363
|
| `Rotulus::MissingTiebreaker` | There is no non-nullable and distinct column in the configured order definition. |
|
363
|
-
| `Rotulus::ConfigurationError` | Generic error for missing/invalid
|
364
|
-
| `Rotulus::OrderChanged` |
|
365
|
-
| `Rotulus::QueryChanged` |
|
364
|
+
| `Rotulus::ConfigurationError` | Generic error for missing/invalid configuration. |
|
365
|
+
| `Rotulus::OrderChanged` | Raised when passing a token to `Page#at` or `Page#at!` methods of a page instance, and the token was generated from a page instance with a different `:order` definition. Can be enabled by setting the `restrict_order_change` to true. |
|
366
|
+
| `Rotulus::QueryChanged` | Raised when passing a token to `Page#at` or `Page#at!` methods of a page instance, and the token was generated from a page instance with a different `:ar_relation` filter/query. Can be enabled by setting the `restrict_query_change` to true. |
|
366
367
|
|
367
368
|
## How it works
|
368
|
-
Cursor-based pagination uses a reference
|
369
|
+
Cursor-based pagination uses a reference record to fetch the relative set of previous or next records. This gem takes care of the SQL query and cursor generation needed for the pagination. To ensure that the pagination results are stable, it requires that:
|
369
370
|
|
370
|
-
* Records are sorted (`ORDER BY`).
|
371
|
+
* Records are sorted (`ORDER BY`) by columns.
|
371
372
|
* In case multiple records with the same column value(s) exists in the result, a unique non-nullable column is needed as tie-breaker. Usually, the table PK suffices for this but for complex queries(e.g. with table joins and with nullable columns, etc.), combining and using multiple columns that would uniquely identify the row in the result is needed.
|
372
373
|
* Columns used in `ORDER BY` would need to be indexed as they will be used in filtering.
|
373
374
|
|
374
375
|
|
375
|
-
#### Sample SQL-generated snippets
|
376
|
+
#### Sample SQL-generated snippets to fetch the next set of records
|
376
377
|
|
377
378
|
##### Example 1: With order by `id` only
|
378
379
|
###### Ruby
|
@@ -459,10 +460,10 @@ WHERE users.first_name >= 'Jane' AND (
|
|
459
460
|
```
|
460
461
|
|
461
462
|
#### Custom Token Format
|
462
|
-
By default, the cursor is encoded as a Base64 token. To customize how the cursor is encoded and decoded, you may just
|
463
|
+
By default, the cursor is encoded as a Base64 token. To customize how the cursor is encoded and decoded, you may just need to subclass `Rotulus::Cursor` with the `.decode` and `.encode` methods implemented.
|
463
464
|
|
464
465
|
##### Example:
|
465
|
-
The implementation below would generate tokens in UUID format where the actual cursor data is stored in memory:
|
466
|
+
The implementation below would generate tokens in UUID format where the actual cursor data is stored in memory(in production, you would use a distributed data store such as Redis):
|
466
467
|
|
467
468
|
```ruby
|
468
469
|
class MyCustomCursor < Rotulus::Cursor
|
@@ -526,7 +527,7 @@ end
|
|
526
527
|
| Environment Variable | Values | Example |
|
527
528
|
| ----------- | ----------- |----------- |
|
528
529
|
| `DB_ADAPTER` | **Default: :sqlite**. `sqlite`,`mysql2`, or `postgresql` | ```DB_ADAPTER=postgresql bundle exec rspec```<br/><br/> ```DB_ADAPTER=postgresql ./bin/console``` |
|
529
|
-
| `RAILS_VERSION` | **Default:
|
530
|
+
| `RAILS_VERSION` | **Default: 8-0** <br/><br/> `4-2`,`5-0`,`5-1`,`5-2`,`6-0`,`6-1`,`7-0`, `7-1`, `7-2`, `8-0` |```RAILS_VERSION=5-2 ./bin/setup```<br/><br/>```RAILS_VERSION=7-2 bundle exec rspec```<br/><br/> ```RAILS_VERSION=7-2 ./bin/console```|
|
530
531
|
|
531
532
|
|
532
533
|
<br/><br/>
|
data/lib/rotulus/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: rotulus
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.1.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Uy Jayson B
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2024-11-24 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: activerecord
|
@@ -19,7 +19,7 @@ dependencies:
|
|
19
19
|
version: '4.2'
|
20
20
|
- - "<"
|
21
21
|
- !ruby/object:Gem::Version
|
22
|
-
version: '
|
22
|
+
version: '8.1'
|
23
23
|
type: :runtime
|
24
24
|
prerelease: false
|
25
25
|
version_requirements: !ruby/object:Gem::Requirement
|
@@ -29,7 +29,7 @@ dependencies:
|
|
29
29
|
version: '4.2'
|
30
30
|
- - "<"
|
31
31
|
- !ruby/object:Gem::Version
|
32
|
-
version: '
|
32
|
+
version: '8.1'
|
33
33
|
- !ruby/object:Gem::Dependency
|
34
34
|
name: activesupport
|
35
35
|
requirement: !ruby/object:Gem::Requirement
|
@@ -39,7 +39,7 @@ dependencies:
|
|
39
39
|
version: '4.2'
|
40
40
|
- - "<"
|
41
41
|
- !ruby/object:Gem::Version
|
42
|
-
version: '
|
42
|
+
version: '8.1'
|
43
43
|
type: :runtime
|
44
44
|
prerelease: false
|
45
45
|
version_requirements: !ruby/object:Gem::Requirement
|
@@ -49,7 +49,7 @@ dependencies:
|
|
49
49
|
version: '4.2'
|
50
50
|
- - "<"
|
51
51
|
- !ruby/object:Gem::Version
|
52
|
-
version: '
|
52
|
+
version: '8.1'
|
53
53
|
- !ruby/object:Gem::Dependency
|
54
54
|
name: multi_json
|
55
55
|
requirement: !ruby/object:Gem::Requirement
|
@@ -113,7 +113,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
113
113
|
- !ruby/object:Gem::Version
|
114
114
|
version: '0'
|
115
115
|
requirements: []
|
116
|
-
rubygems_version: 3.
|
116
|
+
rubygems_version: 3.4.6
|
117
117
|
signing_key:
|
118
118
|
specification_version: 4
|
119
119
|
summary: Cursor-based Rails/ActiveRecord pagination with multiple column sort and
|