rspec-active_record 0.1.0 → 0.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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: dc22932934c76f5aa353ae21165ccccb5dbe12ae02b23384b4bf2be5c3fb2400
4
- data.tar.gz: ceba6e1dd8edd17ba0b732a458e2a71be87a916213b8a6500394d011e3322074
3
+ metadata.gz: c8db7110108ac53ef5b6ef595d04cd711410217b5fce21a27cd94ffebae23ece
4
+ data.tar.gz: bc6c9a9d25c119dd8681f6b45166646a6770693910b802228ecd6552c86f2192
5
5
  SHA512:
6
- metadata.gz: 4895273f1779660e100f207e9497979f07eaf973ddc1133b138ac56d7b1339bc632387a7ac0a1aa8b8fea0903a644fb5d043f76e5e2eb23289ee916240e60085
7
- data.tar.gz: 6ee9a41affc0b83ad13cbb54a0c1da7ac0187c13a7ea0bb5104df6f9ed5e8511572f2bab8a9eb5e45a0960faad7d8e76e55e44274aaf3735feff702d990e9e8b
6
+ metadata.gz: 321aa7bb887b01fb0e893b878acba02068612ea15719e647563652dcba7508bd1673d5765c916258ce278df72bbf9381b3897109a9e213391fb6e5213f609845
7
+ data.tar.gz: 2bd092467cc39b93cd27744c6944a9306ab98be4355abf31e0a5834c21c462ff93332140dd700b05d2fd2fb1e5d0b03fb88d060cd19cb17c796391b1d72400a2
data/CHANGELOG.md CHANGED
@@ -1,5 +1,11 @@
1
1
  ## [Unreleased]
2
2
 
3
+ - Add support for times in create_record
4
+
5
+ ## [0.2.0] - 2024-07-12
6
+
7
+ - Add support for negative matchers
8
+
3
9
  ## [0.1.0] - 2023-04-23
4
10
 
5
11
  - Initial release
data/README.md CHANGED
@@ -4,9 +4,19 @@ Implements helper methods & matchers when working with RSpec & ActiveRecord.
4
4
 
5
5
  ## Installation
6
6
 
7
- Install the gem and add to the application's Gemfile by executing:
7
+ Add it to Gemfile:
8
8
 
9
- $ bundle add rspec-active_record --group test
9
+ ```ruby
10
+ group :test do
11
+ gem "rspec-active_record", require: false
12
+ end
13
+ ```
14
+
15
+ And require it in your `rails_helper` or `spec_helper` after `rspec/rails`:
16
+
17
+ ```ruby
18
+ require "rspec/active_record"
19
+ ```
10
20
 
11
21
  If bundler is not being used to manage dependencies, install the gem by executing:
12
22
 
@@ -20,6 +30,13 @@ Check that block creates a record:
20
30
 
21
31
  ```ruby
22
32
  expect { User.create!(name: "RSpec User") }.to create_record(User)
33
+ expect { User.create!(name: "RSpec User") }.to not_create_record(Company)
34
+ ```
35
+
36
+ Sometimes you also need to match specific count of records:
37
+ ```ruby
38
+ expect { User.create!(name: "RSpec User") }.to create_record(User).once
39
+ expect { User.create!(name: "RSpec User") }.to create_record(User).times(1)
23
40
  ```
24
41
 
25
42
  You can also make sure that attributes match, if it fails you'll get RSpec diff between created record and what you expected:
@@ -44,6 +61,7 @@ expect { user.update!(name: "RSpec User") }.to change_record(user).to(name: "RSp
44
61
  Sometimes it's useful to specify what the attributes should've been initially:
45
62
  ```ruby
46
63
  expect { user.update!(name: "RSpec User") }.to change_record(user).from(name: "Initial Name")
64
+ expect { user.name = "RSpec User" }.to not_change_record(user).from(name: "Initial Name")
47
65
  ```
48
66
 
49
67
  ### destroy_record
@@ -51,6 +69,7 @@ expect { user.update!(name: "RSpec User") }.to change_record(user).from(name: "I
51
69
  Check that code destroys a record:
52
70
  ```ruby
53
71
  expect { user.destroy! }.to destroy_record(user)
72
+ expect { user.save! }.to not_destroy_record(user)
54
73
  ```
55
74
 
56
75
  ### stub_class
@@ -7,13 +7,13 @@ module RSpec
7
7
  include RSpec::Matchers::BuiltIn::BaseMatcher::HashFormatting
8
8
 
9
9
  # Attributes of record should match these after block is executed
10
- def to(**attributes)
10
+ def to(attributes)
11
11
  @to = attributes
12
12
  self
13
13
  end
14
14
 
15
15
  # Attributes of record should match these before block is executed
16
- def from(**attributes)
16
+ def from(attributes)
17
17
  @from = attributes
18
18
  self
19
19
  end
@@ -78,10 +78,6 @@ module RSpec
78
78
 
79
79
  private
80
80
 
81
- def format_hash(hash)
82
- improve_hash_formatting RSpec::Support::ObjectFormatter.format(surface_descriptions_in(hash))
83
- end
84
-
85
81
  def diff_unless_match(record, attributes)
86
82
  diff(record, attributes) unless match_attributes?(record, attributes)
87
83
  end
@@ -3,20 +3,37 @@
3
3
  module RSpec
4
4
  module ActiveRecord
5
5
  # A matcher
6
- class CreateRecord
7
- include RSpec::Matchers::Composable
6
+ class CreateRecord < Matcher
8
7
  include RSpec::Matchers::BuiltIn::BaseMatcher::HashFormatting
9
8
 
10
9
  def initialize(scope)
11
10
  @scope = scope
11
+ super()
12
12
  end
13
13
 
14
14
  # Make sure that created record matches attributes
15
- def matching(**attributes)
15
+ def matching(attributes)
16
16
  @attributes = attributes
17
17
  self
18
18
  end
19
19
 
20
+ def times(times)
21
+ @times = times
22
+ self
23
+ end
24
+
25
+ def once
26
+ times(1)
27
+ end
28
+
29
+ def twice
30
+ times(2)
31
+ end
32
+
33
+ def thrice
34
+ times(3)
35
+ end
36
+
20
37
  def supports_block_expectations?
21
38
  true
22
39
  end
@@ -25,25 +42,25 @@ module RSpec
25
42
  existing_ids = @scope.ids
26
43
  block.call
27
44
 
28
- @new_records = @scope.where.not(@scope.primary_key => existing_ids)
45
+ @new_records = @scope.where.not(@scope.primary_key => existing_ids).to_a
29
46
 
30
- if @attributes
31
- match_attributes?
32
- else
33
- @new_records.present?
34
- end
47
+ match_times? && match_attributes? && @new_records.present?
35
48
  end
36
49
 
37
50
  def description
38
51
  message = "create #{scope_name}"
39
- if @attributes
40
- message += " matching #{RSpec::Support::ObjectFormatter.format(surface_descriptions_in(@attributes))}"
41
- end
52
+ message += " #{@times} time#{"s" if @times != 1}" if @times
53
+ message += " matching #{format_hash(@attributes)}" if @attributes
42
54
  improve_hash_formatting message
43
55
  end
44
56
 
45
57
  def failure_message
46
- add_diff "expected to #{description} but did not"
58
+ failure = "expected to #{description} but"
59
+ if !match_times?
60
+ "#{failure} created #{@new_records.size}"
61
+ else
62
+ add_diff "#{failure} did not"
63
+ end
47
64
  end
48
65
 
49
66
  def failure_message_when_negated
@@ -57,8 +74,13 @@ module RSpec
57
74
  message
58
75
  end
59
76
 
77
+ def match_times?
78
+ @times.nil? || @times == @new_records.size
79
+ end
80
+
60
81
  def match_attributes?
61
- @new_records.any? { |record| RSpec::Matchers::BuiltIn::HaveAttributes.new(@attributes).matches?(record) }
82
+ @attributes.nil? ||
83
+ @new_records.any? { |record| RSpec::Matchers::BuiltIn::HaveAttributes.new(@attributes).matches?(record) }
62
84
  end
63
85
 
64
86
  def scope_name
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RSpec
4
+ module ActiveRecord
5
+ # A matcher
6
+ class Matcher
7
+ include RSpec::Matchers::Composable
8
+
9
+ def supports_block_expectations?
10
+ true
11
+ end
12
+
13
+ private
14
+
15
+ def format_hash(hash)
16
+ improve_hash_formatting RSpec::Support::ObjectFormatter.format(surface_descriptions_in(hash))
17
+ end
18
+ end
19
+ end
20
+ end
@@ -3,17 +3,12 @@
3
3
  module RSpec
4
4
  module ActiveRecord
5
5
  # A matcher
6
- class RecordMatcher
7
- include RSpec::Matchers::Composable
8
-
6
+ class RecordMatcher < Matcher
9
7
  def initialize(record)
8
+ super()
10
9
  @record = record
11
10
  end
12
11
 
13
- def supports_block_expectations?
14
- true
15
- end
16
-
17
12
  private
18
13
 
19
14
  def format_record
@@ -2,6 +2,6 @@
2
2
 
3
3
  module RSpec
4
4
  module ActiveRecord
5
- VERSION = "0.1.0"
5
+ VERSION = "0.3.0"
6
6
  end
7
7
  end
@@ -12,6 +12,7 @@ module RSpec
12
12
  autoload :CreateRecord, "rspec/active_record/create_record"
13
13
  autoload :DestroyRecord, "rspec/active_record/destroy_record"
14
14
  autoload :DiffForMultipleRecords, "rspec/active_record/diff_for_multiple_records"
15
+ autoload :Matcher, "rspec/active_record/matcher"
15
16
  autoload :RecordMatcher, "rspec/active_record/record_matcher"
16
17
  autoload :StubModels, "rspec/active_record/stub_models"
17
18
  autoload :VERSION, "rspec/active_record/version"
@@ -53,3 +54,7 @@ module RSpec
53
54
  end
54
55
  end
55
56
  RSpec.configuration.include RSpec::ActiveRecord
57
+
58
+ RSpec::Matchers.define_negated_matcher :not_create_record, :create_record
59
+ RSpec::Matchers.define_negated_matcher :not_change_record, :change_record
60
+ RSpec::Matchers.define_negated_matcher :not_destroy_record, :destroy_record
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rspec-active_record
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0
4
+ version: 0.3.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Andrius Chamentauskas
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2024-03-13 00:00:00.000000000 Z
11
+ date: 2024-08-02 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activerecord
@@ -141,6 +141,7 @@ files:
141
141
  - lib/rspec/active_record/create_record.rb
142
142
  - lib/rspec/active_record/destroy_record.rb
143
143
  - lib/rspec/active_record/diff_for_multiple_records.rb
144
+ - lib/rspec/active_record/matcher.rb
144
145
  - lib/rspec/active_record/record_matcher.rb
145
146
  - lib/rspec/active_record/stub_models.rb
146
147
  - lib/rspec/active_record/version.rb