minitest-snapshot 0.1.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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: d5a9382c2bcf4cc2c4c57e750249e69956f73ee4b8ee09582f7b9755b1c6055e
4
+ data.tar.gz: 821965c4ad60006195e2535590fdd58f50ae98194201c87fa17bcf17bd23dd2a
5
+ SHA512:
6
+ metadata.gz: e5c939fa50eb1ff4472c95e002137ff959e483bbb02c149dd889c7b748cf1f41d1c3b8fc0b522a433a4a29da18b9d5f9a310f7e6a01a5c669d36d0ec25ffa10b
7
+ data.tar.gz: 1022c4fafa57453cf0ae395eab12c25398e122acbeda053f98e6f7161d1abd32b28e549d9fbb783e2f9b8bded8924a988335a63ee6f754ab73ff87fe0af2a671
data/.rubocop.yml ADDED
@@ -0,0 +1,49 @@
1
+ inherit_from: .rubocop_todo.yml
2
+
3
+ plugins:
4
+ - rubocop-minitest
5
+ - rubocop-performance
6
+ - rubocop-rake
7
+
8
+ # The behavior of RuboCop can be controlled via the .rubocop.yml
9
+ # configuration file. It makes it possible to enable/disable
10
+ # certain cops (checks) and to alter their behavior if they accept
11
+ # any parameters. The file can be placed either in your home
12
+ # directory or in some project directory.
13
+ #
14
+ # RuboCop will start looking for the configuration file in the directory
15
+ # where the inspected file is and continue its way up to the root directory.
16
+ #
17
+ # See https://docs.rubocop.org/rubocop/configuration
18
+
19
+ AllCops:
20
+ TargetRubyVersion: "3.0"
21
+ NewCops: enable
22
+
23
+ Layout/LineLength:
24
+ Enabled: true
25
+ Max: 100
26
+ Exclude:
27
+ - "**/*.gemspec"
28
+
29
+ Metrics/BlockLength:
30
+ Enabled: true
31
+ Exclude:
32
+ - "test/**/*.rb"
33
+ - "spec/**/*.rb"
34
+ - "**/*.gemspec"
35
+
36
+ Metrics/MethodLength:
37
+ Enabled: true
38
+ Max: 15
39
+
40
+ Style/StringLiterals:
41
+ EnforcedStyle: single_quotes
42
+
43
+ Style/StringLiteralsInInterpolation:
44
+ EnforcedStyle: double_quotes
45
+
46
+ Style/Documentation:
47
+ Exclude:
48
+ - "spec/**/*"
49
+ - "test/**/*"
data/.rubocop_todo.yml ADDED
@@ -0,0 +1,7 @@
1
+ # This configuration was generated by
2
+ # `rubocop --auto-gen-config`
3
+ # on 2025-03-14 12:17:15 UTC using RuboCop version 1.73.2.
4
+ # The point is for the user to remove these configuration records
5
+ # one by one as the offenses are removed from the code base.
6
+ # Note that changes in the inspected code, or installation of new
7
+ # versions of RuboCop, may require this file to be generated again.
data/CHANGELOG.md ADDED
@@ -0,0 +1,5 @@
1
+ ## [Unreleased]
2
+
3
+ ## [0.1.0] - 2025-03-14
4
+
5
+ - Initial release
@@ -0,0 +1,132 @@
1
+ # Contributor Covenant Code of Conduct
2
+
3
+ ## Our Pledge
4
+
5
+ We as members, contributors, and leaders pledge to make participation in our
6
+ community a harassment-free experience for everyone, regardless of age, body
7
+ size, visible or invisible disability, ethnicity, sex characteristics, gender
8
+ identity and expression, level of experience, education, socio-economic status,
9
+ nationality, personal appearance, race, caste, color, religion, or sexual
10
+ identity and orientation.
11
+
12
+ We pledge to act and interact in ways that contribute to an open, welcoming,
13
+ diverse, inclusive, and healthy community.
14
+
15
+ ## Our Standards
16
+
17
+ Examples of behavior that contributes to a positive environment for our
18
+ community include:
19
+
20
+ - Demonstrating empathy and kindness toward other people
21
+ - Being respectful of differing opinions, viewpoints, and experiences
22
+ - Giving and gracefully accepting constructive feedback
23
+ - Accepting responsibility and apologizing to those affected by our mistakes,
24
+ and learning from the experience
25
+ - Focusing on what is best not just for us as individuals, but for the overall
26
+ community
27
+
28
+ Examples of unacceptable behavior include:
29
+
30
+ - The use of sexualized language or imagery, and sexual attention or advances of
31
+ any kind
32
+ - Trolling, insulting or derogatory comments, and personal or political attacks
33
+ - Public or private harassment
34
+ - Publishing others' private information, such as a physical or email address,
35
+ without their explicit permission
36
+ - Other conduct which could reasonably be considered inappropriate in a
37
+ professional setting
38
+
39
+ ## Enforcement Responsibilities
40
+
41
+ Community leaders are responsible for clarifying and enforcing our standards of
42
+ acceptable behavior and will take appropriate and fair corrective action in
43
+ response to any behavior that they deem inappropriate, threatening, offensive,
44
+ or harmful.
45
+
46
+ Community leaders have the right and responsibility to remove, edit, or reject
47
+ comments, commits, code, wiki edits, issues, and other contributions that are
48
+ not aligned to this Code of Conduct, and will communicate reasons for moderation
49
+ decisions when appropriate.
50
+
51
+ ## Scope
52
+
53
+ This Code of Conduct applies within all community spaces, and also applies when
54
+ an individual is officially representing the community in public spaces.
55
+ Examples of representing our community include using an official email address,
56
+ posting via an official social media account, or acting as an appointed
57
+ representative at an online or offline event.
58
+
59
+ ## Enforcement
60
+
61
+ Instances of abusive, harassing, or otherwise unacceptable behavior may be
62
+ reported to the community leaders responsible for enforcement at
63
+ [INSERT CONTACT METHOD].
64
+ All complaints will be reviewed and investigated promptly and fairly.
65
+
66
+ All community leaders are obligated to respect the privacy and security of the
67
+ reporter of any incident.
68
+
69
+ ## Enforcement Guidelines
70
+
71
+ Community leaders will follow these Community Impact Guidelines in determining
72
+ the consequences for any action they deem in violation of this Code of Conduct:
73
+
74
+ ### 1. Correction
75
+
76
+ **Community Impact**: Use of inappropriate language or other behavior deemed
77
+ unprofessional or unwelcome in the community.
78
+
79
+ **Consequence**: A private, written warning from community leaders, providing
80
+ clarity around the nature of the violation and an explanation of why the
81
+ behavior was inappropriate. A public apology may be requested.
82
+
83
+ ### 2. Warning
84
+
85
+ **Community Impact**: A violation through a single incident or series of
86
+ actions.
87
+
88
+ **Consequence**: A warning with consequences for continued behavior. No
89
+ interaction with the people involved, including unsolicited interaction with
90
+ those enforcing the Code of Conduct, for a specified period of time. This
91
+ includes avoiding interactions in community spaces as well as external channels
92
+ like social media. Violating these terms may lead to a temporary or permanent
93
+ ban.
94
+
95
+ ### 3. Temporary Ban
96
+
97
+ **Community Impact**: A serious violation of community standards, including
98
+ sustained inappropriate behavior.
99
+
100
+ **Consequence**: A temporary ban from any sort of interaction or public
101
+ communication with the community for a specified period of time. No public or
102
+ private interaction with the people involved, including unsolicited interaction
103
+ with those enforcing the Code of Conduct, is allowed during this period.
104
+ Violating these terms may lead to a permanent ban.
105
+
106
+ ### 4. Permanent Ban
107
+
108
+ **Community Impact**: Demonstrating a pattern of violation of community
109
+ standards, including sustained inappropriate behavior, harassment of an
110
+ individual, or aggression toward or disparagement of classes of individuals.
111
+
112
+ **Consequence**: A permanent ban from any sort of public interaction within the
113
+ community.
114
+
115
+ ## Attribution
116
+
117
+ This Code of Conduct is adapted from the [Contributor Covenant][homepage],
118
+ version 2.1, available at
119
+ [https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1].
120
+
121
+ Community Impact Guidelines were inspired by
122
+ [Mozilla's code of conduct enforcement ladder][Mozilla CoC].
123
+
124
+ For answers to common questions about this code of conduct, see the FAQ at
125
+ [https://www.contributor-covenant.org/faq][FAQ]. Translations are available at
126
+ [https://www.contributor-covenant.org/translations][translations].
127
+
128
+ [homepage]: https://www.contributor-covenant.org
129
+ [v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html
130
+ [Mozilla CoC]: https://github.com/mozilla/diversity
131
+ [FAQ]: https://www.contributor-covenant.org/faq
132
+ [translations]: https://www.contributor-covenant.org/translations
data/Guardfile ADDED
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ # A sample Guardfile
4
+ # More info at https://github.com/guard/guard#readme
5
+
6
+ ## Uncomment and set this to only include directories you want to watch
7
+ # directories %w(app lib config test spec features) \
8
+ # .select{|d| Dir.exist?(d) ? d : UI.warning("Directory #{d} does not exist")}
9
+
10
+ ## Note: if you are using the `directories` clause above and you are not
11
+ ## watching the project directory ('.'), then you will want to move
12
+ ## the Guardfile to a watched dir and symlink it back, e.g.
13
+ #
14
+ # $ mkdir config
15
+ # $ mv Guardfile config/
16
+ # $ ln -s config/Guardfile .
17
+ #
18
+ # and, you'll have to watch "config/Guardfile" instead of "Guardfile"
19
+
20
+ guard :minitest do
21
+ # with Minitest::Spec
22
+ watch(%r{^spec/(.*)_spec\.rb$})
23
+ # watch(%r{^lib/minitest/snapshot/(.+)\.rb$}) { 'spec' }
24
+ watch(%r{^lib/minitest/(.+)\.rb$}) { |m| "spec/minitest/#{m[1]}_spec.rb" }
25
+ watch(%r{^spec/spec_helper\.rb$}) { 'spec' }
26
+ end
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2025 Kematzy
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,202 @@
1
+ <!-- markdownlint-disable MD013 MD033 -->
2
+
3
+ # Minitest::Snapshot - Snapshot testing
4
+
5
+ [![Ruby](https://github.com/kematzy/minitest-snapshot/actions/workflows/ruby.yml/badge.svg?branch=main)](https://github.com/kematzy/minitest-snapshot/actions/workflows/ruby.yml) - [![Gem Version](https://badge.fury.io/rb/minitest-snapshot.svg)](https://badge.fury.io/rb/minitest-snapshot) - [![Minitest Style Guide](https://img.shields.io/badge/code_style-rubocop-brightgreen.svg)](https://github.com/rubocop/rubocop-minitest)
6
+
7
+ Coverage: **100%**
8
+
9
+ Snapshot testing for Minitest - compare your test outputs against saved reference snapshots.
10
+
11
+ ## What is Snapshot Testing?
12
+
13
+ Snapshot testing is a way to verify that your code's output matches a previously saved "snapshot".
14
+ Instead of manually writing expected outputs, you simply:
15
+
16
+ 1. Run the test once to generate the initial snapshot
17
+ 2. Verify the snapshot is correct
18
+ 3. In future test runs, your code's output will be compared against the saved snapshot
19
+
20
+ This is especially useful for testing complex objects, API responses, HTML, JSON,
21
+ and other outputs that would be tedious to write assertions for.
22
+
23
+ This gem makes snapshot testing in Minitest similar to how it works in RSpec
24
+ with [rspec-snapshot](https://github.com/levinmr/rspec-snapshot).
25
+
26
+ ## Installation
27
+
28
+ Add this line to your application's Gemfile:
29
+
30
+ ```ruby
31
+ gem 'minitest-snapshot'
32
+ ```
33
+
34
+ And then execute:
35
+
36
+ ```bash
37
+ bundle install
38
+ ```
39
+
40
+ Or install it yourself as:
41
+
42
+ ```bash
43
+ gem install minitest-snapshot
44
+ ```
45
+
46
+ ## Usage
47
+
48
+ ### Basic Usage
49
+
50
+ ```ruby
51
+ require 'minitest/autorun'
52
+ require 'minitest/snapshot'
53
+
54
+ class MyTest < Minitest::Test
55
+ def test_complex_object_matches_snapshot
56
+ complex_output = {
57
+ user: { name: 'Jane Doe', email: 'jane@example.com' },
58
+ permissions: ['read', 'write', 'admin']
59
+ }
60
+
61
+ assert_snapshot('user_with_permissions', complex_output)
62
+ end
63
+
64
+ def test_response_does_not_match_snapshot
65
+ unexpected_output = { status: 'error' }
66
+
67
+ refute_snapshot('successful_response', unexpected_output)
68
+ end
69
+ end
70
+ ```
71
+
72
+ ### Using Expectation Syntax
73
+
74
+ ```ruby
75
+ require 'minitest/autorun'
76
+ require 'minitest/snapshot'
77
+
78
+ describe 'API Response' do
79
+ it 'matches the expected structure' do
80
+ response = {
81
+ data: {
82
+ id: 123,
83
+ attributes: { title: 'Example', content: 'Content' }
84
+ }
85
+ }
86
+
87
+ _(response).must_match_snapshot('api_response')
88
+ end
89
+
90
+ it 'has changed from the previous version' do
91
+ response = { version: 'v2', data: {...} }
92
+
93
+ _(response).wont_match_snapshot('api_response_v1')
94
+ end
95
+ end
96
+ ```
97
+
98
+ ### Snapshot Naming
99
+
100
+ Snapshot names must follow these rules:
101
+
102
+ - Must not be empty
103
+ - Must not contain invalid filename characters (`< > : " | ? *`)
104
+ - Must not include path traversal (`..`)
105
+ - Must not include the file extension
106
+
107
+ Valid snapshot names:
108
+
109
+ - `'user_profile'`
110
+ - `'api/responses/success'`
111
+
112
+ ### Snapshot Files
113
+
114
+ By default, snapshots are stored in the `__snapshots__` directory at the root of your project.
115
+ Snapshots in nested directories (e.g., `'api/responses/success'`) will be
116
+ stored in matching subdirectories.
117
+
118
+ ### Updating Snapshots
119
+
120
+ To update existing snapshots, run your tests with the `UPDATE_SNAPSHOTS` environment variable:
121
+
122
+ ```bash
123
+ UPDATE_SNAPSHOTS=1 bundle exec rake spec
124
+ ```
125
+
126
+ ## Configuration
127
+
128
+ You can configure Minitest::Snapshot with custom options:
129
+
130
+ ```ruby
131
+ Minitest::Snapshot.configure do |config|
132
+ # Change the directory where snapshots are stored
133
+ config.snapshot_dir = 'test/fixtures/snapshots'
134
+
135
+ # Change the file extension for snapshot files
136
+ config.snapshot_file_extension = 'snap.json'
137
+
138
+ # Use a predefined serializer (:default, :json, or :yaml)
139
+ config.serializer = :json
140
+
141
+ # Configure JSON formatting options
142
+ config.json_options = { indent: ' ', object_nl: "\n" }
143
+
144
+ # Configure YAML formatting options
145
+ config.yaml_options = { line_width: 100 }
146
+
147
+ # Use a custom serializer
148
+ config.serializer = proc { |value| JSON.pretty_generate(value, indent: ' ') }
149
+
150
+ # Add a custom serializer for later use
151
+ config.add_serializer(:xml, proc { |value| value.to_xml })
152
+ # Then use it with:
153
+ # config.serializer = :xml
154
+ end
155
+ ```
156
+
157
+ ### Available Serializers
158
+
159
+ - `:default`: Uses `PP.pp` for pretty printing objects (good for Ruby objects)
160
+ - `:json`: Formats output as JSON
161
+ - `:yaml`: Formats output as YAML
162
+ - Custom serializers can be added with `add_serializer`
163
+
164
+ ## Example Workflow
165
+
166
+ 1. Write a test using `assert_snapshot`
167
+ 2. Run the test for the first time - it will create the snapshot file
168
+ 3. Verify the snapshot file content is correct
169
+ 4. Run the test again - it will compare against the saved snapshot
170
+ 5. If you intentionally change the expected output, update the snapshot with `UPDATE_SNAPSHOTS=1`
171
+
172
+ ## Development
173
+
174
+ After checking out the repo, run `bin/setup` to install dependencies.
175
+
176
+ Then, run `bundle exec rake spec` to run the tests.
177
+
178
+ You can also run `bin/console` for an interactive prompt that will allow you to experiment.
179
+
180
+ ## Contributing
181
+
182
+ Bug reports and pull requests are welcome at [github.com/kematzy/minitest-snapshot](https://github.com/kematzy/minitest-snapshot).
183
+
184
+ ## License
185
+
186
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
187
+
188
+ ## Credits
189
+
190
+ Code inspiration taken from [rspec-snapshot](https://github.com/levinmr/rspec-snapshot)
191
+ by [Mike Levin](https://github.com/levinmr/) released under a MIT License.
192
+
193
+ This gem was essentially built because I wanted to experiment with Zed's AI coding assistant.
194
+
195
+ For that reason, most of the code documentation and a large portion of the tests
196
+ were created by Claude Sonnet via Zed's Assistant UI.
197
+
198
+ Various other code parts were inspired by, or extracted from, Claude, Grok or ChatGPT answers.
199
+
200
+ - [Zed's AI Assistant](https://zed.dev) with Claude Sonnet 3.5 & 3.7
201
+ - [xAI's Grok 3](https://grok.com/)
202
+ - [OpenAI's ChatGPT](https://chatgpt.com)
data/Rakefile ADDED
@@ -0,0 +1,28 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'minitest/test_task'
5
+ require 'rake/testtask'
6
+ require 'rubocop/rake_task'
7
+
8
+ Minitest::TestTask.create
9
+ RuboCop::RakeTask.new
10
+
11
+ Rake::TestTask.new(:spec) do |t|
12
+ # load the `lib` folder, and the `test | spec` folder.
13
+ t.libs = %w[lib spec]
14
+ t.test_files = FileList['spec/**/*_spec.rb']
15
+ # enable extra CLI output, including the CLI command for debugging
16
+ t.verbose = true
17
+ end
18
+
19
+ task default: %i[rubocop coverage]
20
+
21
+ desc 'alias for spec task'
22
+ task test: :spec
23
+
24
+ desc 'Run specs with coverage'
25
+ task :coverage do
26
+ ENV['COVERAGE'] = 'true'
27
+ Rake::Task['spec'].execute
28
+ end
@@ -0,0 +1,27 @@
1
+ # frozen_string_literal: false
2
+
3
+ # Alternative loading version when the `lib` directory is in the `$LOAD_PATH`
4
+ # require 'minitest/snapshot/helpers'
5
+
6
+ # Access the main Minitest module
7
+ module Minitest
8
+ # re-open Assertions
9
+ module Assertions
10
+ # include Minitest::Snapshot::Helpers
11
+ include Minitest::Snapshot
12
+
13
+ def assert_snapshot(snapshot_name, value)
14
+ handle_snapshot(snapshot_name, value)
15
+ end
16
+
17
+ def refute_snapshot(snapshot_name, value)
18
+ handle_snapshot(snapshot_name, value, refute: true)
19
+ end
20
+ end
21
+
22
+ # re-open Expectations
23
+ module Expectations
24
+ infect_an_assertion :assert_snapshot, :must_match_snapshot
25
+ infect_an_assertion :refute_snapshot, :wont_match_snapshot
26
+ end
27
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Minitest
4
+ module Snapshot
5
+ VERSION = '0.1.0'
6
+ end
7
+ end
@@ -0,0 +1,354 @@
1
+ # frozen_string_literal: false
2
+
3
+ require 'pp'
4
+ require 'fileutils'
5
+ require 'minitest'
6
+
7
+ # loading core files using relative paths
8
+ # require_relative 'snapshot/version'
9
+
10
+ # Alternative loading version when the `lib` directory is in the `$LOAD_PATH`
11
+ require 'minitest/snapshot/version'
12
+ require 'minitest/snapshot/assertions'
13
+
14
+ # NOTE! loading files in this manner marks the file as 100% covered in code coverage tests
15
+ # even though it has not been tested
16
+
17
+ module Minitest
18
+ # Contains functionality for snapshot testing in Minitest
19
+ module Snapshot
20
+ # Configuration class for Minitest::Snapshot that manages snapshot
21
+ # directory, file extensions, serializers, and format options
22
+ #
23
+ # @example Basic configuration
24
+ # Minitest::Snapshot.configure do |config|
25
+ # config.snapshot_dir = 'test/snapshots'
26
+ # config.snapshot_file_extension = 'snap.json'
27
+ # end
28
+ #
29
+ # @example Setting a predefined serializer
30
+ # Minitest::Snapshot.configure do |config|
31
+ # config.serializer = :json # Use JSON serialization
32
+ # end
33
+ #
34
+ # @example Setting a custom serializer
35
+ # Minitest::Snapshot.configure do |config|
36
+ # config.serializer = proc { |value| value.to_json }
37
+ # end
38
+ #
39
+ # @example Adding a new serializer
40
+ # Minitest::Snapshot.configure do |config|
41
+ # config.add_serializer(:my_format, proc { |value|
42
+ # MyFormatter.format(value)
43
+ # })
44
+ # config.serializer = :my_format
45
+ # end
46
+ #
47
+ # @example Configuring JSON options
48
+ # Minitest::Snapshot.configure do |config|
49
+ # config.serializer = :json
50
+ # config.json_options = { indent: ' ', object_nl: "\n" }
51
+ # end
52
+ #
53
+ class Configuration
54
+ attr_accessor :snapshot_dir, :snapshot_file_extension, :json_options, :yaml_options
55
+ attr_reader :serializers
56
+
57
+ # Initialize a new Configuration with default values
58
+ #
59
+ # @return [Configuration] A new configuration instance
60
+ def initialize
61
+ @snapshot_dir = '__snapshots__'
62
+ @snapshot_file_extension = 'snap'
63
+ @serializers = {
64
+ default: proc { |value| value.is_a?(String) ? value : PP.pp(value, '') },
65
+ json: proc { |value| JSON.pretty_generate(value) },
66
+ yaml: proc { |value| YAML.dump(value) }
67
+ }
68
+ @current_serializer = :default
69
+
70
+ @json_options = { indent: ' ', object_nl: "\n" }
71
+ @yaml_options = { line_width: 80 }
72
+ end
73
+
74
+ # Sets the serializer to use for snapshots
75
+ #
76
+ # @example Set a built-in serializer
77
+ # config.serializer = :json
78
+ #
79
+ # @example Set a custom serializer
80
+ # config.serializer = proc { |value| CustomFormat.serialize(value) }
81
+ #
82
+ # @param key_or_proc [Symbol, Proc] A symbol referring to a predefined
83
+ # serializer or a custom serializer proc
84
+ #
85
+ # @raise [ArgumentError] If the serializer name doesn't exist
86
+ # @return [Symbol, Proc] The serializer that was set
87
+ #
88
+ def serializer=(key_or_proc)
89
+ if key_or_proc.is_a?(Proc)
90
+ @custom_serializer = key_or_proc
91
+ @current_serializer = :custom
92
+ elsif @serializers.key?(key_or_proc)
93
+ @current_serializer = key_or_proc
94
+ else
95
+ raise ArgumentError, "Unknown serializer: #{key_or_proc}"
96
+ end
97
+ end
98
+
99
+ # Gets the current serializer proc
100
+ #
101
+ # @example Get and use the current serializer
102
+ # serialized_value = config.serializer.call(my_object)
103
+ #
104
+ # @return [Proc] The currently selected serializer proc
105
+ #
106
+ def serializer
107
+ @current_serializer == :custom ? @custom_serializer : @serializers[@current_serializer]
108
+ end
109
+
110
+ # Adds a new serializer to the available serializers
111
+ #
112
+ # @example Adding a new serializer for XML
113
+ # config.add_serializer(:xml, proc { |value| value.to_xml })
114
+ #
115
+ # @param name [Symbol, String] The name for the new serializer
116
+ # @param serializer_proc [Proc] The serializer proc that takes an object and returns a string
117
+ #
118
+ # @return [Hash] The updated serializers hash
119
+ #
120
+ def add_serializer(name, serializer_proc)
121
+ @serializers[name.to_sym] = serializer_proc
122
+ end
123
+ end
124
+
125
+ class << self
126
+ # Returns the configuration object for Minitest::Snapshot
127
+ #
128
+ # @example Getting the current configuration
129
+ # config = Minitest::Snapshot.configuration
130
+ # puts config.snapshot_dir
131
+ #
132
+ # @return [Configuration] The configuration object
133
+ #
134
+ def configuration
135
+ @configuration ||= Configuration.new
136
+ end
137
+
138
+ # Configures Minitest::Snapshot with a block
139
+ #
140
+ # @example Basic configuration
141
+ # Minitest::Snapshot.configure do |config|
142
+ # config.snapshot_dir = 'test/snapshots'
143
+ # config.snapshot_file_extension = 'snap.json'
144
+ # end
145
+ #
146
+ # @example Setting a custom serializer
147
+ # Minitest::Snapshot.configure do |config|
148
+ # config.serializer = proc { |value| JSON.pretty_generate(value) }
149
+ # end
150
+ #
151
+ # @example Using a predefined serializer
152
+ # Minitest::Snapshot.configure do |config|
153
+ # config.serializer = :yaml
154
+ # end
155
+ #
156
+ # @yield [configuration] The configuration object
157
+ # @return [void]
158
+ #
159
+ def configure
160
+ yield(configuration) if block_given?
161
+ end
162
+
163
+ # Gets the directory where snapshots are stored
164
+ #
165
+ # @example Getting the snapshot directory
166
+ # dir = Minitest::Snapshot.snapshot_dir
167
+ # # => "__snapshots__"
168
+ #
169
+ # @return [String] The path to the snapshots directory
170
+ #
171
+ def snapshot_dir
172
+ configuration.snapshot_dir
173
+ end
174
+
175
+ # Sets the directory where snapshots are stored
176
+ #
177
+ # @example Setting the snapshot directory
178
+ # Minitest::Snapshot.snapshot_dir = 'test/fixtures/snapshots'
179
+ #
180
+ # @param value [String] The path to the snapshots directory
181
+ # @return [String] The new path
182
+ #
183
+ def snapshot_dir=(value)
184
+ configuration.snapshot_dir = value
185
+ end
186
+
187
+ # Gets the current serializer used for snapshots
188
+ #
189
+ # @example Getting the current serializer
190
+ # serializer = Minitest::Snapshot.snapshot_serializer
191
+ # serialized_value = serializer.call(my_object)
192
+ #
193
+ # @return [Proc] The serializer proc
194
+ #
195
+ def snapshot_serializer
196
+ configuration.serializer
197
+ end
198
+
199
+ # Sets the serializer used for snapshots
200
+ #
201
+ # @example Setting a custom serializer
202
+ # Minitest::Snapshot.snapshot_serializer = proc { |value|
203
+ # JSON.pretty_generate(value, indent: ' ')
204
+ # }
205
+ #
206
+ # @example Using a predefined serializer
207
+ # Minitest::Snapshot.snapshot_serializer = :json
208
+ #
209
+ # @param value [Proc, Symbol] A proc that takes an object and returns a string,
210
+ # or a symbol referencing a predefined serializer
211
+ #
212
+ # @return [Proc, Symbol] The serializer that was set
213
+ #
214
+ def snapshot_serializer=(value)
215
+ configuration.serializer = value
216
+ end
217
+
218
+ # Gets the file extension used for snapshot files
219
+ #
220
+ # @example Getting the snapshot file extension
221
+ # ext = Minitest::Snapshot.snapshot_file_extension
222
+ # # => "snap"
223
+ #
224
+ # @return [String] The file extension
225
+ #
226
+ def snapshot_file_extension
227
+ configuration.snapshot_file_extension
228
+ end
229
+
230
+ # Sets the file extension used for snapshot files
231
+ #
232
+ # @example Setting the snapshot file extension
233
+ # Minitest::Snapshot.snapshot_file_extension = 'snap.json'
234
+ #
235
+ # @param value [String] The file extension
236
+ #
237
+ # @return [String] The new file extension
238
+ #
239
+ def snapshot_file_extension=(value)
240
+ configuration.snapshot_file_extension = value
241
+ end
242
+ end
243
+
244
+ class Error < StandardError
245
+ end
246
+
247
+ private
248
+
249
+ # Generates a file path for a snapshot based on its name
250
+ #
251
+ # @param snapshot_name [String, Symbol] Name of the snapshot
252
+ #
253
+ # @return [String] The full path where the snapshot file will be stored
254
+ #
255
+ def snapshot_file_path(snapshot_name)
256
+ parts = snapshot_name.to_s.split('/')
257
+ filename = "#{parts.pop}.#{Minitest::Snapshot.snapshot_file_extension}"
258
+ dir = File.join(Minitest::Snapshot.snapshot_dir, *parts)
259
+ FileUtils.mkdir_p(dir)
260
+ File.join(dir, filename)
261
+ end
262
+
263
+ # Loads an existing snapshot from a file
264
+ #
265
+ # @param file [String] Path to the snapshot file
266
+ #
267
+ # @return [String, nil] The content of the snapshot file or nil if the file doesn't exist
268
+ #
269
+ def load_snapshot(file)
270
+ File.read(file) if File.exist?(file)
271
+ end
272
+
273
+ # Saves a value as a snapshot to a file
274
+ #
275
+ # @param file [String] Path where the snapshot will be saved
276
+ # @param value [Object] The value to be serialized and saved
277
+ #
278
+ # @return [Integer] The number of bytes written
279
+ #
280
+ def save_snapshot(file, value)
281
+ serialized = serialize(value)
282
+ File.write(file, serialized)
283
+ end
284
+
285
+ # Serializes a value using the configured serializer
286
+ #
287
+ # @param value [Object] The value to be serialized
288
+ # @return [String] The serialized representation of the value
289
+ #
290
+ def serialize(value)
291
+ Minitest::Snapshot.snapshot_serializer.call(value)
292
+ end
293
+
294
+ # Handles snapshot comparison or creation
295
+ #
296
+ # This method is the main coordinator for snapshot testing. It decides
297
+ # whether to update an existing snapshot, create a new one, or compare
298
+ # against an existing one based on environment settings and file existence.
299
+ #
300
+ # @param snapshot_name [String, Symbol] Name of the snapshot, used to generate the file path
301
+ # @param value [Object] The value to be serialized and compared or saved
302
+ # @param refute [Boolean] When true, test passes if snapshots don't match
303
+ #
304
+ # @raise [Minitest::Snapshot::Error] If any error occurs during snapshot handling
305
+ #
306
+ # @return [void]
307
+ #
308
+ def handle_snapshot(snapshot_name, value, refute: false)
309
+ file = snapshot_file_path(snapshot_name)
310
+
311
+ begin
312
+ if update_snapshots?
313
+ save_snapshot(file, value)
314
+ elsif File.exist?(file)
315
+ compare_with_snapshot(file, value, refute)
316
+ else # rubocop:disable Lint/DuplicateBranch
317
+ save_snapshot(file, value)
318
+ end
319
+ rescue StandardError => e
320
+ raise Minitest::Snapshot::Error,
321
+ "Failed to handle snapshot '#{snapshot_name}': #{e.message}"
322
+ end
323
+ end
324
+
325
+ # Compares a value with a saved snapshot
326
+ #
327
+ # @param file [String] Path to the snapshot file to compare against
328
+ # @param value [Object] The value to be serialized and compared
329
+ # @param refute [Boolean] When true, test passes if snapshots don't match
330
+ #
331
+ # @raise [Minitest::Assertion] If the assertion fails
332
+ #
333
+ # @return [Boolean] true if the assertion passes
334
+ #
335
+ def compare_with_snapshot(file, value, refute)
336
+ expected = load_snapshot(file)
337
+ serialized = serialize(value)
338
+
339
+ if refute
340
+ refute_equal expected, serialized, "Snapshot '#{file}' should not match."
341
+ else
342
+ assert_equal expected, serialized, "Snapshot '#{file}' does not match."
343
+ end
344
+ end
345
+
346
+ # Checks if snapshots should be updated based on environment variable
347
+ #
348
+ # @return [Boolean] true if UPDATE_SNAPSHOTS env var is set, false otherwise
349
+ #
350
+ def update_snapshots?
351
+ !!ENV.fetch('UPDATE_SNAPSHOTS', nil)
352
+ end
353
+ end
354
+ end
@@ -0,0 +1,31 @@
1
+ module Minitest
2
+ module Snapshot
3
+ VERSION: String
4
+ # See the writing guide of rbs: https://github.com/ruby/rbs#guides
5
+
6
+ # Main method used to assert equality with a snapshot
7
+ def assert_snapshot: (untyped actual, ?String name) -> void
8
+
9
+ # Method to update snapshots when they need to be refreshed
10
+ def update_snapshot?: () -> bool
11
+
12
+ # Configuration options for snapshot testing
13
+ class Configuration
14
+ attr_accessor snapshot_directory: String
15
+ attr_accessor update_snapshots: bool
16
+
17
+ def initialize: () -> void
18
+ end
19
+
20
+ # Method to configure snapshot behavior
21
+ def self.configure: () { (Configuration) -> void } -> void
22
+
23
+ # Access to configuration
24
+ def self.configuration: () -> Configuration
25
+ end
26
+
27
+ # Extension to Minitest::Test to include Snapshot functionality
28
+ class Test
29
+ include Snapshot
30
+ end
31
+ end
metadata ADDED
@@ -0,0 +1,90 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: minitest-snapshot
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Kematzy
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2025-03-14 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: minitest
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: 5.20.0
20
+ - - "<"
21
+ - !ruby/object:Gem::Version
22
+ version: '6.0'
23
+ type: :runtime
24
+ prerelease: false
25
+ version_requirements: !ruby/object:Gem::Requirement
26
+ requirements:
27
+ - - ">="
28
+ - !ruby/object:Gem::Version
29
+ version: 5.20.0
30
+ - - "<"
31
+ - !ruby/object:Gem::Version
32
+ version: '6.0'
33
+ description: Snapshot testing for Minitest - compare your test outputs against saved
34
+ reference snapshots
35
+ email:
36
+ - kematzy@gmail.com
37
+ executables: []
38
+ extensions: []
39
+ extra_rdoc_files:
40
+ - README.md
41
+ - LICENSE.txt
42
+ files:
43
+ - ".rubocop.yml"
44
+ - ".rubocop_todo.yml"
45
+ - CHANGELOG.md
46
+ - CODE_OF_CONDUCT.md
47
+ - Guardfile
48
+ - LICENSE.txt
49
+ - README.md
50
+ - Rakefile
51
+ - lib/minitest/snapshot.rb
52
+ - lib/minitest/snapshot/assertions.rb
53
+ - lib/minitest/snapshot/version.rb
54
+ - sig/minitest/snapshot.rbs
55
+ homepage: https://github.com/kematzy/minitest-snapshot
56
+ licenses:
57
+ - MIT
58
+ metadata:
59
+ homepage_uri: https://github.com/kematzy/minitest-snapshot
60
+ source_code_uri: https://github.com/kematzy/minitest-snapshot
61
+ documentation_uri: https://github.com/kematzy/minitest-snapshot
62
+ changelog_uri: https://github.com/kematzy/minitest-snapshot/blob/main/CHANGELOG.md
63
+ rubygems_mfa_required: 'true'
64
+ post_install_message:
65
+ rdoc_options:
66
+ - "--quiet"
67
+ - "--line-numbers"
68
+ - "--inline-source"
69
+ - "--title"
70
+ - 'Minitest::Snapshot: '
71
+ - "--main"
72
+ - README.md
73
+ require_paths:
74
+ - lib
75
+ required_ruby_version: !ruby/object:Gem::Requirement
76
+ requirements:
77
+ - - ">="
78
+ - !ruby/object:Gem::Version
79
+ version: 3.0.0
80
+ required_rubygems_version: !ruby/object:Gem::Requirement
81
+ requirements:
82
+ - - ">="
83
+ - !ruby/object:Gem::Version
84
+ version: '0'
85
+ requirements: []
86
+ rubygems_version: 3.5.22
87
+ signing_key:
88
+ specification_version: 4
89
+ summary: Minitest Snapshot testing - (like "rspec-snapshot")
90
+ test_files: []