bullet 2.2.1 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/.gitignore +1 -0
- data/.travis.yml +4 -1
- data/Gemfile +3 -2
- data/Gemfile.lock +85 -69
- data/Guardfile +8 -0
- data/lib/bullet.rb +13 -8
- data/lib/bullet/action_controller2.rb +4 -4
- data/lib/bullet/active_record2.rb +5 -6
- data/lib/bullet/active_record3.rb +2 -3
- data/lib/bullet/active_record31.rb +6 -8
- data/lib/bullet/detector/association.rb +27 -53
- data/lib/bullet/detector/counter.rb +34 -29
- data/lib/bullet/detector/n_plus_one_query.rb +47 -28
- data/lib/bullet/detector/unused_eager_association.rb +31 -30
- data/lib/bullet/notification/base.rb +14 -12
- data/lib/bullet/notification/n_plus_one_query.rb +6 -10
- data/lib/bullet/notification/unused_eager_loading.rb +1 -2
- data/lib/bullet/notification_collector.rb +1 -2
- data/lib/bullet/rack.rb +6 -3
- data/lib/bullet/registry/association.rb +4 -6
- data/lib/bullet/registry/base.rb +10 -7
- data/lib/bullet/registry/object.rb +6 -6
- data/lib/bullet/version.rb +1 -1
- data/perf/benchmark.rb +30 -12
- data/spec/bullet/detector/association_spec.rb +90 -0
- data/spec/bullet/detector/base_spec.rb +14 -0
- data/spec/bullet/detector/counter_spec.rb +65 -0
- data/spec/bullet/detector/n_plus_one_query_spec.rb +94 -0
- data/spec/bullet/detector/unused_eager_association_spec.rb +62 -0
- data/spec/bullet/notification/base_spec.rb +67 -0
- data/spec/bullet/notification/counter_cache_spec.rb +12 -0
- data/spec/bullet/notification/n_plus_one_query_spec.rb +13 -0
- data/spec/bullet/notification/unused_eager_loading_spec.rb +12 -0
- data/spec/bullet/notification_collector_spec.rb +32 -0
- data/spec/bullet/rack_spec.rb +80 -24
- data/spec/bullet/registry/association_spec.rb +26 -0
- data/spec/bullet/registry/base_spec.rb +44 -0
- data/spec/bullet/registry/object_spec.rb +25 -0
- data/spec/integration/association_for_chris_spec.rb +37 -0
- data/spec/integration/association_for_peschkaj_spec.rb +26 -0
- data/spec/{bullet → integration}/association_spec.rb +1 -359
- data/spec/integration/counter_spec.rb +37 -0
- data/spec/models/address.rb +3 -0
- data/spec/models/author.rb +3 -0
- data/spec/models/base_user.rb +5 -0
- data/spec/models/category.rb +7 -0
- data/spec/models/city.rb +3 -0
- data/spec/models/client.rb +4 -0
- data/spec/models/comment.rb +4 -0
- data/spec/models/company.rb +3 -0
- data/spec/models/contact.rb +3 -0
- data/spec/models/country.rb +3 -0
- data/spec/models/deal.rb +4 -0
- data/spec/models/document.rb +5 -0
- data/spec/models/email.rb +3 -0
- data/spec/models/entry.rb +3 -0
- data/spec/models/firm.rb +4 -0
- data/spec/models/folder.rb +2 -0
- data/spec/models/hotel.rb +4 -0
- data/spec/models/location.rb +3 -0
- data/spec/models/newspaper.rb +3 -0
- data/spec/models/page.rb +2 -0
- data/spec/models/person.rb +3 -0
- data/spec/models/pet.rb +3 -0
- data/spec/models/post.rb +11 -0
- data/spec/models/relationship.rb +4 -0
- data/spec/models/student.rb +3 -0
- data/spec/models/submission.rb +4 -0
- data/spec/models/teacher.rb +3 -0
- data/spec/models/user.rb +4 -0
- data/spec/models/writer.rb +2 -0
- data/spec/spec_helper.rb +16 -106
- data/spec/support/bullet_ext.rb +55 -0
- data/spec/support/rack_double.rb +55 -0
- data/spec/support/seed.rb +256 -0
- metadata +104 -14
- data/.watchr +0 -24
- data/spec/bullet/association_for_chris_spec.rb +0 -96
- data/spec/bullet/association_for_peschkaj_spec.rb +0 -86
- data/spec/bullet/counter_spec.rb +0 -136
data/lib/bullet/version.rb
CHANGED
data/perf/benchmark.rb
CHANGED
@@ -83,16 +83,18 @@ PerfTools::CpuProfiler.start(ARGV[0]|| "benchmark_profile") if defined? PerfTool
|
|
83
83
|
|
84
84
|
Benchmark.bm(70) do |bm|
|
85
85
|
bm.report("Querying & Iterating #{posts_size} Posts with #{comments_size} Comments and #{users_size} Users") do
|
86
|
-
|
87
|
-
|
88
|
-
p
|
89
|
-
|
90
|
-
|
91
|
-
c
|
92
|
-
|
86
|
+
10.times do
|
87
|
+
Bullet.start_request
|
88
|
+
Post.select("SQL_NO_CACHE *").includes(:user, :comments => :user).each do |p|
|
89
|
+
p.title
|
90
|
+
p.user.name
|
91
|
+
p.comments.each do |c|
|
92
|
+
c.body
|
93
|
+
c.user.name
|
94
|
+
end
|
93
95
|
end
|
96
|
+
Bullet.end_request
|
94
97
|
end
|
95
|
-
Bullet.end_request
|
96
98
|
end
|
97
99
|
end
|
98
100
|
|
@@ -100,7 +102,23 @@ PerfTools::CpuProfiler.stop if defined? PerfTools
|
|
100
102
|
puts "End benchmarking..."
|
101
103
|
|
102
104
|
|
103
|
-
#
|
104
|
-
#
|
105
|
-
#
|
106
|
-
|
105
|
+
# Run benchmark with bundler
|
106
|
+
#
|
107
|
+
# bundle exec ruby perf/benchmark.rb
|
108
|
+
#
|
109
|
+
# bullet 2.3.0 with rails 3.2.2
|
110
|
+
# user system total real
|
111
|
+
# Querying & Iterating 1000 Posts with 10000 Comments and 100 Users 16.460000 0.190000 16.650000 ( 16.968246)
|
112
|
+
#
|
113
|
+
# bullet 2.3.0 with rails 3.1.4
|
114
|
+
# user system total real
|
115
|
+
# Querying & Iterating 1000 Posts with 10000 Comments and 100 Users 14.600000 0.130000 14.730000 ( 14.937590)
|
116
|
+
#
|
117
|
+
# bullet 2.3.0 with rails 3.0.12
|
118
|
+
# user system total real
|
119
|
+
# Querying & Iterating 1000 Posts with 10000 Comments and 100 Users 26.120000 0.430000 26.550000 ( 27.179304)
|
120
|
+
#
|
121
|
+
#
|
122
|
+
# bullet 2.2.1 with rails 3.0.12
|
123
|
+
# user system total real
|
124
|
+
# Querying & Iterating 1000 Posts with 10000 Comments and 100 Users 29.970000 0.270000 30.240000 ( 30.452083)
|
@@ -0,0 +1,90 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Bullet
|
4
|
+
module Detector
|
5
|
+
describe Association do
|
6
|
+
before :all do
|
7
|
+
@post1 = Post.first
|
8
|
+
@post2 = Post.last
|
9
|
+
end
|
10
|
+
before(:each) { Association.clear }
|
11
|
+
|
12
|
+
context ".start_request" do
|
13
|
+
it "should set @@checked to false" do
|
14
|
+
Association.start_request
|
15
|
+
Association.class_variable_get(:@@checked).should be_false
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
context ".clear" do
|
20
|
+
it "should clear all class variables" do
|
21
|
+
Association.clear
|
22
|
+
Association.class_variable_get(:@@object_associations).should be_nil
|
23
|
+
Association.class_variable_get(:@@possible_objects).should be_nil
|
24
|
+
Association.class_variable_get(:@@impossible_objects).should be_nil
|
25
|
+
Association.class_variable_get(:@@call_object_associations).should be_nil
|
26
|
+
Association.class_variable_get(:@@eager_loadings).should be_nil
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
context ".add_object_association" do
|
31
|
+
it "should add object, associations pair" do
|
32
|
+
Association.add_object_associations(@post1, :associations)
|
33
|
+
Association.send(:object_associations).should be_include(@post1.ar_key, :associations)
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
context ".add_call_object_associations" do
|
38
|
+
it "should add call object, associations pair" do
|
39
|
+
Association.add_call_object_associations(@post1, :associations)
|
40
|
+
Association.send(:call_object_associations).should be_include(@post1.ar_key, :associations)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
context ".add_possible_objects" do
|
45
|
+
it "should add possible objects" do
|
46
|
+
Association.add_possible_objects([@post1, @post2])
|
47
|
+
Association.send(:possible_objects).should be_include(@post1.ar_key)
|
48
|
+
Association.send(:possible_objects).should be_include(@post2.ar_key)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
context ".add_impossible_object" do
|
53
|
+
it "should add impossible object" do
|
54
|
+
Association.add_impossible_object(@post1)
|
55
|
+
Association.send(:impossible_objects).should be_include(@post1.ar_key)
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
context ".add_eager_loadings" do
|
60
|
+
it "should add objects, associations pair when eager_loadings are empty" do
|
61
|
+
Association.add_eager_loadings([@post1, @post2], :associations)
|
62
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key, @post2.ar_key], :associations)
|
63
|
+
end
|
64
|
+
|
65
|
+
it "should add objects, associations pair for existing eager_loadings" do
|
66
|
+
Association.add_eager_loadings([@post1, @post2], :association1)
|
67
|
+
Association.add_eager_loadings([@post1, @post2], :association2)
|
68
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key, @post2.ar_key], :association1)
|
69
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key, @post2.ar_key], :association2)
|
70
|
+
end
|
71
|
+
|
72
|
+
it "should merge objects, associations pair for existing eager_loadings" do
|
73
|
+
Association.add_eager_loadings(@post1, :association1)
|
74
|
+
Association.add_eager_loadings([@post1, @post2], :association2)
|
75
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key], :association1)
|
76
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key], :association2)
|
77
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key, @post2.ar_key], :association2)
|
78
|
+
end
|
79
|
+
|
80
|
+
it "should delete objects, associations pair for existing eager_loadings" do
|
81
|
+
Association.add_eager_loadings([@post1, @post2], :association1)
|
82
|
+
Association.add_eager_loadings(@post1, :association2)
|
83
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key], :association1)
|
84
|
+
Association.send(:eager_loadings).should be_include([@post1.ar_key], :association2)
|
85
|
+
Association.send(:eager_loadings).should be_include([@post2.ar_key], :association1)
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
89
|
+
end
|
90
|
+
end
|
@@ -0,0 +1,65 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Bullet
|
4
|
+
module Detector
|
5
|
+
describe Counter do
|
6
|
+
before :all do
|
7
|
+
@post1 = Post.first
|
8
|
+
@post2 = Post.last
|
9
|
+
end
|
10
|
+
before(:each) { Counter.clear }
|
11
|
+
|
12
|
+
context ".clear" do
|
13
|
+
it "should clear all class variables" do
|
14
|
+
Counter.clear
|
15
|
+
Counter.class_variable_get(:@@possible_objects).should be_nil
|
16
|
+
Counter.class_variable_get(:@@impossible_objects).should be_nil
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
context ".add_counter_cache" do
|
21
|
+
it "should create notification if conditions met" do
|
22
|
+
Counter.should_receive(:conditions_met?).with(@post1.ar_key, [:comments]).and_return(true)
|
23
|
+
Counter.should_receive(:create_notification).with("Post", [:comments])
|
24
|
+
Counter.add_counter_cache(@post1, [:comments])
|
25
|
+
end
|
26
|
+
|
27
|
+
it "should not create notification if conditions not met" do
|
28
|
+
Counter.should_receive(:conditions_met?).with(@post1.ar_key, [:comments]).and_return(false)
|
29
|
+
Counter.should_receive(:create_notification).never
|
30
|
+
Counter.add_counter_cache(@post1, [:comments])
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
context ".add_possible_objects" do
|
35
|
+
it "should add possible objects" do
|
36
|
+
Counter.add_possible_objects([@post1, @post2])
|
37
|
+
Counter.send(:possible_objects).should be_include(@post1.ar_key)
|
38
|
+
Counter.send(:possible_objects).should be_include(@post2.ar_key)
|
39
|
+
end
|
40
|
+
|
41
|
+
it "should add impossible object" do
|
42
|
+
Counter.add_impossible_object(@post1)
|
43
|
+
Counter.send(:impossible_objects).should be_include(@post1.ar_key)
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
context ".conditions_met?" do
|
48
|
+
it "should be true when object is possible, not impossible" do
|
49
|
+
Counter.add_possible_objects(@post1)
|
50
|
+
Counter.send(:conditions_met?, @post1.ar_key, :associations).should be_true
|
51
|
+
end
|
52
|
+
|
53
|
+
it "should be false when object is not possible" do
|
54
|
+
Counter.send(:conditions_met?, @post1.ar_key, :associations).should be_false
|
55
|
+
end
|
56
|
+
|
57
|
+
it "should be true when object is possible, and impossible" do
|
58
|
+
Counter.add_possible_objects(@post1)
|
59
|
+
Counter.add_impossible_object(@post1)
|
60
|
+
Counter.send(:conditions_met?, @post1.ar_key, :associations).should be_false
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
@@ -0,0 +1,94 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Bullet
|
4
|
+
module Detector
|
5
|
+
describe NPlusOneQuery do
|
6
|
+
before(:all) { @post = Post.first }
|
7
|
+
before(:each) { NPlusOneQuery.clear }
|
8
|
+
|
9
|
+
context ".call_association" do
|
10
|
+
it "should set @@checked to true" do
|
11
|
+
NPlusOneQuery.call_association(@post, :associations)
|
12
|
+
NPlusOneQuery.class_variable_get(:@@checked).should be_true
|
13
|
+
end
|
14
|
+
|
15
|
+
it "should add call_object_associations" do
|
16
|
+
NPlusOneQuery.should_receive(:add_call_object_associations).with(@post, :associations)
|
17
|
+
NPlusOneQuery.call_association(@post, :associations)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
context ".possible?" do
|
22
|
+
it "should be true if possible_objects contain" do
|
23
|
+
NPlusOneQuery.add_possible_objects(@post)
|
24
|
+
NPlusOneQuery.send(:possible?, @post.ar_key).should be_true
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
context ".impossible?" do
|
29
|
+
it "should be true if impossible_objects contain" do
|
30
|
+
NPlusOneQuery.add_impossible_object(@post)
|
31
|
+
NPlusOneQuery.send(:impossible?, @post.ar_key).should be_true
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
context ".association?" do
|
36
|
+
it "should be true if object, associations pair is already existed" do
|
37
|
+
NPlusOneQuery.add_object_associations(@post, :association)
|
38
|
+
NPlusOneQuery.send(:association?, @post.ar_key, :association).should be_true
|
39
|
+
end
|
40
|
+
|
41
|
+
it "should be false if object, association pair is not existed" do
|
42
|
+
NPlusOneQuery.add_object_associations(@post, :association1)
|
43
|
+
NPlusOneQuery.send(:association?, @post.ar_key, :associatio2).should be_false
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
context ".conditions_met?" do
|
48
|
+
it "should be true if object is possible, not impossible and object, associations pair is not already existed" do
|
49
|
+
NPlusOneQuery.stub(:possible?).with(@post.ar_key).and_return(true)
|
50
|
+
NPlusOneQuery.stub(:impossible?).with(@post.ar_key).and_return(false)
|
51
|
+
NPlusOneQuery.stub(:association?).with(@post.ar_key, :associations).and_return(false)
|
52
|
+
NPlusOneQuery.send(:conditions_met?, @post.ar_key, :associations).should be_true
|
53
|
+
end
|
54
|
+
|
55
|
+
it "should be false if object is not possible, not impossible and object, associations pair is not already existed" do
|
56
|
+
NPlusOneQuery.stub(:possible?).with(@post.ar_key).and_return(false)
|
57
|
+
NPlusOneQuery.stub(:impossible?).with(@post.ar_key).and_return(false)
|
58
|
+
NPlusOneQuery.stub(:association?).with(@post.ar_key, :associations).and_return(false)
|
59
|
+
NPlusOneQuery.send(:conditions_met?, @post.ar_key, :associations).should be_false
|
60
|
+
end
|
61
|
+
|
62
|
+
it "should be false if object is possible, but impossible and object, associations pair is not already existed" do
|
63
|
+
NPlusOneQuery.stub(:possible?).with(@post.ar_key).and_return(true)
|
64
|
+
NPlusOneQuery.stub(:impossible?).with(@post.ar_key).and_return(true)
|
65
|
+
NPlusOneQuery.stub(:association?).with(@post.ar_key, :associations).and_return(false)
|
66
|
+
NPlusOneQuery.send(:conditions_met?, @post.ar_key, :associations).should be_false
|
67
|
+
end
|
68
|
+
|
69
|
+
it "should be false if object is possible, not impossible and object, associations pair is already existed" do
|
70
|
+
NPlusOneQuery.stub(:possible?).with(@post.ar_key).and_return(true)
|
71
|
+
NPlusOneQuery.stub(:impossible?).with(@post.ar_key).and_return(false)
|
72
|
+
NPlusOneQuery.stub(:association?).with(@post.ar_key, :associations).and_return(true)
|
73
|
+
NPlusOneQuery.send(:conditions_met?, @post.ar_key, :associations).should be_false
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
context ".call_association" do
|
78
|
+
it "should create notification if conditions met" do
|
79
|
+
NPlusOneQuery.should_receive(:conditions_met?).with(@post.ar_key, :association).and_return(true)
|
80
|
+
NPlusOneQuery.should_receive(:caller_in_project).and_return(["caller"])
|
81
|
+
NPlusOneQuery.should_receive(:create_notification).with(["caller"], "Post", :association)
|
82
|
+
NPlusOneQuery.call_association(@post, :association)
|
83
|
+
end
|
84
|
+
|
85
|
+
it "should not create notification if conditions not met" do
|
86
|
+
NPlusOneQuery.should_receive(:conditions_met?).with(@post.ar_key, :association).and_return(false)
|
87
|
+
NPlusOneQuery.should_not_receive(:caller_in_project!)
|
88
|
+
NPlusOneQuery.should_not_receive(:create_notification).with("Post", :association)
|
89
|
+
NPlusOneQuery.call_association(@post, :association)
|
90
|
+
end
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
@@ -0,0 +1,62 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Bullet
|
4
|
+
module Detector
|
5
|
+
describe UnusedEagerAssociation do
|
6
|
+
before(:all) { @post = Post.first }
|
7
|
+
before(:each) { UnusedEagerAssociation.clear }
|
8
|
+
|
9
|
+
context ".call_associations" do
|
10
|
+
it "should get empty array if eager_loadgins" do
|
11
|
+
UnusedEagerAssociation.send(:call_associations, @post.ar_key, Set.new([:association])).should be_empty
|
12
|
+
end
|
13
|
+
|
14
|
+
it "should get call associations if object and association are both in eager_loadings and call_object_associations" do
|
15
|
+
UnusedEagerAssociation.add_eager_loadings(@post, :association)
|
16
|
+
UnusedEagerAssociation.add_call_object_associations(@post, :association)
|
17
|
+
UnusedEagerAssociation.send(:call_associations, @post.ar_key, Set.new([:association])).should == [:association]
|
18
|
+
end
|
19
|
+
|
20
|
+
it "should not get call associations if not exist in call_object_associations" do
|
21
|
+
UnusedEagerAssociation.add_eager_loadings(@post, :association)
|
22
|
+
UnusedEagerAssociation.send(:call_associations, @post.ar_key, Set.new([:association])).should be_empty
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
context ".diff_object_association" do
|
27
|
+
it "should return associations not exist in call_association" do
|
28
|
+
UnusedEagerAssociation.send(:diff_object_association, @post.ar_key, Set.new([:association])).should == [:association]
|
29
|
+
end
|
30
|
+
|
31
|
+
it "should return empty if associations exist in call_association" do
|
32
|
+
UnusedEagerAssociation.add_eager_loadings(@post, :association)
|
33
|
+
UnusedEagerAssociation.add_call_object_associations(@post, :association)
|
34
|
+
UnusedEagerAssociation.send(:diff_object_association, @post.ar_key, Set.new([:association])).should be_empty
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
context ".check_unused_preload_associations" do
|
39
|
+
it "should set @@checked to true" do
|
40
|
+
UnusedEagerAssociation.check_unused_preload_associations
|
41
|
+
UnusedEagerAssociation.class_variable_get(:@@checked).should be_true
|
42
|
+
end
|
43
|
+
|
44
|
+
it "should create notification if object_association_diff is not empty" do
|
45
|
+
UnusedEagerAssociation.add_object_associations(@post, :association)
|
46
|
+
UnusedEagerAssociation.should_receive(:create_notification).with("Post", [:association])
|
47
|
+
UnusedEagerAssociation.check_unused_preload_associations
|
48
|
+
end
|
49
|
+
|
50
|
+
it "should not create notification if object_association_diff is empty" do
|
51
|
+
UnusedEagerAssociation.clear
|
52
|
+
UnusedEagerAssociation.add_object_associations(@post, :association)
|
53
|
+
UnusedEagerAssociation.add_eager_loadings(@post, :association)
|
54
|
+
UnusedEagerAssociation.add_call_object_associations(@post, :association)
|
55
|
+
UnusedEagerAssociation.send(:diff_object_association, @post.ar_key, Set.new([:association])).should be_empty
|
56
|
+
UnusedEagerAssociation.should_not_receive(:create_notification).with("Post", [:association])
|
57
|
+
UnusedEagerAssociation.check_unused_preload_associations
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
@@ -0,0 +1,67 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
module Bullet
|
4
|
+
module Notification
|
5
|
+
describe Base do
|
6
|
+
subject { Base.new(Post, [:comments, :votes]) }
|
7
|
+
|
8
|
+
context "#title" do
|
9
|
+
it "should raise NoMethodError" do
|
10
|
+
lambda { subject.title }.should raise_error(NoMethodError)
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
context "#body" do
|
15
|
+
it "should raise NoMethodError" do
|
16
|
+
lambda { subject.body }.should raise_error(NoMethodError)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
context "#whoami" do
|
21
|
+
it "should display user name" do
|
22
|
+
user = `whoami`.chomp
|
23
|
+
subject.whoami.should == "user: #{user}"
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
context "#body_with_caller" do
|
28
|
+
it "should return body" do
|
29
|
+
subject.stub(:body => "body")
|
30
|
+
subject.body_with_caller.should == "body"
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
context "#standard_notice" do
|
35
|
+
it "should return title + body" do
|
36
|
+
subject.stub(:title => "title", :body => "body")
|
37
|
+
subject.standard_notice.should == "title\nbody"
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
context "#full_notice" do
|
42
|
+
it "should return whoami + url + title + body_with_caller" do
|
43
|
+
subject.stub(:whoami => "whoami", :url => "url", :title => "title", :body_with_caller => "body_with_caller")
|
44
|
+
subject.full_notice.should == "whoami\nurl\ntitle\nbody_with_caller"
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
context "#notify_inline" do
|
49
|
+
it "should send full_notice to notifier" do
|
50
|
+
notifier = stub
|
51
|
+
subject.stub(:notifier => notifier, :full_notice => "full_notice")
|
52
|
+
notifier.should_receive(:inline_notify).with("full_notice")
|
53
|
+
subject.notify_inline
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
context "#notify_out_of_channel" do
|
58
|
+
it "should send full_out_of_channel to notifier" do
|
59
|
+
notifier = stub
|
60
|
+
subject.stub(:notifier => notifier, :full_notice => "full_notice")
|
61
|
+
notifier.should_receive(:out_of_channel_notify).with("full_notice")
|
62
|
+
subject.notify_out_of_channel
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|