ihoka-friendly 0.6.3 → 0.7.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.document +2 -0
- data/.gitignore +27 -0
- data/.ruby-version +1 -0
- data/APACHE-LICENSE +202 -0
- data/CHANGELOG.md +6 -0
- data/Gemfile +3 -0
- data/Gemfile.lock +38 -0
- data/README.md +13 -22
- data/Rakefile +25 -0
- data/TODO.md +5 -0
- data/VERSION +1 -0
- data/examples/friendly.yml +7 -0
- data/ihoka-friendly.gemspec +31 -0
- data/lib/friendly/document/mixin.rb +2 -2
- data/lib/friendly/indexer.rb +1 -1
- data/lib/tasks/friendly.rake +7 -0
- data/spec/config.yml.example +7 -0
- data/spec/fakes/data_store_fake.rb +29 -0
- data/spec/fakes/database_fake.rb +12 -0
- data/spec/fakes/dataset_fake.rb +28 -0
- data/spec/fakes/document.rb +18 -0
- data/spec/fakes/serializer_fake.rb +12 -0
- data/spec/fakes/time_fake.rb +12 -0
- data/spec/integration/ad_hoc_scopes_spec.rb +42 -0
- data/spec/integration/basic_object_lifecycle_spec.rb +115 -0
- data/spec/integration/batch_insertion_spec.rb +29 -0
- data/spec/integration/convenience_api_spec.rb +25 -0
- data/spec/integration/count_spec.rb +12 -0
- data/spec/integration/default_value_spec.rb +30 -0
- data/spec/integration/dirty_tracking_spec.rb +43 -0
- data/spec/integration/find_via_cache_spec.rb +101 -0
- data/spec/integration/finder_spec.rb +71 -0
- data/spec/integration/has_many_spec.rb +18 -0
- data/spec/integration/index_spec.rb +57 -0
- data/spec/integration/named_scope_spec.rb +34 -0
- data/spec/integration/offline_indexing_spec.rb +53 -0
- data/spec/integration/pagination_spec.rb +63 -0
- data/spec/integration/scope_chaining_spec.rb +22 -0
- data/spec/integration/table_creator_spec.rb +64 -0
- data/spec/integration/write_through_cache_spec.rb +53 -0
- data/spec/spec.opts +1 -0
- data/spec/spec_helper.rb +109 -0
- data/spec/unit/associations/association_spec.rb +57 -0
- data/spec/unit/associations/set_spec.rb +43 -0
- data/spec/unit/attribute_spec.rb +125 -0
- data/spec/unit/cache_by_id_spec.rb +102 -0
- data/spec/unit/cache_spec.rb +21 -0
- data/spec/unit/data_store_spec.rb +201 -0
- data/spec/unit/document/attributes_spec.rb +130 -0
- data/spec/unit/document_spec.rb +318 -0
- data/spec/unit/document_table_spec.rb +126 -0
- data/spec/unit/friendly_spec.rb +25 -0
- data/spec/unit/index_spec.rb +196 -0
- data/spec/unit/memcached_spec.rb +114 -0
- data/spec/unit/query_spec.rb +104 -0
- data/spec/unit/scope_proxy_spec.rb +44 -0
- data/spec/unit/scope_spec.rb +113 -0
- data/spec/unit/storage_factory_spec.rb +59 -0
- data/spec/unit/storage_proxy_spec.rb +244 -0
- data/spec/unit/translator_spec.rb +91 -0
- data/website/index.html +210 -0
- data/website/scripts/clipboard.swf +0 -0
- data/website/scripts/shBrushAS3.js +61 -0
- data/website/scripts/shBrushBash.js +66 -0
- data/website/scripts/shBrushCSharp.js +67 -0
- data/website/scripts/shBrushColdFusion.js +102 -0
- data/website/scripts/shBrushCpp.js +99 -0
- data/website/scripts/shBrushCss.js +93 -0
- data/website/scripts/shBrushDelphi.js +57 -0
- data/website/scripts/shBrushDiff.js +43 -0
- data/website/scripts/shBrushErlang.js +54 -0
- data/website/scripts/shBrushGroovy.js +69 -0
- data/website/scripts/shBrushJScript.js +52 -0
- data/website/scripts/shBrushJava.js +59 -0
- data/website/scripts/shBrushJavaFX.js +60 -0
- data/website/scripts/shBrushPerl.js +74 -0
- data/website/scripts/shBrushPhp.js +91 -0
- data/website/scripts/shBrushPlain.js +35 -0
- data/website/scripts/shBrushPowerShell.js +76 -0
- data/website/scripts/shBrushPython.js +66 -0
- data/website/scripts/shBrushRuby.js +57 -0
- data/website/scripts/shBrushScala.js +53 -0
- data/website/scripts/shBrushSql.js +68 -0
- data/website/scripts/shBrushVb.js +58 -0
- data/website/scripts/shBrushXml.js +71 -0
- data/website/scripts/shCore.js +30 -0
- data/website/scripts/shLegacy.js +30 -0
- data/website/styles/friendly.css +103 -0
- data/website/styles/help.png +0 -0
- data/website/styles/ie.css +35 -0
- data/website/styles/magnifier.png +0 -0
- data/website/styles/page_white_code.png +0 -0
- data/website/styles/page_white_copy.png +0 -0
- data/website/styles/print.css +29 -0
- data/website/styles/printer.png +0 -0
- data/website/styles/screen.css +257 -0
- data/website/styles/shCore.css +330 -0
- data/website/styles/shThemeDefault.css +173 -0
- data/website/styles/shThemeDjango.css +176 -0
- data/website/styles/shThemeEclipse.css +190 -0
- data/website/styles/shThemeEmacs.css +175 -0
- data/website/styles/shThemeFadeToGrey.css +177 -0
- data/website/styles/shThemeMidnight.css +175 -0
- data/website/styles/shThemeRDark.css +175 -0
- metadata +302 -383
@@ -0,0 +1,201 @@
|
|
1
|
+
require File.expand_path("../../spec_helper", __FILE__)
|
2
|
+
|
3
|
+
describe "Friendly::DataStore" do
|
4
|
+
before do
|
5
|
+
@users = DatasetFake.new(:insert => 42)
|
6
|
+
@db = DatabaseFake.new("users" => @users)
|
7
|
+
@datastore = Friendly::DataStore.new(@db)
|
8
|
+
@klass = stub(:table_name => "users")
|
9
|
+
end
|
10
|
+
|
11
|
+
describe "inserting data" do
|
12
|
+
before do
|
13
|
+
@return = @datastore.insert(@klass, :name => "Stewie")
|
14
|
+
end
|
15
|
+
|
16
|
+
it "inserts it in to the table in the datastore" do
|
17
|
+
@users.inserts.length.should == 1
|
18
|
+
@users.inserts.should include(:name => "Stewie")
|
19
|
+
end
|
20
|
+
|
21
|
+
it "returns the id from the dataset" do
|
22
|
+
@return.should == 42
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
describe "retrieving all based on a query" do
|
27
|
+
before do
|
28
|
+
@users.where = {{:name => "Stewie"} => stub(:map => [{:id => 1}])}
|
29
|
+
@return = @datastore.all(@klass, query(:name => "Stewie"))
|
30
|
+
end
|
31
|
+
|
32
|
+
it "gets the data from the dataset for the klass and makes it an arary" do
|
33
|
+
@return.should == [{:id => 1}]
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
describe "retrieving all with a limit" do
|
38
|
+
before do
|
39
|
+
@filtered = stub
|
40
|
+
@filtered.stubs(:limit).with(10, nil).returns(stub(:map => [{:id => 1}]))
|
41
|
+
@users.where = {{:name => "Stewie"} => @filtered}
|
42
|
+
@query = query(:name => "Stewie", :limit! => 10)
|
43
|
+
@return = @datastore.all(@klass, @query)
|
44
|
+
end
|
45
|
+
|
46
|
+
it "limits the filtered dataset and returns the results" do
|
47
|
+
@return.should == [{:id => 1}]
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
describe "retrieving all with a offset" do
|
52
|
+
before do
|
53
|
+
@filtered = stub
|
54
|
+
@filtered.stubs(:limit).with(nil, 10).returns(stub(:map => [{:id => 1}]))
|
55
|
+
@users.where = {{:name => "Stewie"} => @filtered}
|
56
|
+
@query = query(:name => "Stewie", :offset! => 10)
|
57
|
+
@return = @datastore.all(@klass, @query)
|
58
|
+
end
|
59
|
+
|
60
|
+
it "offsets the filtered dataset and returns the results" do
|
61
|
+
@return.should == [{:id => 1}]
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
describe "retrieving all with order" do
|
66
|
+
before do
|
67
|
+
@filtered = stub
|
68
|
+
@filtered.stubs(:order).with(:created_at).returns(stub(:map => [{:id => 1}]))
|
69
|
+
@users.where = {{:name => "Stewie"} => @filtered}
|
70
|
+
@query = query(:name => "Stewie", :order! => :created_at)
|
71
|
+
@return = @datastore.all(@klass, @query)
|
72
|
+
end
|
73
|
+
|
74
|
+
it "orders the filtered dataset and returns the results" do
|
75
|
+
@return.should == [{:id => 1}]
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
describe "all without conditions" do
|
80
|
+
before do
|
81
|
+
@all = stub(:map => [])
|
82
|
+
@users.stubs(:order).with(:created_at).returns(@all)
|
83
|
+
@query = query(:order! => :created_at)
|
84
|
+
@return = @datastore.all(@klass, @query)
|
85
|
+
end
|
86
|
+
|
87
|
+
it "orders the filtered dataset and returns the results" do
|
88
|
+
@return.should == []
|
89
|
+
end
|
90
|
+
end
|
91
|
+
|
92
|
+
describe "retrieving first with conditions" do
|
93
|
+
before do
|
94
|
+
@users.first = {{:id => 1} => {:id => 1}}
|
95
|
+
@return = @datastore.first(@klass, query(:id => 1))
|
96
|
+
end
|
97
|
+
|
98
|
+
it "gets the first object matching the conditions from the dataset" do
|
99
|
+
@return.should == {:id => 1}
|
100
|
+
end
|
101
|
+
end
|
102
|
+
|
103
|
+
describe "updating data" do
|
104
|
+
before do
|
105
|
+
@filtered = DatasetFake.new(:update => true)
|
106
|
+
@users.where = {{:id => 1} => @filtered}
|
107
|
+
@return = @datastore.update(@klass, 1, :name => "Peter")
|
108
|
+
end
|
109
|
+
|
110
|
+
it "filter the dataset by id and update the filtered row" do
|
111
|
+
@filtered.updates.length.should == 1
|
112
|
+
@filtered.updates.should include(:name => "Peter")
|
113
|
+
end
|
114
|
+
end
|
115
|
+
|
116
|
+
describe "deleting data" do
|
117
|
+
before do
|
118
|
+
@filtered = stub
|
119
|
+
@filtered.stubs(:delete)
|
120
|
+
@users.where = {{:id => 1} => @filtered}
|
121
|
+
@datastore.delete(@klass, 1)
|
122
|
+
end
|
123
|
+
|
124
|
+
it "filters the dataset by id and deletes" do
|
125
|
+
@filtered.should have_received(:delete)
|
126
|
+
end
|
127
|
+
end
|
128
|
+
|
129
|
+
describe "when a batch transaction has been started" do
|
130
|
+
before do
|
131
|
+
@datastore.start_batch
|
132
|
+
@persistable = stub(:table_name => "some_table")
|
133
|
+
@datastore.insert(@persistable, {:some => "attrs"})
|
134
|
+
end
|
135
|
+
|
136
|
+
after { Thread.current[:friendly_batch] = nil }
|
137
|
+
|
138
|
+
it "adds the attributes to the batch for that table" do
|
139
|
+
inserts = Thread.current[:friendly_batch]["some_table"]
|
140
|
+
inserts.length.should == 1
|
141
|
+
inserts.should include(:some => "attrs")
|
142
|
+
end
|
143
|
+
end
|
144
|
+
|
145
|
+
describe "starting a batch" do
|
146
|
+
before do
|
147
|
+
@datastore.start_batch
|
148
|
+
end
|
149
|
+
|
150
|
+
after { Thread.current[:friendly_batch] = nil }
|
151
|
+
|
152
|
+
it "sets Thread.current[:friendly_batch] to empty hash" do
|
153
|
+
Thread.current[:friendly_batch].should == {}
|
154
|
+
end
|
155
|
+
end
|
156
|
+
|
157
|
+
describe "resetting a batch transaction" do
|
158
|
+
before do
|
159
|
+
@db.stubs(:from)
|
160
|
+
Thread.current[:friendly_batch] = {"users" => [{:a => "b"}]}
|
161
|
+
@datastore.reset_batch
|
162
|
+
end
|
163
|
+
after { Thread.current[:friendly_batch] = nil }
|
164
|
+
|
165
|
+
it "sets Thread.current[:friendly_batch] to nil without inserting" do
|
166
|
+
Thread.current[:friendly_batch].should be_nil
|
167
|
+
@db.should have_received(:from).never
|
168
|
+
end
|
169
|
+
end
|
170
|
+
|
171
|
+
describe "flushing a batch" do
|
172
|
+
before do
|
173
|
+
@records = [{:name => "Stewie"}, {:name => "Brian"}]
|
174
|
+
Thread.current[:friendly_batch] = {"users" => @records}
|
175
|
+
@users.stubs(:multi_insert)
|
176
|
+
@datastore.flush_batch
|
177
|
+
end
|
178
|
+
after { Thread.current[:friendly_batch] = nil }
|
179
|
+
|
180
|
+
it "performs the multi_insert on each table" do
|
181
|
+
@users.should have_received(:multi_insert).
|
182
|
+
with(@records, :commit_every => 1000)
|
183
|
+
end
|
184
|
+
|
185
|
+
it "resets the batch" do
|
186
|
+
Thread.current[:friendly_batch].should be_nil
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
describe "counting" do
|
191
|
+
before do
|
192
|
+
@filtered = stub(:count => 10)
|
193
|
+
@users.stubs(:where).with(:name => "James").returns(@filtered)
|
194
|
+
end
|
195
|
+
|
196
|
+
it "fitlers and counts in the db" do
|
197
|
+
@datastore.count(@klass, query(:name => "James")).should == 10
|
198
|
+
end
|
199
|
+
end
|
200
|
+
end
|
201
|
+
|
@@ -0,0 +1,130 @@
|
|
1
|
+
require File.expand_path("../../../spec_helper", __FILE__)
|
2
|
+
|
3
|
+
describe "Friendly::Document::Attributes" do
|
4
|
+
before do
|
5
|
+
@klass = Class.new do
|
6
|
+
include Friendly::Document::Attributes
|
7
|
+
|
8
|
+
attribute :name, String
|
9
|
+
end
|
10
|
+
end
|
11
|
+
|
12
|
+
describe "#initialize" do
|
13
|
+
it "sets the attributes using the setters" do
|
14
|
+
@doc = @klass.new :name => "Bond"
|
15
|
+
@doc.name.should == "Bond"
|
16
|
+
end
|
17
|
+
|
18
|
+
it "assigns the default values" do
|
19
|
+
@klass.attribute :id, Friendly::UUID
|
20
|
+
@klass.attributes[:id] = stub(:assign_default_value => nil)
|
21
|
+
@klass.attributes[:name] = stub(:assign_default_value => nil,
|
22
|
+
:typecast => "Bond")
|
23
|
+
@doc = @klass.new :name => "Bond"
|
24
|
+
@klass.attributes[:id].should have_received(:assign_default_value).with(@doc)
|
25
|
+
@klass.attributes[:name].should have_received(:assign_default_value).with(@doc)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
describe "#attributes=" do
|
30
|
+
before do
|
31
|
+
@object = @klass.new
|
32
|
+
@object.attributes = {:name => "Bond"}
|
33
|
+
end
|
34
|
+
|
35
|
+
it "sets the attributes using the setters" do
|
36
|
+
@object.name.should == "Bond"
|
37
|
+
end
|
38
|
+
|
39
|
+
it "raises ArgumentError when there are duplicate keys of differing type" do
|
40
|
+
lambda {
|
41
|
+
@object.attributes = {:name => "Bond", "name" => "Bond"}
|
42
|
+
}.should raise_error(ArgumentError)
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
describe "#to_hash" do
|
47
|
+
before do
|
48
|
+
@object = @klass.new(:name => "Stewie")
|
49
|
+
end
|
50
|
+
|
51
|
+
it "creates a hash that contains its attributes" do
|
52
|
+
@object.to_hash.should == {:name => "Stewie"}
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
describe "#assign" do
|
57
|
+
before do
|
58
|
+
@object = @klass.new
|
59
|
+
@object.assign(:name, "James Bond")
|
60
|
+
end
|
61
|
+
|
62
|
+
it "assigns the value to the attribute" do
|
63
|
+
@object.name.should == "James Bond"
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
describe "#will_change" do
|
68
|
+
before do
|
69
|
+
@klass.send(:attr_accessor, :some_variable)
|
70
|
+
@object = @klass.new
|
71
|
+
@object.some_variable = "Some value"
|
72
|
+
@object.will_change(:some_variable)
|
73
|
+
end
|
74
|
+
|
75
|
+
it "makes the object #changed?" do
|
76
|
+
@object.should be_changed
|
77
|
+
end
|
78
|
+
|
79
|
+
it "returns the value of the variable for #attribute_was" do
|
80
|
+
@object.attribute_was(:some_variable).should == "Some value"
|
81
|
+
end
|
82
|
+
|
83
|
+
it "returns true for attribute_changed?(:some_variable)" do
|
84
|
+
@object.should be_attribute_changed(:some_variable)
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
describe "#reset_changes" do
|
89
|
+
before do
|
90
|
+
@klass.send(:attr_accessor, :some_variable)
|
91
|
+
@object = @klass.new
|
92
|
+
@object.some_variable = "Some value"
|
93
|
+
@object.will_change(:some_variable)
|
94
|
+
@object.reset_changes
|
95
|
+
end
|
96
|
+
|
97
|
+
it "resets the changed status of the object" do
|
98
|
+
@object.should_not be_changed
|
99
|
+
end
|
100
|
+
|
101
|
+
it "returns nil for attribute_was(:some_variable)" do
|
102
|
+
@object.attribute_was(:some_variable).should be_nil
|
103
|
+
end
|
104
|
+
|
105
|
+
it "returns false for attribute_changed?(:some_variable)" do
|
106
|
+
@object.should_not be_attribute_changed(:some_variable)
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
describe "#new_without_change_tracking" do
|
111
|
+
before do
|
112
|
+
@klass = Class.new do
|
113
|
+
attr_reader :name
|
114
|
+
|
115
|
+
def name=(name)
|
116
|
+
will_change(:name)
|
117
|
+
@name = name
|
118
|
+
end
|
119
|
+
|
120
|
+
include Friendly::Document::Attributes
|
121
|
+
end
|
122
|
+
@doc = @klass.new_without_change_tracking(:name => "James")
|
123
|
+
end
|
124
|
+
|
125
|
+
it "initializes and then calls reset_changes" do
|
126
|
+
@doc.name.should == "James"
|
127
|
+
@doc.should_not be_changed
|
128
|
+
end
|
129
|
+
end
|
130
|
+
end
|
@@ -0,0 +1,318 @@
|
|
1
|
+
require File.expand_path("../../spec_helper", __FILE__)
|
2
|
+
|
3
|
+
describe "Friendly::Document" do
|
4
|
+
before do
|
5
|
+
@klass = Class.new { include Friendly::Document }
|
6
|
+
@klass.attribute(:name, String)
|
7
|
+
@storage_proxy = stub
|
8
|
+
@klass.storage_proxy = @storage_proxy
|
9
|
+
end
|
10
|
+
|
11
|
+
it "delegates table_name to it's class" do
|
12
|
+
User.new.table_name.should == User.table_name
|
13
|
+
end
|
14
|
+
|
15
|
+
it "always has an id attribute" do
|
16
|
+
@klass.attributes[:id].type.should == Friendly::UUID
|
17
|
+
end
|
18
|
+
|
19
|
+
it "always has a created_at attribute" do
|
20
|
+
@klass.attributes[:created_at].type.should == Time
|
21
|
+
end
|
22
|
+
|
23
|
+
it "always has a updated_at attribute" do
|
24
|
+
@klass.attributes[:updated_at].type.should == Time
|
25
|
+
end
|
26
|
+
|
27
|
+
describe "saving a new document" do
|
28
|
+
before do
|
29
|
+
@user = @klass.new(:name => "whatever")
|
30
|
+
@storage_proxy.stubs(:create)
|
31
|
+
@user.save
|
32
|
+
end
|
33
|
+
|
34
|
+
it "asks the storage_proxy to create" do
|
35
|
+
@storage_proxy.should have_received(:create).with(@user)
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
39
|
+
describe "saving an existing document" do
|
40
|
+
before do
|
41
|
+
@user = @klass.new(:name => "whatever", :id => 42, :new_record => false)
|
42
|
+
@storage_proxy.stubs(:update)
|
43
|
+
@user.save
|
44
|
+
end
|
45
|
+
|
46
|
+
it "asks the storage_proxy to update" do
|
47
|
+
@storage_proxy.should have_received(:update).with(@user)
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
describe "destroying a document" do
|
52
|
+
before do
|
53
|
+
@user = @klass.new
|
54
|
+
@storage_proxy.stubs(:destroy)
|
55
|
+
@user.destroy
|
56
|
+
end
|
57
|
+
|
58
|
+
it "delegates to the storage proxy" do
|
59
|
+
@storage_proxy.should have_received(:destroy).with(@user)
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
describe "table name" do
|
64
|
+
it "by default: is the class name, converted with pluralize.underscore" do
|
65
|
+
User.table_name.should == "users"
|
66
|
+
end
|
67
|
+
|
68
|
+
it "is overridable" do
|
69
|
+
@klass.table_name = "ASDF"
|
70
|
+
@klass.table_name.should == "ASDF"
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
describe "new record" do
|
75
|
+
before do
|
76
|
+
@object = @klass.new
|
77
|
+
end
|
78
|
+
|
79
|
+
it "is new_record by default" do
|
80
|
+
@object.should be_new_record
|
81
|
+
end
|
82
|
+
|
83
|
+
it "is not new_record when new_record is set to false" do
|
84
|
+
@object.new_record = false
|
85
|
+
@object.should_not be_new_record
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
describe "object equality" do
|
90
|
+
it "is never equal if both objects are new_records" do
|
91
|
+
@klass.new(:name => "x").should_not == @klass.new(:name => "x")
|
92
|
+
end
|
93
|
+
|
94
|
+
it "is equal if both objects have the same id" do
|
95
|
+
uuid = Friendly::UUID.new
|
96
|
+
one = @klass.new(:id => uuid, :new_record => false)
|
97
|
+
one.should == @klass.new(:id => uuid, :new_record => false)
|
98
|
+
end
|
99
|
+
|
100
|
+
it "is equal if the objects point to the same reference" do
|
101
|
+
obj = @klass.new
|
102
|
+
obj.should == obj
|
103
|
+
end
|
104
|
+
|
105
|
+
it "is not equal if two objects are of differing types with the same id" do
|
106
|
+
@klass.new(:id => 1).should_not == User.new(:id => 1)
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
describe "adding an index" do
|
111
|
+
before do
|
112
|
+
@storage_proxy.stubs(:add)
|
113
|
+
@klass = Class.new { include Friendly::Document }
|
114
|
+
@klass.storage_proxy = @storage_proxy
|
115
|
+
@klass.indexes :name
|
116
|
+
end
|
117
|
+
|
118
|
+
it "delegates to the storage_proxy" do
|
119
|
+
@klass.storage_proxy.should have_received(:add).with([:name])
|
120
|
+
end
|
121
|
+
end
|
122
|
+
|
123
|
+
describe "adding a cache" do
|
124
|
+
before do
|
125
|
+
@storage_proxy.stubs(:cache)
|
126
|
+
@klass.caches_by(:name, :created_at)
|
127
|
+
end
|
128
|
+
|
129
|
+
it "delegates to the storage_proxy" do
|
130
|
+
@storage_proxy.should have_received(:cache).with([:name, :created_at], {})
|
131
|
+
end
|
132
|
+
end
|
133
|
+
|
134
|
+
describe "Document.first" do
|
135
|
+
before do
|
136
|
+
@doc = stub
|
137
|
+
@query = stub
|
138
|
+
@query_klass = stub
|
139
|
+
@klass.query_klass = @query_klass
|
140
|
+
@query_klass.stubs(:new).with(:id => 1).returns(@query)
|
141
|
+
@storage_proxy.stubs(:first).with(@query).returns(@doc)
|
142
|
+
end
|
143
|
+
|
144
|
+
it "creates a query object and delegates to the storage proxy" do
|
145
|
+
@klass.first(:id => 1).should == @doc
|
146
|
+
end
|
147
|
+
end
|
148
|
+
|
149
|
+
describe "Document.all" do
|
150
|
+
before do
|
151
|
+
@docs = stub
|
152
|
+
@query = stub
|
153
|
+
@query_klass = stub
|
154
|
+
@klass.query_klass = @query_klass
|
155
|
+
@query_klass.stubs(:new).with(:name => "x").returns(@query)
|
156
|
+
@storage_proxy.stubs(:all).with(@query).returns(@docs)
|
157
|
+
end
|
158
|
+
|
159
|
+
it "delegates to the storage proxy" do
|
160
|
+
@klass.all(:name => "x").should == @docs
|
161
|
+
end
|
162
|
+
end
|
163
|
+
|
164
|
+
describe "Document.find" do
|
165
|
+
describe "when an object is found" do
|
166
|
+
before do
|
167
|
+
@doc = stub
|
168
|
+
@query = stub
|
169
|
+
@query_klass = stub
|
170
|
+
@klass.query_klass = @query_klass
|
171
|
+
@query_klass.stubs(:new).with(:id => 1).returns(@query)
|
172
|
+
@storage_proxy.stubs(:first).with(@query).returns(@doc)
|
173
|
+
end
|
174
|
+
|
175
|
+
it "queries the storage proxy" do
|
176
|
+
@klass.find(1).should == @doc
|
177
|
+
end
|
178
|
+
end
|
179
|
+
|
180
|
+
describe "when no object is found" do
|
181
|
+
before do
|
182
|
+
@storage_proxy.stubs(:first).returns(nil)
|
183
|
+
end
|
184
|
+
|
185
|
+
it "raises RecordNotFound" do
|
186
|
+
lambda {
|
187
|
+
@klass.find(1)
|
188
|
+
}.should raise_error(Friendly::RecordNotFound)
|
189
|
+
end
|
190
|
+
end
|
191
|
+
end
|
192
|
+
|
193
|
+
describe "Document.all" do
|
194
|
+
before do
|
195
|
+
@query = stub
|
196
|
+
@query_klass = stub
|
197
|
+
@klass.query_klass = @query_klass
|
198
|
+
@query_klass.stubs(:new).with(:name => "x").returns(@query)
|
199
|
+
@storage_proxy.stubs(:count).with(@query).returns(25)
|
200
|
+
end
|
201
|
+
|
202
|
+
it "delegates to the storage proxy" do
|
203
|
+
@klass.count(:name => "x").should == 25
|
204
|
+
end
|
205
|
+
end
|
206
|
+
|
207
|
+
describe "Document.create" do
|
208
|
+
before do
|
209
|
+
@storage_proxy.stubs(:create)
|
210
|
+
@doc = @klass.create(:name => "James")
|
211
|
+
end
|
212
|
+
|
213
|
+
it "initializes, then saves the document and returns it" do
|
214
|
+
@storage_proxy.should have_received(:create).with(@doc)
|
215
|
+
@doc.should be_kind_of(@klass)
|
216
|
+
end
|
217
|
+
end
|
218
|
+
|
219
|
+
describe "Document#update_attributes" do
|
220
|
+
before do
|
221
|
+
@storage_proxy.stubs(:update)
|
222
|
+
@doc = @klass.new(:name => "James", :new_record => false)
|
223
|
+
@doc.update_attributes :name => "Steve"
|
224
|
+
end
|
225
|
+
|
226
|
+
it "sets the attributes" do
|
227
|
+
@doc.name.should == "Steve"
|
228
|
+
end
|
229
|
+
|
230
|
+
it "saves the document" do
|
231
|
+
@storage_proxy.should have_received(:update).with(@doc)
|
232
|
+
end
|
233
|
+
end
|
234
|
+
|
235
|
+
describe "when document has been included" do
|
236
|
+
after { Friendly::Document.documents = [] }
|
237
|
+
it "adds the document to the collection" do
|
238
|
+
Friendly::Document.documents.should include(@klass)
|
239
|
+
end
|
240
|
+
end
|
241
|
+
|
242
|
+
describe "Document.paginate" do
|
243
|
+
before do
|
244
|
+
@conditions = {:name => "Stewie", :page! => 1, :per_page! => 20}
|
245
|
+
@query = stub(:page => 1, :per_page => 20)
|
246
|
+
@docs = stub
|
247
|
+
@query_klass = stub
|
248
|
+
@klass.query_klass = @query_klass
|
249
|
+
@count = 10
|
250
|
+
@collection_klass = stub
|
251
|
+
@collection = stub
|
252
|
+
@klass.collection_klass = @collection_klass
|
253
|
+
@collection.stubs(:replace).returns(@collection)
|
254
|
+
@collection_klass.stubs(:new).with(1, 20, @count).returns(@collection)
|
255
|
+
@query_klass.stubs(:new).returns(@query)
|
256
|
+
@storage_proxy.stubs(:count).with(@query).returns(@count)
|
257
|
+
@storage_proxy.stubs(:all).with(@query).returns(@docs)
|
258
|
+
|
259
|
+
@pagination = @klass.paginate(@conditions)
|
260
|
+
end
|
261
|
+
|
262
|
+
it "creates an instance of the collection klass and returns it" do
|
263
|
+
@pagination.should == @collection
|
264
|
+
end
|
265
|
+
|
266
|
+
it "fills the collection with objects from the datastore" do
|
267
|
+
@collection.should have_received(:replace).with(@docs)
|
268
|
+
end
|
269
|
+
end
|
270
|
+
|
271
|
+
describe "creating a named_scope" do
|
272
|
+
before do
|
273
|
+
@scope_proxy = stub(:add_named => nil)
|
274
|
+
@klass.scope_proxy = @scope_proxy
|
275
|
+
@klass.named_scope(:by_name, :order => :name)
|
276
|
+
end
|
277
|
+
|
278
|
+
it "asks the named_scope_set to add it" do
|
279
|
+
@klass.scope_proxy.should have_received(:add_named).
|
280
|
+
with(:by_name, :order => :name)
|
281
|
+
end
|
282
|
+
end
|
283
|
+
|
284
|
+
describe "Document.has_named_scope?" do
|
285
|
+
it "delegates to the scope_proxy" do
|
286
|
+
@scope_proxy = stub
|
287
|
+
@scope_proxy.stubs(:has_named_scope?).with(:whatever).returns(true)
|
288
|
+
@klass.scope_proxy = @scope_proxy
|
289
|
+
@klass.has_named_scope?(:whatever).should be_true
|
290
|
+
end
|
291
|
+
end
|
292
|
+
|
293
|
+
describe "accessing an ad-hoc scope" do
|
294
|
+
before do
|
295
|
+
@scope = stub
|
296
|
+
@scope_proxy = stub
|
297
|
+
@scope_proxy.stubs(:ad_hoc).with(:order! => :name).returns(@scope)
|
298
|
+
@klass.scope_proxy = @scope_proxy
|
299
|
+
end
|
300
|
+
|
301
|
+
it "asks the named_scope_set to add it" do
|
302
|
+
@klass.scope(:order! => :name).should == @scope
|
303
|
+
end
|
304
|
+
end
|
305
|
+
|
306
|
+
describe "adding an association" do
|
307
|
+
before do
|
308
|
+
@assoc_set = stub(:add => nil)
|
309
|
+
@klass.association_set = @assoc_set
|
310
|
+
@klass.has_many :addresses
|
311
|
+
end
|
312
|
+
|
313
|
+
it "asks the association set to add it" do
|
314
|
+
@assoc_set.should have_received(:add).with(:addresses, {})
|
315
|
+
end
|
316
|
+
end
|
317
|
+
end
|
318
|
+
|