terrestrial 0.1.0 → 0.1.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.gitignore +1 -9
- data/.rspec +2 -0
- data/.ruby-version +1 -0
- data/CODE_OF_CONDUCT.md +28 -0
- data/Gemfile.lock +73 -0
- data/LICENSE.txt +22 -0
- data/MissingFeatures.md +64 -0
- data/README.md +161 -16
- data/Rakefile +30 -0
- data/TODO.md +41 -0
- data/features/env.rb +60 -0
- data/features/example.feature +120 -0
- data/features/step_definitions/example_steps.rb +46 -0
- data/lib/terrestrial/abstract_record.rb +99 -0
- data/lib/terrestrial/association_loaders.rb +52 -0
- data/lib/terrestrial/collection_mutability_proxy.rb +81 -0
- data/lib/terrestrial/configurations/conventional_association_configuration.rb +186 -0
- data/lib/terrestrial/configurations/conventional_configuration.rb +302 -0
- data/lib/terrestrial/dataset.rb +49 -0
- data/lib/terrestrial/deleted_record.rb +20 -0
- data/lib/terrestrial/dirty_map.rb +42 -0
- data/lib/terrestrial/graph_loader.rb +63 -0
- data/lib/terrestrial/graph_serializer.rb +91 -0
- data/lib/terrestrial/identity_map.rb +22 -0
- data/lib/terrestrial/lazy_collection.rb +74 -0
- data/lib/terrestrial/lazy_object_proxy.rb +55 -0
- data/lib/terrestrial/many_to_many_association.rb +138 -0
- data/lib/terrestrial/many_to_one_association.rb +66 -0
- data/lib/terrestrial/mapper_facade.rb +137 -0
- data/lib/terrestrial/one_to_many_association.rb +66 -0
- data/lib/terrestrial/public_conveniencies.rb +139 -0
- data/lib/terrestrial/query_order.rb +32 -0
- data/lib/terrestrial/relation_mapping.rb +50 -0
- data/lib/terrestrial/serializer.rb +18 -0
- data/lib/terrestrial/short_inspection_string.rb +18 -0
- data/lib/terrestrial/struct_factory.rb +17 -0
- data/lib/terrestrial/subset_queries_proxy.rb +11 -0
- data/lib/terrestrial/upserted_record.rb +15 -0
- data/lib/terrestrial/version.rb +1 -1
- data/lib/terrestrial.rb +5 -2
- data/sequel_mapper.gemspec +31 -0
- data/spec/config_override_spec.rb +193 -0
- data/spec/custom_serializers_spec.rb +49 -0
- data/spec/deletion_spec.rb +101 -0
- data/spec/graph_persistence_spec.rb +313 -0
- data/spec/graph_traversal_spec.rb +121 -0
- data/spec/new_graph_persistence_spec.rb +71 -0
- data/spec/object_identity_spec.rb +70 -0
- data/spec/ordered_association_spec.rb +51 -0
- data/spec/persistence_efficiency_spec.rb +224 -0
- data/spec/predefined_queries_spec.rb +62 -0
- data/spec/proxying_spec.rb +88 -0
- data/spec/querying_spec.rb +48 -0
- data/spec/readme_examples_spec.rb +35 -0
- data/spec/sequel_mapper/abstract_record_spec.rb +244 -0
- data/spec/sequel_mapper/collection_mutability_proxy_spec.rb +135 -0
- data/spec/sequel_mapper/deleted_record_spec.rb +59 -0
- data/spec/sequel_mapper/dirty_map_spec.rb +214 -0
- data/spec/sequel_mapper/lazy_collection_spec.rb +119 -0
- data/spec/sequel_mapper/lazy_object_proxy_spec.rb +140 -0
- data/spec/sequel_mapper/public_conveniencies_spec.rb +58 -0
- data/spec/sequel_mapper/upserted_record_spec.rb +59 -0
- data/spec/spec_helper.rb +36 -0
- data/spec/support/blog_schema.rb +38 -0
- data/spec/support/have_persisted_matcher.rb +19 -0
- data/spec/support/mapper_setup.rb +221 -0
- data/spec/support/mock_sequel.rb +193 -0
- data/spec/support/object_graph_setup.rb +139 -0
- data/spec/support/seed_data_setup.rb +165 -0
- data/spec/support/sequel_persistence_setup.rb +19 -0
- data/spec/support/sequel_test_support.rb +166 -0
- metadata +207 -13
- data/.travis.yml +0 -4
- data/bin/console +0 -14
- data/bin/setup +0 -7
- data/terrestrial.gemspec +0 -23
@@ -0,0 +1,135 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
require "terrestrial/collection_mutability_proxy"
|
4
|
+
|
5
|
+
RSpec.describe Terrestrial::CollectionMutabilityProxy do
|
6
|
+
let(:proxy) {
|
7
|
+
Terrestrial::CollectionMutabilityProxy.new(lazy_enum)
|
8
|
+
}
|
9
|
+
|
10
|
+
let(:lazy_enum) { data_set.each.lazy }
|
11
|
+
let(:data_set) { (0..9) }
|
12
|
+
|
13
|
+
def id
|
14
|
+
->(x) { x }
|
15
|
+
end
|
16
|
+
|
17
|
+
it "is Enumerable" do
|
18
|
+
expect(proxy).to be_a(Enumerable)
|
19
|
+
end
|
20
|
+
|
21
|
+
describe "#to_a" do
|
22
|
+
it "is equivalent to the original enumeration" do
|
23
|
+
expect(proxy.map(&id)).to eq(data_set.to_a)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
describe "#to_ary" do
|
28
|
+
it "is equivalent to the original enumeration" do
|
29
|
+
expect(proxy.to_ary).to eq(data_set.to_a)
|
30
|
+
end
|
31
|
+
|
32
|
+
it "implicitly coerces to Array" do
|
33
|
+
expect([-1].concat(proxy)).to eq([-1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9])
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
describe "#each" do
|
38
|
+
context "when called with a block" do
|
39
|
+
it "returns self" do
|
40
|
+
expect(proxy.each(&id)).to eq(proxy)
|
41
|
+
end
|
42
|
+
|
43
|
+
it "yields each element to the block" do
|
44
|
+
yielded = []
|
45
|
+
|
46
|
+
proxy.each do |element|
|
47
|
+
yielded.push(element)
|
48
|
+
end
|
49
|
+
|
50
|
+
expect(yielded).to eq(data_set.to_a)
|
51
|
+
end
|
52
|
+
|
53
|
+
context "when calling each more than once" do
|
54
|
+
before do
|
55
|
+
proxy.each { |x| nil }
|
56
|
+
proxy.each { |x| nil }
|
57
|
+
end
|
58
|
+
|
59
|
+
it "rewinds the enumeration on each call" do
|
60
|
+
expect(proxy.map(&id)).to eq(data_set.to_a)
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
context "when a new element is pushed into the collection" do
|
66
|
+
let(:new_element) { double(:new_element) }
|
67
|
+
|
68
|
+
before do
|
69
|
+
proxy.push(new_element)
|
70
|
+
end
|
71
|
+
|
72
|
+
it "adds the new element to the enumeration" do
|
73
|
+
expect(proxy.to_a.last).to eq(new_element)
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
describe "#each_loaded" do
|
79
|
+
context "when called with a block" do
|
80
|
+
it "returns self" do
|
81
|
+
expect(proxy.each(&id)).to eq(proxy)
|
82
|
+
end
|
83
|
+
|
84
|
+
it "yields each element to the block" do
|
85
|
+
yielded = []
|
86
|
+
|
87
|
+
proxy.each do |element|
|
88
|
+
yielded.push(element)
|
89
|
+
end
|
90
|
+
|
91
|
+
expect(yielded).to eq(data_set.to_a)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
context "when called without a block" do
|
96
|
+
it "returns an enumerator" do
|
97
|
+
expect(proxy.each).to be_a(Enumerator)
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
101
|
+
|
102
|
+
describe "#delete" do
|
103
|
+
it "returns self" do
|
104
|
+
expect(proxy.delete(3)).to be(proxy)
|
105
|
+
end
|
106
|
+
|
107
|
+
context "after removing a element from the enumeration" do
|
108
|
+
before do
|
109
|
+
proxy.delete(3)
|
110
|
+
end
|
111
|
+
|
112
|
+
it "skips that element in the enumeration" do
|
113
|
+
expect(proxy.map(&id)).to eq([0,1,2,4,5,6,7,8,9])
|
114
|
+
end
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
describe "#push" do
|
119
|
+
context "after pushing another element into the enumeration" do
|
120
|
+
before do
|
121
|
+
proxy.push(new_value)
|
122
|
+
end
|
123
|
+
|
124
|
+
let(:new_value) { double(:new_value) }
|
125
|
+
|
126
|
+
it "does not alter the other elements" do
|
127
|
+
expect(proxy.map(&id)[0..-2]).to eq([0,1,2,3,4,5,6,7,8,9])
|
128
|
+
end
|
129
|
+
|
130
|
+
it "appends the element to the enumeration" do
|
131
|
+
expect(proxy.map(&id).last).to eq(new_value)
|
132
|
+
end
|
133
|
+
end
|
134
|
+
end
|
135
|
+
end
|
@@ -0,0 +1,59 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
require "terrestrial/deleted_record"
|
4
|
+
|
5
|
+
RSpec.describe Terrestrial::DeletedRecord do
|
6
|
+
subject(:record) {
|
7
|
+
Terrestrial::DeletedRecord.new(namespace, identity, raw_data)
|
8
|
+
}
|
9
|
+
|
10
|
+
let(:namespace) { double(:namespace) }
|
11
|
+
|
12
|
+
let(:identity) {
|
13
|
+
{ id: id }
|
14
|
+
}
|
15
|
+
|
16
|
+
let(:raw_data) {
|
17
|
+
{
|
18
|
+
name: name,
|
19
|
+
}
|
20
|
+
}
|
21
|
+
|
22
|
+
let(:id) { double(:id) }
|
23
|
+
let(:name) { double(:name) }
|
24
|
+
|
25
|
+
describe "#if_delete" do
|
26
|
+
it "invokes the callback" do
|
27
|
+
expect { |callback|
|
28
|
+
record.if_delete(&callback)
|
29
|
+
}.to yield_with_args(record)
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
describe "#==" do
|
34
|
+
context "with another record that deletes" do
|
35
|
+
let(:comparitor) {
|
36
|
+
record.merge({})
|
37
|
+
}
|
38
|
+
|
39
|
+
it "is equal" do
|
40
|
+
expect(record.==(comparitor)).to be(true)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
context "with another record that does not delete" do
|
45
|
+
let(:comparitor) {
|
46
|
+
Class.new(Terrestrial::AbstractRecord) do
|
47
|
+
protected
|
48
|
+
def operation
|
49
|
+
:something_else
|
50
|
+
end
|
51
|
+
end
|
52
|
+
}
|
53
|
+
|
54
|
+
it "is not equal" do
|
55
|
+
expect(record.==(comparitor)).to be(false)
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
@@ -0,0 +1,214 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
require "terrestrial/dirty_map"
|
4
|
+
require "terrestrial/upserted_record"
|
5
|
+
require "terrestrial/deleted_record"
|
6
|
+
|
7
|
+
RSpec.describe Terrestrial::DirtyMap do
|
8
|
+
subject(:dirty_map) {
|
9
|
+
Terrestrial::DirtyMap.new(storage)
|
10
|
+
}
|
11
|
+
|
12
|
+
let(:storage) { {} }
|
13
|
+
|
14
|
+
let(:loaded_record) {
|
15
|
+
create_record( namespace, identity_fields, attributes, depth)
|
16
|
+
}
|
17
|
+
|
18
|
+
let(:namespace) { :table_name }
|
19
|
+
let(:identity_fields) { [:id] }
|
20
|
+
let(:depth) { 0 }
|
21
|
+
|
22
|
+
let(:attributes) {
|
23
|
+
{
|
24
|
+
id: "record/id",
|
25
|
+
name: "record/name",
|
26
|
+
email: "record/email",
|
27
|
+
}
|
28
|
+
}
|
29
|
+
|
30
|
+
describe "#load" do
|
31
|
+
it "adds the record to its storage" do
|
32
|
+
dirty_map.load(loaded_record)
|
33
|
+
|
34
|
+
expect(storage.values).to include(loaded_record)
|
35
|
+
end
|
36
|
+
|
37
|
+
it "returns the loaded record" do
|
38
|
+
expect(dirty_map.load(loaded_record)).to eq(loaded_record)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
describe "#dirty" do
|
43
|
+
let(:clean_record) {
|
44
|
+
create_record(namespace, identity_fields, attributes, depth)
|
45
|
+
}
|
46
|
+
|
47
|
+
let(:dirty_record) {
|
48
|
+
create_record(
|
49
|
+
namespace,
|
50
|
+
identity_fields,
|
51
|
+
attributes.merge(name: "record/dirty_name"),
|
52
|
+
depth,
|
53
|
+
)
|
54
|
+
}
|
55
|
+
|
56
|
+
context "when the record has not been loaded (new record)" do
|
57
|
+
it "return true" do
|
58
|
+
expect(dirty_map.dirty?(clean_record)).to be(true)
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
context "when a record with same identity has been loaded (existing record)" do
|
63
|
+
before do
|
64
|
+
dirty_map.load(loaded_record)
|
65
|
+
end
|
66
|
+
|
67
|
+
context "when the record is unchanged" do
|
68
|
+
it "returns false" do
|
69
|
+
expect(dirty_map.dirty?(clean_record)).to be(false)
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
context "when the record's attributes are changed" do
|
74
|
+
it "returns true" do
|
75
|
+
expect(dirty_map.dirty?(dirty_record)).to be(true)
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
context "when the record is deleted" do
|
80
|
+
let(:deleted_record) {
|
81
|
+
Terrestrial::DeletedRecord.new(
|
82
|
+
namespace,
|
83
|
+
identity_fields,
|
84
|
+
attributes,
|
85
|
+
depth,
|
86
|
+
)
|
87
|
+
}
|
88
|
+
|
89
|
+
it "is always dirty" do
|
90
|
+
expect(dirty_map.dirty?(deleted_record)).to be(true)
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
context "when the record's attributes hash is mutated" do
|
95
|
+
before do
|
96
|
+
attributes.merge!(name: "new_value")
|
97
|
+
end
|
98
|
+
|
99
|
+
it "returns true" do
|
100
|
+
expect(dirty_map.dirty?(clean_record)).to be(true)
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
context "when a record's string value is mutated" do
|
105
|
+
before do
|
106
|
+
attributes.fetch(:name) << "MUTANT"
|
107
|
+
end
|
108
|
+
|
109
|
+
it "returns true" do
|
110
|
+
expect(dirty_map.dirty?(clean_record)).to be(true)
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
114
|
+
context "when record contains an unchanged subset of the fields loaded" do
|
115
|
+
let(:partial_record) {
|
116
|
+
create_record(
|
117
|
+
namespace,
|
118
|
+
identity_fields,
|
119
|
+
partial_clean_attrbiutes,
|
120
|
+
depth,
|
121
|
+
)
|
122
|
+
}
|
123
|
+
|
124
|
+
let(:partial_clean_attrbiutes) {
|
125
|
+
attributes.reject { |k, _v| k == :email }
|
126
|
+
}
|
127
|
+
|
128
|
+
it "return false" do
|
129
|
+
expect(dirty_map.dirty?(partial_record)).to be(false)
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
context "when record contains a changed subset of the fields loaded" do
|
134
|
+
let(:partial_record) {
|
135
|
+
create_record(
|
136
|
+
namespace,
|
137
|
+
identity_fields,
|
138
|
+
partial_dirty_attrbiutes,
|
139
|
+
depth,
|
140
|
+
)
|
141
|
+
}
|
142
|
+
|
143
|
+
let(:partial_dirty_attrbiutes) {
|
144
|
+
attributes
|
145
|
+
.reject { |k, _v| k == :email }
|
146
|
+
.merge(name: "record/changed_name")
|
147
|
+
}
|
148
|
+
|
149
|
+
it "return false" do
|
150
|
+
expect(dirty_map.dirty?(partial_record)).to be(true)
|
151
|
+
end
|
152
|
+
end
|
153
|
+
|
154
|
+
context "when record contains an unchanged superset of the fields loaded" do
|
155
|
+
let(:super_record) {
|
156
|
+
create_record(
|
157
|
+
namespace,
|
158
|
+
identity_fields,
|
159
|
+
super_clean_attributes,
|
160
|
+
depth,
|
161
|
+
)
|
162
|
+
}
|
163
|
+
|
164
|
+
let(:super_clean_attributes) {
|
165
|
+
attributes.merge(unknown_key: "record/unknown_value")
|
166
|
+
}
|
167
|
+
|
168
|
+
it "return true" do
|
169
|
+
expect(dirty_map.dirty?(super_record)).to be(true)
|
170
|
+
end
|
171
|
+
end
|
172
|
+
end
|
173
|
+
|
174
|
+
context "#reject_unchanged_fields" do
|
175
|
+
context "when the record has not been loaded (new record)" do
|
176
|
+
it "returns an eqiuivalent record" do
|
177
|
+
expect(dirty_map.reject_unchanged_fields(dirty_record))
|
178
|
+
.to eq(dirty_record)
|
179
|
+
end
|
180
|
+
end
|
181
|
+
|
182
|
+
context "when a record with same identity has been loaded (existing record)" do
|
183
|
+
before do
|
184
|
+
dirty_map.load(loaded_record)
|
185
|
+
end
|
186
|
+
|
187
|
+
context "with a equivalent record" do
|
188
|
+
it "returns an empty record" do
|
189
|
+
expect(dirty_map.reject_unchanged_fields(clean_record)).to be_empty
|
190
|
+
end
|
191
|
+
end
|
192
|
+
|
193
|
+
context "a record with a changed field" do
|
194
|
+
it "returns a record containing just that field" do
|
195
|
+
expect(
|
196
|
+
dirty_map
|
197
|
+
.reject_unchanged_fields(dirty_record)
|
198
|
+
.non_identity_attributes
|
199
|
+
).to eq( name: "record/dirty_name" )
|
200
|
+
end
|
201
|
+
end
|
202
|
+
end
|
203
|
+
end
|
204
|
+
end
|
205
|
+
|
206
|
+
def create_record(namespace, identity_fields, attributes, depth)
|
207
|
+
Terrestrial::UpsertedRecord.new(
|
208
|
+
namespace,
|
209
|
+
identity_fields,
|
210
|
+
attributes,
|
211
|
+
depth,
|
212
|
+
)
|
213
|
+
end
|
214
|
+
end
|
@@ -0,0 +1,119 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
require "terrestrial/lazy_collection"
|
4
|
+
|
5
|
+
RSpec.describe Terrestrial::LazyCollection do
|
6
|
+
let(:proxy) {
|
7
|
+
Terrestrial::LazyCollection.new(
|
8
|
+
database_enum,
|
9
|
+
loader,
|
10
|
+
mapper,
|
11
|
+
)
|
12
|
+
}
|
13
|
+
|
14
|
+
let(:row1) { double(:row1) }
|
15
|
+
let(:row2) { double(:row2) }
|
16
|
+
let(:object1) { double(:object1) }
|
17
|
+
let(:object2) { double(:object2) }
|
18
|
+
let(:row_object_map) {
|
19
|
+
{
|
20
|
+
row1 => object1,
|
21
|
+
row2 => object2,
|
22
|
+
}
|
23
|
+
}
|
24
|
+
let(:collection_size) { row_object_map.size }
|
25
|
+
|
26
|
+
let(:database_enum) { [row1, row2].each.lazy }
|
27
|
+
|
28
|
+
let(:mapper) { double(:mapper) }
|
29
|
+
|
30
|
+
let(:loader_count) { @loader_count }
|
31
|
+
let(:loader) {
|
32
|
+
@loader_count = 0
|
33
|
+
|
34
|
+
->(row) {
|
35
|
+
@loader_count = @loader_count + 1
|
36
|
+
row_object_map.fetch(row)
|
37
|
+
}
|
38
|
+
}
|
39
|
+
|
40
|
+
it "is Enumerable" do
|
41
|
+
expect(proxy).to be_a(Enumerable)
|
42
|
+
end
|
43
|
+
|
44
|
+
describe "#to_ary" do
|
45
|
+
it "is equivalent to the original enumeration" do
|
46
|
+
expect(proxy.to_ary).to eq([object1, object2])
|
47
|
+
end
|
48
|
+
|
49
|
+
it "implicitly coerces to Array" do
|
50
|
+
new_object = Object.new
|
51
|
+
|
52
|
+
expect([new_object].concat(proxy)).to eq([new_object, object1, object2])
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
describe "#each" do
|
57
|
+
it "iterates over all elements of the database_enum" do
|
58
|
+
elements = []
|
59
|
+
proxy.each { |x| elements.push(x) }
|
60
|
+
|
61
|
+
expect(elements).to eq([object1, object2])
|
62
|
+
end
|
63
|
+
|
64
|
+
context "when the collection is not loaded" do
|
65
|
+
it "loads the collection on first call" do
|
66
|
+
proxy.each { |x| x }
|
67
|
+
|
68
|
+
expect(loader_count).to eq(collection_size)
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
context "when the collection has already loaded (second call to #each)" do
|
73
|
+
before do
|
74
|
+
proxy.each { |x| x }
|
75
|
+
end
|
76
|
+
|
77
|
+
it "does not load a second time" do
|
78
|
+
expect {
|
79
|
+
proxy.each { |x| x }
|
80
|
+
}.not_to change { loader_count }
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
context "when #first has been called beforehand" do
|
85
|
+
before do
|
86
|
+
proxy.first
|
87
|
+
end
|
88
|
+
|
89
|
+
it "does not reload the first element of the collection" do
|
90
|
+
proxy.each { |x| x }
|
91
|
+
|
92
|
+
expect(loader_count).to eq(collection_size)
|
93
|
+
end
|
94
|
+
|
95
|
+
it "iterates over all elements" do
|
96
|
+
elements = []
|
97
|
+
proxy.each { |x| elements.push(x) }
|
98
|
+
|
99
|
+
expect(elements).to eq([object1, object2])
|
100
|
+
end
|
101
|
+
end
|
102
|
+
|
103
|
+
context "when drop has been called beforehand" do
|
104
|
+
it "loads each object just once" do
|
105
|
+
proxy.drop(1).each { |x| x }
|
106
|
+
|
107
|
+
expect(loader_count).to eq(collection_size)
|
108
|
+
end
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
describe "#first" do
|
113
|
+
it "loads only the first object" do
|
114
|
+
proxy.first
|
115
|
+
|
116
|
+
expect(loader_count).to eq(1)
|
117
|
+
end
|
118
|
+
end
|
119
|
+
end
|
@@ -0,0 +1,140 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
require "terrestrial/lazy_object_proxy"
|
4
|
+
|
5
|
+
RSpec.describe Terrestrial::LazyObjectProxy do
|
6
|
+
subject(:proxy) {
|
7
|
+
Terrestrial::LazyObjectProxy.new(
|
8
|
+
object_loader,
|
9
|
+
key_fields,
|
10
|
+
)
|
11
|
+
}
|
12
|
+
|
13
|
+
let(:id) { double(:id) }
|
14
|
+
let(:key_fields) { { id: id } }
|
15
|
+
let(:object_loader) { double(:object_loader, call: proxied_object) }
|
16
|
+
let(:proxied_object) { double(:proxied_object, name: name) }
|
17
|
+
let(:name) { double(:name) }
|
18
|
+
|
19
|
+
describe "#__getobj__" do
|
20
|
+
it "loads the object" do
|
21
|
+
proxy.__getobj__
|
22
|
+
|
23
|
+
expect(object_loader).to have_received(:call)
|
24
|
+
end
|
25
|
+
|
26
|
+
it "returns the proxied object" do
|
27
|
+
expect(proxy.__getobj__).to be(proxied_object)
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
context "when no method is called on it" do
|
32
|
+
it "does not call the loader" do
|
33
|
+
proxy
|
34
|
+
|
35
|
+
expect(object_loader).not_to have_received(:call)
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
39
|
+
context "when a missing method is called on the proxy" do
|
40
|
+
it "is a true decorator" do
|
41
|
+
expect(proxied_object).to receive(:arbitrary_message)
|
42
|
+
|
43
|
+
proxy.arbitrary_message
|
44
|
+
end
|
45
|
+
|
46
|
+
it "loads the object" do
|
47
|
+
proxy.name
|
48
|
+
|
49
|
+
expect(object_loader).to have_received(:call)
|
50
|
+
end
|
51
|
+
|
52
|
+
it "returns delegates the message to the object" do
|
53
|
+
args = [ double, double ]
|
54
|
+
proxy.name(*args)
|
55
|
+
|
56
|
+
expect(proxied_object).to have_received(:name).with(*args)
|
57
|
+
end
|
58
|
+
|
59
|
+
it "returns the objects return value" do
|
60
|
+
expect(proxy.name).to eq(name)
|
61
|
+
end
|
62
|
+
|
63
|
+
context "when calling a method twice" do
|
64
|
+
it "loads the object once" do
|
65
|
+
proxy.name
|
66
|
+
proxy.name
|
67
|
+
|
68
|
+
expect(object_loader).to have_received(:call)
|
69
|
+
end
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
describe "#loaded?" do
|
74
|
+
context "before the object is loaded" do
|
75
|
+
it "returns false" do
|
76
|
+
expect(proxy).not_to be_loaded
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
context "after the object is loaded" do
|
81
|
+
def force_object_load(object)
|
82
|
+
object.__getobj__
|
83
|
+
end
|
84
|
+
|
85
|
+
before { force_object_load(proxy) }
|
86
|
+
|
87
|
+
it "returns true" do
|
88
|
+
expect(proxy).to be_loaded
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
92
|
+
|
93
|
+
describe "key fields" do
|
94
|
+
context "when key fields are provided before load (such as from foreign key)" do
|
95
|
+
it "does not load the object when that field is accessed" do
|
96
|
+
proxy.id
|
97
|
+
|
98
|
+
expect(proxy).not_to be_loaded
|
99
|
+
end
|
100
|
+
|
101
|
+
it "returns the given value" do
|
102
|
+
expect(proxy.id).to be(id)
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
106
|
+
|
107
|
+
describe "#respond_to?" do
|
108
|
+
context "when method corresponds to a key field" do
|
109
|
+
it "does not the load the object" do
|
110
|
+
proxy.respond_to?(:id)
|
111
|
+
|
112
|
+
expect(proxy).not_to be_loaded
|
113
|
+
end
|
114
|
+
|
115
|
+
it "repsonds to the method" do
|
116
|
+
expect(proxy).to respond_to(:id)
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
context "when the method is not a key field" do
|
121
|
+
it "loads the object" do
|
122
|
+
proxy.respond_to?(:something_arbitrary)
|
123
|
+
|
124
|
+
expect(proxy).to be_loaded
|
125
|
+
end
|
126
|
+
|
127
|
+
context "when lazy proxied object does respond to the method" do
|
128
|
+
it "responds to the method" do
|
129
|
+
expect(proxy).to respond_to(:name)
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
context "when lazy proxied object does not respond to the method" do
|
134
|
+
it "does not respond to the method" do
|
135
|
+
expect(proxy).not_to respond_to(:something_arbitrary)
|
136
|
+
end
|
137
|
+
end
|
138
|
+
end
|
139
|
+
end
|
140
|
+
end
|