dm-core 0.9.2
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +144 -0
- data/FAQ +74 -0
- data/MIT-LICENSE +22 -0
- data/QUICKLINKS +12 -0
- data/README +143 -0
- data/lib/dm-core.rb +213 -0
- data/lib/dm-core/adapters.rb +4 -0
- data/lib/dm-core/adapters/abstract_adapter.rb +202 -0
- data/lib/dm-core/adapters/data_objects_adapter.rb +701 -0
- data/lib/dm-core/adapters/mysql_adapter.rb +132 -0
- data/lib/dm-core/adapters/postgres_adapter.rb +179 -0
- data/lib/dm-core/adapters/sqlite3_adapter.rb +105 -0
- data/lib/dm-core/associations.rb +172 -0
- data/lib/dm-core/associations/many_to_many.rb +138 -0
- data/lib/dm-core/associations/many_to_one.rb +101 -0
- data/lib/dm-core/associations/one_to_many.rb +275 -0
- data/lib/dm-core/associations/one_to_one.rb +61 -0
- data/lib/dm-core/associations/relationship.rb +116 -0
- data/lib/dm-core/associations/relationship_chain.rb +74 -0
- data/lib/dm-core/auto_migrations.rb +64 -0
- data/lib/dm-core/collection.rb +604 -0
- data/lib/dm-core/hook.rb +11 -0
- data/lib/dm-core/identity_map.rb +45 -0
- data/lib/dm-core/is.rb +16 -0
- data/lib/dm-core/logger.rb +233 -0
- data/lib/dm-core/migrations/destructive_migrations.rb +17 -0
- data/lib/dm-core/migrator.rb +29 -0
- data/lib/dm-core/model.rb +399 -0
- data/lib/dm-core/naming_conventions.rb +52 -0
- data/lib/dm-core/property.rb +611 -0
- data/lib/dm-core/property_set.rb +158 -0
- data/lib/dm-core/query.rb +590 -0
- data/lib/dm-core/repository.rb +159 -0
- data/lib/dm-core/resource.rb +618 -0
- data/lib/dm-core/scope.rb +35 -0
- data/lib/dm-core/support.rb +7 -0
- data/lib/dm-core/support/array.rb +13 -0
- data/lib/dm-core/support/assertions.rb +8 -0
- data/lib/dm-core/support/errors.rb +23 -0
- data/lib/dm-core/support/kernel.rb +7 -0
- data/lib/dm-core/support/symbol.rb +41 -0
- data/lib/dm-core/transaction.rb +267 -0
- data/lib/dm-core/type.rb +160 -0
- data/lib/dm-core/type_map.rb +80 -0
- data/lib/dm-core/types.rb +19 -0
- data/lib/dm-core/types/boolean.rb +7 -0
- data/lib/dm-core/types/discriminator.rb +32 -0
- data/lib/dm-core/types/object.rb +20 -0
- data/lib/dm-core/types/paranoid_boolean.rb +23 -0
- data/lib/dm-core/types/paranoid_datetime.rb +22 -0
- data/lib/dm-core/types/serial.rb +9 -0
- data/lib/dm-core/types/text.rb +10 -0
- data/spec/integration/association_spec.rb +1215 -0
- data/spec/integration/association_through_spec.rb +150 -0
- data/spec/integration/associations/many_to_many_spec.rb +171 -0
- data/spec/integration/associations/many_to_one_spec.rb +123 -0
- data/spec/integration/associations/one_to_many_spec.rb +66 -0
- data/spec/integration/auto_migrations_spec.rb +398 -0
- data/spec/integration/collection_spec.rb +1015 -0
- data/spec/integration/data_objects_adapter_spec.rb +32 -0
- data/spec/integration/model_spec.rb +68 -0
- data/spec/integration/mysql_adapter_spec.rb +85 -0
- data/spec/integration/postgres_adapter_spec.rb +732 -0
- data/spec/integration/property_spec.rb +224 -0
- data/spec/integration/query_spec.rb +376 -0
- data/spec/integration/repository_spec.rb +57 -0
- data/spec/integration/resource_spec.rb +324 -0
- data/spec/integration/sqlite3_adapter_spec.rb +352 -0
- data/spec/integration/sti_spec.rb +185 -0
- data/spec/integration/transaction_spec.rb +75 -0
- data/spec/integration/type_spec.rb +149 -0
- data/spec/lib/mock_adapter.rb +27 -0
- data/spec/spec_helper.rb +112 -0
- data/spec/unit/adapters/abstract_adapter_spec.rb +133 -0
- data/spec/unit/adapters/adapter_shared_spec.rb +15 -0
- data/spec/unit/adapters/data_objects_adapter_spec.rb +627 -0
- data/spec/unit/adapters/postgres_adapter_spec.rb +125 -0
- data/spec/unit/associations/many_to_many_spec.rb +14 -0
- data/spec/unit/associations/many_to_one_spec.rb +138 -0
- data/spec/unit/associations/one_to_many_spec.rb +385 -0
- data/spec/unit/associations/one_to_one_spec.rb +7 -0
- data/spec/unit/associations/relationship_spec.rb +67 -0
- data/spec/unit/associations_spec.rb +205 -0
- data/spec/unit/auto_migrations_spec.rb +110 -0
- data/spec/unit/collection_spec.rb +174 -0
- data/spec/unit/data_mapper_spec.rb +21 -0
- data/spec/unit/identity_map_spec.rb +126 -0
- data/spec/unit/is_spec.rb +80 -0
- data/spec/unit/migrator_spec.rb +33 -0
- data/spec/unit/model_spec.rb +339 -0
- data/spec/unit/naming_conventions_spec.rb +28 -0
- data/spec/unit/property_set_spec.rb +96 -0
- data/spec/unit/property_spec.rb +447 -0
- data/spec/unit/query_spec.rb +485 -0
- data/spec/unit/repository_spec.rb +93 -0
- data/spec/unit/resource_spec.rb +557 -0
- data/spec/unit/scope_spec.rb +131 -0
- data/spec/unit/transaction_spec.rb +493 -0
- data/spec/unit/type_map_spec.rb +114 -0
- data/spec/unit/type_spec.rb +119 -0
- metadata +187 -0
@@ -0,0 +1,125 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), '..', "..", 'spec_helper'))
|
2
|
+
|
3
|
+
if HAS_POSTGRES
|
4
|
+
describe DataMapper::Adapters::PostgresAdapter do
|
5
|
+
before :all do
|
6
|
+
@adapter = repository(:postgres).adapter
|
7
|
+
end
|
8
|
+
|
9
|
+
describe '#upgrade_model_storage' do
|
10
|
+
before do
|
11
|
+
@repository = mock('repository', :kind_of? => true, :name => :postgres)
|
12
|
+
@model = mock('model', :kind_of? => true, :storage_name => 'models')
|
13
|
+
@property = mock('property', :kind_of? => true, :model => @model, :serial? => true, :field => 'property')
|
14
|
+
|
15
|
+
@model.should_receive(:properties).with(:postgres).any_number_of_times.and_return([@property])
|
16
|
+
|
17
|
+
@command = mock('command')
|
18
|
+
@connection = mock('connection', :create_command => @command, :close => true)
|
19
|
+
@result = mock('result', :to_i => 0)
|
20
|
+
|
21
|
+
DataObjects::Connection.stub!(:new).and_return(@connection)
|
22
|
+
|
23
|
+
@adapter.stub!(:execute).and_return(@result)
|
24
|
+
@adapter.stub!(:storage_exists?).and_return(true)
|
25
|
+
@adapter.stub!(:query).and_return([ 0 ])
|
26
|
+
|
27
|
+
@original_method = @adapter.class.superclass.instance_method(:upgrade_model_storage)
|
28
|
+
@adapter.class.superclass.send(:define_method, :upgrade_model_storage) {}
|
29
|
+
end
|
30
|
+
|
31
|
+
after do
|
32
|
+
method = @original_method
|
33
|
+
@adapter.class.superclass.send(:define_method, :upgrade_model_storage) do |*args|
|
34
|
+
method.bind(self).call(*args)
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
it 'should check to make sure the sequences exist' do
|
39
|
+
statement = %q[SELECT COUNT(*) FROM "pg_class" WHERE "relkind" = 'S' AND "relname" = ?]
|
40
|
+
@adapter.should_receive(:query).with(statement, 'models_property_seq').and_return([ 0 ])
|
41
|
+
@adapter.upgrade_model_storage(@repository, @model)
|
42
|
+
end
|
43
|
+
|
44
|
+
it 'should add sequences' do
|
45
|
+
statement = %q[CREATE SEQUENCE "models_property_seq"]
|
46
|
+
@adapter.should_receive(:execute).with(statement)
|
47
|
+
@adapter.upgrade_model_storage(@repository, @model)
|
48
|
+
end
|
49
|
+
|
50
|
+
it 'should execute the superclass upgrade_model_storage' do
|
51
|
+
rv = mock('inside super')
|
52
|
+
@adapter.class.superclass.send(:define_method, :upgrade_model_storage) { rv }
|
53
|
+
@adapter.upgrade_model_storage(@repository, @model).should == rv
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
describe '#create_model_storage' do
|
58
|
+
before do
|
59
|
+
@repository = mock('repository', :kind_of? => true, :name => :postgres)
|
60
|
+
@model = mock('model', :kind_of? => true, :storage_name => 'models')
|
61
|
+
@property = mock('property', :kind_of? => true, :model => @model, :serial? => true, :field => 'property')
|
62
|
+
|
63
|
+
@model.should_receive(:properties).with(:postgres).any_number_of_times.and_return([@property])
|
64
|
+
|
65
|
+
@adapter.stub!(:execute).and_return(@result)
|
66
|
+
@adapter.stub!(:storage_exists?).and_return(true)
|
67
|
+
@adapter.stub!(:query).and_return([ 0 ])
|
68
|
+
|
69
|
+
@original_method = @adapter.class.superclass.instance_method(:create_table_statement)
|
70
|
+
@adapter.class.superclass.send(:define_method, :create_table_statement) {}
|
71
|
+
end
|
72
|
+
|
73
|
+
after do
|
74
|
+
method = @original_method
|
75
|
+
@adapter.class.superclass.send(:define_method, :create_table_statement) do |*args|
|
76
|
+
method.bind(self).call(*args)
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
it 'should check to make sure the sequences exist' do
|
81
|
+
statement = %q[SELECT COUNT(*) FROM "pg_class" WHERE "relkind" = 'S' AND "relname" = ?]
|
82
|
+
@adapter.should_receive(:query).with(statement, 'models_property_seq').and_return([ 0 ])
|
83
|
+
@adapter.create_model_storage(@repository, @model)
|
84
|
+
end
|
85
|
+
|
86
|
+
it 'should add sequences' do
|
87
|
+
statement = %q[CREATE SEQUENCE "models_property_seq"]
|
88
|
+
@adapter.should_receive(:execute).with(statement)
|
89
|
+
@adapter.create_model_storage(@repository, @model)
|
90
|
+
end
|
91
|
+
|
92
|
+
it 'should execute the superclass upgrade_model_storage' do
|
93
|
+
rv = mock('inside super')
|
94
|
+
@adapter.class.superclass.send(:define_method, :create_table_statement) { rv }
|
95
|
+
@adapter.create_table_statement(@repository, @model).should == rv
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
describe '#destroy_model_storage' do
|
100
|
+
before do
|
101
|
+
@repository = mock('repository', :kind_of? => true, :name => :postgres)
|
102
|
+
@model = mock('model', :kind_of? => true, :storage_name => 'models')
|
103
|
+
@property = mock('property', :kind_of? => true, :model => @model, :serial? => true, :field => 'property')
|
104
|
+
|
105
|
+
@model.should_receive(:properties).with(:postgres).any_number_of_times.and_return([@property])
|
106
|
+
|
107
|
+
@original_method = @adapter.class.superclass.instance_method(:destroy_model_storage)
|
108
|
+
@adapter.class.superclass.send(:define_method, :destroy_model_storage) {}
|
109
|
+
end
|
110
|
+
|
111
|
+
after do
|
112
|
+
method = @original_method
|
113
|
+
@adapter.class.superclass.send(:define_method, :destroy_model_storage) do |*args|
|
114
|
+
method.bind(self).call(*args)
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
it 'should execute the superclass destroy_model_storage' do
|
119
|
+
rv = mock('inside super')
|
120
|
+
@adapter.class.superclass.send(:define_method, :destroy_model_storage) { rv }
|
121
|
+
@adapter.destroy_model_storage(@repository, @model).should == rv
|
122
|
+
end
|
123
|
+
end
|
124
|
+
end
|
125
|
+
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), '..', '..', 'spec_helper'))
|
2
|
+
|
3
|
+
describe DataMapper::Associations::ManyToMany do
|
4
|
+
it 'should allow a declaration' do
|
5
|
+
lambda do
|
6
|
+
class Supplier
|
7
|
+
has n, :manufacturers, :through => Resource
|
8
|
+
end
|
9
|
+
end.should_not raise_error
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
describe DataMapper::Associations::ManyToMany::Proxy do
|
14
|
+
end
|
@@ -0,0 +1,138 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), '..', '..', 'spec_helper'))
|
2
|
+
|
3
|
+
describe DataMapper::Associations::ManyToOne do
|
4
|
+
it 'should allow a declaration' do
|
5
|
+
lambda do
|
6
|
+
class Vehicle
|
7
|
+
belongs_to :manufacturer
|
8
|
+
end
|
9
|
+
end.should_not raise_error
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
describe DataMapper::Associations::ManyToOne::Proxy do
|
14
|
+
before do
|
15
|
+
@child = mock('child', :kind_of? => true)
|
16
|
+
@parent = mock('parent')
|
17
|
+
@relationship = mock('relationship', :kind_of? => true, :repository_name => :default, :get_parent => @parent, :attach_parent => nil)
|
18
|
+
@association = DataMapper::Associations::ManyToOne::Proxy.new(@relationship, @child)
|
19
|
+
|
20
|
+
@association.replace(@parent)
|
21
|
+
end
|
22
|
+
|
23
|
+
it 'should provide #replace' do
|
24
|
+
@association.should respond_to(:replace)
|
25
|
+
end
|
26
|
+
|
27
|
+
describe '#replace' do
|
28
|
+
before do
|
29
|
+
@other = mock('other parent')
|
30
|
+
end
|
31
|
+
|
32
|
+
before do
|
33
|
+
@relationship.should_receive(:attach_parent).with(@child, @other)
|
34
|
+
end
|
35
|
+
|
36
|
+
it 'should remove the resource from the collection' do
|
37
|
+
@association.should == @parent
|
38
|
+
@association.replace(@other)
|
39
|
+
@association.should == @other
|
40
|
+
end
|
41
|
+
|
42
|
+
it 'should not automatically save that the resource was removed from the association' do
|
43
|
+
@other.should_not_receive(:save)
|
44
|
+
@association.replace(@other)
|
45
|
+
end
|
46
|
+
|
47
|
+
it 'should return the association' do
|
48
|
+
@association.replace(@other).object_id.should == @association.object_id
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
it 'should provide #save' do
|
53
|
+
@association.should respond_to(:replace)
|
54
|
+
end
|
55
|
+
|
56
|
+
describe '#save' do
|
57
|
+
describe 'when the parent is nil' do
|
58
|
+
before do
|
59
|
+
@parent.should_receive(:nil?).with(no_args).and_return(true)
|
60
|
+
end
|
61
|
+
|
62
|
+
it 'should not save the parent' do
|
63
|
+
@association.save
|
64
|
+
end
|
65
|
+
|
66
|
+
it 'should return false' do
|
67
|
+
@association.save.should == false
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
describe 'when the parent is not a new record' do
|
72
|
+
before do
|
73
|
+
@parent.should_receive(:new_record?).with(no_args).and_return(false)
|
74
|
+
end
|
75
|
+
|
76
|
+
it 'should not save the parent' do
|
77
|
+
@parent.should_not_receive(:save)
|
78
|
+
@association.save
|
79
|
+
end
|
80
|
+
|
81
|
+
it 'should return true' do
|
82
|
+
@association.save.should == true
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
describe 'when the parent is a new record' do
|
87
|
+
before do
|
88
|
+
@parent.should_receive(:new_record?).with(no_args).and_return(true)
|
89
|
+
end
|
90
|
+
|
91
|
+
it 'should save the parent' do
|
92
|
+
@parent.should_receive(:save).with(no_args)
|
93
|
+
@association.save
|
94
|
+
end
|
95
|
+
|
96
|
+
it 'should return the result of the save' do
|
97
|
+
save_results = mock('save results')
|
98
|
+
@parent.should_receive(:save).with(no_args).and_return(save_results)
|
99
|
+
@association.save.object_id.should == save_results.object_id
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
it 'should provide #reload' do
|
105
|
+
@association.should respond_to(:reload)
|
106
|
+
end
|
107
|
+
|
108
|
+
describe '#reload' do
|
109
|
+
before(:each) do
|
110
|
+
@mock_parent = mock('#reload test parent')
|
111
|
+
@association.replace(@mock_parent)
|
112
|
+
end
|
113
|
+
|
114
|
+
it 'should set the @parent ivar to nil' do
|
115
|
+
@association.__send__(:parent).should == @mock_parent # Sanity check.
|
116
|
+
|
117
|
+
# We can't test the value of the instance variable since
|
118
|
+
# #instance_variable_get will be run on the @parent (thanks to
|
119
|
+
# Proxy#method_missing). Instead, test that Relationship#get_parent is
|
120
|
+
# run -- if @parent wasn't set to nil, this expectation should fail.
|
121
|
+
@relationship.should_receive(:get_parent).once.and_return(@mock_parent)
|
122
|
+
@association.reload
|
123
|
+
|
124
|
+
# Trigger #get_parent on the relationship.
|
125
|
+
@association.__send__(:parent)
|
126
|
+
end
|
127
|
+
|
128
|
+
it 'should not change the foreign key in the child' do
|
129
|
+
@relationship.should_not_receive(:attach_parent)
|
130
|
+
@association.reload
|
131
|
+
end
|
132
|
+
|
133
|
+
it 'should return self' do
|
134
|
+
@association.reload.should be_kind_of(DataMapper::Associations::ManyToOne::Proxy)
|
135
|
+
@association.reload.object_id.should == @association.object_id
|
136
|
+
end
|
137
|
+
end
|
138
|
+
end
|
@@ -0,0 +1,385 @@
|
|
1
|
+
require File.expand_path(File.join(File.dirname(__FILE__), '..', '..', 'spec_helper'))
|
2
|
+
|
3
|
+
describe DataMapper::Associations::OneToMany do
|
4
|
+
before do
|
5
|
+
@class = Class.new do
|
6
|
+
def self.name
|
7
|
+
'User'
|
8
|
+
end
|
9
|
+
|
10
|
+
include DataMapper::Resource
|
11
|
+
|
12
|
+
property :user_id, Integer, :key => true
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
it 'should provide #has' do
|
17
|
+
@class.should respond_to(:has)
|
18
|
+
end
|
19
|
+
|
20
|
+
describe '#has' do
|
21
|
+
it 'should return a Relationship' do
|
22
|
+
@class.has(@class.n, :orders).should be_kind_of(DataMapper::Associations::Relationship)
|
23
|
+
end
|
24
|
+
|
25
|
+
describe 'relationship' do
|
26
|
+
before do
|
27
|
+
@relationship = mock('relationship')
|
28
|
+
DataMapper::Associations::Relationship.stub!(:new).and_return(@relationship)
|
29
|
+
end
|
30
|
+
|
31
|
+
it 'should receive the name' do
|
32
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |name,_,_,_,_|
|
33
|
+
name.should == :user
|
34
|
+
end
|
35
|
+
@class.has(@class.n, :orders)
|
36
|
+
end
|
37
|
+
|
38
|
+
it 'should receive the repository name' do
|
39
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |_,repository_name,_,_,_|
|
40
|
+
repository_name.should == :mock
|
41
|
+
end
|
42
|
+
repository(:mock) do
|
43
|
+
@class.has(@class.n, :orders)
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
it 'should receive the child model name when passed in as class_name' do
|
48
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |_,_,child_model_name,_,_|
|
49
|
+
child_model_name.should == 'Company::Order'
|
50
|
+
end
|
51
|
+
@class.has(@class.n, :orders, :class_name => 'Company::Order')
|
52
|
+
end
|
53
|
+
|
54
|
+
it 'should receive the child model name when class_name not passed in' do
|
55
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |_,_,child_model_name,_,_|
|
56
|
+
child_model_name.should == 'Order'
|
57
|
+
end
|
58
|
+
@class.has(@class.n, :orders)
|
59
|
+
end
|
60
|
+
|
61
|
+
it 'should receive the parent model name' do
|
62
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |_,_,_,parent_model_name,_|
|
63
|
+
parent_model_name.should == 'User'
|
64
|
+
end
|
65
|
+
@class.has(@class.n, :orders)
|
66
|
+
end
|
67
|
+
|
68
|
+
it 'should receive the parent model name' do
|
69
|
+
options = { :min => 0, :max => 100 }
|
70
|
+
DataMapper::Associations::Relationship.should_receive(:new) do |_,_,_,parent_model_name,_|
|
71
|
+
options.object_id.should == options.object_id
|
72
|
+
end
|
73
|
+
@class.has(@class.n, :orders, options)
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
it 'should add an accessor for the proxy' do
|
78
|
+
@class.new.should_not respond_to(:orders)
|
79
|
+
@class.has(@class.n, :orders)
|
80
|
+
@class.new.should respond_to(:orders)
|
81
|
+
end
|
82
|
+
|
83
|
+
describe 'proxy accessor' do
|
84
|
+
before :all do
|
85
|
+
class User
|
86
|
+
include DataMapper::Resource
|
87
|
+
end
|
88
|
+
|
89
|
+
class Order
|
90
|
+
include DataMapper::Resource
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
it 'should return a OneToMany::Proxy' do
|
95
|
+
@class.has(@class.n, :orders)
|
96
|
+
@class.new.orders.should be_kind_of(DataMapper::Associations::OneToMany::Proxy)
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
it 'should work with classes inside modules'
|
102
|
+
end
|
103
|
+
|
104
|
+
describe DataMapper::Associations::OneToMany::Proxy do
|
105
|
+
before do
|
106
|
+
@parent = mock('parent', :new_record? => true, :kind_of? => true)
|
107
|
+
@resource = mock('resource', :null_object => true)
|
108
|
+
@collection = []
|
109
|
+
@repository = mock('repository', :save => nil, :kind_of? => true)
|
110
|
+
@relationship = mock('relationship', :get_children => @collection, :repository_name => :mock, :query => {}, :kind_of? => true)
|
111
|
+
@association = DataMapper::Associations::OneToMany::Proxy.new(@relationship, @parent)
|
112
|
+
end
|
113
|
+
|
114
|
+
describe 'a method that relates the resource', :shared => true do
|
115
|
+
it 'should add the resource to the collection' do
|
116
|
+
@association.should_not include(@resource)
|
117
|
+
do_add.should == return_value
|
118
|
+
@association.should include(@resource)
|
119
|
+
end
|
120
|
+
|
121
|
+
it 'should not automatically save that the resource was added to the association' do
|
122
|
+
@relationship.should_not_receive(:attach_parent)
|
123
|
+
do_add.should == return_value
|
124
|
+
end
|
125
|
+
|
126
|
+
it 'should persist the addition after saving the association' do
|
127
|
+
do_add.should == return_value
|
128
|
+
@relationship.should_receive(:attach_parent).with(@resource, @parent)
|
129
|
+
@association.save
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
describe 'a method that orphans the resource', :shared => true do
|
134
|
+
before do
|
135
|
+
@association << @resource
|
136
|
+
end
|
137
|
+
|
138
|
+
it 'should remove the resource from the collection' do
|
139
|
+
@association.should include(@resource)
|
140
|
+
do_remove.should == return_value
|
141
|
+
@association.should_not include(@resource)
|
142
|
+
end
|
143
|
+
|
144
|
+
it 'should not automatically save that the resource was removed from the association' do
|
145
|
+
@relationship.should_not_receive(:attach_parent)
|
146
|
+
do_remove.should == return_value
|
147
|
+
end
|
148
|
+
|
149
|
+
it 'should persist the removal after saving the association' do
|
150
|
+
do_remove.should == return_value
|
151
|
+
@relationship.should_receive(:attach_parent).with(@resource, nil)
|
152
|
+
@association.save
|
153
|
+
end
|
154
|
+
end
|
155
|
+
|
156
|
+
it 'should provide #<<' do
|
157
|
+
@association.should respond_to(:<<)
|
158
|
+
end
|
159
|
+
|
160
|
+
describe '#<<' do
|
161
|
+
def do_add
|
162
|
+
@association << @resource
|
163
|
+
end
|
164
|
+
|
165
|
+
def return_value
|
166
|
+
@association
|
167
|
+
end
|
168
|
+
|
169
|
+
it_should_behave_like 'a method that relates the resource'
|
170
|
+
end
|
171
|
+
|
172
|
+
it 'should provide #push' do
|
173
|
+
@association.should respond_to(:push)
|
174
|
+
end
|
175
|
+
|
176
|
+
describe '#push' do
|
177
|
+
def do_add
|
178
|
+
@association.push(@resource)
|
179
|
+
end
|
180
|
+
|
181
|
+
def return_value
|
182
|
+
@association
|
183
|
+
end
|
184
|
+
|
185
|
+
it_should_behave_like 'a method that relates the resource'
|
186
|
+
end
|
187
|
+
|
188
|
+
it 'should provide #unshift' do
|
189
|
+
@association.should respond_to(:unshift)
|
190
|
+
end
|
191
|
+
|
192
|
+
describe '#unshift' do
|
193
|
+
def do_add
|
194
|
+
@association.unshift(@resource)
|
195
|
+
end
|
196
|
+
|
197
|
+
def return_value
|
198
|
+
@association
|
199
|
+
end
|
200
|
+
|
201
|
+
it_should_behave_like 'a method that relates the resource'
|
202
|
+
end
|
203
|
+
|
204
|
+
it 'should provide #replace' do
|
205
|
+
@association.should respond_to(:replace)
|
206
|
+
end
|
207
|
+
|
208
|
+
describe '#replace' do
|
209
|
+
before do
|
210
|
+
@children = [
|
211
|
+
mock('child 1', :save => true),
|
212
|
+
mock('child 2', :save => true),
|
213
|
+
]
|
214
|
+
@collection << @resource
|
215
|
+
@collection.stub!(:loaded?).and_return(true)
|
216
|
+
@relationship.stub!(:attach_parent)
|
217
|
+
end
|
218
|
+
|
219
|
+
def do_replace
|
220
|
+
@association.replace(@children)
|
221
|
+
end
|
222
|
+
|
223
|
+
def return_value
|
224
|
+
@association
|
225
|
+
end
|
226
|
+
|
227
|
+
it 'should remove the resource from the collection' do
|
228
|
+
@association.should include(@resource)
|
229
|
+
do_replace.should == return_value
|
230
|
+
@association.should_not include(@resource)
|
231
|
+
end
|
232
|
+
|
233
|
+
it 'should not automatically save that the resource was removed from the association' do
|
234
|
+
@relationship.should_not_receive(:attach_parent)
|
235
|
+
do_replace.should == return_value
|
236
|
+
end
|
237
|
+
|
238
|
+
it 'should persist the removal after saving the association' do
|
239
|
+
do_replace.should == return_value
|
240
|
+
@relationship.should_receive(:attach_parent).with(@resource, nil)
|
241
|
+
@association.save
|
242
|
+
end
|
243
|
+
|
244
|
+
it 'should not automatically save that the children were added to the association' do
|
245
|
+
@relationship.should_not_receive(:attach_parent)
|
246
|
+
do_replace.should == return_value
|
247
|
+
end
|
248
|
+
|
249
|
+
it 'should persist the addition after saving the association' do
|
250
|
+
do_replace.should == return_value
|
251
|
+
@relationship.should_receive(:attach_parent).with(@children[0], @parent)
|
252
|
+
@relationship.should_receive(:attach_parent).with(@children[1], @parent)
|
253
|
+
@association.save
|
254
|
+
end
|
255
|
+
end
|
256
|
+
|
257
|
+
it 'should provide #pop' do
|
258
|
+
@association.should respond_to(:pop)
|
259
|
+
end
|
260
|
+
|
261
|
+
describe '#pop' do
|
262
|
+
def do_remove
|
263
|
+
@association.pop
|
264
|
+
end
|
265
|
+
|
266
|
+
def return_value
|
267
|
+
@resource
|
268
|
+
end
|
269
|
+
|
270
|
+
it_should_behave_like 'a method that orphans the resource'
|
271
|
+
end
|
272
|
+
|
273
|
+
it 'should provide #shift' do
|
274
|
+
@association.should respond_to(:shift)
|
275
|
+
end
|
276
|
+
|
277
|
+
describe '#shift' do
|
278
|
+
def do_remove
|
279
|
+
@association.shift
|
280
|
+
end
|
281
|
+
|
282
|
+
def return_value
|
283
|
+
@resource
|
284
|
+
end
|
285
|
+
|
286
|
+
it_should_behave_like 'a method that orphans the resource'
|
287
|
+
end
|
288
|
+
|
289
|
+
it 'should provide #delete' do
|
290
|
+
@association.should respond_to(:delete)
|
291
|
+
end
|
292
|
+
|
293
|
+
describe '#delete' do
|
294
|
+
def do_remove
|
295
|
+
@association.delete(@resource)
|
296
|
+
end
|
297
|
+
|
298
|
+
def return_value
|
299
|
+
@resource
|
300
|
+
end
|
301
|
+
|
302
|
+
it_should_behave_like 'a method that orphans the resource'
|
303
|
+
end
|
304
|
+
|
305
|
+
it 'should provide #delete_at' do
|
306
|
+
@association.should respond_to(:delete_at)
|
307
|
+
end
|
308
|
+
|
309
|
+
describe '#delete_at' do
|
310
|
+
def do_remove
|
311
|
+
@association.delete_at(0)
|
312
|
+
end
|
313
|
+
|
314
|
+
def return_value
|
315
|
+
@resource
|
316
|
+
end
|
317
|
+
|
318
|
+
it_should_behave_like 'a method that orphans the resource'
|
319
|
+
end
|
320
|
+
|
321
|
+
it 'should provide #clear' do
|
322
|
+
@association.should respond_to(:clear)
|
323
|
+
end
|
324
|
+
|
325
|
+
describe '#clear' do
|
326
|
+
def do_remove
|
327
|
+
@association.clear
|
328
|
+
end
|
329
|
+
|
330
|
+
def return_value
|
331
|
+
@association
|
332
|
+
end
|
333
|
+
|
334
|
+
it_should_behave_like 'a method that orphans the resource'
|
335
|
+
|
336
|
+
it 'should empty the collection' do
|
337
|
+
@association << mock('other resource')
|
338
|
+
@association.should have(2).entries
|
339
|
+
do_remove
|
340
|
+
@association.should be_empty
|
341
|
+
end
|
342
|
+
end
|
343
|
+
|
344
|
+
it 'should provide #reload' do
|
345
|
+
@association.should respond_to(:reload)
|
346
|
+
end
|
347
|
+
|
348
|
+
describe '#reload' do
|
349
|
+
before do
|
350
|
+
@children = [ mock('child 1', :save => true), mock('child 2', :save => true) ]
|
351
|
+
@relationship.stub!(:get_children).and_return(@children)
|
352
|
+
end
|
353
|
+
|
354
|
+
it 'should set the @children ivar to nil' do
|
355
|
+
@association.__send__(:children).should == @children # Sanity check.
|
356
|
+
|
357
|
+
# We can't test the value of the @children instance variable since
|
358
|
+
# #instance_variable_get will be run on @children (thanks to
|
359
|
+
# Proxy#method_missing). Instead, test that Relationship#get_children is
|
360
|
+
# run -- if @children wasn't set to nil, this expectation should fail.
|
361
|
+
@relationship.should_receive(:get_children).once.and_return(@children)
|
362
|
+
@association.reload
|
363
|
+
|
364
|
+
# Trigger #get_children on the relationship.
|
365
|
+
@association.__send__(:children).should == @children
|
366
|
+
end
|
367
|
+
|
368
|
+
it 'should return self' do
|
369
|
+
@association.reload.should be_kind_of(DataMapper::Associations::OneToMany::Proxy)
|
370
|
+
@association.reload.object_id.should == @association.object_id
|
371
|
+
end
|
372
|
+
end
|
373
|
+
|
374
|
+
describe 'when deleting the parent' do
|
375
|
+
it 'should delete all the children without calling destroy if relationship :dependent is :delete_all'
|
376
|
+
|
377
|
+
it 'should destroy all the children if relationship :dependent is :destroy'
|
378
|
+
|
379
|
+
it 'should set the parent key for each child to nil if relationship :dependent is :nullify'
|
380
|
+
|
381
|
+
it 'should restrict the parent from being deleted if a child remains if relationship :dependent is restrict'
|
382
|
+
|
383
|
+
it 'should be restrict by default if relationship :dependent is not specified'
|
384
|
+
end
|
385
|
+
end
|