acts_as_approvable 0.1.1 → 0.7.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (97) hide show
  1. data/.gitignore +4 -0
  2. data/.rspec +2 -0
  3. data/.travis.yml +5 -0
  4. data/Appraisals +22 -0
  5. data/CHANGELOG +76 -0
  6. data/Gemfile +3 -0
  7. data/Gemfile.lock +84 -0
  8. data/MIT-LICENSE +2 -2
  9. data/README.md +146 -0
  10. data/Rakefile +90 -7
  11. data/TODO.md +30 -0
  12. data/VERSION +1 -0
  13. data/acts_as_approvable.gemspec +40 -0
  14. data/features/create_approval.feature +36 -0
  15. data/features/destroy_approval.feature +19 -0
  16. data/features/reset_approval.feature +13 -0
  17. data/features/step_definitions/cucumber_steps.rb +132 -0
  18. data/features/support/env.rb +14 -0
  19. data/features/support/large.txt +29943 -0
  20. data/features/support/second_large.txt +31798 -0
  21. data/features/update_approval.feature +48 -0
  22. data/gemfiles/Gemfile.ci +14 -0
  23. data/gemfiles/Gemfile.ci.lock +98 -0
  24. data/gemfiles/mysql2.gemfile +7 -0
  25. data/gemfiles/mysql2.gemfile.lock +86 -0
  26. data/gemfiles/rails2.gemfile +8 -0
  27. data/gemfiles/rails2.gemfile.lock +86 -0
  28. data/gemfiles/rails30.gemfile +9 -0
  29. data/gemfiles/rails30.gemfile.lock +124 -0
  30. data/gemfiles/rails31.gemfile +9 -0
  31. data/gemfiles/rails31.gemfile.lock +135 -0
  32. data/gemfiles/sqlite.gemfile +7 -0
  33. data/generators/acts_as_approvable/USAGE +3 -0
  34. data/generators/acts_as_approvable/acts_as_approvable_generator.rb +81 -0
  35. data/generators/acts_as_approvable/templates/approvals.js +71 -0
  36. data/generators/acts_as_approvable/templates/approvals_controller.rb +91 -0
  37. data/generators/acts_as_approvable/templates/create_approvals.rb +27 -0
  38. data/generators/acts_as_approvable/templates/initializer.rb +3 -0
  39. data/generators/acts_as_approvable/templates/jquery.form.js +101 -0
  40. data/generators/acts_as_approvable/templates/views/erb/_owner_select.html.erb +4 -0
  41. data/generators/acts_as_approvable/templates/views/erb/_table.html.erb +26 -0
  42. data/generators/acts_as_approvable/templates/views/erb/index.html.erb +17 -0
  43. data/generators/acts_as_approvable/templates/views/haml/_owner_select.html.haml +3 -0
  44. data/generators/acts_as_approvable/templates/views/haml/_table.html.haml +19 -0
  45. data/generators/acts_as_approvable/templates/views/haml/index.html.haml +15 -0
  46. data/init.rb +1 -0
  47. data/lib/acts_as_approvable.rb +96 -2
  48. data/lib/acts_as_approvable/approval.rb +205 -11
  49. data/lib/acts_as_approvable/error.rb +34 -0
  50. data/lib/acts_as_approvable/model.rb +60 -0
  51. data/lib/acts_as_approvable/model/class_methods.rb +63 -0
  52. data/lib/acts_as_approvable/model/create_instance_methods.rb +88 -0
  53. data/lib/acts_as_approvable/model/destroy_instance_methods.rb +38 -0
  54. data/lib/acts_as_approvable/model/instance_methods.rb +107 -0
  55. data/lib/acts_as_approvable/model/update_instance_methods.rb +61 -0
  56. data/lib/acts_as_approvable/ownership.rb +141 -0
  57. data/lib/acts_as_approvable/railtie.rb +7 -0
  58. data/lib/acts_as_approvable/version.rb +1 -8
  59. data/lib/generators/acts_as_approvable/USAGE +1 -0
  60. data/lib/generators/acts_as_approvable/acts_as_approvable_generator.rb +68 -0
  61. data/lib/generators/acts_as_approvable/base.rb +30 -0
  62. data/lib/generators/acts_as_approvable/templates/approvals.js +71 -0
  63. data/lib/generators/acts_as_approvable/templates/approvals_controller.rb +91 -0
  64. data/lib/generators/acts_as_approvable/templates/create_approvals.rb +27 -0
  65. data/lib/generators/acts_as_approvable/templates/jquery.form.js +101 -0
  66. data/lib/generators/erb/acts_as_approvable_generator.rb +33 -0
  67. data/lib/generators/erb/templates/_owner_select.html.erb +4 -0
  68. data/lib/generators/erb/templates/_table.html.erb +26 -0
  69. data/lib/generators/erb/templates/index.html.erb +17 -0
  70. data/lib/generators/haml/acts_as_approvable_generator.rb +33 -0
  71. data/lib/generators/haml/templates/_owner_select.html.haml +3 -0
  72. data/lib/generators/haml/templates/_table.html.haml +19 -0
  73. data/lib/generators/haml/templates/index.html.haml +15 -0
  74. data/lib/tasks/acts_as_approvable.rake +4 -0
  75. data/rails/init.rb +1 -0
  76. data/spec/acts_as_approvable/approval_spec.rb +614 -0
  77. data/spec/acts_as_approvable/model/class_methods_spec.rb +219 -0
  78. data/spec/acts_as_approvable/model/create_instance_methods_spec.rb +169 -0
  79. data/spec/acts_as_approvable/model/destroy_instance_methods_spec.rb +71 -0
  80. data/spec/acts_as_approvable/model/instance_methods_spec.rb +328 -0
  81. data/spec/acts_as_approvable/model/update_instance_methods_spec.rb +111 -0
  82. data/spec/acts_as_approvable/model_spec.rb +113 -0
  83. data/spec/acts_as_approvable/ownership/class_methods_spec.rb +134 -0
  84. data/spec/acts_as_approvable/ownership/instance_methods_spec.rb +32 -0
  85. data/spec/acts_as_approvable/ownership_spec.rb +52 -0
  86. data/spec/acts_as_approvable_spec.rb +31 -0
  87. data/spec/spec_helper.rb +51 -0
  88. data/spec/support/database.rb +49 -0
  89. data/spec/support/database.yml +12 -0
  90. data/spec/support/matchers.rb +87 -0
  91. data/spec/support/models.rb +67 -0
  92. data/spec/support/schema.rb +54 -0
  93. metadata +375 -58
  94. data/README.rdoc +0 -38
  95. data/lib/acts_as_approvable/approver.rb +0 -76
  96. data/lib/generators/acts_as_approvable/install_generator.rb +0 -28
  97. data/lib/generators/acts_as_approvable/templates/install.rb +0 -16
@@ -0,0 +1,134 @@
1
+ require 'spec_helper'
2
+
3
+ describe ActsAsApprovable::Ownership::ClassMethods do
4
+ before(:all) do
5
+ ActsAsApprovable::Ownership.configure
6
+ end
7
+
8
+ before(:each) do
9
+ @user1 = User.create
10
+ @user2 = User.create
11
+ end
12
+
13
+ subject { Approval }
14
+
15
+ describe '.owner_class' do
16
+ it 'proxies to ActsAsApprovable' do
17
+ ActsAsApprovable.should_receive(:owner_class)
18
+ subject.owner_class
19
+ end
20
+ end
21
+
22
+ describe '.owner_source' do
23
+ it 'proxies to ActsAsApprovable' do
24
+ ActsAsApprovable.should_receive(:owner_source)
25
+ subject.owner_source
26
+ end
27
+ end
28
+
29
+ describe '.available_owners' do
30
+ it 'selects all records from #owner_class' do
31
+ subject.available_owners.should include(@user1)
32
+ subject.available_owners.should include(@user2)
33
+ end
34
+
35
+ context 'when an owner source is configured' do
36
+ before(:each) do
37
+ class FakeSource; end
38
+ ActsAsApprovable.owner_source = FakeSource
39
+ end
40
+
41
+ it 'proxies to the configured source' do
42
+ FakeSource.should_receive(:available_owners)
43
+ subject.available_owners
44
+ end
45
+ end
46
+ end
47
+
48
+ describe '.options_for_available_owners' do
49
+ it 'returns an array usable by #options_for_select' do
50
+ subject.options_for_available_owners.should be_an_options_array
51
+ end
52
+
53
+ it 'uses .available_owners as its source' do
54
+ subject.should_receive(:available_owners).and_return([])
55
+ subject.options_for_available_owners
56
+ end
57
+
58
+ it 'uses .options_for_owner to format each record' do
59
+ subject.should_receive(:option_for_owner).with(@user1).once
60
+ subject.should_receive(:option_for_owner).with(@user2).once
61
+ subject.options_for_available_owners
62
+ end
63
+
64
+ it 'includes a prompt if requested' do
65
+ subject.options_for_available_owners(true).should include(['(none)', nil])
66
+ end
67
+
68
+ it 'does not includes a prompt by default' do
69
+ subject.options_for_available_owners.should_not include(['(none)', nil])
70
+ end
71
+ end
72
+
73
+ context 'when an owner source is configured' do
74
+ before(:each) do
75
+ class FakeSource; end
76
+ ActsAsApprovable.owner_source = FakeSource
77
+ end
78
+
79
+ describe '.assigned_owners' do
80
+ it 'proxies to the configured source' do
81
+ FakeSource.should_receive(:assigned_owners)
82
+ subject.assigned_owners
83
+ end
84
+ end
85
+ end
86
+
87
+ context 'when no users are assigned' do
88
+ describe '.assigned_owners' do
89
+ it 'should be empty' do
90
+ subject.assigned_owners.should be_empty
91
+ end
92
+ end
93
+ end
94
+
95
+ context 'when some users are assigned' do
96
+ before(:each) do
97
+ CreatesApprovable.create.approval.update_attribute(:owner_id, @user1.id)
98
+ end
99
+
100
+ describe '.assigned_owners' do
101
+ it 'selects all owners with an assigned approval' do
102
+ subject.assigned_owners.should include(@user1)
103
+ end
104
+
105
+ it 'does not include owners without an assignment' do
106
+ subject.assigned_owners.should_not include(@user2)
107
+ end
108
+ end
109
+
110
+ describe '.options_for_assigned_owners' do
111
+ it 'returns an array usable by #options_for_select' do
112
+ subject.options_for_assigned_owners.should be_an_options_array
113
+ end
114
+
115
+ it 'uses .assigned_owners as its source' do
116
+ subject.should_receive(:assigned_owners).and_return([@user1])
117
+ subject.options_for_assigned_owners
118
+ end
119
+
120
+ it 'uses .options_for_owner to format each record' do
121
+ subject.should_receive(:option_for_owner).with(@user1)
122
+ subject.options_for_assigned_owners
123
+ end
124
+
125
+ it 'includes a prompt if requested' do
126
+ subject.options_for_assigned_owners(true).should include(['All Users', nil])
127
+ end
128
+
129
+ it 'does not includes a prompt by default' do
130
+ subject.options_for_assigned_owners.should_not include(['All Users', nil])
131
+ end
132
+ end
133
+ end
134
+ end
@@ -0,0 +1,32 @@
1
+ require 'spec_helper'
2
+
3
+ describe ActsAsApprovable::Ownership::InstanceMethods do
4
+ before(:all) do
5
+ ActsAsApprovable::Ownership.configure
6
+ end
7
+
8
+ before(:each) do
9
+ @user1 = User.create
10
+ @user2 = User.new
11
+ subject.stub(:save => true)
12
+ end
13
+
14
+ subject { Approval.new }
15
+
16
+ describe '#assign' do
17
+ it 'sets the owner' do
18
+ expect { subject.assign(@user1) }.to change{subject.owner}.from(nil).to(@user1)
19
+ end
20
+
21
+ it 'raises an InvalidOwner error if the owner is not valid' do
22
+ expect { subject.assign(@user2) }.to raise_error(ActsAsApprovable::Error::InvalidOwner)
23
+ end
24
+ end
25
+
26
+ describe '#unassign' do
27
+ it 'removes the assigned owner' do
28
+ subject.owner = @user1
29
+ expect { subject.unassign }.to change{subject.owner}.from(@user1).to(nil)
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,52 @@
1
+ require 'spec_helper'
2
+
3
+ describe ActsAsApprovable::Ownership do
4
+ describe '.configure' do
5
+ it 'defaults to Approval as the approval model' do
6
+ Approval.should_receive(:include).with(ActsAsApprovable::Ownership)
7
+ subject.configure
8
+ end
9
+
10
+ it 'defaults to User as the owner model' do
11
+ ActsAsApprovable.should_receive(:owner_class=).with(User)
12
+ subject.configure
13
+ end
14
+
15
+ it 'adds a belongs_to(:owner) association' do
16
+ subject.configure
17
+ Approval.new.should belong_to(:owner)
18
+ end
19
+
20
+ it 'uses the given approval :model' do
21
+ class FakeApproval < ActiveRecord::Base; end
22
+ FakeApproval.should_receive(:include).with(ActsAsApprovable::Ownership)
23
+ subject.configure(:model => FakeApproval)
24
+ end
25
+
26
+ it 'uses the given :owner' do
27
+ class FakeUser; end
28
+ ActsAsApprovable.should_receive(:owner_class=).with(FakeUser)
29
+ subject.configure(:owner => FakeUser)
30
+ end
31
+
32
+ it 'uses the given :source' do
33
+ class FakeSource; end
34
+ ActsAsApprovable.should_receive(:owner_source=).with(FakeSource)
35
+ subject.configure(:source => FakeSource)
36
+ end
37
+ end
38
+
39
+ describe '.included' do
40
+ before(:all) do
41
+ class IncludedOwnership; include ActsAsApprovable::Ownership; end
42
+ end
43
+
44
+ it 'should extend ClassMethods' do
45
+ IncludedOwnership.should extend(ActsAsApprovable::Ownership::ClassMethods)
46
+ end
47
+
48
+ it 'should extend InstanceMethods' do
49
+ IncludedOwnership.should extend(ActsAsApprovable::Ownership::InstanceMethods)
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,31 @@
1
+ require 'spec_helper'
2
+
3
+ describe ActsAsApprovable do
4
+ it { should respond_to(:owner_class) }
5
+ it { should respond_to(:owner_class=) }
6
+ it { should respond_to(:view_language) }
7
+ it { should respond_to(:view_language=) }
8
+ it { should respond_to(:stale_check=) }
9
+ it { should respond_to(:stale_check?) }
10
+
11
+ describe '.enabled?' do
12
+ it 'returns true by default' do
13
+ subject.enabled?.should be_true
14
+ end
15
+ end
16
+
17
+ describe '.disable' do
18
+ it 'disables the approval queue' do
19
+ subject.disable
20
+ subject.enabled?.should be_false
21
+ end
22
+ end
23
+
24
+ describe '.enable' do
25
+ it 'enables the approval queue' do
26
+ subject.enable
27
+ subject.enabled?.should be_true
28
+ end
29
+ end
30
+
31
+ end
@@ -0,0 +1,51 @@
1
+ if RUBY_VERSION =~ /^1\.9/
2
+ begin
3
+ require 'simplecov'
4
+ SimpleCov.start do
5
+ add_filter '/spec/'
6
+ end if ENV['COVERAGE']
7
+ rescue LoadError
8
+ end
9
+ end
10
+
11
+ ENV['RAILS_ENV'] = 'test'
12
+ ENV['RAILS_ROOT'] ||= File.expand_path(File.join(File.dirname(__FILE__), '..', '..', '..', '..'))
13
+
14
+ require 'rspec'
15
+ require 'shoulda'
16
+ require 'timecop'
17
+ require 'active_record'
18
+
19
+ begin
20
+ require 'plymouth'
21
+ rescue LoadError
22
+ end if ENV['PRYABLE']
23
+
24
+ require File.expand_path('../lib/acts_as_approvable', File.dirname(__FILE__))
25
+
26
+ require File.expand_path('support/database', File.dirname(__FILE__))
27
+ require File.expand_path('support/models', File.dirname(__FILE__))
28
+ require File.expand_path('support/matchers', File.dirname(__FILE__))
29
+
30
+ RSpec.configure do |config|
31
+ config.before(:suite) do
32
+ LOGGER = Support::Database.setup_log unless defined?(LOGGER)
33
+ Support::Database.load_schema
34
+ end
35
+
36
+ config.before(:each) do
37
+ Object.send(:remove_const, :CleanApprovable) if defined?(CleanApprovable)
38
+ class CleanApprovable < ActiveRecord::Base
39
+ def self.table_name; 'nots'; end
40
+ def self.primary_key; 'id'; end
41
+ end
42
+
43
+ freeze_at = Time.parse('2012-01-01')
44
+ Timecop.freeze(freeze_at)
45
+ end
46
+
47
+ config.after(:each) do
48
+ Timecop.return
49
+ Support::Database.truncate
50
+ end
51
+ end
@@ -0,0 +1,49 @@
1
+ module Support
2
+ module Database
3
+ def self.setup_log(logfile = nil)
4
+ logfile ||= File.expand_path('../../debug.log', File.dirname(__FILE__))
5
+ ActiveRecord::Base.logger = if defined?(ActiveSupport::BufferedLogger)
6
+ ActiveSupport::BufferedLogger.new(logfile)
7
+ else
8
+ Logger.new(logfile)
9
+ end
10
+ end
11
+
12
+ def self.load_schema
13
+ config = YAML::load(IO.read(File.expand_path('database.yml', File.dirname(__FILE__))))
14
+
15
+ unless db_adapter = ENV['DB']
16
+ %w(sqlite3 mysql2 sqlite).each do |gem|
17
+ begin
18
+ require gem
19
+ db_adapter = gem
20
+ break
21
+ rescue MissingSourceFile
22
+ end
23
+ end
24
+ end
25
+
26
+ if db_adapter.nil?
27
+ puts
28
+ puts
29
+ puts "You must run tests using one of the available appraisals. `bundle exec rake -T` to list, `bundle exec rake appraisal:rails31` to run against Rails 3.1 and SQLite."
30
+ exit
31
+ end
32
+
33
+ ActiveRecord::Base.establish_connection(config[db_adapter])
34
+ ActiveRecord::Migration.suppress_messages do
35
+ load(File.expand_path('schema.rb', File.dirname(__FILE__)))
36
+ end
37
+
38
+ [User, Approval, NotApprovable, DefaultApprovable, CreatesApprovable, CreatesWithStateApprovable, UpdatesApprovable, UpdatesIgnoreFieldsApprovable, UpdatesOnlyFieldsApprovable, DestroysApprovable].each do |klass|
39
+ klass.reset_column_information
40
+ end
41
+ end
42
+
43
+ def self.truncate
44
+ [User, Approval, NotApprovable, DefaultApprovable, CreatesApprovable, UpdatesApprovable, DestroysApprovable].each do |klass|
45
+ klass.delete_all
46
+ end
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,12 @@
1
+ sqlite:
2
+ :adapter: sqlite
3
+ :database: spec/acts_as_approvable_plugin.sqlite.db
4
+
5
+ sqlite3:
6
+ :adapter: sqlite3
7
+ :database: spec/acts_as_approvable_plugin.sqlite3.db
8
+
9
+ mysql2:
10
+ :adapter: mysql2
11
+ :database: acts_as_approvable_test
12
+ :user: root
@@ -0,0 +1,87 @@
1
+ require 'rspec/expectations'
2
+ require 'shoulda/active_record/matchers'
3
+
4
+ class Module
5
+ # Return any modules we +extend+
6
+ def extended_modules
7
+ (class << self; self end).included_modules
8
+ end
9
+ end
10
+
11
+ # Truthfully this checks both extend *and* include. :include is already used as a matcher for Arrays :-/
12
+ RSpec::Matchers.define :extend do |expected|
13
+ match do |actual|
14
+ extended = (actual.extended_modules - Module.extended_modules) + actual.included_modules
15
+ extended.include?(expected)
16
+ end
17
+ description do
18
+ "extend #{expected}"
19
+ end
20
+ failure_message_for_should do |actual|
21
+ "expected #{actual} to extend #{expected}"
22
+ end
23
+ failure_message_for_should_not do |actual|
24
+ "expected #{actual} not to extend #{expected}"
25
+ end
26
+ end
27
+
28
+ RSpec::Matchers.define :be_an_options_array do
29
+ match do |actual|
30
+ actual.should be_an(Array)
31
+ actual.each do |option|
32
+ unless option.is_a?(String)
33
+ option.should be_an(Array)
34
+ option.length.should be(2)
35
+ end
36
+ end
37
+ end
38
+ description do
39
+ 'returns an array usable by #options_for_select'
40
+ end
41
+ failure_message_for_should do |actual|
42
+ "expected #{actual} to map to a valid #options_for_select array"
43
+ end
44
+ failure_message_for_should_not do |actual|
45
+ "expected #{actual} not to map to a valid #options_for_select array"
46
+ end
47
+ end
48
+
49
+ module Shoulda
50
+ module ActiveRecord
51
+ module Matchers
52
+ def validate_inclusion_of(attr)
53
+ ValidatesInclusionOfMatcher.new(attr)
54
+ end
55
+
56
+ class ValidatesInclusionOfMatcher < ValidationMatcher
57
+ def in(values)
58
+ @values = values
59
+ self
60
+ end
61
+
62
+ def description
63
+ "require #{@attribute} to be one of #{@values.inspect}"
64
+ end
65
+
66
+ def matches?(subject)
67
+ super(subject)
68
+
69
+ allows_given_values && disallows_other_values
70
+ end
71
+
72
+ private
73
+ def allows_given_values
74
+ @values.each do |value|
75
+ allows_value_of(value, @message)
76
+ end unless @values.empty?
77
+ end
78
+
79
+ def disallows_other_values
80
+ @values.each do |value|
81
+ disallows_value_of("#{value}s", @message)
82
+ end unless @values.empty?
83
+ end
84
+ end
85
+ end
86
+ end
87
+ end