├── .gitignore ├── Gemfile ├── MIT-LICENSE ├── README.md ├── Rakefile ├── clean_pagination.gemspec ├── lib ├── clean_pagination.rb ├── clean_pagination │ └── version.rb └── tasks │ └── clean_pagination_tasks.rake └── test ├── clean_pagination_test.rb ├── dummy ├── README.rdoc ├── Rakefile ├── app │ ├── assets │ │ ├── images │ │ │ └── .keep │ │ ├── javascripts │ │ │ └── application.js │ │ └── stylesheets │ │ │ └── application.css │ ├── controllers │ │ ├── application_controller.rb │ │ └── concerns │ │ │ └── .keep │ ├── helpers │ │ └── application_helper.rb │ ├── mailers │ │ └── .keep │ ├── models │ │ ├── .keep │ │ └── concerns │ │ │ └── .keep │ └── views │ │ └── layouts │ │ └── application.html.erb ├── bin │ ├── bundle │ ├── rails │ └── rake ├── config.ru ├── config │ ├── application.rb │ ├── boot.rb │ ├── database.yml │ ├── environment.rb │ ├── environments │ │ ├── development.rb │ │ ├── production.rb │ │ └── test.rb │ ├── initializers │ │ ├── backtrace_silencers.rb │ │ ├── filter_parameter_logging.rb │ │ ├── inflections.rb │ │ ├── mime_types.rb │ │ ├── secret_token.rb │ │ ├── session_store.rb │ │ └── wrap_parameters.rb │ ├── locales │ │ └── en.yml │ ├── routes.rb │ └── secrets.yml ├── lib │ └── assets │ │ └── .keep ├── log │ └── .keep └── public │ ├── 404.html │ ├── 422.html │ ├── 500.html │ └── favicon.ico └── test_helper.rb /.gitignore: -------------------------------------------------------------------------------- 1 | Gemfile.lock 2 | .bundle/ 3 | log/*.log 4 | pkg/ 5 | test/dummy/db/*.sqlite3 6 | test/dummy/db/*.sqlite3-journal 7 | test/dummy/log/*.log 8 | test/dummy/tmp/ 9 | test/dummy/.sass-cache 10 | -------------------------------------------------------------------------------- /Gemfile: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | # Declare your gem's dependencies in clean_pagination.gemspec. 4 | # Bundler will treat runtime dependencies like base dependencies, and 5 | # development dependencies will be added by default to the :development group. 6 | gemspec 7 | 8 | # Declare any dependencies that are still in development here instead of in 9 | # your gemspec. These might include edge Rails or gems from your path or 10 | # Git. Remember to move these dependencies to your gemspec before releasing 11 | # your gem to rubygems.org. 12 | 13 | # To use debugger 14 | # gem 'debugger' 15 | -------------------------------------------------------------------------------- /MIT-LICENSE: -------------------------------------------------------------------------------- 1 | Copyright 2014 YOURNAME 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## Clean Pagination 2 | 3 | [![Build Status](https://travis-ci.org/begriffs/clean_pagination.png?branch=master)](https://travis-ci.org/begriffs/clean_pagination) 4 | 5 | The simplest, most flexible, most standards-compliant 6 | pagination gem there is. Pairs nicely with 7 | [begriffs/angular-paginate-anything](https://github.com/begriffs/angular-paginate-anything). 8 | 9 | ### Usage 10 | 11 | ```ruby 12 | class ApplicationController < ActionController::Base 13 | include CleanPagination 14 | 15 | # Using activemodel 16 | def index 17 | max_per_page = 100 18 | 19 | paginate Bla.count, max_per_page do |limit, offset| 20 | render json: Bla.limit(limit).offset(offset) 21 | end 22 | end 23 | 24 | # Using some custom data 25 | def numbers 26 | paginate Float::INFINITY, 100 do |limit, offset| 27 | render json: (offset...offset+limit).to_a 28 | end 29 | end 30 | 31 | # Using optional settings 32 | def options 33 | begin 34 | paginate Foo.scope.count, max_per_page, allow_render: false, raise_errors: true do |limit, offset| 35 | respond_with Foo.scope.limit(limit).offset(offset) 36 | end 37 | rescue RangeError 38 | render json: { error: { message: "invalid pagination range" } } 39 | end 40 | end 41 | end 42 | ``` 43 | 44 | ### Benefits 45 | 46 | * **HTTP Content-Type agnoticism.** Information about total items, 47 | selected ranges, and next- previous-links are sent through headers. 48 | It works without modifying your API payload in any way. 49 | * **Graceful degredation.** Both client and server specify the maximum 50 | page size they accept and communication gracefully degrades to 51 | accomodate the lesser. 52 | * **Expressive retrieval.** This approach, unlike the use of `per_page` and 53 | `page` parameters, allows the client to request any (possibly unbounded) 54 | interval of items. 55 | * **Semantic HTTP.** Built in strict conformance to RFCs 2616 and 5988. 56 | 57 | ### Under the hood 58 | 59 | To prevent this gem from rendering while still allowing it to set 60 | headers and response codes, pass `allow_render: false` to `paginate`. 61 | This may be used to avoid `DoubleRenderError` situations. 62 | 63 | To handle the invalid request range condition in your app, pass the 64 | `raise_errors: true` option. This will raise a `RangeError` which you can 65 | rescue (and thus control what is rendered). Headers will still be set. 66 | 67 | [TODO: explain what the headers mean.] Until this is written you can consult 68 | the tests for an idea how it works, or use a client that is compatible, such as 69 | [begriffs/angular-paginate-anything](https://github.com/begriffs/angular-paginate-anything). 70 | -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | begin 2 | require 'bundler/setup' 3 | rescue LoadError 4 | puts 'You must `gem install bundler` and `bundle install` to run rake tasks' 5 | end 6 | 7 | require 'rdoc/task' 8 | 9 | RDoc::Task.new(:rdoc) do |rdoc| 10 | rdoc.rdoc_dir = 'rdoc' 11 | rdoc.title = 'CleanPagination' 12 | rdoc.options << '--line-numbers' 13 | rdoc.rdoc_files.include('README.rdoc') 14 | rdoc.rdoc_files.include('lib/**/*.rb') 15 | end 16 | 17 | 18 | 19 | 20 | Bundler::GemHelper.install_tasks 21 | 22 | require 'rake/testtask' 23 | 24 | Rake::TestTask.new(:test) do |t| 25 | t.libs << 'lib' 26 | t.libs << 'test' 27 | t.pattern = 'test/**/*_test.rb' 28 | t.verbose = false 29 | end 30 | 31 | 32 | task default: :test 33 | -------------------------------------------------------------------------------- /clean_pagination.gemspec: -------------------------------------------------------------------------------- 1 | $:.push File.expand_path("../lib", __FILE__) 2 | 3 | # Maintain your gem's version: 4 | require "clean_pagination/version" 5 | 6 | # Describe your gem and declare its dependencies: 7 | Gem::Specification.new do |s| 8 | s.name = "clean_pagination" 9 | s.version = CleanPagination::VERSION 10 | s.authors = ["Joe Nelson"] 11 | s.email = ["cred+github@begriffs.com"] 12 | s.description = "API pagination the way RFC2616 intended it" 13 | s.summary = "Mix into controllers to get pagination helpers" 14 | s.homepage = "https://github.com/begriffs/clean_pagination" 15 | s.license = "MIT" 16 | 17 | s.files = Dir["{app,config,db,lib}/**/*", "MIT-LICENSE", "Rakefile", "README.rdoc"] 18 | s.test_files = Dir["test/**/*"] 19 | 20 | s.add_development_dependency "rails", ">= 3.2", "< 4.2" 21 | s.add_development_dependency "sqlite3" 22 | s.add_development_dependency "pry" 23 | s.add_development_dependency "mocha" 24 | end 25 | -------------------------------------------------------------------------------- /lib/clean_pagination.rb: -------------------------------------------------------------------------------- 1 | module CleanPagination 2 | 3 | def paginate total_items, max_range_size, options = {} 4 | options[:allow_render] = true if options[:allow_render].nil? 5 | options[:raise_errors] ||= false 6 | 7 | headers['Accept-Ranges'] = 'items' 8 | headers['Range-Unit'] = 'items' 9 | 10 | requested_from, requested_to = 0, [0, total_items - 1].max 11 | 12 | if request.headers['Range-Unit'] == 'items' && 13 | request.headers['Range'].present? 14 | if request.headers['Range'] =~ /(\d+)-(\d*)/ 15 | requested_from, requested_to = $1.to_i, ($2.present? ? $2.to_i : Float::INFINITY) 16 | end 17 | end 18 | 19 | if (requested_from > requested_to) || 20 | (requested_from > 0 && requested_from >= total_items) 21 | response.status = 416 22 | headers['Content-Range'] = "*/#{total_items}" 23 | message = 'invalid pagination range' 24 | raise RangeError, message if options[:raise_errors] 25 | render text: message if options[:allow_render] 26 | return 27 | end 28 | 29 | available_to = [requested_to, 30 | total_items - 1, 31 | requested_from + max_range_size - 1 32 | ].min 33 | available_limit = available_to - requested_from + 1 34 | 35 | if available_limit == 0 36 | headers['Content-Range'] = "*/0" 37 | else 38 | headers['Content-Range'] = "#{ 39 | requested_from 40 | }-#{ 41 | available_to 42 | }/#{ 43 | total_items < Float::INFINITY ? total_items : '*' 44 | }" 45 | end 46 | 47 | yield available_limit, requested_from 48 | if available_limit < total_items && response.status == 200 49 | response.status = 206 50 | end 51 | 52 | requested_limit = requested_to - requested_from + 1 53 | 54 | links = [] 55 | if available_to < total_items - 1 56 | links << "<#{request.url}>; rel=\"next\"; items=\"#{ 57 | available_to + 1 58 | }-#{ 59 | suppress_infinity(available_to + requested_limit) 60 | }\"" 61 | 62 | if total_items < Float::INFINITY 63 | links << "<#{request.url}>; rel=\"last\"; items=\"#{ 64 | # let rounding do the work 65 | ((total_items-1) / available_limit) * available_limit 66 | }-#{ 67 | (((total_items-1) / available_limit) * available_limit) + requested_limit - 1 68 | }\"" 69 | end 70 | end 71 | if requested_from > 0 72 | previous_from = [0, requested_from - [requested_limit, max_range_size].min].max 73 | 74 | links << "<#{request.url}>; rel=\"prev\"; items=\"#{ 75 | previous_from 76 | }-#{ 77 | suppress_infinity(previous_from + requested_limit - 1) 78 | }\"" 79 | 80 | links << "<#{request.url}>; rel=\"first\"; items=\"0-#{suppress_infinity(requested_limit-1)}\"" 81 | end 82 | 83 | headers['Link'] = links.join ', ' unless links.empty? 84 | end 85 | 86 | private 87 | 88 | def suppress_infinity n 89 | n < Float::INFINITY ? n : '' 90 | end 91 | 92 | end 93 | -------------------------------------------------------------------------------- /lib/clean_pagination/version.rb: -------------------------------------------------------------------------------- 1 | module CleanPagination 2 | VERSION = "1.0.0" 3 | end 4 | -------------------------------------------------------------------------------- /lib/tasks/clean_pagination_tasks.rake: -------------------------------------------------------------------------------- 1 | # desc "Explaining what the task does" 2 | # task :clean_pagination do 3 | # # Task goes here 4 | # end 5 | -------------------------------------------------------------------------------- /test/clean_pagination_test.rb: -------------------------------------------------------------------------------- 1 | require 'test_helper' 2 | 3 | def parse_link_ranges header 4 | links = Hash.new 5 | (header || '').split(',').each do |link| 6 | link =~ /rel="(.*)".*items="(.*)"/ 7 | links[$1] = $2 8 | end 9 | links 10 | end 11 | 12 | class ApplicationControllerTest < ActionController::TestCase 13 | setup do 14 | @controller.stubs(:total_items).returns 101 15 | @controller.stubs(:max_range).returns 100 16 | end 17 | 18 | test 'rangeless request range works normally if max_range >= total' do 19 | @controller.stubs(:total_items).returns 100 20 | 21 | @controller.expects(:action).with(100, 0) 22 | get :index 23 | assert_equal 200, response.status 24 | assert_equal 'items', response.headers['Accept-Ranges'] 25 | end 26 | 27 | test 'rangeless request truncates if max_range < total' do 28 | @controller.expects(:action).with(100, 0) 29 | get :index 30 | assert_equal 206, response.status 31 | assert_equal 'items', response.headers['Accept-Ranges'] 32 | assert_equal '0-99/101', response.headers['Content-Range'] 33 | end 34 | 35 | test 'an acceptable range succeeds' do 36 | @request.headers['Range-Unit'] = 'items' 37 | @request.headers['Range'] = "0-99" 38 | 39 | @controller.expects(:action).with(100, 0) 40 | get :index 41 | assert_equal 206, response.status 42 | assert_equal 'items', response.headers['Range-Unit'] 43 | assert_equal '0-99/101', response.headers['Content-Range'] 44 | end 45 | 46 | test 'an oversized range is truncated' do 47 | @request.headers['Range-Unit'] = 'items' 48 | @request.headers['Range'] = "0-100" 49 | 50 | get :index 51 | assert_equal 206, response.status 52 | assert_equal 'items', response.headers['Range-Unit'] 53 | assert_equal '0-99/101', response.headers['Content-Range'] 54 | end 55 | 56 | test "passes along exceptional status codes" do 57 | @request.headers['Range-Unit'] = 'items' 58 | @request.headers['Range'] = "0-99" 59 | 60 | [100, 301, 404, 500].each do |code| 61 | @controller.stubs(:index_status).returns code 62 | get :index 63 | assert_equal code, response.status 64 | end 65 | end 66 | 67 | test "reports infinite/unknown collection" do 68 | @controller.stubs(:total_items).returns Float::INFINITY 69 | 70 | @request.headers['Range-Unit'] = 'items' 71 | @request.headers['Range'] = "0-9" 72 | get :index 73 | assert_equal '0-9/*', response.headers['Content-Range'] 74 | end 75 | 76 | test "refuses offside ranges" do 77 | @request.headers['Range-Unit'] = 'items' 78 | @request.headers['Range'] = "1-0" 79 | 80 | @controller.expects(:action).never 81 | get :index 82 | assert_equal 416, response.status 83 | assert_equal '*/101', response.headers['Content-Range'] 84 | end 85 | 86 | test "returns 200 when no items found, delegates response to action" do 87 | @controller.stubs(:total_items).returns 0 88 | @request.headers['Range-Unit'] = 'items' 89 | @request.headers['Range'] = "0-9" 90 | 91 | @controller.expects(:action).with(0, 0) 92 | get :index 93 | assert_equal 200, response.status 94 | assert_equal '*/0', response.headers['Content-Range'] 95 | end 96 | 97 | test "refuses a range with nonzero start when there are no items" do 98 | @controller.stubs(:total_items).returns 0 99 | @request.headers['Range-Unit'] = 'items' 100 | @request.headers['Range'] = "1-10" 101 | 102 | @controller.expects(:action).never 103 | get :index 104 | assert_equal 416, response.status 105 | assert_equal '*/0', response.headers['Content-Range'] 106 | end 107 | 108 | test "refuses range start past end" do 109 | @request.headers['Range-Unit'] = 'items' 110 | @request.headers['Range'] = "101-" 111 | 112 | @controller.expects(:action).never 113 | get :index 114 | assert_equal 416, response.status 115 | assert_equal '*/101', response.headers['Content-Range'] 116 | end 117 | 118 | test "optionally raise exception when range is invalid" do 119 | @request.headers['Range-Unit'] = 'items' 120 | @request.headers['Range'] = "1-0" 121 | @controller.stubs(:raise_errors).returns true 122 | 123 | @controller.expects(:action).never 124 | assert_raises(RangeError) do 125 | get :index 126 | end 127 | end 128 | 129 | test "optionally prevent rendering anything when range is invalid" do 130 | @request.headers['Range-Unit'] = 'items' 131 | @request.headers['Range'] = "1-0" 132 | @controller.stubs(:allow_render).returns false 133 | @controller.expects(:action).never 134 | assert_raises(ActionView::MissingTemplate) do 135 | get :index 136 | end 137 | end 138 | 139 | test "allows one-item requests" do 140 | @request.headers['Range-Unit'] = 'items' 141 | @request.headers['Range'] = "0-0" 142 | 143 | @controller.expects(:action).with(1, 0) 144 | get :index 145 | assert_equal 206, response.status 146 | assert_equal '0-0/101', response.headers['Content-Range'] 147 | end 148 | 149 | test "allows one-item responses" do 150 | @request.headers['Range-Unit'] = 'items' 151 | @request.headers['Range'] = "0-9" 152 | @controller.stubs(:total_items).returns 1 153 | @controller.stubs(:max_range).returns 100 154 | 155 | @controller.expects(:action).with(1, 0) 156 | get :index 157 | assert_equal 200, response.status 158 | assert_equal '0-0/1', response.headers['Content-Range'] 159 | end 160 | 161 | test "handles ranges beyond collection length via truncation" do 162 | @request.headers['Range-Unit'] = 'items' 163 | @request.headers['Range'] = "50-200" 164 | 165 | @controller.expects(:action).with(51, 50) 166 | get :index 167 | 168 | assert_equal 206, response.status 169 | assert_equal '50-100/101', response.headers['Content-Range'] 170 | end 171 | 172 | test "includes link headers" do 173 | @controller.stubs(:total_items).returns 100 174 | 175 | @request.headers['Range-Unit'] = 'items' 176 | @request.headers['Range'] = "20-29" 177 | 178 | get :index 179 | links = parse_link_ranges response.headers['Link'] 180 | 181 | assert_equal '30-39', links['next'] 182 | assert_equal '10-19', links['prev'] 183 | assert_equal '0-9', links['first'] 184 | assert_equal '90-99', links['last'] 185 | end 186 | 187 | test "next page range can extend beyond last item" do 188 | @controller.stubs(:total_items).returns 100 189 | 190 | @request.headers['Range-Unit'] = 'items' 191 | @request.headers['Range'] = "50-89" 192 | 193 | get :index 194 | links = parse_link_ranges response.headers['Link'] 195 | 196 | assert_equal '90-129', links['next'] 197 | end 198 | 199 | test "previous page range cannot go negative" do 200 | @controller.stubs(:total_items).returns 100 201 | 202 | @request.headers['Range-Unit'] = 'items' 203 | @request.headers['Range'] = "10-99" 204 | 205 | get :index 206 | links = parse_link_ranges response.headers['Link'] 207 | 208 | assert_equal '0-89', links['prev'] 209 | end 210 | 211 | test "first page range always starts at zero" do 212 | @controller.stubs(:total_items).returns 100 213 | 214 | @request.headers['Range-Unit'] = 'items' 215 | @request.headers['Range'] = "63-72" 216 | 217 | get :index 218 | links = parse_link_ranges response.headers['Link'] 219 | 220 | assert_equal '0-9', links['first'] 221 | end 222 | 223 | test "last page range can extend beyond the last item" do 224 | @controller.stubs(:total_items).returns 100 225 | 226 | @request.headers['Range-Unit'] = 'items' 227 | @request.headers['Range'] = "0-6" 228 | 229 | get :index 230 | links = parse_link_ranges response.headers['Link'] 231 | 232 | assert_equal '98-104', links['last'] 233 | end 234 | 235 | test "infinite collections have no last page" do 236 | @controller.stubs(:total_items).returns Float::INFINITY 237 | 238 | @request.headers['Range-Unit'] = 'items' 239 | @request.headers['Range'] = "0-9" 240 | 241 | get :index 242 | links = parse_link_ranges response.headers['Link'] 243 | 244 | assert_nil links['last'] 245 | end 246 | 247 | test "omitting the end number asks for everything" do 248 | @controller.stubs(:total_items).returns Float::INFINITY 249 | @controller.stubs(:max_range).returns 1000000 250 | @request.headers['Range-Unit'] = 'items' 251 | @request.headers['Range'] = "50-" 252 | 253 | @controller.expects(:action).with(1000000, 50) 254 | get :index 255 | end 256 | 257 | test "omitting the end number omits in first link too" do 258 | @controller.stubs(:total_items).returns Float::INFINITY 259 | @controller.stubs(:max_range).returns 1000000 260 | @request.headers['Range-Unit'] = 'items' 261 | @request.headers['Range'] = "50-" 262 | 263 | get :index 264 | links = parse_link_ranges response.headers['Link'] 265 | assert_equal '0-', links['first'] 266 | end 267 | 268 | test "next link with omitted end number shifts by max page" do 269 | @controller.stubs(:total_items).returns Float::INFINITY 270 | @controller.stubs(:max_range).returns 1000000 271 | @request.headers['Range-Unit'] = 'items' 272 | @request.headers['Range'] = "50-" 273 | 274 | get :index 275 | links = parse_link_ranges response.headers['Link'] 276 | assert_equal '1000050-', links['next'] 277 | end 278 | 279 | test "prev link with omitted end number shifts by max page" do 280 | @controller.stubs(:total_items).returns Float::INFINITY 281 | @controller.stubs(:max_range).returns 25 282 | @request.headers['Range-Unit'] = 'items' 283 | @request.headers['Range'] = "50-" 284 | 285 | get :index 286 | links = parse_link_ranges response.headers['Link'] 287 | assert_equal '25-', links['prev'] 288 | end 289 | 290 | test "shifts penultimate page to beginning, preserving length" do 291 | @controller.stubs(:total_items).returns 100 292 | 293 | @request.headers['Range-Unit'] = 'items' 294 | @request.headers['Range'] = "10-49" 295 | 296 | get :index 297 | links = parse_link_ranges response.headers['Link'] 298 | 299 | assert_equal '0-39', links['prev'] 300 | assert_equal '0-39', links['first'] 301 | end 302 | 303 | test "prev is the left inverse of next" do 304 | @request.headers['Range-Unit'] = 'items' 305 | 306 | 100.times do 307 | total_items = rand(1..20) 308 | if total_items == 20 309 | total_items = Float::INFINITY 310 | end 311 | to = rand(1..[total_items, 20].min) 312 | from = rand(0...to) 313 | max_range = rand(1..20) 314 | 315 | msg = "#{from}-#{to}/#{total_items} max_range=#{max_range}" 316 | 317 | @controller.stubs(:total_items).returns total_items 318 | @controller.stubs(:max_range).returns max_range 319 | @request.headers['Range'] = "#{from}-#{to}" 320 | get :index 321 | 322 | links = parse_link_ranges response.headers['Link'] 323 | if links['next'] 324 | msg += " thence to #{links['next']}/#{total_items}" 325 | @request.headers['Range'] = links['next'] 326 | get :index 327 | 328 | links = parse_link_ranges response.headers['Link'] 329 | assert_equal "#{from}-#{to}", links['prev'], msg 330 | end 331 | end 332 | end 333 | 334 | test "for from > to-from, next is the right inverse of prev" do 335 | @request.headers['Range-Unit'] = 'items' 336 | 337 | 100.times do 338 | total_items = rand(1..20) 339 | if total_items == 20 340 | total_items = Float::INFINITY 341 | end 342 | to = rand(1..[total_items, 20].min) 343 | from = rand(to/2+1...to) 344 | max_range = rand(1..20) 345 | 346 | msg = "#{from}-#{to}/#{total_items} max_range=#{max_range}" 347 | 348 | @controller.stubs(:total_items).returns total_items 349 | @controller.stubs(:max_range).returns max_range 350 | @request.headers['Range'] = "#{from}-#{to}" 351 | get :index 352 | 353 | links = parse_link_ranges response.headers['Link'] 354 | if links['prev'] 355 | msg += " thence to #{links['prev']}/#{total_items}" 356 | @request.headers['Range'] = links['prev'] 357 | get :index 358 | 359 | links = parse_link_ranges response.headers['Link'] 360 | assert_equal "#{from}-#{to}", links['next'], msg 361 | end 362 | end 363 | end 364 | 365 | test "omits prev and first links at start" do 366 | @request.headers['Range-Unit'] = 'items' 367 | @request.headers['Range'] = "0-9" 368 | 369 | get :index 370 | links = parse_link_ranges response.headers['Link'] 371 | 372 | assert_nil links['first'] 373 | assert_nil links['prev'] 374 | end 375 | 376 | test "omits next and last links at end" do 377 | @controller.stubs(:total_items).returns 100 378 | @request.headers['Range-Unit'] = 'items' 379 | @request.headers['Range'] = "90-99" 380 | 381 | get :index 382 | links = parse_link_ranges response.headers['Link'] 383 | 384 | assert_nil links['last'] 385 | assert_nil links['next'] 386 | end 387 | 388 | test "preserves query parameters in link headers" do 389 | @request.headers['Range-Unit'] = 'items' 390 | @request.headers['Range'] = "20-29" 391 | get :index, foo: 'bar' 392 | 393 | response.headers['Link'].scan(/<[^>]+>/).each do |link| 394 | assert_match(/\?foo=bar/, link) 395 | end 396 | end 397 | 398 | test "omits empty link header" do 399 | @controller.stubs(:total_items).returns 2 400 | 401 | get :index 402 | 403 | assert_equal false, response.headers.has_key?('Link') 404 | end 405 | end 406 | -------------------------------------------------------------------------------- /test/dummy/README.rdoc: -------------------------------------------------------------------------------- 1 | == README 2 | 3 | This README would normally document whatever steps are necessary to get the 4 | application up and running. 5 | 6 | Things you may want to cover: 7 | 8 | * Ruby version 9 | 10 | * System dependencies 11 | 12 | * Configuration 13 | 14 | * Database creation 15 | 16 | * Database initialization 17 | 18 | * How to run the test suite 19 | 20 | * Services (job queues, cache servers, search engines, etc.) 21 | 22 | * Deployment instructions 23 | 24 | * ... 25 | 26 | 27 | Please feel free to use a different markup language if you do not plan to run 28 | rake doc:app. 29 | -------------------------------------------------------------------------------- /test/dummy/Rakefile: -------------------------------------------------------------------------------- 1 | # Add your own tasks in files placed in lib/tasks ending in .rake, 2 | # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake. 3 | 4 | require File.expand_path('../config/application', __FILE__) 5 | 6 | Rails.application.load_tasks 7 | -------------------------------------------------------------------------------- /test/dummy/app/assets/images/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/app/assets/images/.keep -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/application.js: -------------------------------------------------------------------------------- 1 | // This is a manifest file that'll be compiled into application.js, which will include all the files 2 | // listed below. 3 | // 4 | // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts, 5 | // or vendor/assets/javascripts of plugins, if any, can be referenced here using a relative path. 6 | // 7 | // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the 8 | // compiled file. 9 | // 10 | // Read Sprockets README (https://github.com/sstephenson/sprockets#sprockets-directives) for details 11 | // about supported directives. 12 | // 13 | //= require_tree . 14 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/application.css: -------------------------------------------------------------------------------- 1 | /* 2 | * This is a manifest file that'll be compiled into application.css, which will include all the files 3 | * listed below. 4 | * 5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets, 6 | * or vendor/assets/stylesheets of plugins, if any, can be referenced here using a relative path. 7 | * 8 | * You're free to add application-wide styles to this file and they'll appear at the bottom of the 9 | * compiled file so the styles you add here take precedence over styles defined in any styles 10 | * defined in the other CSS/SCSS files in this directory. It is generally better to create a new 11 | * file per style scope. 12 | * 13 | *= require_tree . 14 | *= require_self 15 | */ 16 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/application_controller.rb: -------------------------------------------------------------------------------- 1 | class ApplicationController < ActionController::Base 2 | # Prevent CSRF attacks by raising an exception. 3 | # For APIs, you may want to use :null_session instead. 4 | protect_from_forgery with: :exception 5 | 6 | include CleanPagination 7 | 8 | def index 9 | paginate total_items, max_range, 10 | allow_render: allow_render, raise_errors: raise_errors do |limit, offset| 11 | 12 | action limit, offset 13 | render json: [limit, offset], status: index_status 14 | end 15 | end 16 | 17 | def allow_render 18 | true # stub me 19 | end 20 | 21 | def raise_errors 22 | false # stub me 23 | end 24 | 25 | def total_items 26 | raise 'stub me' 27 | end 28 | 29 | def max_range 30 | raise 'stub me' 31 | end 32 | 33 | def action limit, offset 34 | # gets spied on 35 | end 36 | 37 | def index_status 38 | 200 39 | end 40 | end 41 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/concerns/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/app/controllers/concerns/.keep -------------------------------------------------------------------------------- /test/dummy/app/helpers/application_helper.rb: -------------------------------------------------------------------------------- 1 | module ApplicationHelper 2 | end 3 | -------------------------------------------------------------------------------- /test/dummy/app/mailers/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/app/mailers/.keep -------------------------------------------------------------------------------- /test/dummy/app/models/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/app/models/.keep -------------------------------------------------------------------------------- /test/dummy/app/models/concerns/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/app/models/concerns/.keep -------------------------------------------------------------------------------- /test/dummy/app/views/layouts/application.html.erb: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Dummy 5 | <%= stylesheet_link_tag "application", media: "all", "data-turbolinks-track" => true %> 6 | <%= javascript_include_tag "application", "data-turbolinks-track" => true %> 7 | <%= csrf_meta_tags %> 8 | 9 | 10 | 11 | <%= yield %> 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /test/dummy/bin/bundle: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__) 3 | load Gem.bin_path('bundler', 'bundle') 4 | -------------------------------------------------------------------------------- /test/dummy/bin/rails: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | APP_PATH = File.expand_path('../../config/application', __FILE__) 3 | require_relative '../config/boot' 4 | require 'rails/commands' 5 | -------------------------------------------------------------------------------- /test/dummy/bin/rake: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | require_relative '../config/boot' 3 | require 'rake' 4 | Rake.application.run 5 | -------------------------------------------------------------------------------- /test/dummy/config.ru: -------------------------------------------------------------------------------- 1 | # This file is used by Rack-based servers to start the application. 2 | 3 | require ::File.expand_path('../config/environment', __FILE__) 4 | run Rails.application 5 | -------------------------------------------------------------------------------- /test/dummy/config/application.rb: -------------------------------------------------------------------------------- 1 | require File.expand_path('../boot', __FILE__) 2 | 3 | require 'rails/all' 4 | 5 | Bundler.require(*Rails.groups) 6 | require "clean_pagination" 7 | 8 | module Dummy 9 | class Application < Rails::Application 10 | # Settings in config/environments/* take precedence over those specified here. 11 | # Application configuration should go into files in config/initializers 12 | # -- all .rb files in that directory are automatically loaded. 13 | 14 | # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone. 15 | # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC. 16 | # config.time_zone = 'Central Time (US & Canada)' 17 | 18 | # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded. 19 | # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s] 20 | # config.i18n.default_locale = :de 21 | end 22 | end 23 | 24 | -------------------------------------------------------------------------------- /test/dummy/config/boot.rb: -------------------------------------------------------------------------------- 1 | # Set up gems listed in the Gemfile. 2 | ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../../../Gemfile', __FILE__) 3 | 4 | require 'bundler/setup' if File.exist?(ENV['BUNDLE_GEMFILE']) 5 | $LOAD_PATH.unshift File.expand_path('../../../../lib', __FILE__) 6 | -------------------------------------------------------------------------------- /test/dummy/config/database.yml: -------------------------------------------------------------------------------- 1 | # SQLite version 3.x 2 | # gem install sqlite3 3 | # 4 | # Ensure the SQLite 3 gem is defined in your Gemfile 5 | # gem 'sqlite3' 6 | # 7 | default: &default 8 | adapter: sqlite3 9 | pool: 5 10 | timeout: 5000 11 | 12 | development: 13 | <<: *default 14 | database: db/development.sqlite3 15 | 16 | # Warning: The database defined as "test" will be erased and 17 | # re-generated from your development database when you run "rake". 18 | # Do not set this db to the same as development or production. 19 | test: 20 | <<: *default 21 | database: db/test.sqlite3 22 | 23 | production: 24 | <<: *default 25 | database: db/production.sqlite3 26 | -------------------------------------------------------------------------------- /test/dummy/config/environment.rb: -------------------------------------------------------------------------------- 1 | # Load the Rails application. 2 | require File.expand_path('../application', __FILE__) 3 | 4 | # Initialize the Rails application. 5 | Rails.application.initialize! 6 | -------------------------------------------------------------------------------- /test/dummy/config/environments/development.rb: -------------------------------------------------------------------------------- 1 | Rails.application.configure do 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # In the development environment your application's code is reloaded on 5 | # every request. This slows down response time but is perfect for development 6 | # since you don't have to restart the web server when you make code changes. 7 | config.cache_classes = false 8 | 9 | # Do not eager load code on boot. 10 | config.eager_load = false 11 | 12 | # Show full error reports and disable caching. 13 | config.consider_all_requests_local = true 14 | config.action_controller.perform_caching = false 15 | 16 | # Don't care if the mailer can't send. 17 | config.action_mailer.raise_delivery_errors = false 18 | 19 | # Print deprecation notices to the Rails logger. 20 | config.active_support.deprecation = :log 21 | 22 | # Raise an error on page load if there are pending migrations. 23 | config.active_record.migration_error = :page_load 24 | 25 | # Debug mode disables concatenation and preprocessing of assets. 26 | # This option may cause significant delays in view rendering with a large 27 | # number of complex assets. 28 | config.assets.debug = true 29 | end 30 | -------------------------------------------------------------------------------- /test/dummy/config/environments/production.rb: -------------------------------------------------------------------------------- 1 | Rails.application.configure do 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # Code is not reloaded between requests. 5 | config.cache_classes = true 6 | 7 | # Eager load code on boot. This eager loads most of Rails and 8 | # your application in memory, allowing both thread web servers 9 | # and those relying on copy on write to perform better. 10 | # Rake tasks automatically ignore this option for performance. 11 | config.eager_load = true 12 | 13 | # Full error reports are disabled and caching is turned on. 14 | config.consider_all_requests_local = false 15 | config.action_controller.perform_caching = true 16 | 17 | # Enable Rack::Cache to put a simple HTTP cache in front of your application 18 | # Add `rack-cache` to your Gemfile before enabling this. 19 | # For large-scale production use, consider using a caching reverse proxy like nginx, varnish or squid. 20 | # config.action_dispatch.rack_cache = true 21 | 22 | # Disable Rails's static asset server (Apache or nginx will already do this). 23 | config.serve_static_assets = false 24 | 25 | # Compress JavaScripts and CSS. 26 | config.assets.js_compressor = :uglifier 27 | # config.assets.css_compressor = :sass 28 | 29 | # Do not fallback to assets pipeline if a precompiled asset is missed. 30 | config.assets.compile = false 31 | 32 | # Generate digests for assets URLs. 33 | config.assets.digest = true 34 | 35 | # Version of your assets, change this if you want to expire all your assets. 36 | config.assets.version = '1.0' 37 | 38 | # Specifies the header that your server uses for sending files. 39 | # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for apache 40 | # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for nginx 41 | 42 | # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies. 43 | # config.force_ssl = true 44 | 45 | # Set to :debug to see everything in the log. 46 | config.log_level = :info 47 | 48 | # Prepend all log lines with the following tags. 49 | # config.log_tags = [ :subdomain, :uuid ] 50 | 51 | # Use a different logger for distributed setups. 52 | # config.logger = ActiveSupport::TaggedLogging.new(SyslogLogger.new) 53 | 54 | # Use a different cache store in production. 55 | # config.cache_store = :mem_cache_store 56 | 57 | # Enable serving of images, stylesheets, and JavaScripts from an asset server. 58 | # config.action_controller.asset_host = "http://assets.example.com" 59 | 60 | # Precompile additional assets. 61 | # application.js, application.css, and all non-JS/CSS in app/assets folder are already added. 62 | # config.assets.precompile += %w( search.js ) 63 | 64 | # Ignore bad email addresses and do not raise email delivery errors. 65 | # Set this to true and configure the email server for immediate delivery to raise delivery errors. 66 | # config.action_mailer.raise_delivery_errors = false 67 | 68 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to 69 | # the I18n.default_locale when a translation can not be found). 70 | config.i18n.fallbacks = true 71 | 72 | # Send deprecation notices to registered listeners. 73 | config.active_support.deprecation = :notify 74 | 75 | # Disable automatic flushing of the log to improve performance. 76 | # config.autoflush_log = false 77 | 78 | # Use default logging formatter so that PID and timestamp are not suppressed. 79 | config.log_formatter = ::Logger::Formatter.new 80 | end 81 | -------------------------------------------------------------------------------- /test/dummy/config/environments/test.rb: -------------------------------------------------------------------------------- 1 | Rails.application.config do |config| 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # The test environment is used exclusively to run your application's 5 | # test suite. You never need to work with it otherwise. Remember that 6 | # your test database is "scratch space" for the test suite and is wiped 7 | # and recreated between test runs. Don't rely on the data there! 8 | config.cache_classes = true 9 | 10 | # Do not eager load code on boot. This avoids loading your whole application 11 | # just for the purpose of running a single test. If you are using a tool that 12 | # preloads Rails for running tests, you may have to set it to true. 13 | config.eager_load = false 14 | 15 | # Configure static asset server for tests with Cache-Control for performance. 16 | config.serve_static_assets = true 17 | config.static_cache_control = "public, max-age=3600" 18 | 19 | # Show full error reports and disable caching. 20 | config.consider_all_requests_local = true 21 | config.action_controller.perform_caching = false 22 | 23 | # Raise exceptions instead of rendering exception templates. 24 | config.action_dispatch.show_exceptions = false 25 | 26 | # Disable request forgery protection in test environment. 27 | config.action_controller.allow_forgery_protection = false 28 | 29 | # Tell Action Mailer not to deliver emails to the real world. 30 | # The :test delivery method accumulates sent emails in the 31 | # ActionMailer::Base.deliveries array. 32 | config.action_mailer.delivery_method = :test 33 | 34 | # Print deprecation notices to the stderr. 35 | config.active_support.deprecation = :stderr 36 | end 37 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/backtrace_silencers.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces. 4 | # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ } 5 | 6 | # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code. 7 | # Rails.backtrace_cleaner.remove_silencers! 8 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/filter_parameter_logging.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Configure sensitive parameters which will be filtered from the log file. 4 | Rails.application.config.filter_parameters += [:password] 5 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/inflections.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Add new inflection rules using the following format. Inflections 4 | # are locale specific, and you may define rules for as many different 5 | # locales as you wish. All of these examples are active by default: 6 | # ActiveSupport::Inflector.inflections(:en) do |inflect| 7 | # inflect.plural /^(ox)$/i, '\1en' 8 | # inflect.singular /^(ox)en/i, '\1' 9 | # inflect.irregular 'person', 'people' 10 | # inflect.uncountable %w( fish sheep ) 11 | # end 12 | 13 | # These inflection rules are supported but not enabled by default: 14 | # ActiveSupport::Inflector.inflections(:en) do |inflect| 15 | # inflect.acronym 'RESTful' 16 | # end 17 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/mime_types.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Add new mime types for use in respond_to blocks: 4 | # Mime::Type.register "text/richtext", :rtf 5 | # Mime::Type.register_alias "text/html", :iphone 6 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/secret_token.rb: -------------------------------------------------------------------------------- 1 | Rails.application.config.secret_token = 'deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef' 2 | Rails.application.config.secret_key_base = 'deadbeef' if Rails.application.config.respond_to?(:secret_key_base) 3 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/session_store.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | Rails.application.config.session_store :cookie_store, key: '_dummy_session' 4 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/wrap_parameters.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # This file contains settings for ActionController::ParamsWrapper which 4 | # is enabled by default. 5 | 6 | # Enable parameter wrapping for JSON. You can disable this by setting :format to an empty array. 7 | ActiveSupport.on_load(:action_controller) do 8 | wrap_parameters format: [:json] if respond_to?(:wrap_parameters) 9 | end 10 | 11 | # To enable root element in JSON for ActiveRecord objects. 12 | # ActiveSupport.on_load(:active_record) do 13 | # self.include_root_in_json = true 14 | # end 15 | -------------------------------------------------------------------------------- /test/dummy/config/locales/en.yml: -------------------------------------------------------------------------------- 1 | # Files in the config/locales directory are used for internationalization 2 | # and are automatically loaded by Rails. If you want to use locales other 3 | # than English, add the necessary files in this directory. 4 | # 5 | # To use the locales, use `I18n.t`: 6 | # 7 | # I18n.t 'hello' 8 | # 9 | # In views, this is aliased to just `t`: 10 | # 11 | # <%= t('hello') %> 12 | # 13 | # To use a different locale, set it with `I18n.locale`: 14 | # 15 | # I18n.locale = :es 16 | # 17 | # This would use the information in config/locales/es.yml. 18 | # 19 | # To learn more, please read the Rails Internationalization guide 20 | # available at http://guides.rubyonrails.org/i18n.html. 21 | 22 | en: 23 | hello: "Hello world" 24 | -------------------------------------------------------------------------------- /test/dummy/config/routes.rb: -------------------------------------------------------------------------------- 1 | Rails.application.routes.draw do 2 | # The priority is based upon order of creation: first created -> highest priority. 3 | # See how all your routes lay out with "rake routes". 4 | 5 | # You can have the root of your site routed with "root" 6 | root 'application#index' 7 | 8 | # Example of regular route: 9 | # get 'products/:id' => 'catalog#view' 10 | 11 | # Example of named route that can be invoked with purchase_url(id: product.id) 12 | # get 'products/:id/purchase' => 'catalog#purchase', as: :purchase 13 | 14 | # Example resource route (maps HTTP verbs to controller actions automatically): 15 | # resources :products 16 | 17 | # Example resource route with options: 18 | # resources :products do 19 | # member do 20 | # get 'short' 21 | # post 'toggle' 22 | # end 23 | # 24 | # collection do 25 | # get 'sold' 26 | # end 27 | # end 28 | 29 | # Example resource route with sub-resources: 30 | # resources :products do 31 | # resources :comments, :sales 32 | # resource :seller 33 | # end 34 | 35 | # Example resource route with more complex sub-resources: 36 | # resources :products do 37 | # resources :comments 38 | # resources :sales do 39 | # get 'recent', on: :collection 40 | # end 41 | # end 42 | 43 | # Example resource route with concerns: 44 | # concern :toggleable do 45 | # post 'toggle' 46 | # end 47 | # resources :posts, concerns: :toggleable 48 | # resources :photos, concerns: :toggleable 49 | 50 | # Example resource route within a namespace: 51 | # namespace :admin do 52 | # # Directs /admin/products/* to Admin::ProductsController 53 | # # (app/controllers/admin/products_controller.rb) 54 | # resources :products 55 | # end 56 | end 57 | -------------------------------------------------------------------------------- /test/dummy/config/secrets.yml: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Your secret key is used for verifying the integrity of signed cookies. 4 | # If you change this key, all old signed cookies will become invalid! 5 | 6 | # Make sure the secret is at least 30 characters and all random, 7 | # no regular words or you'll be exposed to dictionary attacks. 8 | # You can use `rake secret` to generate a secure secret key. 9 | 10 | # Make sure the secrets in this file are kept private 11 | # if you're sharing your code publicly. 12 | 13 | development: 14 | secret_key_base: 4b880c6ba2c6f3aac0c087b76b8f430043d50b64e2ee7814d96e4c9244687a0f18e11e9a5f39a6ea2eadc6646c8f3602c29a7012b32bf726b3edee2c6e511a8f 15 | 16 | test: 17 | secret_key_base: 0edce341206491f05dac056c58f7188661b6a004bdee0c1833de6599e402ea807742e05bcb5bb29488d336f150c8e79b64127ab215b1fe437d94ac7db9e7bb7d 18 | 19 | production: 20 | secret_key_base: 15ebd1686c467852264569f55ec1101e5f5b78e67135d7bdedc1759aa58ada70a866e67e475e96e34d5546401e743f290b409e005ffe2d6d7f5dbce2f8ca9534 21 | -------------------------------------------------------------------------------- /test/dummy/lib/assets/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/lib/assets/.keep -------------------------------------------------------------------------------- /test/dummy/log/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/log/.keep -------------------------------------------------------------------------------- /test/dummy/public/404.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The page you were looking for doesn't exist (404) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

The page you were looking for doesn't exist.

62 |

You may have mistyped the address or the page may have moved.

63 |
64 |

If you are the application owner check the logs for more information.

65 |
66 | 67 | 68 | -------------------------------------------------------------------------------- /test/dummy/public/422.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The change you wanted was rejected (422) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

The change you wanted was rejected.

62 |

Maybe you tried to change something you didn't have access to.

63 |
64 |

If you are the application owner check the logs for more information.

65 |
66 | 67 | 68 | -------------------------------------------------------------------------------- /test/dummy/public/500.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | We're sorry, but something went wrong (500) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

We're sorry, but something went wrong.

62 |
63 |

If you are the application owner check the logs for more information.

64 |
65 | 66 | 67 | -------------------------------------------------------------------------------- /test/dummy/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/begriffs/clean_pagination/2b45eafde5a409aabc02affdb37396b528550c92/test/dummy/public/favicon.ico -------------------------------------------------------------------------------- /test/test_helper.rb: -------------------------------------------------------------------------------- 1 | # Configure Rails Environment 2 | ENV["RAILS_ENV"] = "test" 3 | 4 | require File.expand_path("../dummy/config/environment.rb", __FILE__) 5 | require "rails/test_help" 6 | 7 | Rails.backtrace_cleaner.remove_silencers! 8 | 9 | # Load support files 10 | Dir["#{File.dirname(__FILE__)}/support/**/*.rb"].each { |f| require f } 11 | 12 | # Load fixtures from the engine 13 | if ActiveSupport::TestCase.method_defined?(:fixture_path=) 14 | ActiveSupport::TestCase.fixture_path = File.expand_path("../fixtures", __FILE__) 15 | end 16 | --------------------------------------------------------------------------------