rack-reducer 1.0.1 → 1.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (54) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +112 -252
  3. data/lib/rack/reducer/middleware.rb +17 -2
  4. data/lib/rack/reducer/reduction.rb +19 -16
  5. data/lib/rack/reducer/refinements.rb +13 -1
  6. data/lib/rack/reducer/version.rb +3 -1
  7. data/lib/rack/reducer/warnings.rb +27 -0
  8. data/lib/rack/reducer.rb +51 -20
  9. data/spec/benchmarks.rb +51 -21
  10. data/spec/fixtures.rb +30 -0
  11. data/spec/middleware_spec.rb +55 -23
  12. data/spec/rails_spec.rb +33 -3
  13. data/spec/reducer_spec.rb +104 -0
  14. data/spec/spec_helper.rb +6 -15
  15. metadata +34 -136
  16. data/lib/rack/reducer/parser.rb +0 -26
  17. data/spec/_hanami_example/apps/web/application.rb +0 -326
  18. data/spec/_hanami_example/apps/web/config/routes.rb +0 -4
  19. data/spec/_hanami_example/apps/web/controllers/artists/index.rb +0 -12
  20. data/spec/_hanami_example/apps/web/views/application_layout.rb +0 -7
  21. data/spec/_hanami_example/config/boot.rb +0 -2
  22. data/spec/_hanami_example/config/environment.rb +0 -29
  23. data/spec/_hanami_example/lib/hanami_example/entities/artist.rb +0 -2
  24. data/spec/_hanami_example/lib/hanami_example/repositories/artist_repository.rb +0 -9
  25. data/spec/_hanami_example/lib/hanami_example.rb +0 -5
  26. data/spec/_rails_example/app/channels/application_cable/channel.rb +0 -4
  27. data/spec/_rails_example/app/channels/application_cable/connection.rb +0 -4
  28. data/spec/_rails_example/app/controllers/application_controller.rb +0 -2
  29. data/spec/_rails_example/app/controllers/artists_controller.rb +0 -8
  30. data/spec/_rails_example/app/jobs/application_job.rb +0 -2
  31. data/spec/_rails_example/app/mailers/application_mailer.rb +0 -4
  32. data/spec/_rails_example/app/models/application_record.rb +0 -3
  33. data/spec/_rails_example/app/models/rails_example/artist.rb +0 -21
  34. data/spec/_rails_example/config/application.rb +0 -35
  35. data/spec/_rails_example/config/boot.rb +0 -3
  36. data/spec/_rails_example/config/environment.rb +0 -5
  37. data/spec/_rails_example/config/environments/development.rb +0 -47
  38. data/spec/_rails_example/config/environments/production.rb +0 -83
  39. data/spec/_rails_example/config/environments/test.rb +0 -42
  40. data/spec/_rails_example/config/initializers/application_controller_renderer.rb +0 -8
  41. data/spec/_rails_example/config/initializers/backtrace_silencers.rb +0 -7
  42. data/spec/_rails_example/config/initializers/cors.rb +0 -16
  43. data/spec/_rails_example/config/initializers/filter_parameter_logging.rb +0 -4
  44. data/spec/_rails_example/config/initializers/inflections.rb +0 -16
  45. data/spec/_rails_example/config/initializers/mime_types.rb +0 -4
  46. data/spec/_rails_example/config/initializers/wrap_parameters.rb +0 -14
  47. data/spec/_rails_example/config/puma.rb +0 -56
  48. data/spec/_rails_example/config/routes.rb +0 -4
  49. data/spec/_rails_example/db/seeds.rb +0 -7
  50. data/spec/behavior.rb +0 -51
  51. data/spec/hanami_spec.rb +0 -6
  52. data/spec/roda_spec.rb +0 -13
  53. data/spec/sinatra_functional_spec.rb +0 -26
  54. data/spec/sinatra_mixin_spec.rb +0 -20
@@ -1,29 +0,0 @@
1
- require 'bundler/setup'
2
- require 'hanami/setup'
3
- require 'hanami/model'
4
- require_relative '../lib/hanami_example'
5
- require_relative '../apps/web/application'
6
-
7
- Hanami.configure do
8
- mount Web::Application, at: '/'
9
-
10
- model do
11
- ##
12
- # Database adapter
13
- #
14
- # Available options:
15
- #
16
- # * SQL adapter
17
- # adapter :sql, 'sqlite://db/hanami_example_development.sqlite3'
18
- # adapter :sql, 'postgresql://localhost/hanami_example_development'
19
- # adapter :sql, 'mysql://localhost/hanami_example_development'
20
- #
21
- adapter :sql, "sqlite://#{__dir__}/../../fixtures.sqlite"
22
-
23
- ##
24
- # Migrations
25
- #
26
- migrations 'db/migrations'
27
- schema 'db/schema.sql'
28
- end
29
- end
@@ -1,2 +0,0 @@
1
- class Artist < Hanami::Entity
2
- end
@@ -1,9 +0,0 @@
1
- class ArtistRepository < Hanami::Repository
2
- def query
3
- { dataset: artists.dataset, filters: SEQUEL_QUERY[:filters] }
4
- end
5
-
6
- def reduce(params)
7
- Rack::Reducer.call(params, query)
8
- end
9
- end
@@ -1,5 +0,0 @@
1
- require_relative 'hanami_example/entities/artist'
2
- require_relative 'hanami_example/repositories/artist_repository'
3
-
4
- module HanamiExample
5
- end
@@ -1,4 +0,0 @@
1
- module ApplicationCable
2
- class Channel < ActionCable::Channel::Base
3
- end
4
- end
@@ -1,4 +0,0 @@
1
- module ApplicationCable
2
- class Connection < ActionCable::Connection::Base
3
- end
4
- end
@@ -1,2 +0,0 @@
1
- class ApplicationController < ActionController::API
2
- end
@@ -1,8 +0,0 @@
1
- class ArtistsController < ApplicationController
2
- # GET /artists
3
- def index
4
- @artists = RailsExample::Artist.reduce(params)
5
-
6
- render json: @artists
7
- end
8
- end
@@ -1,2 +0,0 @@
1
- class ApplicationJob < ActiveJob::Base
2
- end
@@ -1,4 +0,0 @@
1
- class ApplicationMailer < ActionMailer::Base
2
- default from: 'from@example.com'
3
- layout 'mailer'
4
- end
@@ -1,3 +0,0 @@
1
- class ApplicationRecord < ActiveRecord::Base
2
- self.abstract_class = true
3
- end
@@ -1,21 +0,0 @@
1
- module RailsExample
2
- class Artist < ApplicationRecord
3
- scope :by_name, lambda { |name|
4
- where('lower(name) like ?', "%#{name.downcase}%")
5
- }
6
- def self.search_genre(genre)
7
- where('lower(genre) like ?', "%#{genre.downcase}%")
8
- end
9
-
10
- extend Rack::Reducer
11
- reduces all, filters: [
12
- # filters can call class methods...
13
- ->(genre:) { search_genre(genre) },
14
- # or scopes...
15
- ->(name:) { by_name(name) },
16
- # or inline ActiveRecord queries
17
- ->(order:) { order(order.to_sym) },
18
- ->(releases: ) { where(release_count: releases.to_i) },
19
- ]
20
- end
21
- end
@@ -1,35 +0,0 @@
1
- require_relative 'boot'
2
-
3
- require "rails"
4
- # Pick the frameworks you want:
5
- require "active_model/railtie"
6
- #require "active_job/railtie"
7
- require "active_record/railtie"
8
- require "action_controller/railtie"
9
- require "action_mailer/railtie"
10
- require "action_view/railtie"
11
- #require "action_cable/engine"
12
- # require "sprockets/railtie"
13
- #require "rails/test_unit/railtie"
14
-
15
- # Require the gems listed in Gemfile, including any gems
16
- # you've limited to :test, :development, or :production.
17
- Bundler.require(*Rails.groups)
18
-
19
- module ExRails
20
- class Application < Rails::Application
21
- # Initialize configuration defaults for originally generated Rails version.
22
- config.load_defaults 5.1
23
-
24
-
25
- # Settings in config/environments/* take precedence over those specified here.
26
- # Application configuration should go into files in config/initializers
27
- # -- all .rb files in that directory are automatically loaded.
28
-
29
- # Only loads a smaller set of middleware suitable for API only apps.
30
- # Middleware like session, flash, cookies can be added back manually.
31
- # Skip views, helpers and assets when generating a new resource.
32
- config.api_only = true
33
-
34
- end
35
- end
@@ -1,3 +0,0 @@
1
- ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../Gemfile', __dir__)
2
-
3
- require 'bundler/setup' # Set up gems listed in the Gemfile.
@@ -1,5 +0,0 @@
1
- # Load the Rails application.
2
- require_relative 'application'
3
-
4
- # Initialize the Rails application.
5
- Rails.application.initialize!
@@ -1,47 +0,0 @@
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.
13
- config.consider_all_requests_local = true
14
-
15
- # Enable/disable caching. By default caching is disabled.
16
- if Rails.root.join('tmp/caching-dev.txt').exist?
17
- config.action_controller.perform_caching = true
18
-
19
- config.cache_store = :memory_store
20
- config.public_file_server.headers = {
21
- 'Cache-Control' => "public, max-age=#{2.days.seconds.to_i}"
22
- }
23
- else
24
- config.action_controller.perform_caching = false
25
-
26
- config.cache_store = :null_store
27
- end
28
-
29
- # Don't care if the mailer can't send.
30
- config.action_mailer.raise_delivery_errors = false
31
-
32
- config.action_mailer.perform_caching = false
33
-
34
- # Print deprecation notices to the Rails logger.
35
- config.active_support.deprecation = :log
36
-
37
- # Raise an error on page load if there are pending migrations.
38
- config.active_record.migration_error = :page_load
39
-
40
-
41
- # Raises error for missing translations
42
- # config.action_view.raise_on_missing_translations = true
43
-
44
- # Use an evented file watcher to asynchronously detect changes in source code,
45
- # routes, locales, etc. This feature depends on the listen gem.
46
- config.file_watcher = ActiveSupport::EventedFileUpdateChecker
47
- end
@@ -1,83 +0,0 @@
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 threaded 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
- # Attempt to read encrypted secrets from `config/secrets.yml.enc`.
18
- # Requires an encryption key in `ENV["RAILS_MASTER_KEY"]` or
19
- # `config/secrets.yml.key`.
20
- config.read_encrypted_secrets = true
21
-
22
- # Disable serving static files from the `/public` folder by default since
23
- # Apache or NGINX already handles this.
24
- config.public_file_server.enabled = ENV['RAILS_SERVE_STATIC_FILES'].present?
25
-
26
-
27
- # Enable serving of images, stylesheets, and JavaScripts from an asset server.
28
- # config.action_controller.asset_host = 'http://assets.example.com'
29
-
30
- # Specifies the header that your server uses for sending files.
31
- # config.action_dispatch.x_sendfile_header = 'X-Sendfile' # for Apache
32
- # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for NGINX
33
-
34
- # Mount Action Cable outside main process or domain
35
- # config.action_cable.mount_path = nil
36
- # config.action_cable.url = 'wss://example.com/cable'
37
- # config.action_cable.allowed_request_origins = [ 'http://example.com', /http:\/\/example.*/ ]
38
-
39
- # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
40
- # config.force_ssl = true
41
-
42
- # Use the lowest log level to ensure availability of diagnostic information
43
- # when problems arise.
44
- config.log_level = :debug
45
-
46
- # Prepend all log lines with the following tags.
47
- config.log_tags = [ :request_id ]
48
-
49
- # Use a different cache store in production.
50
- # config.cache_store = :mem_cache_store
51
-
52
- # Use a real queuing backend for Active Job (and separate queues per environment)
53
- # config.active_job.queue_adapter = :resque
54
- # config.active_job.queue_name_prefix = "ex_rails_#{Rails.env}"
55
- config.action_mailer.perform_caching = false
56
-
57
- # Ignore bad email addresses and do not raise email delivery errors.
58
- # Set this to true and configure the email server for immediate delivery to raise delivery errors.
59
- # config.action_mailer.raise_delivery_errors = false
60
-
61
- # Enable locale fallbacks for I18n (makes lookups for any locale fall back to
62
- # the I18n.default_locale when a translation cannot be found).
63
- config.i18n.fallbacks = true
64
-
65
- # Send deprecation notices to registered listeners.
66
- config.active_support.deprecation = :notify
67
-
68
- # Use default logging formatter so that PID and timestamp are not suppressed.
69
- config.log_formatter = ::Logger::Formatter.new
70
-
71
- # Use a different logger for distributed setups.
72
- # require 'syslog/logger'
73
- # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new 'app-name')
74
-
75
- if ENV["RAILS_LOG_TO_STDOUT"].present?
76
- logger = ActiveSupport::Logger.new(STDOUT)
77
- logger.formatter = config.log_formatter
78
- config.logger = ActiveSupport::TaggedLogging.new(logger)
79
- end
80
-
81
- # Do not dump schema after migrations.
82
- config.active_record.dump_schema_after_migration = false
83
- end
@@ -1,42 +0,0 @@
1
- Rails.application.configure do
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 public file server for tests with Cache-Control for performance.
16
- config.public_file_server.enabled = true
17
- config.public_file_server.headers = {
18
- 'Cache-Control' => "public, max-age=#{1.hour.seconds.to_i}"
19
- }
20
-
21
- # Show full error reports and disable caching.
22
- config.consider_all_requests_local = true
23
- config.action_controller.perform_caching = false
24
-
25
- # Raise exceptions instead of rendering exception templates.
26
- config.action_dispatch.show_exceptions = false
27
-
28
- # Disable request forgery protection in test environment.
29
- config.action_controller.allow_forgery_protection = false
30
- config.action_mailer.perform_caching = false
31
-
32
- # Tell Action Mailer not to deliver emails to the real world.
33
- # The :test delivery method accumulates sent emails in the
34
- # ActionMailer::Base.deliveries array.
35
- config.action_mailer.delivery_method = :test
36
-
37
- # Print deprecation notices to the stderr.
38
- config.active_support.deprecation = :stderr
39
-
40
- # Raises error for missing translations
41
- # config.action_view.raise_on_missing_translations = true
42
- end
@@ -1,8 +0,0 @@
1
- # Be sure to restart your server when you modify this file.
2
-
3
- # ActiveSupport::Reloader.to_prepare do
4
- # ApplicationController.renderer.defaults.merge!(
5
- # http_host: 'example.org',
6
- # https: false
7
- # )
8
- # end
@@ -1,7 +0,0 @@
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!
@@ -1,16 +0,0 @@
1
- # Be sure to restart your server when you modify this file.
2
-
3
- # Avoid CORS issues when API is called from the frontend app.
4
- # Handle Cross-Origin Resource Sharing (CORS) in order to accept cross-origin AJAX requests.
5
-
6
- # Read more: https://github.com/cyu/rack-cors
7
-
8
- # Rails.application.config.middleware.insert_before 0, Rack::Cors do
9
- # allow do
10
- # origins 'example.com'
11
- #
12
- # resource '*',
13
- # headers: :any,
14
- # methods: [:get, :post, :put, :patch, :delete, :options, :head]
15
- # end
16
- # end
@@ -1,4 +0,0 @@
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]
@@ -1,16 +0,0 @@
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
@@ -1,4 +0,0 @@
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
@@ -1,14 +0,0 @@
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]
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
@@ -1,56 +0,0 @@
1
- # Puma can serve each request in a thread from an internal thread pool.
2
- # The `threads` method setting takes two numbers: a minimum and maximum.
3
- # Any libraries that use thread pools should be configured to match
4
- # the maximum value specified for Puma. Default is set to 5 threads for minimum
5
- # and maximum; this matches the default thread size of Active Record.
6
- #
7
- threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 }
8
- threads threads_count, threads_count
9
-
10
- # Specifies the `port` that Puma will listen on to receive requests; default is 3000.
11
- #
12
- port ENV.fetch("PORT") { 3000 }
13
-
14
- # Specifies the `environment` that Puma will run in.
15
- #
16
- environment ENV.fetch("RAILS_ENV") { "development" }
17
-
18
- # Specifies the number of `workers` to boot in clustered mode.
19
- # Workers are forked webserver processes. If using threads and workers together
20
- # the concurrency of the application would be max `threads` * `workers`.
21
- # Workers do not work on JRuby or Windows (both of which do not support
22
- # processes).
23
- #
24
- # workers ENV.fetch("WEB_CONCURRENCY") { 2 }
25
-
26
- # Use the `preload_app!` method when specifying a `workers` number.
27
- # This directive tells Puma to first boot the application and load code
28
- # before forking the application. This takes advantage of Copy On Write
29
- # process behavior so workers use less memory. If you use this option
30
- # you need to make sure to reconnect any threads in the `on_worker_boot`
31
- # block.
32
- #
33
- # preload_app!
34
-
35
- # If you are preloading your application and using Active Record, it's
36
- # recommended that you close any connections to the database before workers
37
- # are forked to prevent connection leakage.
38
- #
39
- # before_fork do
40
- # ActiveRecord::Base.connection_pool.disconnect! if defined?(ActiveRecord)
41
- # end
42
-
43
- # The code in the `on_worker_boot` will be called if you are using
44
- # clustered mode by specifying a number of `workers`. After each worker
45
- # process is booted, this block will be run. If you are using the `preload_app!`
46
- # option, you will want to use this block to reconnect to any threads
47
- # or connections that may have been created at application boot, as Ruby
48
- # cannot share connections between processes.
49
- #
50
- # on_worker_boot do
51
- # ActiveRecord::Base.establish_connection if defined?(ActiveRecord)
52
- # end
53
- #
54
-
55
- # Allow puma to be restarted by `rails restart` command.
56
- plugin :tmp_restart
@@ -1,4 +0,0 @@
1
- Rails.application.routes.draw do
2
- resources :artists
3
- # For details on the DSL available within this file, see http://guides.rubyonrails.org/routing.html
4
- end
@@ -1,7 +0,0 @@
1
- # This file should contain all the record creation needed to seed the database with its default values.
2
- # The data can then be loaded with the rails db:seed command (or created alongside the database with db:setup).
3
- #
4
- # Examples:
5
- #
6
- # movies = Movie.create([{ name: 'Star Wars' }, { name: 'Lord of the Rings' }])
7
- # Character.create(name: 'Luke', movie: movies.first)
data/spec/behavior.rb DELETED
@@ -1,51 +0,0 @@
1
- RSpec.shared_examples_for Rack::Reducer do
2
- let(:app) { described_class }
3
-
4
- it 'responds with unfiltered data when filter params are empty' do
5
- get('/artists') do |res|
6
- DB[:artists].each { |artist| expect(res.body).to include(artist[:name]) }
7
- end
8
- end
9
-
10
- it 'filters by a single param, e.g. name' do
11
- get('/artists?name=Blake') do |response|
12
- expect(response.body).to include('Blake Mills')
13
- expect(response.body).to include('James Blake')
14
- expect(response.body).not_to include('SZA')
15
- end
16
- end
17
-
18
- it 'filters by a single param, e.g. genre' do
19
- get('/artists?genre=electronic') do |response|
20
- expect(response.body).to include('Björk')
21
- expect(response.body).to include('James Blake')
22
- expect(response.body).not_to include('Blake Mills')
23
- end
24
-
25
- get '/artists?genre=soul' do |response|
26
- expect(response.body).to include('Janelle Monae')
27
- expect(response.body).not_to include('Björk')
28
- end
29
- end
30
-
31
- it 'chains multiple filters' do
32
- get('/artists?genre=electronic&name=blake') do |response|
33
- expect(response.body).to include('James Blake')
34
- expect(response.body).not_to include('Blake Mills')
35
- end
36
- end
37
-
38
- it 'handles falsy values' do
39
- get('/artists?releases=0') do |response|
40
- expect(response.body).to include('Chris Frank')
41
- expect(JSON.parse(response.body).length).to eq(1)
42
- end
43
- end
44
-
45
- it 'can sort as well as filter' do
46
- get '/artists?order=genre' do |response|
47
- genre = JSON.parse(response.body)[0]['genre']
48
- expect(genre).to eq('alt-soul')
49
- end
50
- end
51
- end
data/spec/hanami_spec.rb DELETED
@@ -1,6 +0,0 @@
1
- require 'spec_helper'
2
- require_relative '_hanami_example/config/boot'
3
-
4
- describe Hanami.app do
5
- it_behaves_like Rack::Reducer
6
- end
data/spec/roda_spec.rb DELETED
@@ -1,13 +0,0 @@
1
- require 'spec_helper'
2
- require 'roda'
3
-
4
- class RodaTest < Roda
5
- plugin :json
6
- route do |r|
7
- r.get('artists') { Rack::Reducer.call(r.params, SEQUEL_QUERY).to_a }
8
- end
9
- end
10
-
11
- describe RodaTest do
12
- it_behaves_like Rack::Reducer
13
- end
@@ -1,26 +0,0 @@
1
- require 'spec_helper'
2
- require 'sinatra/base'
3
- require 'json'
4
-
5
- class SinatraFunctional < Sinatra::Base
6
- class Artist < Sequel::Model
7
- plugin :json_serializer
8
- end
9
-
10
- get '/artists' do
11
- @artists = Rack::Reducer.call(params, SEQUEL_QUERY)
12
- @artists.to_a.to_json
13
- end
14
- end
15
-
16
- describe SinatraFunctional do
17
- let(:app) { described_class }
18
- it_behaves_like Rack::Reducer
19
-
20
- it 'applies a default order' do
21
- get '/artists' do |response|
22
- genre = JSON.parse(response.body)[0]['genre']
23
- expect(genre).to eq('alt-soul')
24
- end
25
- end
26
- end
@@ -1,20 +0,0 @@
1
- require 'spec_helper'
2
- require 'sinatra/base'
3
- require 'json'
4
-
5
- class SinatraMixin < Sinatra::Base
6
- class Artist < Sequel::Model
7
- plugin :json_serializer
8
- extend Rack::Reducer
9
- reduces dataset, filters: SEQUEL_QUERY[:filters]
10
- end
11
-
12
- get '/artists' do
13
- @artists = Artist.reduce(params)
14
- @artists.all.to_json
15
- end
16
- end
17
-
18
- describe SinatraMixin do
19
- it_behaves_like Rack::Reducer
20
- end