search_object 1.1.3 → 1.2.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.projections.json +8 -0
- data/CHANGELOG.md +47 -1
- data/README.md +32 -1
- data/example/app/assets/stylesheets/application.css.scss +2 -0
- data/example/app/models/post_search.rb +17 -5
- data/example/app/views/posts/index.html.slim +1 -0
- data/lib/search_object.rb +1 -0
- data/lib/search_object/base.rb +8 -1
- data/lib/search_object/helper.rb +13 -0
- data/lib/search_object/plugin/enum.rb +70 -0
- data/lib/search_object/search.rb +1 -12
- data/lib/search_object/version.rb +1 -1
- data/search_object.gemspec +1 -0
- data/spec/search_object/base_spec.rb +15 -0
- data/spec/search_object/helper_spec.rb +31 -6
- data/spec/search_object/plugin/enum_spec.rb +130 -0
- data/spec/search_object/plugin/kaminari_spec.rb +1 -2
- metadata +20 -4
- data/spec/support/kaminari_setup.rb +0 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 4cce323b42516f0df8bff1b13275f60dcf0439cf
|
4
|
+
data.tar.gz: 35c0d32d6011eae2946e2758ebcf4222b3392f32
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 6df95c9bff5882b28585bf3de86623d2015bdb633fabc8eac0b82e330ce11671ef09f581e7bc25a81e1157db828d0334e513a859bed0971dcfe36e1b906bbace
|
7
|
+
data.tar.gz: 9ef427a40467a42bfde8499f6066de5a98481c66d92c7830b021c992386adf0172bca7df56795ab11c442659103d2902fffcf858f00e8e0608463b47460443b7
|
data/.projections.json
ADDED
data/CHANGELOG.md
CHANGED
@@ -1,6 +1,52 @@
|
|
1
1
|
# Changelog
|
2
2
|
|
3
|
-
## Version 1.
|
3
|
+
## Version 1.2.0 (unreleased)
|
4
|
+
|
5
|
+
* __[feature]__ `enum` plugin added (@rstankov)
|
6
|
+
|
7
|
+
```ruby
|
8
|
+
class ProductSearch
|
9
|
+
include SearchObject.module(:enum)
|
10
|
+
|
11
|
+
scope { Product.all }
|
12
|
+
|
13
|
+
option :order, enum: %w(popular date)
|
14
|
+
|
15
|
+
private
|
16
|
+
|
17
|
+
# Gets called when order with 'popular' is given
|
18
|
+
def apply_order_with_popular(scope)
|
19
|
+
scope.by_popularity
|
20
|
+
end
|
21
|
+
|
22
|
+
# Gets called when order with 'date' is given
|
23
|
+
def apply_order_with_date(scope)
|
24
|
+
scope.by_date
|
25
|
+
end
|
26
|
+
|
27
|
+
# Gets called when invalid enum is given
|
28
|
+
def handle_invalid_order(scope, invalid_value)
|
29
|
+
scope
|
30
|
+
end
|
31
|
+
end
|
32
|
+
```
|
33
|
+
|
34
|
+
* __[feature]__ Scope is executed in context of SearchObject::Base context (@rstankov)
|
35
|
+
|
36
|
+
```ruby
|
37
|
+
class ProductSearch
|
38
|
+
include SearchObject.module
|
39
|
+
|
40
|
+
scope { @shop.products }
|
41
|
+
|
42
|
+
def initialize(shop)
|
43
|
+
@shop = shop
|
44
|
+
super
|
45
|
+
end
|
46
|
+
end
|
47
|
+
```
|
48
|
+
|
49
|
+
## Version 1.1.3
|
4
50
|
|
5
51
|
* __[feature]__ Passing nil as `scope` in constructor, falls back to default scope (@rstankov)
|
6
52
|
|
data/README.md
CHANGED
@@ -104,9 +104,40 @@ include SearchObject.module(:will_paginate)
|
|
104
104
|
include SearchObject.module(:kaminari)
|
105
105
|
```
|
106
106
|
|
107
|
+
### Enum plugin
|
108
|
+
|
109
|
+
Gives you filter with pre-defined options.
|
110
|
+
|
111
|
+
```ruby
|
112
|
+
class ProductSearch
|
113
|
+
include SearchObject.module(:enum)
|
114
|
+
|
115
|
+
scope { Product.all }
|
116
|
+
|
117
|
+
option :order, enum: %w(popular date)
|
118
|
+
|
119
|
+
private
|
120
|
+
|
121
|
+
# Gets called when order with 'popular' is given
|
122
|
+
def apply_order_with_popular(scope)
|
123
|
+
scope.by_popularity
|
124
|
+
end
|
125
|
+
|
126
|
+
# Gets called when order with 'date' is given
|
127
|
+
def apply_order_with_date(scope)
|
128
|
+
scope.by_date
|
129
|
+
end
|
130
|
+
|
131
|
+
# (optional) Gets called when invalid enum is given
|
132
|
+
def handle_invalid_order(scope, invalid_value)
|
133
|
+
scope
|
134
|
+
end
|
135
|
+
end
|
136
|
+
```
|
137
|
+
|
107
138
|
### Model plugin
|
108
139
|
|
109
|
-
Extends your search object with ```ActiveModel```, so you can use it in
|
140
|
+
Extends your search object with ```ActiveModel```, so you can use it in Rails forms.
|
110
141
|
|
111
142
|
```ruby
|
112
143
|
class ProductSearch
|
@@ -31,7 +31,9 @@ body { font-family: "Helvetica Neue", Helvetica, Arial, sans-serif !important; f
|
|
31
31
|
fieldset { margin-bottom: 20px; padding: 10px; border: 1px solid #e7e7e7; border-radius: 4px; background-color: #f8f8f8; }
|
32
32
|
fieldset input[type="submit"] { @extend .btn; @extend .btn-default; }
|
33
33
|
fieldset input[type="search"],
|
34
|
+
fieldset select,
|
34
35
|
fieldset input[type="date"] { vertical-align: middle; height: 34px; padding: 6px 12px; margin-right: 10px; }
|
36
|
+
fieldset select { width: 100px; }
|
35
37
|
|
36
38
|
table thead label { color: #428bca; }
|
37
39
|
table thead .active { font-weight: bold; color: black; }
|
@@ -1,5 +1,5 @@
|
|
1
1
|
class PostSearch
|
2
|
-
include SearchObject.module(:model, :sorting, :will_paginate)
|
2
|
+
include SearchObject.module(:model, :sorting, :will_paginate, :enum)
|
3
3
|
|
4
4
|
scope { Post.all }
|
5
5
|
|
@@ -13,6 +13,10 @@ class PostSearch
|
|
13
13
|
option :user_id
|
14
14
|
option :category_name
|
15
15
|
|
16
|
+
option :term, with: :apply_term
|
17
|
+
|
18
|
+
option :rating, enum: %i(low high)
|
19
|
+
|
16
20
|
option :title do |scope, value|
|
17
21
|
scope.where 'title LIKE ?', escape_search_term(value)
|
18
22
|
end
|
@@ -21,10 +25,6 @@ class PostSearch
|
|
21
25
|
scope.where published: true if value.present?
|
22
26
|
end
|
23
27
|
|
24
|
-
option :term do |scope, value|
|
25
|
-
scope.where 'title LIKE :term OR body LIKE :term', term: escape_search_term(value)
|
26
|
-
end
|
27
|
-
|
28
28
|
option :created_after do |scope, value|
|
29
29
|
date = parse_date value
|
30
30
|
scope.where('DATE(created_at) >= ?', date) if date.present?
|
@@ -37,6 +37,18 @@ class PostSearch
|
|
37
37
|
|
38
38
|
private
|
39
39
|
|
40
|
+
def apply_term(scope, value)
|
41
|
+
scope.where 'title LIKE :term OR body LIKE :term', term: escape_search_term(value)
|
42
|
+
end
|
43
|
+
|
44
|
+
def apply_rating_with_low(scope)
|
45
|
+
scope.where 'views_count < 100'
|
46
|
+
end
|
47
|
+
|
48
|
+
def apply_rating_with_high(scope)
|
49
|
+
scope.where 'views_count > 500'
|
50
|
+
end
|
51
|
+
|
40
52
|
def parse_date(value)
|
41
53
|
Date.parse(value).strftime('%Y-%m-%d')
|
42
54
|
rescue
|
@@ -7,6 +7,7 @@
|
|
7
7
|
= form.search_field :term, placeholder: 'Search term'
|
8
8
|
= date_field_tag 'f[created_after]', @search.created_after
|
9
9
|
= date_field_tag 'f[created_before]', @search.created_before
|
10
|
+
= select_tag 'f[rating]', options_for_select([['All', ''], ['Popular', 'high'], ['Unpopular', 'low']], @search.rating)
|
10
11
|
= form.submit 'Search'
|
11
12
|
|
12
13
|
div.pull-right
|
data/lib/search_object.rb
CHANGED
data/lib/search_object/base.rb
CHANGED
@@ -12,7 +12,14 @@ module SearchObject
|
|
12
12
|
end
|
13
13
|
|
14
14
|
def initialize(options = {})
|
15
|
-
|
15
|
+
config = self.class.config
|
16
|
+
scope = options[:scope] || (config[:scope] && instance_eval(&config[:scope]))
|
17
|
+
actions = config[:actions] || {}
|
18
|
+
params = Helper.normalize_params(config[:defaults], options[:filters], actions.keys)
|
19
|
+
|
20
|
+
raise MissingScopeError unless scope
|
21
|
+
|
22
|
+
@search = Search.new(scope, params, actions)
|
16
23
|
end
|
17
24
|
|
18
25
|
def results
|
data/lib/search_object/helper.rb
CHANGED
@@ -21,6 +21,15 @@ module SearchObject
|
|
21
21
|
text.to_s.gsub(/(?:^|_)(.)/) { Regexp.last_match[1].upcase }
|
22
22
|
end
|
23
23
|
|
24
|
+
def underscore(text)
|
25
|
+
text.to_s
|
26
|
+
.tr('::', '_')
|
27
|
+
.gsub(/([A-Z]+)([A-Z][a-z])/) { "#{Regexp.last_match[1]}_#{Regexp.last_match[2]}" }
|
28
|
+
.gsub(/([a-z\d])([A-Z])/) { "#{Regexp.last_match[1]}_#{Regexp.last_match[2]}" }
|
29
|
+
.tr('-', '_')
|
30
|
+
.downcase
|
31
|
+
end
|
32
|
+
|
24
33
|
def ensure_included(item, collection)
|
25
34
|
if collection.include? item
|
26
35
|
item
|
@@ -45,6 +54,10 @@ module SearchObject
|
|
45
54
|
end
|
46
55
|
end
|
47
56
|
|
57
|
+
def normalize_params(defaults, filters, keys)
|
58
|
+
(defaults || {}).merge(slice_keys(stringify_keys(filters || {}), keys || []))
|
59
|
+
end
|
60
|
+
|
48
61
|
def deep_copy(object) # rubocop:disable Metrics/MethodLength
|
49
62
|
case object
|
50
63
|
when Array
|
@@ -0,0 +1,70 @@
|
|
1
|
+
module SearchObject
|
2
|
+
module Plugin
|
3
|
+
module Enum
|
4
|
+
def self.included(base)
|
5
|
+
base.extend ClassMethods
|
6
|
+
end
|
7
|
+
|
8
|
+
module ClassMethods
|
9
|
+
def option(name, options = nil, &block)
|
10
|
+
return super unless options.is_a?(Hash) && options[:enum]
|
11
|
+
|
12
|
+
raise BlockIgnoredError if block
|
13
|
+
raise WithIgnoredError if options[:with]
|
14
|
+
|
15
|
+
handler = Handler.build(name, options[:enum])
|
16
|
+
|
17
|
+
super(name, options, &handler)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
module Handler
|
22
|
+
module_function
|
23
|
+
|
24
|
+
def build(name, enums)
|
25
|
+
enums = enums.map(&:to_s)
|
26
|
+
handler = self
|
27
|
+
->(scope, value) { handler.apply_filter(object: self, option: name, enums: enums, scope: scope, value: value) }
|
28
|
+
end
|
29
|
+
|
30
|
+
def apply_filter(object:, option:, enums:, scope:, value:)
|
31
|
+
return if value.nil? || value == ''
|
32
|
+
|
33
|
+
unless enums.include? value
|
34
|
+
return handle_invalid_value(object: object, option: option, enums: enums, scope: scope, value: value)
|
35
|
+
end
|
36
|
+
|
37
|
+
object.send("apply_#{option}_with_#{Helper.underscore(value)}", scope)
|
38
|
+
end
|
39
|
+
|
40
|
+
def handle_invalid_value(object:, option:, enums:, scope:, value:)
|
41
|
+
specific = "handle_invalid_#{option}"
|
42
|
+
return object.send(specific, scope, value) if object.respond_to? specific, true
|
43
|
+
|
44
|
+
catch_all = 'handle_invalid_enum'
|
45
|
+
return object.send(catch_all, option, scope, value) if object.respond_to? catch_all, true
|
46
|
+
|
47
|
+
raise InvalidEnumValueError.new(option, enums, value)
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
class InvalidEnumValueError < ArgumentError
|
52
|
+
def initialize(option, enums, value)
|
53
|
+
super "Wrong value '#{value}' used for enum #{option} (expected one of #{enums.join(', ')})"
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
class BlockIgnoredError < ArgumentError
|
58
|
+
def initialize(message = "Enum options don't accept blocks")
|
59
|
+
super message
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
class WithIgnoredError < ArgumentError
|
64
|
+
def initialize(message = "Enum options don't accept :with")
|
65
|
+
super message
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
70
|
+
end
|
data/lib/search_object/search.rb
CHANGED
@@ -1,19 +1,8 @@
|
|
1
1
|
module SearchObject
|
2
|
+
# :api: private
|
2
3
|
class Search
|
3
4
|
attr_reader :params
|
4
5
|
|
5
|
-
class << self
|
6
|
-
def build_for(config, options)
|
7
|
-
scope = options[:scope] || (config[:scope] && config[:scope].call)
|
8
|
-
filters = Helper.stringify_keys(options.fetch(:filters, {}))
|
9
|
-
params = config[:defaults].merge Helper.slice_keys(filters, config[:actions].keys)
|
10
|
-
|
11
|
-
raise MissingScopeError unless scope
|
12
|
-
|
13
|
-
new scope, params, config[:actions]
|
14
|
-
end
|
15
|
-
end
|
16
|
-
|
17
6
|
def initialize(scope, params, actions)
|
18
7
|
@scope = scope
|
19
8
|
@actions = actions
|
data/search_object.gemspec
CHANGED
@@ -28,6 +28,7 @@ Gem::Specification.new do |spec|
|
|
28
28
|
spec.add_development_dependency 'coveralls'
|
29
29
|
spec.add_development_dependency 'will_paginate'
|
30
30
|
spec.add_development_dependency 'kaminari'
|
31
|
+
spec.add_development_dependency 'kaminari-activerecord'
|
31
32
|
spec.add_development_dependency 'rubocop', '0.46.0'
|
32
33
|
spec.add_development_dependency 'rubocop-rspec', '1.8.0'
|
33
34
|
end
|
@@ -101,6 +101,21 @@ module SearchObject
|
|
101
101
|
expect(search_class.new(scope: 'other scope').results).to eq 'other scope'
|
102
102
|
end
|
103
103
|
|
104
|
+
it 'scope block is exectued in context of search object' do
|
105
|
+
search_class = define_search_class do
|
106
|
+
scope { inner_scope }
|
107
|
+
|
108
|
+
attr_reader :inner_scope
|
109
|
+
|
110
|
+
def initialize
|
111
|
+
@inner_scope = 'scope'
|
112
|
+
super
|
113
|
+
end
|
114
|
+
end
|
115
|
+
|
116
|
+
expect(search_class.new.results).to eq 'scope'
|
117
|
+
end
|
118
|
+
|
104
119
|
it 'passing nil as scope in constructor, falls back to default scope' do
|
105
120
|
search_class = define_search_class do
|
106
121
|
scope { 'scope' }
|
@@ -5,32 +5,57 @@ module SearchObject
|
|
5
5
|
describe Helper do
|
6
6
|
describe '.stringify_keys' do
|
7
7
|
it 'converts hash keys to strings' do
|
8
|
-
hash =
|
8
|
+
hash = described_class.stringify_keys a: 1, b: nil, c: false
|
9
9
|
expect(hash).to eq 'a' => 1, 'b' => nil, 'c' => false
|
10
10
|
end
|
11
11
|
|
12
12
|
it 'converts ActionController::Parameters to hash' do
|
13
13
|
params = ::ActionController::Parameters.new a: 1, b: nil, c: false
|
14
|
-
hash =
|
14
|
+
hash = described_class.stringify_keys params
|
15
15
|
expect(hash).to eq 'a' => 1, 'b' => nil, 'c' => false
|
16
16
|
end
|
17
17
|
end
|
18
18
|
|
19
19
|
describe '.slice_keys' do
|
20
20
|
it 'selects only given keys' do
|
21
|
-
hash =
|
21
|
+
hash = described_class.slice_keys({ a: 1, b: 2, c: 3 }, [:a, :b])
|
22
22
|
expect(hash).to eq a: 1, b: 2
|
23
23
|
end
|
24
24
|
|
25
25
|
it 'ignores not existing keys' do
|
26
|
-
hash =
|
26
|
+
hash = described_class.slice_keys({}, [:a, :b])
|
27
27
|
expect(hash).to eq({})
|
28
28
|
end
|
29
29
|
end
|
30
30
|
|
31
31
|
describe 'camelize' do
|
32
32
|
it "transforms :paging to 'Paging'" do
|
33
|
-
expect(
|
33
|
+
expect(described_class.camelize(:paging)).to eq 'Paging'
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
describe 'underscore' do
|
38
|
+
it "transforms 'veryPopular' to 'very_popular'" do
|
39
|
+
expect(described_class.underscore(:veryPopular)).to eq 'very_popular'
|
40
|
+
expect(described_class.underscore('VERY_POPULAR')).to eq 'very_popular'
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
describe '.normalize_filters' do
|
45
|
+
it 'combines defaults and filters' do
|
46
|
+
expect(described_class.normalize_params({ 'a' => 1, 'b' => 2 }, { a: 2 }, %w(a b))).to eq 'a' => 2, 'b' => 2
|
47
|
+
end
|
48
|
+
|
49
|
+
it 'excludes non specified keys' do
|
50
|
+
expect(described_class.normalize_params({ 'a' => 1 }, { b: 2 }, %w(a))).to eq 'a' => 1
|
51
|
+
end
|
52
|
+
|
53
|
+
it 'handles missing defaults' do
|
54
|
+
expect(described_class.normalize_params(nil, { a: 1 }, %w(a))).to eq 'a' => 1
|
55
|
+
end
|
56
|
+
|
57
|
+
it 'handles missing filters' do
|
58
|
+
expect(described_class.normalize_params(nil, nil, ['a'])).to eq({})
|
34
59
|
end
|
35
60
|
end
|
36
61
|
|
@@ -44,7 +69,7 @@ module SearchObject
|
|
44
69
|
null: nil
|
45
70
|
}
|
46
71
|
|
47
|
-
deep_copy =
|
72
|
+
deep_copy = described_class.deep_copy(original)
|
48
73
|
|
49
74
|
original[:array][0] = 42
|
50
75
|
original[:hash][:key] = 'other value'
|
@@ -0,0 +1,130 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
require 'ostruct'
|
3
|
+
|
4
|
+
module SearchObject
|
5
|
+
module Plugin
|
6
|
+
describe Enum do
|
7
|
+
class TestSearch
|
8
|
+
include SearchObject.module(:enum)
|
9
|
+
|
10
|
+
scope { [1, 2, 3, 4, 5] }
|
11
|
+
|
12
|
+
option :filter, enum: %w(odd even)
|
13
|
+
|
14
|
+
private
|
15
|
+
|
16
|
+
def apply_filter_with_odd(scope)
|
17
|
+
scope.select(&:odd?)
|
18
|
+
end
|
19
|
+
|
20
|
+
def apply_filter_with_even(scope)
|
21
|
+
scope.select(&:even?)
|
22
|
+
end
|
23
|
+
|
24
|
+
def handle_invalid_filter(_scope, value)
|
25
|
+
"invalid filter - #{value}"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
it 'can filter by enum values' do
|
30
|
+
expect(TestSearch.results(filters: { filter: 'odd' })).to eq [1, 3, 5]
|
31
|
+
expect(TestSearch.results(filters: { filter: 'even' })).to eq [2, 4]
|
32
|
+
end
|
33
|
+
|
34
|
+
it 'ignores blank values' do
|
35
|
+
expect(TestSearch.results(filters: { filter: nil })).to eq [1, 2, 3, 4, 5]
|
36
|
+
expect(TestSearch.results(filters: { filter: '' })).to eq [1, 2, 3, 4, 5]
|
37
|
+
end
|
38
|
+
|
39
|
+
it 'handles wrong enum values' do
|
40
|
+
expect(TestSearch.results(filters: { filter: 'foo' })).to eq 'invalid filter - foo'
|
41
|
+
end
|
42
|
+
|
43
|
+
it 'raises when block is passed with enum option' do
|
44
|
+
expect do
|
45
|
+
Class.new do
|
46
|
+
include SearchObject.module(:enum)
|
47
|
+
|
48
|
+
option(:filter, enum: %w(a b)) { |_scope, _value| nil }
|
49
|
+
end
|
50
|
+
end.to raise_error Enum::BlockIgnoredError
|
51
|
+
end
|
52
|
+
|
53
|
+
it 'raises when :with is passed with enum option' do
|
54
|
+
expect do
|
55
|
+
Class.new do
|
56
|
+
include SearchObject.module(:enum)
|
57
|
+
|
58
|
+
option :filter, enum: %w(a b), with: :method_name
|
59
|
+
end
|
60
|
+
end.to raise_error Enum::WithIgnoredError
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
describe Enum::Handler do
|
65
|
+
describe 'apply_filter' do
|
66
|
+
def new_object(&block)
|
67
|
+
klass = Class.new(&block)
|
68
|
+
klass.new
|
69
|
+
end
|
70
|
+
|
71
|
+
def call(object: nil, option: nil, enums: nil, scope: nil, value:)
|
72
|
+
described_class.apply_filter(
|
73
|
+
object: object || new_object,
|
74
|
+
option: option || 'option',
|
75
|
+
enums: enums || [value],
|
76
|
+
scope: scope || [],
|
77
|
+
value: value
|
78
|
+
)
|
79
|
+
end
|
80
|
+
|
81
|
+
it 'filters by methods based on the enum value' do
|
82
|
+
object = new_object do
|
83
|
+
private
|
84
|
+
|
85
|
+
def apply_select_with_name(scope)
|
86
|
+
scope.select { |value| value == 'name' }
|
87
|
+
end
|
88
|
+
|
89
|
+
def apply_select_with_age(scope)
|
90
|
+
scope.select { |value| value == 'age' }
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
scope = %w(name age location)
|
95
|
+
|
96
|
+
expect(call(object: object, option: 'select', scope: scope, value: 'name')).to eq %w(name)
|
97
|
+
expect(call(object: object, option: 'select', scope: scope, value: 'age')).to eq %w(age)
|
98
|
+
end
|
99
|
+
|
100
|
+
it 'raises NoMethodError when object can not handle enum method' do
|
101
|
+
expect { call(enums: ['a'], value: 'a') }.to raise_error NoMethodError
|
102
|
+
end
|
103
|
+
|
104
|
+
it 'raises error when value is not an enum' do
|
105
|
+
expect { call(enums: [], value: 'invalid') }.to raise_error Enum::InvalidEnumValueError
|
106
|
+
end
|
107
|
+
|
108
|
+
it 'can delegate missing enum value to object' do
|
109
|
+
object = new_object do
|
110
|
+
def handle_invalid_option(_scope, value)
|
111
|
+
"handles #{value} value"
|
112
|
+
end
|
113
|
+
end
|
114
|
+
|
115
|
+
expect(call(object: object, enums: [], value: 'invalid')).to eq 'handles invalid value'
|
116
|
+
end
|
117
|
+
|
118
|
+
it 'can delegate missing enum value to object (cath all)' do
|
119
|
+
object = new_object do
|
120
|
+
def handle_invalid_enum(option, _scope, value)
|
121
|
+
"handles #{value} value for #{option}"
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
expect(call(object: object, enums: [], value: 'invalid')).to eq 'handles invalid value for option'
|
126
|
+
end
|
127
|
+
end
|
128
|
+
end
|
129
|
+
end
|
130
|
+
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: search_object
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.
|
4
|
+
version: 1.2.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Radoslav Stankov
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2017-04-
|
11
|
+
date: 2017-04-27 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: bundler
|
@@ -150,6 +150,20 @@ dependencies:
|
|
150
150
|
- - ">="
|
151
151
|
- !ruby/object:Gem::Version
|
152
152
|
version: '0'
|
153
|
+
- !ruby/object:Gem::Dependency
|
154
|
+
name: kaminari-activerecord
|
155
|
+
requirement: !ruby/object:Gem::Requirement
|
156
|
+
requirements:
|
157
|
+
- - ">="
|
158
|
+
- !ruby/object:Gem::Version
|
159
|
+
version: '0'
|
160
|
+
type: :development
|
161
|
+
prerelease: false
|
162
|
+
version_requirements: !ruby/object:Gem::Requirement
|
163
|
+
requirements:
|
164
|
+
- - ">="
|
165
|
+
- !ruby/object:Gem::Version
|
166
|
+
version: '0'
|
153
167
|
- !ruby/object:Gem::Dependency
|
154
168
|
name: rubocop
|
155
169
|
requirement: !ruby/object:Gem::Requirement
|
@@ -186,6 +200,7 @@ extensions: []
|
|
186
200
|
extra_rdoc_files: []
|
187
201
|
files:
|
188
202
|
- ".gitignore"
|
203
|
+
- ".projections.json"
|
189
204
|
- ".rspec"
|
190
205
|
- ".rubocop.yml"
|
191
206
|
- ".travis.yml"
|
@@ -238,6 +253,7 @@ files:
|
|
238
253
|
- lib/search_object/base.rb
|
239
254
|
- lib/search_object/errors.rb
|
240
255
|
- lib/search_object/helper.rb
|
256
|
+
- lib/search_object/plugin/enum.rb
|
241
257
|
- lib/search_object/plugin/kaminari.rb
|
242
258
|
- lib/search_object/plugin/model.rb
|
243
259
|
- lib/search_object/plugin/paging.rb
|
@@ -248,6 +264,7 @@ files:
|
|
248
264
|
- search_object.gemspec
|
249
265
|
- spec/search_object/base_spec.rb
|
250
266
|
- spec/search_object/helper_spec.rb
|
267
|
+
- spec/search_object/plugin/enum_spec.rb
|
251
268
|
- spec/search_object/plugin/kaminari_spec.rb
|
252
269
|
- spec/search_object/plugin/model_spec.rb
|
253
270
|
- spec/search_object/plugin/paging_spec.rb
|
@@ -256,7 +273,6 @@ files:
|
|
256
273
|
- spec/search_object/search_spec.rb
|
257
274
|
- spec/spec_helper.rb
|
258
275
|
- spec/spec_helper_active_record.rb
|
259
|
-
- spec/support/kaminari_setup.rb
|
260
276
|
- spec/support/paging_shared_example.rb
|
261
277
|
homepage: https://github.com/RStankov/SearchObject
|
262
278
|
licenses:
|
@@ -285,6 +301,7 @@ summary: Provides DSL for creating search objects
|
|
285
301
|
test_files:
|
286
302
|
- spec/search_object/base_spec.rb
|
287
303
|
- spec/search_object/helper_spec.rb
|
304
|
+
- spec/search_object/plugin/enum_spec.rb
|
288
305
|
- spec/search_object/plugin/kaminari_spec.rb
|
289
306
|
- spec/search_object/plugin/model_spec.rb
|
290
307
|
- spec/search_object/plugin/paging_spec.rb
|
@@ -293,5 +310,4 @@ test_files:
|
|
293
310
|
- spec/search_object/search_spec.rb
|
294
311
|
- spec/spec_helper.rb
|
295
312
|
- spec/spec_helper_active_record.rb
|
296
|
-
- spec/support/kaminari_setup.rb
|
297
313
|
- spec/support/paging_shared_example.rb
|