From 70190bb309eddf0e280f7389cd1d6280a77b98a3 Mon Sep 17 00:00:00 2001 From: Domizio Demichelis Date: Sat, 14 Dec 2024 17:31:51 +0700 Subject: [PATCH] Shorten the cutoff string by using an array in place of a hash --- docs/api/keyset.md | 30 ++++---- docs/api/keyset_for_ui.md | 8 +-- docs/extras/keyset_for_ui.md | 18 ++--- docs/how-to.md | 9 +-- gem/lib/pagy/keyset.rb | 10 ++- test/pagy/extras/headers_test.rb | 5 +- test/pagy/extras/headers_test.rb.yaml | 100 +++++++++++++------------- test/pagy/extras/jsonapi_test.rb | 2 +- test/pagy/extras/jsonapi_test.rb.yaml | 28 ++++---- test/pagy/extras/keyset_test.rb | 16 ++--- test/pagy/keyset_for_ui_test.rb | 26 +++---- test/pagy/keyset_test.rb | 70 ++++++++---------- 12 files changed, 149 insertions(+), 173 deletions(-) diff --git a/docs/api/keyset.md b/docs/api/keyset.md index 381cac9ed..acdad8a11 100644 --- a/docs/api/keyset.md +++ b/docs/api/keyset.md @@ -52,7 +52,7 @@ If you want the best of the two worlds, check out the [keyset_for_ui extra](/doc | `set` | The `uniquely ordered` `ActiveRecord::Relation` or `Sequel::Dataset` collection to paginate. | | `keyset` | The hash of column/direction pairs. Pagy extracts it from the order of the `set`. | | `keyset attributes` | The hash of keyset-column/record-value pairs of a record. | -| `cutoff` | A point in the `set` that separates the records of two contiguous `page`s. It's the encoded string of the `keyset attributes` of the last record of a `page`. | +| `cutoff` | A point in the `set` where a `page` ended. Its value is a `Base64` encoded URL-safe string. | | `page` | The current `page`, i.e. the page of records beginning after the `cutoff` of the previous page. Also the `:page` variable, which is set to the `cutoff` of the previous page | | `next` | The next `page`, i.e. the page of records beginning after the `cutoff`. Also the `cutoff` value retured by the `next` method. | @@ -164,29 +164,24 @@ If you need a specific order: #### Understanding the Cutoffs -A `cutoff` defines a point in the `set`, right AFTER the last record of a `page`. +A `cutoff` defines a point in the `set` where a `page` ended. All the records AFTER that point are or will be part of the `next` page. Let's consider an example of a simple `set`. In order to avoid confusion with numeric ids and number of records, let's assume that -it that has an `id` column that is actually a unique alphanumeric code, and its order is: -`order(:id)`. +it has an `id` column populated by unique alphanumeric codes, and its order is: `order(:id)`. -Assuming a LIMIT of 6, the first page will include the first 6 records in the set: no `cutoff` required so -far... +Assuming a LIMIT of 6, the first page will include the first 6 records in the set: no `cutoff` required so far... ``` | page | not yet paginated | beginning ->|. . . . . .|. . . . . . . . . . . . . . . . . . . . . . . . . . .|<- end of set ``` -After we pull the 6 records, we read the `id` of the last one in the page, which is `F`. So our `cutoff` can be defined like: _" -the point after the value `F` in the `id` column"_. +After we pull the first 6 records from the beginning of the `set`, we read the `id` of the last one, which is `F`. So our `cutoff` can be defined like: _"the point up to the value `F` in the `id` column"_. -Notice that this is not like saying _"after the record `F`"_. It's important to understand that a `cutoff` refers just to a value -in a column (or multiple column in case of muti-columns keysets) after which there will be the next page. - -Indeed, that very record could be deleted right after we read it, and our `cutoff` will still be the valid start for the next page -of 6 records after the `cutoff-F`... +Notice that this is not like saying _"up to the record `F`"_. It's important to understand that a `cutoff` refers just to a value +in a column (or a combination of multiple column, in case of muti-columns keysets). +Indeed, that very record could be deleted right after we read it, and our `cutoff` will still be the valid reference that _"we paginated the `set`, up to the "F" value"_... ``` | page | page | not yet paginated | beginning ->|. . . . . F]. . . . . .|. . . . . . . . . . . . . . . . . . . . .|<- end of set @@ -194,8 +189,7 @@ beginning ->|. . . . . F]. . . . . .|. . . . . . . . . . . . . . . . . . . . .|< cutoff-F ``` -Again, after we pull the next 6 records, we read the `id` of the last one in the page, which is `L`: so we have our new -`cutoff-L`, which is the start of the next `page`... +For getting the `next` page of records - this time - we pull the `next` 6 records AFTER the `cutoff-F`. Again, we read the `id` of the last one, which is `L`: so we have our new `cutoff-L`, which is the end of the current `page`, and the `next` will go AFTER it... ``` | page | page | page | not yet paginated | @@ -204,7 +198,7 @@ beginning ->|. . . . . F]. . . . . L]. . . . . .|. . . . . . . . . . . . . . .|< cutoff-F cutoff-L ``` -Pagy encodes the values of the `cutoffs` in a `Base64` URL-safe string that is used in the request as a param. +Pagy encodes the values of the `cutoffs` in a `Base64` URL-safe string that is sent as a param in the `request`. ## ORMs @@ -254,7 +248,7 @@ Default `nil`. ==- `:jsonify_keyset_attributes` -A lambda to override the generic json encoding of the `keyset` attributes. Use it when the generic `to_json` method would lose +A lambda to override the generic json encoding of the `keyset` attributes. It receives the keyset attributes to jsonify, and it should return a JSON string of the `attributes.values` array. Use it when the generic `to_json` method would lose some information when decoded. For example: `Time` objects may lose or round the fractional seconds through the encoding/decoding cycle, causing the ordering to @@ -266,7 +260,7 @@ etc.). Here is what you can do: jsonify_keyset_attributes = lambda do |attributes| # Convert it to a string matching the stored value/format in SQLite DB attributes[:created_at] = attributes[:created_at].strftime('%F %T.%6N') - attributes.to_json + attributes.values.to_json # remember to return an array of the values only end Pagy::Keyset(set, jsonify_keyset_attributes:) diff --git a/docs/api/keyset_for_ui.md b/docs/api/keyset_for_ui.md index 860a929d3..1dc99661f 100644 --- a/docs/api/keyset_for_ui.md +++ b/docs/api/keyset_for_ui.md @@ -33,11 +33,11 @@ You should also familiarize with the [Pagy::Keyset](keyset.md) class. This section integrates the [Keyset Glossary](keyset_for_ui.md#glossary) -| Term | Description | -|----------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| Term | Description | +|----------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | `keyset pagination for UI` | The pagy exclusive technique to use `keyset pagination` with numeric pages, supporting `pagy_*navs` and the other Frontend helpers.
The best technique for performance AND functionality! | -| `page` | The current page **number** | -| `cutoffs` | The `cutoff`s of the pagination known so far, used to keep track of the visited pages. | +| `page` | The current page **number** | +| `cutoffs` | The `cutoff`s of the known pagination state, used to keep track of the visited pages during the navigation. | ## How Pagy Keyset For UI works diff --git a/docs/extras/keyset_for_ui.md b/docs/extras/keyset_for_ui.md index 6e6dd9d9d..6d024e63f 100644 --- a/docs/extras/keyset_for_ui.md +++ b/docs/extras/keyset_for_ui.md @@ -17,11 +17,10 @@ and the other Frontend helpers. ## Overview -This extra manages the cache used by the `Pagy::KeysetForUI` instance, allowing easy customization and integration with your -app. +This extra manages the cache used by the `Pagy::KeysetForUI` instance, allowing easy customization and integration with your app. -It also adds a `pagy_keyset_for_ui` constructor method that can be used in your controllers, and provides the automatic setting -of the variables from the request `params`. +It also adds a `pagy_keyset_for_ui` constructor method that can be used in your controllers, and provides the automatic setting of +the variables from the request `params`. Please refer to the following resource: @@ -53,8 +52,8 @@ def pagy_cache_new_key = my_custom_cache.generate_key ## Understanding the cache -This extra uses the `session` object as the cache for the `cutoffs` (not the records!) by default, because it's simple and works in any app, at least for -prototyping. +This extra uses the `session` object as the cache for the `cutoffs` (not for the records!) by default, because it's simple and +works in any app, at least for prototyping. Notice that the `cutoffs` array can potentially grow big if you don't use `:max_pages`, especially if your `keyset` contains multiple ordered columns and more if their size is big. You must be aware of it. @@ -69,8 +68,8 @@ session as the cache (e.g. `ActiveRecord::SessionStore`). !!!warning -Besides writing and reading from it, Pagy does not expire nor handle the cache in any way. Your app should manage it (e.g. like it does -with the `session` object). +Besides writing and reading from it, Pagy does not expire nor handle the cache in any way. Your app should manage it (e.g. like it +does with the `session` object). !!! This extra uses only 3 simple methods to handle the cache: @@ -86,7 +85,8 @@ handling other aspects of it (e.g. expiration, etc.) We are considering implementing a client-side cache using the Browser's `sessionStorage`. -It might considerably simplify the handling of the cache, but it will require some time to design it properly, so please, hang tight and cheer for us! +It might considerably simplify the handling of the cache, but it will require some time to design it properly, so please, hang +tight and cheer for us! !!! ## Variables diff --git a/docs/how-to.md b/docs/how-to.md index 9313ccae1..5d952f44b 100644 --- a/docs/how-to.md +++ b/docs/how-to.md @@ -10,14 +10,7 @@ This page contains the practical tips and examples to get the job done with Pagy You can also [Ask any question to the Pagy trained AI](https://gurubase.io/g/pagy) for instant answers not covered in this page. -## Choose between Offset, Countless, Keyset and Keyset Numeric pagination - -| Type | Tech | Queries | UI Support | -|---------------| ------ |---------|------------| -| Regular | Offset | 2 slow | Complete | -| Countless | Offset | 1 slow | Partial | -| Keyset | Keyset | 1 fast | None | -| Keyset Numeric | Keyset | 1 fast | Partial | +## Choose the right pagination type [AI-powered answer](https://gurubase.io/g/pagy/choose-between-pagy-offset-countless-keyset) diff --git a/gem/lib/pagy/keyset.rb b/gem/lib/pagy/keyset.rb index 8bc3dfe69..94dcfb623 100644 --- a/gem/lib/pagy/keyset.rb +++ b/gem/lib/pagy/keyset.rb @@ -105,10 +105,8 @@ def after_cutoff_sql(prefix = nil) # Decode a cutoff, check its consistency and returns the cutoff args def cutoff_to_args(cutoff) - args = JSON.parse(B64.urlsafe_decode(cutoff)).transform_keys(&:to_sym) - raise InternalError, 'cutoff and keyset are not consistent' \ - unless args.keys == @keyset.keys - + values = JSON.parse(B64.urlsafe_decode(cutoff)) + args = @keyset.keys.zip(values).to_h typecast_args(args) end @@ -123,8 +121,8 @@ def default # Derive the cutoff from the last record def derive_cutoff - hash = keyset_attributes_from(@records.last) - json = @vars[:jsonify_keyset_attributes]&.(hash) || hash.to_json + attr = keyset_attributes_from(@records.last) + json = @vars[:jsonify_keyset_attributes]&.(attr) || attr.values.to_json B64.urlsafe_encode(json) end diff --git a/test/pagy/extras/headers_test.rb b/test/pagy/extras/headers_test.rb index 5ddf4d825..147642f34 100644 --- a/test/pagy/extras/headers_test.rb +++ b/test/pagy/extras/headers_test.rb @@ -79,10 +79,13 @@ it 'returns custom headers hash' do pagy, _records = app.send(:pagy_keyset, Pet.order(:id), - page: 'eyJpZCI6MjB9', + page: 'WzIwXQ', headers: { limit: 'Per-Page', page: 'Page', count: 'Total', pages: false }) _(app.send(:pagy_headers, pagy)).must_rematch :headers end + # -{"link"=>"; rel=\"first\", ; rel=\"next\"", "Page"=>"eyJpZCI6MjB9", "Per-Page"=>"20"} + # +{"link"=>"; rel=\"first\"", "Page"=>"eyJpZCI6MjB9", "Per-Page"=>"20"} + it 'omit next on last page' do pagy, _records = app.send(:pagy_keyset, Pet.order(:id), limit: 50) _(app.send(:pagy_headers, pagy)).must_rematch :headers diff --git a/test/pagy/extras/headers_test.rb.yaml b/test/pagy/extras/headers_test.rb.yaml index 6bcbf00b5..cd583904e 100644 --- a/test/pagy/extras/headers_test.rb.yaml +++ b/test/pagy/extras/headers_test.rb.yaml @@ -1,47 +1,10 @@ --- -pagy/extras/headers___pagy_headers_merge_test_0001_returns_the_full_headers_hash: - :response: !ruby/hash:Rack::Headers - link: ; rel="first", ; - rel="prev", ; rel="next", ; - rel="last" - current-page: '3' - page-items: '20' - total-pages: '50' - total-count: '1000' -pagy/extras/headers___pagy_headers_merge_with_Calendar_test_0001_returns_the_full_headers_hash: - :response: !ruby/hash:Rack::Headers - link: ; rel="first", ; - rel="prev", ; rel="next", ; - rel="last" - current-page: '3' - page-items: '20' - total-pages: '26' - total-count: '505' -pagy/extras/headers___pagy_headers_with_Keyset_test_0003_omit_next_on_last_page: - :headers: - link: ; rel="first" - page-items: '50' -pagy/extras/headers___pagy_headers_with_Keyset_test_0002_returns_custom_headers_hash: - :headers: - link: ; rel="first", ; - rel="next" - Page: eyJpZCI6MjB9 - Per-Page: '20' -pagy/extras/headers___pagy_headers_with_Keyset_test_0001_returns_the_full_headers_hash: - :headers: - link: ; rel="first", ; - rel="next" - page-items: '20' -pagy/extras/headers___pagy_headers_test_0004_returns_the_countless_headers_hash: - :headers: - link: ; rel="first", ; - rel="next" - current-page: '1' - page-items: '20' -pagy/extras/headers___pagy_headers_test_0003_returns_custom_headers_hash: +pagy/extras/headers___pagy_headers_test_0002_returns_custom_headers_hash: :headers: link: ; rel="first", ; rel="next", ; rel="last" + Per-Page: '20' + Total: '1000' pagy/extras/headers___pagy_headers_test_0006_omit_next_on_last_page: :headers: link: ; rel="first", ; @@ -50,6 +13,12 @@ pagy/extras/headers___pagy_headers_test_0006_omit_next_on_last_page: page-items: '20' total-pages: '50' total-count: '1000' +pagy/extras/headers___pagy_headers_test_0004_returns_the_countless_headers_hash: + :headers: + link: ; rel="first", ; + rel="next" + current-page: '1' + page-items: '20' pagy/extras/headers___pagy_headers_test_0001_returns_the_full_headers_hash: :headers: link: ; rel="first", ; @@ -58,12 +27,10 @@ pagy/extras/headers___pagy_headers_test_0001_returns_the_full_headers_hash: page-items: '20' total-pages: '50' total-count: '1000' -pagy/extras/headers___pagy_headers_test_0002_returns_custom_headers_hash: +pagy/extras/headers___pagy_headers_test_0003_returns_custom_headers_hash: :headers: link: ; rel="first", ; rel="next", ; rel="last" - Per-Page: '20' - Total: '1000' pagy/extras/headers___pagy_headers_test_0005_omit_prev_on_first_page: :headers: link: ; rel="first", ; @@ -72,16 +39,45 @@ pagy/extras/headers___pagy_headers_test_0005_omit_prev_on_first_page: page-items: '20' total-pages: '50' total-count: '1000' -pagy/extras/headers___pagy_headers_with_Calendar_test_0004_returns_the_countless_headers_hash: +pagy/extras/headers___pagy_headers_merge_test_0001_returns_the_full_headers_hash: + :response: !ruby/hash:Rack::Headers + link: ; rel="first", ; + rel="prev", ; rel="next", ; + rel="last" + current-page: '3' + page-items: '20' + total-pages: '50' + total-count: '1000' +pagy/extras/headers___pagy_headers_with_Keyset_test_0002_returns_custom_headers_hash: :headers: - link: ; rel="first", ; + link: ; rel="first", ; + rel="next" + Page: WzIwXQ + Per-Page: '20' +pagy/extras/headers___pagy_headers_with_Keyset_test_0003_omit_next_on_last_page: + :headers: + link: ; rel="first" + page-items: '50' +pagy/extras/headers___pagy_headers_with_Keyset_test_0001_returns_the_full_headers_hash: + :headers: + link: ; rel="first", ; rel="next" - current-page: '1' page-items: '20' -pagy/extras/headers___pagy_headers_with_Calendar_test_0003_returns_custom_headers_hash: +pagy/extras/headers___pagy_headers_merge_with_Calendar_test_0001_returns_the_full_headers_hash: + :response: !ruby/hash:Rack::Headers + link: ; rel="first", ; + rel="prev", ; rel="next", ; + rel="last" + current-page: '3' + page-items: '20' + total-pages: '26' + total-count: '505' +pagy/extras/headers___pagy_headers_with_Calendar_test_0002_returns_custom_headers_hash: :headers: link: ; rel="first", ; rel="next", ; rel="last" + Per-Page: '20' + Total: '505' pagy/extras/headers___pagy_headers_with_Calendar_test_0006_omit_next_on_last_page: :headers: link: ; rel="first", ; @@ -90,6 +86,12 @@ pagy/extras/headers___pagy_headers_with_Calendar_test_0006_omit_next_on_last_pag page-items: '20' total-pages: '26' total-count: '505' +pagy/extras/headers___pagy_headers_with_Calendar_test_0004_returns_the_countless_headers_hash: + :headers: + link: ; rel="first", ; + rel="next" + current-page: '1' + page-items: '20' pagy/extras/headers___pagy_headers_with_Calendar_test_0001_returns_the_full_headers_hash: :headers: link: ; rel="first", ; @@ -98,12 +100,10 @@ pagy/extras/headers___pagy_headers_with_Calendar_test_0001_returns_the_full_head page-items: '20' total-pages: '26' total-count: '505' -pagy/extras/headers___pagy_headers_with_Calendar_test_0002_returns_custom_headers_hash: +pagy/extras/headers___pagy_headers_with_Calendar_test_0003_returns_custom_headers_hash: :headers: link: ; rel="first", ; rel="next", ; rel="last" - Per-Page: '20' - Total: '505' pagy/extras/headers___pagy_headers_with_Calendar_test_0005_omit_prev_on_first_page: :headers: link: ; rel="first", ; diff --git a/test/pagy/extras/jsonapi_test.rb b/test/pagy/extras/jsonapi_test.rb index 0109b0a2d..1a7e20783 100644 --- a/test/pagy/extras/jsonapi_test.rb +++ b/test/pagy/extras/jsonapi_test.rb @@ -95,7 +95,7 @@ end describe '#pagy_jsonapi_links (keyset)' do it 'returns the ordered links' do - app = MockApp.new(params: { page: { latest: 'eyJpZCI6MTB9', size: 10 } }) + app = MockApp.new(params: { page: { latest: 'WzIwXQ', size: 10 } }) pagy, _records = app.send(:pagy_keyset, Pet.order(:id), page_param: :latest, diff --git a/test/pagy/extras/jsonapi_test.rb.yaml b/test/pagy/extras/jsonapi_test.rb.yaml index 5ada83907..032aa25fa 100644 --- a/test/pagy/extras/jsonapi_test.rb.yaml +++ b/test/pagy/extras/jsonapi_test.rb.yaml @@ -1,27 +1,27 @@ --- -pagy/extras/jsonapi___pagy_jsonapi_links_(keyset)_test_0002_sets_the_next_value_to_null_when_the_link_is_unavailable: +pagy/extras/jsonapi___pagy_jsonapi_links_(keyset)_test_0001_returns_the_ordered_links: :keyset_result: - :first: "/foo?page%5Bsize%5D=50&page%5Blatest%5D" + :first: "/foo?page%5Blatest%5D&page%5Bsize%5D=10" :last: :prev: - :next: -pagy/extras/jsonapi___pagy_jsonapi_links_(keyset)_test_0001_returns_the_ordered_links: + :next: "/foo?page%5Blatest%5D=WzMwXQ&page%5Bsize%5D=10" +pagy/extras/jsonapi___pagy_jsonapi_links_(keyset)_test_0002_sets_the_next_value_to_null_when_the_link_is_unavailable: :keyset_result: - :first: "/foo?page%5Blatest%5D&page%5Bsize%5D=10" + :first: "/foo?page%5Bsize%5D=50&page%5Blatest%5D" :last: :prev: - :next: "/foo?page%5Blatest%5D=eyJpZCI6MjB9&page%5Bsize%5D=10" + :next: +pagy/extras/jsonapi__JsonApi_test_0001_uses_the__jsonapi_with_page_nil: + :url_1: "/foo?page%5Bpage%5D=1" + :url_2: "/foo?page%5Bpage%5D=1&page%5Blimit%5D=20" +pagy/extras/jsonapi__JsonApi_test_0002_uses_the__jsonapi_with_page_3: + :url_1: "/foo?page%5Bpage%5D=2" + :url_2: "/foo?page%5Bpage%5D=2&page%5Blimit%5D=20" +pagy/extras/jsonapi__JsonApi_with_custom_named_params_test_0002_sets_custom_named_params: + :url: "/foo?page%5Bnumber%5D=4&page%5Bsize%5D=10" pagy/extras/jsonapi___pagy_jsonapi_links_test_0001_returns_the_ordered_links: :result: :first: "/foo?page%5Bnumber%5D=1&page%5Bsize%5D=10" :last: "/foo?page%5Bnumber%5D=100&page%5Bsize%5D=10" :prev: "/foo?page%5Bnumber%5D=2&page%5Bsize%5D=10" :next: "/foo?page%5Bnumber%5D=4&page%5Bsize%5D=10" -pagy/extras/jsonapi__JsonApi_with_custom_named_params_test_0002_sets_custom_named_params: - :url: "/foo?page%5Bnumber%5D=4&page%5Bsize%5D=10" -pagy/extras/jsonapi__JsonApi_test_0002_uses_the__jsonapi_with_page_3: - :url_1: "/foo?page%5Bpage%5D=2" - :url_2: "/foo?page%5Bpage%5D=2&page%5Blimit%5D=20" -pagy/extras/jsonapi__JsonApi_test_0001_uses_the__jsonapi_with_page_nil: - :url_1: "/foo?page%5Bpage%5D=1" - :url_2: "/foo?page%5Bpage%5D=1&page%5Blimit%5D=20" diff --git a/test/pagy/extras/keyset_test.rb b/test/pagy/extras/keyset_test.rb index 28afb416d..c928b103f 100644 --- a/test/pagy/extras/keyset_test.rb +++ b/test/pagy/extras/keyset_test.rb @@ -18,32 +18,32 @@ limit: 10) _(pagy).must_be_kind_of Pagy::Keyset _(records.size).must_equal 10 - _(pagy.next).must_equal "eyJhbmltYWwiOiJjYXQiLCJuYW1lIjoiRWxsYSIsImlkIjoxOH0" + _(pagy.next).must_equal "WyJjYXQiLCJFbGxhIiwxOF0" end it 'pulls the page from params' do - app = MockApp.new(params: { page: "eyJpZCI6MTB9", limit: 10 }) + app = MockApp.new(params: { page: "WzEwXQ", limit: 10 }) pagy, records = app.send(:pagy_keyset, model.order(:id), tuple_comparison: true) _(records.first.id).must_equal 11 - _(pagy.next).must_equal "eyJpZCI6MjB9" + _(pagy.next).must_equal "WzIwXQ" end end describe 'URL helpers' do it 'returns the URLs for first page' do - app = MockApp.new(params: { page: "eyJpZCI6MTB9", limit: 10 }) + app = MockApp.new(params: { page: nil, limit: 10 }) pagy, _records = app.send(:pagy_keyset, model.order(:id)) _(app.send(:pagy_keyset_first_url, pagy)).must_equal "/foo?page&limit=10" - _(app.send(:pagy_keyset_next_url, pagy)).must_equal "/foo?page=eyJpZCI6MjB9&limit=10" + _(app.send(:pagy_keyset_next_url, pagy)).must_equal "/foo?page=WzEwXQ&limit=10" end it 'returns the URLs for second page' do - app = MockApp.new(params: { page: "eyJpZCI6MjB9", limit: 10 }) + app = MockApp.new(params: { page: "WzEwXQ", limit: 10 }) pagy, _records = app.send(:pagy_keyset, model.order(:id)) _(app.send(:pagy_keyset_first_url, pagy)).must_equal "/foo?page&limit=10" - _(app.send(:pagy_keyset_next_url, pagy)).must_equal "/foo?page=eyJpZCI6MzB9&limit=10" + _(app.send(:pagy_keyset_next_url, pagy)).must_equal "/foo?page=WzIwXQ&limit=10" end it 'returns the URLs for last page' do - app = MockApp.new(params: { page: "eyJpZCI6NDB9", limit: 10 }) + app = MockApp.new(params: { page: "WzQwXQ", limit: 10 }) pagy, _records = app.send(:pagy_keyset, model.order(:id)) _(app.send(:pagy_keyset_first_url, pagy)).must_equal "/foo?page&limit=10" _(app.send(:pagy_keyset_next_url, pagy)).must_be_nil diff --git a/test/pagy/keyset_for_ui_test.rb b/test/pagy/keyset_for_ui_test.rb index 88e9f4ea9..ef7380645 100644 --- a/test/pagy/keyset_for_ui_test.rb +++ b/test/pagy/keyset_for_ui_test.rb @@ -11,7 +11,7 @@ describe 'uses optional variables' do it 'use the :tuple_comparison' do pagy = Pagy::KeysetForUI.new(model.order(:animal, :name, :id), - cutoffs: [nil, nil, "eyJhbmltYWwiOiJjYXQiLCJuYW1lIjoiRWxsYSIsImlkIjoxOH0"], + cutoffs: [nil, nil, "WyJjYXQiLCJFbGxhIiwxOF0"], page: 2, limit: 10, tuple_comparison: true) @@ -21,10 +21,10 @@ end it 'uses :jsonify_keyset_attributes' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9"], + cutoffs: [nil, nil, "WzEwXQ"], page: 2, limit: 10, - jsonify_keyset_attributes: lambda(&:to_json)) + jsonify_keyset_attributes: ->(attr) { attr.values.to_json }) _(pagy.next).must_equal(3) _(pagy.instance_variable_get(:@cutoff_args)).must_equal(id: 10) end @@ -39,19 +39,19 @@ end it 'handles the page/cut for the second page' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9"], + cutoffs: [nil, nil, "WzEwXQ"], limit: 10, page: 2) _(pagy.instance_variable_get(:@cutoff_args)).must_equal(id: 10) _(pagy.records.first.id).must_equal 11 _(pagy.next).must_equal 3 - _(pagy.cutoffs).must_equal [nil, nil, "eyJpZCI6MTB9", "eyJpZCI6MjB9"] + _(pagy.cutoffs).must_equal [nil, nil, "WzEwXQ", "WzIwXQ"] end it 'handles the page/cut for the last page' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6NDB9"], + cutoffs: [nil, nil, "WzEwXQ", "WzIwXQ", "WzMwXQ", "WzQwXQ"], limit: 10, - page: 2) + page: 5) _(pagy.instance_variable_get(:@cutoff_args)).must_equal(id: 40) _(pagy.records.first.id).must_equal 41 _(pagy.next).must_be_nil @@ -61,27 +61,27 @@ it 'raises OverflowError' do _ do Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9"], + cutoffs: [nil, nil, "WzEwXQ"], limit: 10, page: 3) end.must_raise Pagy::OverflowError end it 'resets overflow' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9", "eyJpZCI6MjB9"], + cutoffs: [nil, nil, "WzEwXQ", "WzIwXQ"], reset_overflow: true, limit: 10, page: 4) _(pagy.instance_variable_get(:@cutoff_args)).must_be_nil _(pagy.records.first.id).must_equal 1 _(pagy.next).must_equal 2 - _(pagy.cutoffs).must_equal [nil, nil, "eyJpZCI6MTB9"] + _(pagy.cutoffs).must_equal [nil, nil, "WzEwXQ"] end end describe 'handles the jumping back' do it 'handles the assign_cut_args jump back to the first page' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9"], # last visited 2 + cutoffs: [nil, nil, "WzEwXQ"], # last visited 2 page: 1, limit: 10) _(pagy.instance_variable_get(:@cut)).must_be_nil @@ -90,13 +90,13 @@ end it 'handles the assign_cut_args jump back to the second page' do pagy = Pagy::KeysetForUI.new(model.order(:id), - cutoffs: [nil, nil, "eyJpZCI6MTB9", "eyJpZCI6MjB9"], + cutoffs: [nil, nil, "WzEwXQ", "WzIwXQ"], page: 2, limit: 10) _(pagy.instance_variable_get(:@cutoff_args)).must_equal({ :id => 10, :cutoff_id => 20 }) _(pagy.records.first.id).must_equal 11 _(pagy.next).must_equal 3 - _(pagy.cutoffs).must_equal [nil, nil, "eyJpZCI6MTB9", "eyJpZCI6MjB9"] + _(pagy.cutoffs).must_equal [nil, nil, "WzEwXQ", "WzIwXQ"] end end describe 'other requirements' do diff --git a/test/pagy/keyset_test.rb b/test/pagy/keyset_test.rb index e81a6c12e..4d8d21426 100644 --- a/test/pagy/keyset_test.rb +++ b/test/pagy/keyset_test.rb @@ -24,8 +24,8 @@ _(Pagy::Keyset.new(model.order(:id))).must_be_kind_of Pagy::Keyset end it 'raises Pagy::InternalError for inconsistent page/keyset' do - page_animal_id = Pagy::B64.urlsafe_encode({animal: 'dog', id: 23}.to_json) - err = assert_raises(Pagy::InternalError) do + page_animal_id = Pagy::B64.urlsafe_encode({ animal: 'dog', id: 23 }.to_json) + err = assert_raises(Pagy::InternalError) do Pagy::Keyset.new(model.order(:id), limit: 10, page: page_animal_id) end assert_match(/cutoff and keyset are not consistent/, err.message) @@ -33,52 +33,39 @@ end describe 'uses optional variables' do it 'use the :tuple_comparison' do - pagy = Pagy::Keyset.new(model.order(:animal, :name, :id), - page: "eyJhbmltYWwiOiJjYXQiLCJuYW1lIjoiRWxsYSIsImlkIjoxOH0", - limit: 10, - tuple_comparison: true) + pagy = Pagy::Keyset.new(model.order(:animal, :name, :id), + page: "WyJjYXQiLCJFbGxhIiwxOF0", + limit: 10, + tuple_comparison: true) records = pagy.records _(records.size).must_equal 10 _(records.first.id).must_equal 13 end it 'uses :jsonify_keyset_attributes' do pagy = Pagy::Keyset.new(model.order(:id), - page: "eyJpZCI6MTB9", - limit: 10, - jsonify_keyset_attributes: lambda(&:to_json)) + page: "WzEwXQ", + limit: 10, + jsonify_keyset_attributes: ->(attr) { attr.values.to_json }) _(pagy.next).must_equal("eyJpZCI6MjB9") - _(pagy.instance_variable_get(:@cutoff_args)).must_equal({id: 10}) - end - it 'uses :filter_records' do - filter_records = if model == Pet - ->(set, filter_args, _keyset) { set.where('id > :id', **filter_args) } - else - ->(set, filter_args, _keyset) { set.where(Sequel.lit('id > :id', **filter_args)) } - end - pagy = Pagy::Keyset.new(model.order(:id), - page: "eyJpZCI6MTB9", - limit: 10, - filter_records:) - records = pagy.records - _(records.first.id).must_equal(11) + _(pagy.instance_variable_get(:@cutoff_args)).must_equal({ id: 10 }) end end describe '#extract_keyset' do it 'extracts the keyset from the set order (single column)' do pagy = Pagy::Keyset.new(model.order(:id)) - _(pagy.instance_variable_get(:@keyset)).must_equal({:id => :asc}) + _(pagy.instance_variable_get(:@keyset)).must_equal({ :id => :asc }) set = model == Pet ? model.order(id: :desc) : model.order(Sequel.desc(:id)) pagy = Pagy::Keyset.new(set) - _(pagy.instance_variable_get(:@keyset)).must_equal({:id => :desc}) + _(pagy.instance_variable_get(:@keyset)).must_equal({ :id => :desc }) end it 'extracts the keyset from the set order (multiple columns)' do - set = if model == Pet - model.order(animal: :desc, id: :asc) - else - model.order(Sequel.desc(:animal), Sequel.asc(:id)) - end + set = if model == Pet + model.order(animal: :desc, id: :asc) + else + model.order(Sequel.desc(:animal), Sequel.asc(:id)) + end pagy = Pagy::Keyset.new(set) - _(pagy.instance_variable_get(:@keyset)).must_equal({animal: :desc, :id => :asc}) + _(pagy.instance_variable_get(:@keyset)).must_equal({ animal: :desc, :id => :asc }) end if model == PetSequel it 'raises TypeError for unknown order type' do @@ -87,12 +74,12 @@ it 'skips unrestricted primary keys' do model.unrestrict_primary_key Pagy::Keyset.new(model.order(:id), - page: "eyJpZCI6MTB9", + page: "WzEwXQ", limit: 10) _(model.restrict_primary_key?).must_equal false model.restrict_primary_key Pagy::Keyset.new(model.order(:id), - page: "eyJpZCI6MTB9", + page: "WzEwXQ", limit: 10) _(model.restrict_primary_key?).must_equal true end @@ -102,16 +89,16 @@ it 'handles the page/cut for the first page' do pagy = Pagy::Keyset.new(model.order(:id), limit: 10) _(pagy.instance_variable_get(:@cut)).must_be_nil - _(pagy.next).must_equal "eyJpZCI6MTB9" + _(pagy.next).must_equal "WzEwXQ" end it 'handles the page/cut for the second page' do - pagy = Pagy::Keyset.new(model.order(:id), limit: 10, page: "eyJpZCI6MTB9") + pagy = Pagy::Keyset.new(model.order(:id), limit: 10, page: "WzEwXQ") _(pagy.instance_variable_get(:@cutoff_args)).must_equal(id: 10) _(pagy.records.first.id).must_equal 11 - _(pagy.next).must_equal "eyJpZCI6MjB9" + _(pagy.next).must_equal "WzIwXQ" end it 'handles the page/cut for the last page' do - pagy = Pagy::Keyset.new(model.order(:id), limit: 10, page: "eyJpZCI6NDB9") + pagy = Pagy::Keyset.new(model.order(:id), limit: 10, page: "WzQwXQ") _(pagy.instance_variable_get(:@cutoff_args)).must_equal(id: 40) _(pagy.records.first.id).must_equal 41 _(pagy.next).must_be_nil @@ -119,8 +106,8 @@ end describe 'other requirements' do it 'adds the required columns to the selected values' do - set = model.order(:animal, :name, :id).select(:name) - pagy = Pagy::Keyset.new(set, limit: 10) + set = model.order(:animal, :name, :id).select(:name) + pagy = Pagy::Keyset.new(set, limit: 10) pagy.records set = pagy.instance_variable_get(:@set) _((model == Pet ? set.select_values : set.opts[:select]).sort).must_equal %i[animal id name] @@ -132,6 +119,7 @@ def slurp_by_page(page: nil, records: [], &block) records << result[:records] result[:page] ? slurp_by_page(page: result[:page], records:, &block) : records end + mixed_set = if model == Pet model.order(animal: :asc, birthdate: :desc, id: :asc) elsif model == PetSequel @@ -141,9 +129,9 @@ def slurp_by_page(page: nil, records: [], &block) model.order(:animal, :name, :id), mixed_set].each_with_index do |set, i| it "pulls all the records in set#{i} without repetions" do - pages = slurp_by_page do |page| + pages = slurp_by_page do |page| pagy = Pagy::Keyset.new(set, page:, limit: 9) - {records: pagy.records, page: pagy.next} + { records: pagy.records, page: pagy.next } end collection = set.to_a _(collection.size).must_equal 50