filterameter 0.10.0 → 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +106 -56
- data/lib/filterameter/configuration.rb +18 -14
- data/lib/filterameter/declaration_errors/cannot_be_inline_scope_error.rb +1 -1
- data/lib/filterameter/declaration_errors/filter_scope_argument_error.rb +1 -1
- data/lib/filterameter/declaration_errors/no_such_attribute_error.rb +1 -1
- data/lib/filterameter/declaration_errors/not_a_scope_error.rb +1 -1
- data/lib/filterameter/declaration_errors/sort_scope_requires_one_argument_error.rb +1 -1
- data/lib/filterameter/declaration_errors/unexpected_error.rb +1 -1
- data/lib/filterameter/declaration_errors.rb +1 -1
- data/lib/filterameter/declarations_validator.rb +5 -4
- data/lib/filterameter/declarative_filters.rb +133 -3
- data/lib/filterameter/errors.rb +1 -1
- data/lib/filterameter/exceptions/cannot_determine_model_error.rb +1 -1
- data/lib/filterameter/exceptions/collection_association_sort_error.rb +3 -3
- data/lib/filterameter/exceptions/invalid_association_declaration_error.rb +1 -1
- data/lib/filterameter/exceptions/undeclared_parameter_error.rb +1 -1
- data/lib/filterameter/exceptions/validation_error.rb +1 -1
- data/lib/filterameter/filter_coordinator.rb +1 -1
- data/lib/filterameter/filter_declaration.rb +1 -1
- data/lib/filterameter/filter_factory.rb +1 -1
- data/lib/filterameter/filters/arel_filter.rb +3 -2
- data/lib/filterameter/filters/attribute_filter.rb +1 -1
- data/lib/filterameter/filters/attribute_validator.rb +1 -1
- data/lib/filterameter/filters/conditional_scope_filter.rb +1 -1
- data/lib/filterameter/filters/matches_filter.rb +1 -1
- data/lib/filterameter/filters/maximum_filter.rb +1 -1
- data/lib/filterameter/filters/minimum_filter.rb +1 -1
- data/lib/filterameter/filters/nested_collection_filter.rb +1 -1
- data/lib/filterameter/filters/nested_filter.rb +1 -1
- data/lib/filterameter/filters/scope_filter.rb +1 -1
- data/lib/filterameter/helpers/declaration_with_model.rb +1 -1
- data/lib/filterameter/helpers/joins_values_builder.rb +1 -1
- data/lib/filterameter/helpers/requested_sort.rb +1 -1
- data/lib/filterameter/log_subscriber.rb +1 -1
- data/lib/filterameter/options/partial_options.rb +9 -8
- data/lib/filterameter/parameters_base.rb +1 -1
- data/lib/filterameter/query_builder.rb +10 -8
- data/lib/filterameter/registries/filter_registry.rb +1 -1
- data/lib/filterameter/registries/registry.rb +1 -1
- data/lib/filterameter/registries/sort_registry.rb +1 -1
- data/lib/filterameter/registries/sub_registry.rb +1 -2
- data/lib/filterameter/sort_declaration.rb +1 -1
- data/lib/filterameter/sort_factory.rb +1 -1
- data/lib/filterameter/sorts/attribute_sort.rb +1 -1
- data/lib/filterameter/sorts/scope_sort.rb +1 -1
- data/lib/filterameter/validators/inclusion_validator.rb +3 -3
- data/lib/filterameter/version.rb +1 -1
- data/lib/filterameter.rb +1 -1
- metadata +7 -12
- data/lib/filterameter/filterable.rb +0 -55
- data/lib/filterameter/sortable.rb +0 -40
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a250b39d9c80245166bcfe14a7133cd5ec83f9746989d7e461fa0c121358f57a
|
4
|
+
data.tar.gz: 5d7c64cb79fd73ee94e4e94316b0b5964fac440b14fb0bc58c0c1d805d0f9d60
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d170eb98f80c24173fa92d0251aa68af7d5f3d002d4c3ff473e086aba8d16f07fd05422af6dced018331e883431c93292b4d0b7e5bf3476587b40df8dbeab8d3
|
7
|
+
data.tar.gz: e2342236255b79da8ee96aa7eec0e33c6e9f0131473eaadffd31f833c7212d01d2123eeb6a70cdff8d619ff8678f4d6aafc8d9b52151de9dc6583983f3accd92
|
data/README.md
CHANGED
@@ -4,13 +4,78 @@
|
|
4
4
|
[](https://codeclimate.com/github/RockSolt/filterameter/maintainability)
|
5
5
|
|
6
6
|
# Filterameter
|
7
|
-
|
7
|
+
Filterameter provides declarative filters for Rails controllers to reduce boilerplate code and increase readability. How many times have you seen (or written) this controller action?
|
8
8
|
|
9
|
-
|
10
|
-
|
9
|
+
```ruby
|
10
|
+
def index
|
11
|
+
@films = Films.all
|
12
|
+
@films = @films.where(name: params[:name]) if params[:name]
|
13
|
+
@films = @films.joins(:film_locations).merge(FilmLocations.where(location_id: params[:location_id])) if params[:location_id]
|
14
|
+
@films = @films.directed_by(params[:director_id]) if params[:director_id]
|
15
|
+
@films = @films.written_by(params[:writer_id]) if params[:writer_id]
|
16
|
+
@films = @films.acted_by(params[:actor_id]) if params[:actor_id]
|
17
|
+
end
|
18
|
+
```
|
11
19
|
|
12
|
-
|
20
|
+
It's redundant code and a bit of a pain to write and maintain. Not to mention what RuboCop is going to say about it. Wouldn't it be nice if you could just declare the filters that the controller accepts?
|
13
21
|
|
22
|
+
```ruby
|
23
|
+
filter :name, partial: true
|
24
|
+
filter :location_id, association: :film_locations
|
25
|
+
filter :director_id, name: :directed_by
|
26
|
+
filter :writer_id, name: :written_by
|
27
|
+
filter :actor_id, name: :acted_by
|
28
|
+
|
29
|
+
def index
|
30
|
+
@films = build_query_from_filters
|
31
|
+
end
|
32
|
+
```
|
33
|
+
|
34
|
+
Simplify and speed development of Rails controllers by making filter parameters declarative with Filterameter.
|
35
|
+
|
36
|
+
## Table of Contents
|
37
|
+
- [Getting Started](#getting-started)
|
38
|
+
- [Usage](#usage)
|
39
|
+
- [Filtering Options](#filtering-options)
|
40
|
+
- [Name](#name)
|
41
|
+
- [Association](#association)
|
42
|
+
- [Validates](#validates)
|
43
|
+
- [Partial](#partial)
|
44
|
+
- [Range](#range)
|
45
|
+
- [Sortable](#sortable)
|
46
|
+
- [Scope Filters](#scope-filters)
|
47
|
+
- [Sorting](#sorting)
|
48
|
+
- [Building the Query](#building-the-query)
|
49
|
+
- [Specifying the Model](#specifying-the-model)
|
50
|
+
- [Configuration](#configuration)
|
51
|
+
- [Testing Declarations](#testing-declarations)
|
52
|
+
- [Forms and Query Parameters](#forms-and-query-parameters)
|
53
|
+
- [Contribute](#contribute)
|
54
|
+
- [License](#license)
|
55
|
+
|
56
|
+
## Getting Started
|
57
|
+
|
58
|
+
This gem requires Rails 6.1+, and works with ActiveRecord.
|
59
|
+
|
60
|
+
### Installation
|
61
|
+
Add this line to your application's Gemfile:
|
62
|
+
|
63
|
+
```ruby
|
64
|
+
gem 'filterameter'
|
65
|
+
```
|
66
|
+
|
67
|
+
And then execute:
|
68
|
+
```bash
|
69
|
+
$ bundle install
|
70
|
+
```
|
71
|
+
|
72
|
+
Or install it yourself as:
|
73
|
+
```bash
|
74
|
+
$ gem install filterameter
|
75
|
+
```
|
76
|
+
|
77
|
+
## Usage
|
78
|
+
Include module `Filterameter::DeclarativeFilters` in the controller to provide the filter DSL. It can be included in the `ApplicationController` to make the functionality available to all controllers or it can be mixed in on a case-by-case basis.
|
14
79
|
|
15
80
|
```ruby
|
16
81
|
filter :color
|
@@ -190,18 +255,6 @@ default_sort updated_at: :desc, :description
|
|
190
255
|
|
191
256
|
In order to provide consistent results, a sort is always applied. If no default is specified, it will use primary key descending.
|
192
257
|
|
193
|
-
### Specifying the Model
|
194
|
-
|
195
|
-
Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace, then with the namespace.
|
196
|
-
|
197
|
-
**If the conventions do not provide the correct model**, the model can be named explicitly with the following:
|
198
|
-
|
199
|
-
```ruby
|
200
|
-
filter_model 'Picture'
|
201
|
-
```
|
202
|
-
|
203
|
-
_Important:_ If the `filter_model` declaration is used, it must be before any filter or sort declarations.
|
204
|
-
|
205
258
|
### Building the Query
|
206
259
|
|
207
260
|
There are two ways to apply the filters and build the query, depending on how much control and/or visibility is desired:
|
@@ -273,34 +326,29 @@ This method optionally takes a starting query. If there was a controller for Act
|
|
273
326
|
end
|
274
327
|
```
|
275
328
|
|
276
|
-
|
277
|
-
|
278
|
-
### Query Parameters
|
279
|
-
|
280
|
-
The query parameters are pulled from the controller parameters, nested under the key `filter`. For example a request for large, blue widgets might have the following url:
|
281
|
-
|
282
|
-
`/widgets?filter[size]=large&filter[color]=blue`
|
329
|
+
The starting query is also a good place to provide any includes to enable eager loading:
|
283
330
|
|
284
|
-
|
285
|
-
|
286
|
-
|
287
|
-
|
288
|
-
|
331
|
+
```ruby
|
332
|
+
def index
|
333
|
+
@widgets = build_query_from_filters(Widgets.includes(:manufacturer))
|
334
|
+
end
|
335
|
+
```
|
289
336
|
|
290
|
-
|
337
|
+
Note that the starting query provides the model, so the model is not looked up and the `model_name` declaration in not needed.
|
291
338
|
|
292
|
-
|
339
|
+
### Specifying the Model
|
293
340
|
|
294
|
-
|
341
|
+
Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace, then with the namespace.
|
295
342
|
|
296
|
-
|
297
|
-
- `-` descending
|
343
|
+
**If the conventions do not provide the correct model**, the model can be named explicitly with the following:
|
298
344
|
|
299
|
-
|
345
|
+
```ruby
|
346
|
+
filter_model 'Picture'
|
347
|
+
```
|
300
348
|
|
301
|
-
|
349
|
+
_Important:_ If the `filter_model` declaration is used, it must be before any filter or sort declarations.
|
302
350
|
|
303
|
-
|
351
|
+
## Configuration
|
304
352
|
|
305
353
|
There are three configuration options:
|
306
354
|
|
@@ -340,7 +388,7 @@ If the filter parameters are NOT nested, set this to false. Doing so will restri
|
|
340
388
|
those that have been declared, meaning undeclared parameters are ignored (and the action_on_undeclared_parameters
|
341
389
|
configuration option does not come into play).
|
342
390
|
|
343
|
-
|
391
|
+
## Testing Declarations
|
344
392
|
|
345
393
|
The declarations can be tested for each controller, catching typos, incorrectly defined scopes, or any other issues. Method `declarations_validator` is added to each controller, and a single controller test can be added to validate all the declarations for that controller.
|
346
394
|
|
@@ -357,26 +405,9 @@ validator = WidgetsController.declarations_validator
|
|
357
405
|
assert_predicate validator, :valid?, -> { validator.errors }
|
358
406
|
```
|
359
407
|
|
360
|
-
## Installation
|
361
|
-
Add this line to your application's Gemfile:
|
362
|
-
|
363
|
-
```ruby
|
364
|
-
gem 'filterameter'
|
365
|
-
```
|
366
|
-
|
367
|
-
And then execute:
|
368
|
-
```bash
|
369
|
-
$ bundle
|
370
|
-
```
|
371
|
-
|
372
|
-
Or install it yourself as:
|
373
|
-
```bash
|
374
|
-
$ gem install filterameter
|
375
|
-
```
|
376
|
-
|
377
408
|
## Forms and Query Parameters
|
378
409
|
|
379
|
-
The
|
410
|
+
The filter parameters are pulled from the controller parameters, nested under the key `filter` (by default; see [Configuration](#configuration) to change the filter key). For example a request for large, blue widgets might have the following query parameters on the url:
|
380
411
|
|
381
412
|
```
|
382
413
|
?filter[size]=large&filter[color]=blue
|
@@ -394,7 +425,26 @@ On [a generic search form](https://guides.rubyonrails.org/form_helpers.html#a-ge
|
|
394
425
|
<% end %>
|
395
426
|
```
|
396
427
|
|
397
|
-
|
428
|
+
#### Sort Parameters
|
429
|
+
|
430
|
+
The sort is also nested underneath the filter key:
|
431
|
+
|
432
|
+
`/widgets?filter[sort]=size`
|
433
|
+
|
434
|
+
Use an array to pass multiple sorts. The order of the parameters is the order the sorts will be applied. For example, the following sorts first by size then by color:
|
435
|
+
|
436
|
+
`/widgets?filter[sort]=size&filter[sort]=color`
|
437
|
+
|
438
|
+
Sorts are ascending by default, but can use a prefix can be added to control the sort:
|
439
|
+
|
440
|
+
- `+` ascending (the default)
|
441
|
+
- `-` descending
|
442
|
+
|
443
|
+
For example, the following sorts by size descending:
|
444
|
+
|
445
|
+
`/widgets?filter[sort]=-size`
|
446
|
+
|
447
|
+
## Contribute
|
398
448
|
|
399
449
|
Feedback, feature requests, and proposed changes are welcomed. Please use the [issue tracker](https://github.com/RockSolt/filterameter/issues)
|
400
450
|
for feedback and feature requests. To propose a change directly, please fork the repo and open a pull request. Keep an eye on the actions to make
|
@@ -1,29 +1,33 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Filterameter
|
4
|
-
#
|
4
|
+
# # Configuration
|
5
5
|
#
|
6
6
|
# Class Configuration stores the following settings:
|
7
|
-
#
|
8
|
-
#
|
9
|
-
#
|
7
|
+
# * action_on_undeclared_parameters
|
8
|
+
# * action_on_validation_failure
|
9
|
+
# * filter_key
|
10
10
|
#
|
11
|
-
#
|
11
|
+
# ## Action on Undeclared Parameters
|
12
12
|
#
|
13
|
-
# Occurs when the filter parameter contains any keys that are not defined. Valid
|
14
|
-
# false (do not take action). By default,
|
13
|
+
# Occurs when the filter parameter contains any keys that are not defined. Valid
|
14
|
+
# actions are :log, :raise, and false (do not take action). By default,
|
15
|
+
# development will log, test will raise, and production will do nothing.
|
15
16
|
#
|
16
|
-
#
|
17
|
+
# ## Action on Validation Failure
|
17
18
|
#
|
18
|
-
# Occurs when a filter parameter fails a validation. Valid actions are :log,
|
19
|
-
# By default, development will log, test
|
19
|
+
# Occurs when a filter parameter fails a validation. Valid actions are :log,
|
20
|
+
# :raise, and false (do not take action). By default, development will log, test
|
21
|
+
# will raise, and production will do nothing.
|
20
22
|
#
|
21
|
-
#
|
23
|
+
# ## Filter Key
|
22
24
|
#
|
23
|
-
# By default, the filter parameters are nested under the key :filter. Use this
|
25
|
+
# By default, the filter parameters are nested under the key :filter. Use this
|
26
|
+
# setting to override the key.
|
24
27
|
#
|
25
|
-
# If the filter parameters are NOT nested, set this to false. Doing so will
|
26
|
-
# those that have been declared, meaning
|
28
|
+
# If the filter parameters are NOT nested, set this to false. Doing so will
|
29
|
+
# restrict the filter parameters to only those that have been declared, meaning
|
30
|
+
# undeclared parameters are ignored (and the action_on_undeclared_parameters
|
27
31
|
# configuration option does not come into play).
|
28
32
|
class Configuration
|
29
33
|
attr_accessor :action_on_undeclared_parameters, :action_on_validation_failure, :filter_key
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module DeclarationErrors
|
5
|
-
#
|
5
|
+
# # Cannot Be Inline Scope Error
|
6
6
|
#
|
7
7
|
# Error CannotBeInlineScopeError occurs when an inline scope has been used to define a filter that takes a
|
8
8
|
# parameter. This is not valid for use as a Filterameter filter because an inline scope always has an arity of -1
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module DeclarationErrors
|
5
|
-
#
|
5
|
+
# # Filter Scope Argument Error
|
6
6
|
#
|
7
7
|
# Error FilterScopeArgumentError occurs when a scope used as a filter but does not have either zero or one
|
8
8
|
# arument. A conditional scope filter should take zero arguments; other scope filters should take one argument.
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module DeclarationErrors
|
5
|
-
#
|
5
|
+
# # No Such Attribute Error
|
6
6
|
#
|
7
7
|
# Error NoSuchAttributeError occurs when a filter or sort references an attribute that does not exist on the model.
|
8
8
|
# The most likely case of this is a typo. Note that if the typo was supposed to reference a scope, this error is
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module DeclarationErrors
|
5
|
-
#
|
5
|
+
# # Not A Scope Error
|
6
6
|
#
|
7
7
|
# Error NotAScopeError flags a class method that has been used as a filter but is not a scope. This could occur if
|
8
8
|
# there is a class method of the same name an attribute, in which case the class method is going to block the
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module DeclarationErrors
|
5
|
-
#
|
5
|
+
# # Sort Scope Requires One Argument Error
|
6
6
|
#
|
7
7
|
# Error SortScopeRequiresOneArgumentError occurs when a sort has been declared for a scope that does not take
|
8
8
|
# exactly one argument. Sort scopes must take a single argument and will receive either :asc or :desc to indicate
|
@@ -1,7 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Filterameter
|
4
|
-
#
|
4
|
+
# # Declarations Validator
|
5
5
|
#
|
6
6
|
# Class DeclarationsValidtor fetches each filter and sort from the registry to validate the declaration. This class
|
7
7
|
# can be accessed from the controller as `declarations_validator` (via the FilterCoordinator) and be used in tests.
|
@@ -10,11 +10,12 @@ module Filterameter
|
|
10
10
|
#
|
11
11
|
# A test in RSpec might look like this:
|
12
12
|
#
|
13
|
-
#
|
13
|
+
# expect(WidgetsController.declarations_validator).to be_valid
|
14
14
|
#
|
15
15
|
# In Minitest it might look like this:
|
16
16
|
#
|
17
|
-
#
|
17
|
+
# validator = WidgetsController.declarations_validator
|
18
|
+
# assert_predicate validator, :valid?, -> { validator.errors }
|
18
19
|
class DeclarationsValidator
|
19
20
|
include Filterameter::Errors
|
20
21
|
|
@@ -69,7 +70,7 @@ module Filterameter
|
|
69
70
|
"\nInvalid #{type} for '#{name}':\n #{errors.join("\n ")}"
|
70
71
|
end
|
71
72
|
|
72
|
-
#
|
73
|
+
# # Factory Errors
|
73
74
|
#
|
74
75
|
# Class FactoryErrors is swapped in if the fetch from a factory fails. It is always invalid and provides the reason.
|
75
76
|
class FactoryErrors
|
@@ -1,22 +1,134 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Filterameter
|
4
|
-
#
|
4
|
+
# # Declarative Controller Filters
|
5
5
|
#
|
6
6
|
# Mixin DeclarativeFilters should be included in controllers to enable the filter DSL.
|
7
7
|
module DeclarativeFilters
|
8
8
|
extend ActiveSupport::Concern
|
9
|
-
include Filterameter::Filterable
|
10
|
-
include Filterameter::Sortable
|
11
9
|
|
12
10
|
class_methods do
|
13
11
|
delegate :declarations_validator, to: :filter_coordinator
|
14
12
|
|
13
|
+
# Declares a filter that can be read from the parameters and applied to the
|
14
|
+
# ActiveRecord query. The `name` identifies the name of the parameter and is the
|
15
|
+
# default value to determine the criteria to be applied. The name can be either
|
16
|
+
# an attribute or a scope.
|
17
|
+
#
|
18
|
+
# ## Options
|
19
|
+
#
|
20
|
+
# The declaration can also include an `options` hash to specialize the behavior of the filter.
|
21
|
+
#
|
22
|
+
# * **name**: Specify the attribute or scope name if the parameter name is not the same. The default value is the
|
23
|
+
# parameter name, so if the two match this can be left out.
|
24
|
+
#
|
25
|
+
# * **association**: Specify the name of the association if the attribute or scope is nested. Use an array if the
|
26
|
+
# asociation is more than one level.
|
27
|
+
#
|
28
|
+
# * **validates**: Specify a validation if the parameter value should be validated. This uses ActiveModel
|
29
|
+
# validations; please review those for types of validations and usage.
|
30
|
+
#
|
31
|
+
# * **partial**: Specify the partial option if the filter should do a partial search (SQL's `LIKE`). The partial
|
32
|
+
# option accepts a hash to specify the search behavior.
|
33
|
+
#
|
34
|
+
# Here are the available options:
|
35
|
+
# * match: `:anywhere` (default), `:from_start`, `:dynamic`
|
36
|
+
# * case_sensitive: `true`, `false` (default)
|
37
|
+
#
|
38
|
+
# There are two shortcuts: the partial option can be declared with `true`,
|
39
|
+
# which just uses the defaults; or the partial option can be declared with
|
40
|
+
# the match option directly, such as `partial: :from_start`.
|
41
|
+
#
|
42
|
+
# * **range**: Specify a range option if the filter also allows ranges to be searched. The range option accepts
|
43
|
+
# the following options:
|
44
|
+
# * true: enables two additional parameters with attribute name plus
|
45
|
+
# suffixes `_min` and `_max`
|
46
|
+
# * min_only: enables additional parameter with attribute name plus
|
47
|
+
# suffix `_min`
|
48
|
+
# * max_only: enables additional parameter with attribute name plus
|
49
|
+
# suffix `_max`
|
50
|
+
#
|
51
|
+
# * **sortable**: By default most filters are sortable. To prevent an attribute filter from being sortable, set
|
52
|
+
# the option to `false`.
|
53
|
+
#
|
54
|
+
# Options examples:
|
55
|
+
#
|
56
|
+
# filter :status, name: :current_status
|
57
|
+
# filter :manager_id, association: :department
|
58
|
+
# filter :business_unit_name, name: :name, association: [:department, :business_unit]
|
59
|
+
# filter :size, validates: { inclusion: { in: %w[Small Medium Large], allow_multiple_values: true } }
|
60
|
+
# filter :description, partial: true
|
61
|
+
# filter :department_name, partial: :from_start
|
62
|
+
# filter :reason, partial: { match: :dynamic, case_sensitive: true }
|
63
|
+
# filter :price, range: true
|
64
|
+
def filter(name, options = {})
|
65
|
+
filter_coordinator.add_filter(name, options)
|
66
|
+
end
|
67
|
+
|
68
|
+
# Declares a list of filters without options. Filters that require options must be declared with `filter`.
|
69
|
+
def filters(*names)
|
70
|
+
names.each { |name| filter(name) }
|
71
|
+
end
|
72
|
+
|
73
|
+
# Declares a sort that can be read from the parameters and applied to the
|
74
|
+
# ActiveRecord query. The `parameter_name` identifies the name of the parameter
|
75
|
+
# and is the default value for the attribute name when none is specified in the
|
76
|
+
# options.
|
77
|
+
#
|
78
|
+
# ## Options
|
79
|
+
#
|
80
|
+
# The declaration can also include an `options` hash to specialize the behavior of the sort.
|
81
|
+
#
|
82
|
+
# * **name**: Specify the attribute or scope name if the parameter name is not the same. The default value is the
|
83
|
+
# parameter name, so if the two match this can be left out.
|
84
|
+
#
|
85
|
+
# * **association**: Specify the name of the association if the attribute or scope is nested.
|
86
|
+
#
|
87
|
+
# Options example:
|
88
|
+
#
|
89
|
+
# sort :project_created_at, name: :created_at, association: :project
|
90
|
+
def sort(parameter_name, options = {})
|
91
|
+
filter_coordinator.add_sort(parameter_name, options)
|
92
|
+
end
|
93
|
+
|
94
|
+
# Declares a list of sorts without options. Sorts that require options must be declared with `sort`.
|
95
|
+
def sorts(*parameter_names)
|
96
|
+
parameter_names.each { |parameter_name| filter(parameter_name) }
|
97
|
+
end
|
98
|
+
|
99
|
+
def default_sort(sort_and_direction_pairs)
|
100
|
+
filter_coordinator.default_sort = sort_and_direction_pairs
|
101
|
+
end
|
102
|
+
|
103
|
+
# Rails conventions are used to determine the controller's model. For example, the PhotosController builds a query
|
104
|
+
# against the Photo model. If a controller is namespaced, the model will first be looked up without the namespace,
|
105
|
+
# then with the namespace.
|
106
|
+
#
|
107
|
+
# **If the conventions do not provide the correct model**, the model can be named explicitly with the following:
|
108
|
+
#
|
109
|
+
# ```ruby
|
110
|
+
# filter_model 'Picture'
|
111
|
+
# ```
|
112
|
+
#
|
113
|
+
# An optional second parameter can be used to specify the variable name. Both the model and the variable name can
|
114
|
+
# be specified with this short-cut. For example, to use the Picture model and store the results as `@data`, use
|
115
|
+
# the following:
|
116
|
+
#
|
117
|
+
# filter_model 'Picture', :data
|
118
|
+
#
|
119
|
+
# **Important:** If the `filter_model` declaration is used, it must be before any filter or sort declarations.
|
15
120
|
def filter_model(model_class, query_var_name = nil)
|
16
121
|
filter_coordinator.model_class = model_class
|
17
122
|
filter_query_var_name(query_var_name) if query_var_name.present?
|
18
123
|
end
|
19
124
|
|
125
|
+
# When using the before action callback `build_filtered_query`, the query is assigned to an instance variable with
|
126
|
+
# the name of the model pluralized. For example, the Photo model will use the variable `@photos`.
|
127
|
+
#
|
128
|
+
# To use a different variable name with the callback, assign it with `filter_query_var_name`. For example, if the
|
129
|
+
# query is stored as `@data`, use the following:
|
130
|
+
#
|
131
|
+
# filter_query_var_name :data
|
20
132
|
def filter_query_var_name(query_variable_name)
|
21
133
|
filter_coordinator.query_variable_name = query_variable_name
|
22
134
|
end
|
@@ -26,6 +138,24 @@ module Filterameter
|
|
26
138
|
end
|
27
139
|
end
|
28
140
|
|
141
|
+
# Returns an ActiveRecord query from the filter parameters.
|
142
|
+
#
|
143
|
+
# def index
|
144
|
+
# @widgets = build_query_from_filters
|
145
|
+
# end
|
146
|
+
#
|
147
|
+
# The method optionally takes a starting query. For example, this restricts the results
|
148
|
+
# to only active widgets:
|
149
|
+
#
|
150
|
+
# def index
|
151
|
+
# @widgets = build_query_from_filters(Widgets.where(active: true))
|
152
|
+
# end
|
153
|
+
#
|
154
|
+
# The starting query can also be used to provide eager loading:
|
155
|
+
#
|
156
|
+
# def index
|
157
|
+
# @widgets = build_query_from_filters(Widgets.includes(:manufacturer))
|
158
|
+
# end
|
29
159
|
def build_query_from_filters(starting_query = nil)
|
30
160
|
self.class.filter_coordinator.build_query(filter_parameters, starting_query)
|
31
161
|
end
|
data/lib/filterameter/errors.rb
CHANGED
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Exceptions
|
5
|
-
#
|
5
|
+
# # Cannot Determine Model Error
|
6
6
|
#
|
7
7
|
# Class CannotDetermineModelError is raised when the model class cannot be determined from either the controller
|
8
8
|
# name or controller path. This is a setup issue; the resolution is for the controller to specify the model class
|
@@ -2,10 +2,10 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Exceptions
|
5
|
-
#
|
5
|
+
# # Collection Association Sort Error
|
6
6
|
#
|
7
|
-
# Class CollectionAssociationSortError is raised when a sort is attempted on a
|
8
|
-
# only valid on
|
7
|
+
# Class CollectionAssociationSortError is raised when a sort is attempted on a
|
8
|
+
# collection association. (Sorting is only valid on *singular* associations.)
|
9
9
|
class CollectionAssociationSortError < FilterameterError
|
10
10
|
def initialize(declaration)
|
11
11
|
super("Sorting is not allowed on collection associations: \n\t\t#{declaration}")
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Exceptions
|
5
|
-
#
|
5
|
+
# # Invalid Association Declaration Error
|
6
6
|
#
|
7
7
|
# Class InvalidAssociationDeclarationError is raised when the declared association(s) are not valid.
|
8
8
|
class InvalidAssociationDeclarationError < FilterameterError
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Exceptions
|
5
|
-
#
|
5
|
+
# # Undeclared Parameter Error
|
6
6
|
#
|
7
7
|
# Class UndeclaredParameterError is raised when a request contains filter parameters that have not been declared.
|
8
8
|
# Configuration setting `action_on_undeclared_parameters` determines whether or not the exception is raised.
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Exceptions
|
5
|
-
#
|
5
|
+
# # Validation Error
|
6
6
|
#
|
7
7
|
# Class ValidationError is raised when a specified parameter fails a validation. Configuration setting
|
8
8
|
# `action_on_validation_failure` determines whether or not the exception is raised.
|
@@ -8,7 +8,7 @@ require 'action_controller/metal/live'
|
|
8
8
|
require 'action_controller/metal/strong_parameters'
|
9
9
|
|
10
10
|
module Filterameter
|
11
|
-
#
|
11
|
+
# # Filter Coordinator
|
12
12
|
#
|
13
13
|
# Class FilterCoordinator stores the configuration declared via class-level method calls such as the list of
|
14
14
|
# filters and the optionally declared model class. Each controller will have one instance of the coordinator
|
@@ -2,9 +2,10 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Filters
|
5
|
-
#
|
5
|
+
# # Arel Filter
|
6
6
|
#
|
7
|
-
# Class ArelFilter is a base class for arel queries. It does not implement
|
7
|
+
# Class ArelFilter is a base class for arel queries. It does not implement
|
8
|
+
# `apply`.
|
8
9
|
class ArelFilter
|
9
10
|
include Filterameter::Errors
|
10
11
|
include Filterameter::Filters::AttributeValidator
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Helpers
|
5
|
-
#
|
5
|
+
# # Joins Values Builder
|
6
6
|
#
|
7
7
|
# Class JoinsValuesBuilder evaluates an array of names to return either the single entry when there is only
|
8
8
|
# one element in the array or a nested hash when there is more than one element. This is the argument that is
|
@@ -2,19 +2,20 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Options
|
5
|
-
#
|
5
|
+
# # Partial Options
|
6
6
|
#
|
7
|
-
# Class PartialOptions parses the options passed in as partial, then exposes
|
8
|
-
# their valid values:
|
9
|
-
#
|
10
|
-
#
|
7
|
+
# Class PartialOptions parses the options passed in as partial, then exposes
|
8
|
+
# those. Here are the options along with their valid values:
|
9
|
+
# * match: anywhere (default), from_start, dynamic
|
10
|
+
# * case_sensitive: true, false (default)
|
11
11
|
#
|
12
12
|
# Options may be specified by passing a hash with the option keys:
|
13
13
|
#
|
14
|
-
#
|
14
|
+
# partial: { match: :from_start, case_sensitive: true }
|
15
15
|
#
|
16
|
-
# There are two shortcuts: the partial option can be declared with `true`, which
|
17
|
-
# partial option can be declared with the match
|
16
|
+
# There are two shortcuts: the partial option can be declared with `true`, which
|
17
|
+
# just uses the defaults; or the partial option can be declared with the match
|
18
|
+
# option directly, such as partial: :from_start.
|
18
19
|
class PartialOptions
|
19
20
|
VALID_OPTIONS = %i[match case_sensitive].freeze
|
20
21
|
VALID_MATCH_OPTIONS = %w[anywhere from_start dynamic].freeze
|
@@ -1,17 +1,18 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Filterameter
|
4
|
-
#
|
4
|
+
# # Query Builder
|
5
5
|
#
|
6
6
|
# Class Query Builder turns filter parameters into a query.
|
7
7
|
#
|
8
8
|
# The query builder is instantiated by the filter coordinator. The default query currently is simple `all`. The
|
9
9
|
# default sort comes for the controller declaration of the same name; it is optional and the value may be nil.
|
10
10
|
#
|
11
|
-
# If the request includes a sort, it is always applied. If not, the following
|
12
|
-
#
|
13
|
-
#
|
14
|
-
#
|
11
|
+
# If the request includes a sort, it is always applied. If not, the following
|
12
|
+
# logic kicks in to provide a sort:
|
13
|
+
# * if the starting query includes a sort, no additional sort is applied
|
14
|
+
# * if a default sort has been declared, it is applied
|
15
|
+
# * if neither of those provides a sort, then the fallback is primary key desc
|
15
16
|
class QueryBuilder
|
16
17
|
def initialize(default_query, default_sort, filter_registry)
|
17
18
|
@default_query = default_query
|
@@ -87,9 +88,10 @@ module Filterameter
|
|
87
88
|
end
|
88
89
|
end
|
89
90
|
|
90
|
-
# TODO: this handles any runtime exceptions, not just undeclared parameter
|
91
|
-
#
|
92
|
-
#
|
91
|
+
# TODO: this handles any runtime exceptions, not just undeclared parameter
|
92
|
+
# errors:
|
93
|
+
# * should the config option be more generalized?
|
94
|
+
# * or should there be a config option for each type of error?
|
93
95
|
def handle_undeclared_parameter(exception)
|
94
96
|
action = Filterameter.configuration.action_on_undeclared_parameters
|
95
97
|
return unless action
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Registries
|
5
|
-
#
|
5
|
+
# # Filter Registry
|
6
6
|
#
|
7
7
|
# Class FilterRegistry is a collection of the filters. It captures the filter declarations when classes are loaded,
|
8
8
|
# then uses the injected FilterFactory to build the filters on demand as they are needed.
|
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Registries
|
5
|
-
# Sort Registry
|
5
|
+
# # Sort Registry
|
6
6
|
#
|
7
7
|
# Class SortRegistry is a collection of the sorts. It captures the declarations when classes are loaded,
|
8
8
|
# then uses the injected SortFactory to build the sorts on demand as they are needed.
|
@@ -2,12 +2,11 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Registries
|
5
|
-
# SubRegistry
|
5
|
+
# # SubRegistry
|
6
6
|
#
|
7
7
|
# Class SubRegistry provides add and fetch methods as well as the initialization for sub-registries.
|
8
8
|
#
|
9
9
|
# Subclasses must implement build_declaration.
|
10
|
-
#
|
11
10
|
class SubRegistry
|
12
11
|
def initialize(factory)
|
13
12
|
@factory = factory
|
@@ -1,7 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Filterameter
|
4
|
-
#
|
4
|
+
# # Sort Declaration
|
5
5
|
#
|
6
6
|
# Class SortDeclaration captures the sort declaration within the controller. A sort declaration is also generated
|
7
7
|
# from a FilterDeclaration when it is `sortable?`.
|
@@ -2,14 +2,14 @@
|
|
2
2
|
|
3
3
|
module Filterameter
|
4
4
|
module Validators
|
5
|
-
#
|
5
|
+
# # Inclusion Validator
|
6
6
|
#
|
7
7
|
# Class InclusionValidator extends ActiveModel::Validations::InclusionValidator to enable validations of multiple
|
8
8
|
# values.
|
9
9
|
#
|
10
|
-
#
|
10
|
+
# ## Example
|
11
11
|
#
|
12
|
-
#
|
12
|
+
# validates: { inclusion: { in: %w[Small Medium Large], allow_multiple_values: true } }
|
13
13
|
#
|
14
14
|
class InclusionValidator < ActiveModel::Validations::InclusionValidator
|
15
15
|
def validate_each(record, attribute, value)
|
data/lib/filterameter/version.rb
CHANGED
data/lib/filterameter.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: filterameter
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 1.0.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Todd Kummer
|
8
|
-
autorequire:
|
9
8
|
bindir: bin
|
10
9
|
cert_chain: []
|
11
|
-
date:
|
10
|
+
date: 2025-01-09 00:00:00.000000000 Z
|
12
11
|
dependencies:
|
13
12
|
- !ruby/object:Gem::Dependency
|
14
13
|
name: rails
|
@@ -100,14 +99,14 @@ dependencies:
|
|
100
99
|
requirements:
|
101
100
|
- - "~>"
|
102
101
|
- !ruby/object:Gem::Version
|
103
|
-
version: '
|
102
|
+
version: '7.0'
|
104
103
|
type: :development
|
105
104
|
prerelease: false
|
106
105
|
version_requirements: !ruby/object:Gem::Requirement
|
107
106
|
requirements:
|
108
107
|
- - "~>"
|
109
108
|
- !ruby/object:Gem::Version
|
110
|
-
version: '
|
109
|
+
version: '7.0'
|
111
110
|
- !ruby/object:Gem::Dependency
|
112
111
|
name: rubocop
|
113
112
|
requirement: !ruby/object:Gem::Requirement
|
@@ -156,14 +155,14 @@ dependencies:
|
|
156
155
|
requirements:
|
157
156
|
- - "~>"
|
158
157
|
- !ruby/object:Gem::Version
|
159
|
-
version: 3.0
|
158
|
+
version: 3.2.0
|
160
159
|
type: :development
|
161
160
|
prerelease: false
|
162
161
|
version_requirements: !ruby/object:Gem::Requirement
|
163
162
|
requirements:
|
164
163
|
- - "~>"
|
165
164
|
- !ruby/object:Gem::Version
|
166
|
-
version: 3.0
|
165
|
+
version: 3.2.0
|
167
166
|
- !ruby/object:Gem::Dependency
|
168
167
|
name: rubocop-rspec_rails
|
169
168
|
requirement: !ruby/object:Gem::Requirement
|
@@ -223,7 +222,6 @@ files:
|
|
223
222
|
- lib/filterameter/filter_coordinator.rb
|
224
223
|
- lib/filterameter/filter_declaration.rb
|
225
224
|
- lib/filterameter/filter_factory.rb
|
226
|
-
- lib/filterameter/filterable.rb
|
227
225
|
- lib/filterameter/filters/arel_filter.rb
|
228
226
|
- lib/filterameter/filters/attribute_filter.rb
|
229
227
|
- lib/filterameter/filters/attribute_validator.rb
|
@@ -247,7 +245,6 @@ files:
|
|
247
245
|
- lib/filterameter/registries/sub_registry.rb
|
248
246
|
- lib/filterameter/sort_declaration.rb
|
249
247
|
- lib/filterameter/sort_factory.rb
|
250
|
-
- lib/filterameter/sortable.rb
|
251
248
|
- lib/filterameter/sorts/attribute_sort.rb
|
252
249
|
- lib/filterameter/sorts/scope_sort.rb
|
253
250
|
- lib/filterameter/validators/inclusion_validator.rb
|
@@ -256,7 +253,6 @@ homepage: https://github.com/RockSolt/filterameter
|
|
256
253
|
licenses:
|
257
254
|
- MIT
|
258
255
|
metadata: {}
|
259
|
-
post_install_message:
|
260
256
|
rdoc_options: []
|
261
257
|
require_paths:
|
262
258
|
- lib
|
@@ -271,8 +267,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
271
267
|
- !ruby/object:Gem::Version
|
272
268
|
version: '0'
|
273
269
|
requirements: []
|
274
|
-
rubygems_version: 3.
|
275
|
-
signing_key:
|
270
|
+
rubygems_version: 3.6.2
|
276
271
|
specification_version: 4
|
277
272
|
summary: Declarative Filter Parameters
|
278
273
|
test_files: []
|
@@ -1,55 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Filterameter
|
4
|
-
# = Declarative Filters
|
5
|
-
#
|
6
|
-
# Mixin Filterable provides class methods <tt>filter</tt> and <tt>filters</tt>.
|
7
|
-
module Filterable
|
8
|
-
extend ActiveSupport::Concern
|
9
|
-
|
10
|
-
class_methods do
|
11
|
-
# Declares a filter that can be read from the parameters and applied to the ActiveRecord query. The <tt>name</tt>
|
12
|
-
# identifies the name of the parameter and is the default value to determine the criteria to be applied. The name
|
13
|
-
# can be either an attribute or a scope.
|
14
|
-
#
|
15
|
-
# === Options
|
16
|
-
#
|
17
|
-
# [:name]
|
18
|
-
# Specify the attribute or scope name if the parameter name is not the same. The default value
|
19
|
-
# is the parameter name, so if the two match this can be left out.
|
20
|
-
#
|
21
|
-
# [:association]
|
22
|
-
# Specify the name of the association if the attribute or scope is nested.
|
23
|
-
#
|
24
|
-
# [:validates]
|
25
|
-
# Specify a validation if the parameter value should be validated. This uses ActiveModel validations;
|
26
|
-
# please review those for types of validations and usage.
|
27
|
-
#
|
28
|
-
# [:partial]
|
29
|
-
# Specify the partial option if the filter should do a partial search (SQL's `LIKE`). The partial
|
30
|
-
# option accepts a hash to specify the search behavior. Here are the available options:
|
31
|
-
# - match: anywhere (default), from_start, dynamic
|
32
|
-
# - case_sensitive: true, false (default)
|
33
|
-
#
|
34
|
-
# There are two shortcuts: : the partial option can be declared with `true`, which just uses the
|
35
|
-
# defaults; or the partial option can be declared with the match option directly,
|
36
|
-
# such as `partial: :from_start`.
|
37
|
-
#
|
38
|
-
# [:range]
|
39
|
-
# Specify a range option if the filter also allows ranges to be searched. The range option accepts
|
40
|
-
# the following options:
|
41
|
-
# - true: enables two additional parameters with attribute name plus suffixes <tt>_min</tt> and <tt>_max</tt>
|
42
|
-
# - :min_only: enables additional parameter with attribute name plus suffix <tt>_min</tt>
|
43
|
-
# - :max_only: enables additional parameter with attribute name plus suffix <tt>_max</tt>
|
44
|
-
def filter(name, options = {})
|
45
|
-
filter_coordinator.add_filter(name, options)
|
46
|
-
end
|
47
|
-
|
48
|
-
# Declares a list of filters that can be read from the parameters and applied to the query. The name can be either
|
49
|
-
# an attribute or a scope. Declare filters individually with <tt>filter</tt> if more options are required.
|
50
|
-
def filters(*names)
|
51
|
-
names.each { |name| filter(name) }
|
52
|
-
end
|
53
|
-
end
|
54
|
-
end
|
55
|
-
end
|
@@ -1,40 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Filterameter
|
4
|
-
# = Sortable
|
5
|
-
#
|
6
|
-
# Mixin Sortable provides class methods <tt>sort</tt> and <tt>sorts</tt>.
|
7
|
-
module Sortable
|
8
|
-
extend ActiveSupport::Concern
|
9
|
-
|
10
|
-
class_methods do
|
11
|
-
# Declares a sort that can be read from the parameters and applied to the ActiveRecord query. The
|
12
|
-
# <tt>parameter_name</tt> identifies the name of the parameter and is the default value for the attribute
|
13
|
-
# name when none is specified in the options.
|
14
|
-
#
|
15
|
-
# The including class must implement `filter_coordinator`
|
16
|
-
#
|
17
|
-
# === Options
|
18
|
-
#
|
19
|
-
# [:name]
|
20
|
-
# Specify the attribute or scope name if the parameter name is not the same. The default value
|
21
|
-
# is the parameter name, so if the two match this can be left out.
|
22
|
-
#
|
23
|
-
# [:association]
|
24
|
-
# Specify the name of the association if the attribute or scope is nested.
|
25
|
-
def sort(parameter_name, options = {})
|
26
|
-
filter_coordinator.add_sort(parameter_name, options)
|
27
|
-
end
|
28
|
-
|
29
|
-
# Declares a list of filters that can be read from the parameters and applied to the query. The name can be either
|
30
|
-
# an attribute or a scope. Declare filters individually with <tt>filter</tt> if more options are required.
|
31
|
-
def sorts(*parameter_names)
|
32
|
-
parameter_names.each { |parameter_name| filter(parameter_name) }
|
33
|
-
end
|
34
|
-
|
35
|
-
def default_sort(sort_and_direction_pairs)
|
36
|
-
filter_coordinator.default_sort = sort_and_direction_pairs
|
37
|
-
end
|
38
|
-
end
|
39
|
-
end
|
40
|
-
end
|