active_record-json_associations 0.6.9 → 0.9.1

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
- SHA1:
3
- metadata.gz: 6dcf607cd60319296ad2e04a4f96de08273b36bd
4
- data.tar.gz: b35848ecd6ca8e044a390889a45aaba54192f600
2
+ SHA256:
3
+ metadata.gz: e03d284df9e4faf9319475962d17aee50025fc58407f1cb641819c070958eca6
4
+ data.tar.gz: 026dff5c42712728a8c85a59e413265f3c1c09922ec1fc366fee5db3404fb42a
5
5
  SHA512:
6
- metadata.gz: b8d2c2ecf2271d0b37988c4653a8f1e6e81bee6ac9842c72a2e3c9b098c7d44ee58aaa64d010038ee5ab4b20d17e00ef250dd2ceade22baecd4bcc67f6322b61
7
- data.tar.gz: ab14750a0370ac2532ecab1c7f13bb2b1d48db6a8731178247457b142b0807c902483e07da6850943053231c7b336931f3a4d034643a24f3be88cd8eb04a3ad0
6
+ metadata.gz: 449f81549f3ed177a88ce17070a92b2f6b07f252193ab1cf64d706d51c85d9065f636af21242bd3a689f1b2402384d2d0487c58cc74de1f07b1c3f441da3b34d
7
+ data.tar.gz: 5dea4c68f936961403ca768f7775795fe810df3e811cadb147901faa0e2e5313512853aed0d5bdaaabefafaeb60f9cdc3375d5d5d986ca2cde42be6759b94125
data/.gitignore CHANGED
@@ -6,6 +6,7 @@
6
6
  .ruby-version
7
7
  .ruby-gemset
8
8
  Gemfile.lock
9
+ gemfiles/*.lock
9
10
  InstalledFiles
10
11
  _yardoc
11
12
  coverage
@@ -1,6 +1,18 @@
1
1
  language: ruby
2
2
  rvm:
3
- - 2.2.2
4
- - 2.3.3
5
- - 2.4.0
3
+ - 2.4
4
+ - 2.5
5
+ - 2.6
6
+ - 2.7
7
+ gemfile:
8
+ - gemfiles/rails_5.0.gemfile
9
+ - gemfiles/rails_5.1.gemfile
10
+ - gemfiles/rails_5.2.gemfile
11
+ - gemfiles/rails_6.0.gemfile
12
+ jobs:
13
+ exclude:
14
+ - rvm: 2.4
15
+ gemfile: gemfiles/rails_6.0.gemfile
16
+ before_install:
17
+ - gem install bundler -v"~>1.17"
6
18
 
@@ -0,0 +1,21 @@
1
+ appraise "rails-5.0" do
2
+ gem "rails", "~>5.0.0"
3
+ gem "sqlite3", "~>1.3.13" # 1.4 seems to break rails 5.0?
4
+ end
5
+
6
+ appraise "rails-5.1" do
7
+ gem "rails", "~>5.1.0"
8
+ end
9
+
10
+ appraise "rails-5.2" do
11
+ gem "rails", "~>5.2.0"
12
+ end
13
+
14
+ appraise "rails-6.0" do
15
+ gem "rails", "~>6.0.0"
16
+ end
17
+
18
+ appraise "rails-6.1" do
19
+ gem "rails", "~>6.1.0"
20
+ end
21
+
data/README.md CHANGED
@@ -37,6 +37,20 @@ parent.child_ids #=> [1,2]
37
37
  parent.children? #=> true
38
38
  ```
39
39
 
40
+ And a scope method for finding records assocatied with an id:
41
+
42
+ ```ruby
43
+ Parent.child_ids_including(2) # => [<Parent child_ids: [1,2,3]>]
44
+ ```
45
+
46
+ Or any of specified array of ids:
47
+
48
+ ```ruby
49
+ Parent.child_ids_including([2,4,5]) # => [<Parent child_ids: [1,2,3]>]
50
+ ```
51
+
52
+ `touch: true` can be specified on belongs_to_many to touch the associated records' timestamps when the record is modified.
53
+
40
54
  It also adds an `json_foreign_key` option to `has_many` for specifying that the foreign keys are in a json array.
41
55
 
42
56
  ```ruby
@@ -20,10 +20,12 @@ Gem::Specification.new do |spec|
20
20
 
21
21
  spec.add_dependency "activerecord"
22
22
 
23
- spec.add_development_dependency "bundler", "~> 1.6"
23
+ spec.add_development_dependency "bundler"
24
+ spec.add_development_dependency "appraisal"
24
25
  spec.add_development_dependency "rake"
25
26
  spec.add_development_dependency "rspec"
26
27
  spec.add_development_dependency "sqlite3"
27
28
  spec.add_development_dependency "byebug"
29
+ spec.add_development_dependency "timecop"
28
30
  end
29
31
 
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
@@ -0,0 +1,8 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rails", "~>5.0.0"
6
+ gem "sqlite3", "~>1.3.13"
7
+
8
+ gemspec path: "../"
@@ -0,0 +1,7 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rails", "~>5.1.0"
6
+
7
+ gemspec path: "../"
@@ -0,0 +1,7 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rails", "~>5.2.0"
6
+
7
+ gemspec path: "../"
@@ -0,0 +1,7 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rails", "~>6.0.0"
6
+
7
+ gemspec path: "../"
@@ -0,0 +1,7 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rails", "~>6.1.0"
6
+
7
+ gemspec path: "../"
@@ -3,7 +3,15 @@ require "json"
3
3
 
4
4
  module ActiveRecord
5
5
  module JsonAssociations
6
- def belongs_to_many(many, class_name: nil)
6
+ FIELD_INCLUDE_SCOPE_BUILDER_PROC = proc do |context, field, id|
7
+ context.where("#{field}='[#{id}]'").or(
8
+ context.where("#{field} LIKE '[#{id},%'")).or(
9
+ context.where("#{field} LIKE '%,#{id},%'")).or(
10
+ context.where("#{field} LIKE '%,#{id}]'"))
11
+ end
12
+ private_constant :FIELD_INCLUDE_SCOPE_BUILDER_PROC
13
+
14
+ def belongs_to_many(many, class_name: nil, touch: nil)
7
15
  one = many.to_s.singularize
8
16
  one_ids = :"#{one}_ids"
9
17
  one_ids_equals = :"#{one_ids}="
@@ -14,9 +22,44 @@ module ActiveRecord
14
22
 
15
23
  serialize one_ids, JSON
16
24
 
25
+ if touch
26
+ after_commit do
27
+ method = respond_to?(:saved_changes) ? :saved_changes : :previous_changes
28
+ old_ids, new_ids = send(method)[one_ids.to_s]
29
+ ids = Array(send(one_ids)) | Array(old_ids) | Array(new_ids)
30
+ scope = class_name.constantize.where(self.class.primary_key => ids)
31
+
32
+ if scope.respond_to?(:touch) # AR 6.0+
33
+ scope.touch_all
34
+ elsif self.class.respond_to?(:touch_attributes_with_time) # AR 5.1+
35
+ scope.update_all self.class.touch_attributes_with_time
36
+ else # AR 5.0
37
+ attributes = timestamp_attributes_for_update_in_model.inject({}) do |attributes, key|
38
+ attributes.merge(key => current_time_from_proper_timezone)
39
+ end
40
+ scope.update_all attributes
41
+ end
42
+ end
43
+ end
44
+
45
+ extend Module.new {
46
+ define_method :"#{one_ids}_including" do |id|
47
+ raise "can't query for a record that does not have an id!" if id.blank?
48
+ if id.is_a?(Hash)
49
+ Array(id[:any]).inject(none) do |context, id|
50
+ context.or(FIELD_INCLUDE_SCOPE_BUILDER_PROC.call(self, one_ids, id))
51
+ end
52
+ else
53
+ FIELD_INCLUDE_SCOPE_BUILDER_PROC.call(self, one_ids, id)
54
+ end
55
+ end
56
+ }
57
+
17
58
  include Module.new {
18
59
  define_method one_ids do
19
- super() || []
60
+ super().tap do |value|
61
+ return send(one_ids_equals, []) if value.nil?
62
+ end
20
63
  end
21
64
 
22
65
  define_method one_ids_equals do |ids|
@@ -47,7 +90,7 @@ module ActiveRecord
47
90
  }
48
91
  end
49
92
 
50
- def has_many many, scope = nil, options = {}, &extension
93
+ def has_many many, scope = nil, **options, &extension
51
94
  unless (scope.is_a?(Hash) && scope[:json_foreign_key]) || (options.is_a?(Hash) && options[:json_foreign_key])
52
95
  return super
53
96
  end
@@ -81,10 +124,7 @@ module ActiveRecord
81
124
 
82
125
  define_method many do
83
126
  klass = class_name.constantize
84
- klass.where("#{foreign_key} LIKE '[#{id}]'").or(
85
- klass.where("#{foreign_key} LIKE '[#{id},%'")).or(
86
- klass.where("#{foreign_key} LIKE '%,#{id},%'")).or(
87
- klass.where("#{foreign_key} LIKE '%,#{id}]'"))
127
+ FIELD_INCLUDE_SCOPE_BUILDER_PROC.call(klass, foreign_key, id)
88
128
  end
89
129
 
90
130
  define_method many_equals do |collection|
@@ -1,5 +1,5 @@
1
1
  module ActiveRecord
2
2
  module JsonAssociations
3
- VERSION = "0.6.9"
3
+ VERSION = "0.9.1"
4
4
  end
5
5
  end
@@ -9,16 +9,21 @@ describe ActiveRecord::JsonAssociations do
9
9
  create_table :parents do |t|
10
10
  t.text :child_ids
11
11
  t.text :fuzzy_ids
12
+ t.timestamps
12
13
  end
13
14
 
14
- create_table :children
15
+ create_table :children do |t|
16
+ t.timestamps
17
+ end
15
18
 
16
- create_table :pets
19
+ create_table :pets do |t|
20
+ t.timestamps
21
+ end
17
22
  end
18
23
  end
19
24
 
20
25
  class Parent < ActiveRecord::Base
21
- belongs_to_many :children
26
+ belongs_to_many :children, touch: true
22
27
  belongs_to_many :fuzzies, class_name: "Pet"
23
28
  end
24
29
 
@@ -38,6 +43,148 @@ describe ActiveRecord::JsonAssociations do
38
43
 
39
44
  describe ".belongs_to_many :children" do
40
45
  subject { Parent.new }
46
+ let!(:winner) { Parent.create! }
47
+
48
+ describe ".child_ids_including" do
49
+ context "finds records with the specified id" do
50
+ let(:child) { Child.create! }
51
+
52
+ it "as the whole json array" do
53
+ parent = Parent.create(children: [child])
54
+ expect(Parent.child_ids_including(child.id)).to eq [parent]
55
+ end
56
+
57
+ it "at the beginning of the json array" do
58
+ parent = Parent.create(children: [child, Child.create!])
59
+ expect(Parent.child_ids_including(child.id)).to eq [parent]
60
+ end
61
+
62
+ it "in the middle of the json array" do
63
+ parent = Parent.create(children: [Child.create!, child, Child.create!])
64
+ expect(Parent.child_ids_including(child.id)).to eq [parent]
65
+ end
66
+
67
+ it "at the end of the json array" do
68
+ parent = Parent.create(children: [Child.create!, child])
69
+ expect(Parent.child_ids_including(child.id)).to eq [parent]
70
+ end
71
+ end
72
+
73
+ context "finds records including any of the specified array of ids" do
74
+ let(:peter) { Child.create! }
75
+ let(:paul) { Child.create! }
76
+
77
+ it "both as the whole json array" do
78
+ parent = Parent.create(children: [peter, paul])
79
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
80
+ end
81
+
82
+ it "one as the whole json array" do
83
+ parent = Parent.create(children: [peter])
84
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
85
+ end
86
+
87
+ it "the other as the whole json array" do
88
+ parent = Parent.create(children: [paul])
89
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
90
+ end
91
+
92
+ it "both at the beginning of the json array" do
93
+ parent = Parent.create(children: [peter, paul, Child.create!])
94
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
95
+ end
96
+
97
+ it "one at the beginning of the json array" do
98
+ parent = Parent.create(children: [peter, Child.create!])
99
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
100
+ end
101
+
102
+ it "the other at the beginning of the json array" do
103
+ parent = Parent.create(children: [paul, Child.create!])
104
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
105
+ end
106
+
107
+ it "both in the middle of the json array" do
108
+ parent = Parent.create(children: [Child.create!, peter, paul, Child.create!])
109
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
110
+ end
111
+
112
+ it "one in the middle of the json array" do
113
+ parent = Parent.create(children: [Child.create!, peter, Child.create!])
114
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
115
+ end
116
+
117
+ it "the other in the middle of the json array" do
118
+ parent = Parent.create(children: [Child.create!, paul, Child.create!])
119
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
120
+ end
121
+
122
+ it "both at the end of the json array" do
123
+ parent = Parent.create(children: [Child.create!, peter, paul])
124
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
125
+ end
126
+
127
+ it "one at the end of the json array" do
128
+ parent = Parent.create(children: [Child.create!, peter])
129
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
130
+ end
131
+
132
+ it "the other at the end of the json array" do
133
+ parent = Parent.create(children: [Child.create!, paul])
134
+ expect(Parent.child_ids_including(any: [peter.id, paul.id])).to eq [parent]
135
+ end
136
+ end
137
+ end
138
+
139
+ describe "touch: true" do
140
+ around do |example|
141
+ old_zone = Time.zone
142
+ Time.zone = "UTC"
143
+ example.run
144
+ Time.zone = old_zone
145
+ end
146
+
147
+ let(:old_time) { 1.year.ago.round }
148
+ let(:new_time) { 1.second.ago.round }
149
+
150
+ around do |example|
151
+ Timecop.freeze(new_time) do
152
+ example.run
153
+ end
154
+ end
155
+
156
+ it "touches records associated upon creation" do
157
+ children = [Child.create!(updated_at: old_time), Child.create!(updated_at: old_time)]
158
+ fuzzies = [Pet.create!(updated_at: old_time), Pet.create!(updated_at: old_time)]
159
+ parent = Parent.create!(children: children, fuzzies: fuzzies)
160
+ expect(children.each(&:reload).map(&:updated_at)).to eq [new_time, new_time] # touch: true
161
+ expect(fuzzies.each(&:reload).map(&:updated_at)).to eq [old_time, old_time] # touch: nil
162
+ end
163
+
164
+ it "touches exising association records" do
165
+ children = [Child.create!, Child.create!]
166
+ parent = Parent.create!(children: children)
167
+ children.each { |child| child.update!(updated_at: old_time) }
168
+ parent.save!
169
+ expect(children.each(&:reload).map(&:updated_at)).to eq [new_time, new_time]
170
+ end
171
+
172
+ it "touches removed associated records" do
173
+ peter, paul, mary = Child.create!, Child.create!, Child.create!
174
+ parent = Parent.create!(children: [peter, paul, mary])
175
+ [peter, paul, mary].each { |child| child.update_column :updated_at, old_time }
176
+ parent.update!(children: [peter, paul])
177
+ expect([peter, paul, mary].each(&:reload).map(&:updated_at)).to eq [new_time, new_time, new_time]
178
+ end
179
+
180
+ it "touches added associated records" do
181
+ peter, paul, mary = Child.create!, Child.create!, Child.create!
182
+ parent = Parent.create!(children: [peter, paul])
183
+ [peter, paul, mary].each { |child| child.update_column :updated_at, old_time }
184
+ parent.update!(children: [peter, paul, mary])
185
+ expect([peter, paul, mary].each(&:reload).map(&:updated_at)).to eq [new_time, new_time, new_time]
186
+ end
187
+ end
41
188
 
42
189
  describe "#child_ids" do
43
190
  it "is empty by default" do
@@ -48,6 +195,13 @@ describe ActiveRecord::JsonAssociations do
48
195
  subject.child_ids = [1,2,3]
49
196
  expect(subject.child_ids).to eq [1,2,3]
50
197
  end
198
+
199
+ it "can be pushed to" do
200
+ subject.child_ids << 1
201
+ subject.child_ids << 2
202
+ subject.child_ids << 3
203
+ expect(subject.child_ids).to eq [1,2,3]
204
+ end
51
205
  end
52
206
 
53
207
  describe "#child_ids=" do
@@ -1,4 +1,5 @@
1
1
  require "byebug"
2
+ require "timecop"
2
3
 
3
4
  RSpec.configure do |config|
4
5
  config.filter_run focus: true
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: active_record-json_associations
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.6.9
4
+ version: 0.9.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Micah Geisel
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2018-08-01 00:00:00.000000000 Z
11
+ date: 2021-01-24 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activerecord
@@ -28,16 +28,30 @@ dependencies:
28
28
  name: bundler
29
29
  requirement: !ruby/object:Gem::Requirement
30
30
  requirements:
31
- - - "~>"
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
32
39
  - !ruby/object:Gem::Version
33
- version: '1.6'
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: appraisal
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
34
48
  type: :development
35
49
  prerelease: false
36
50
  version_requirements: !ruby/object:Gem::Requirement
37
51
  requirements:
38
- - - "~>"
52
+ - - ">="
39
53
  - !ruby/object:Gem::Version
40
- version: '1.6'
54
+ version: '0'
41
55
  - !ruby/object:Gem::Dependency
42
56
  name: rake
43
57
  requirement: !ruby/object:Gem::Requirement
@@ -94,21 +108,43 @@ dependencies:
94
108
  - - ">="
95
109
  - !ruby/object:Gem::Version
96
110
  version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: timecop
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - ">="
116
+ - !ruby/object:Gem::Version
117
+ version: '0'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - ">="
123
+ - !ruby/object:Gem::Version
124
+ version: '0'
97
125
  description: Instead of a many-to-many join table, serialize the ids into a JSON array.
98
126
  email:
99
127
  - micah@botandrose.com
100
- executables: []
128
+ executables:
129
+ - setup
101
130
  extensions: []
102
131
  extra_rdoc_files: []
103
132
  files:
104
133
  - ".gitignore"
105
134
  - ".rspec"
106
135
  - ".travis.yml"
136
+ - Appraisals
107
137
  - Gemfile
108
138
  - LICENSE.txt
109
139
  - README.md
110
140
  - Rakefile
111
141
  - active_record-json_associations.gemspec
142
+ - bin/setup
143
+ - gemfiles/rails_5.0.gemfile
144
+ - gemfiles/rails_5.1.gemfile
145
+ - gemfiles/rails_5.2.gemfile
146
+ - gemfiles/rails_6.0.gemfile
147
+ - gemfiles/rails_6.1.gemfile
112
148
  - lib/active_record/json_associations.rb
113
149
  - lib/active_record/json_associations/version.rb
114
150
  - spec/json_associations_spec.rb
@@ -132,8 +168,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
132
168
  - !ruby/object:Gem::Version
133
169
  version: '0'
134
170
  requirements: []
135
- rubyforge_project:
136
- rubygems_version: 2.4.8
171
+ rubygems_version: 3.0.3
137
172
  signing_key:
138
173
  specification_version: 4
139
174
  summary: Instead of a many-to-many join table, serialize the ids into a JSON array.