Skip to content

Commit

Permalink
Feature/Migration analytics rake task (#13)
Browse files Browse the repository at this point in the history
* Changed CLI to printer
* Added rake task, tests
* Updated gem documentation
* Updated gem version, changelog
  • Loading branch information
bestwebua authored Nov 9, 2024
1 parent 6923474 commit 2484a51
Show file tree
Hide file tree
Showing 14 changed files with 131 additions and 136 deletions.
1 change: 1 addition & 0 deletions .circleci/gemspecs/compatible
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ Gem::Specification.new do |spec|
spec.require_paths = %w[lib]

spec.add_runtime_dependency 'colorize', '>= 0.8.1'
spec.add_runtime_dependency 'rake', '~> 13.2', '>= 13.2.1'
spec.add_runtime_dependency 'rspec-core', '~> 3.10'
spec.add_runtime_dependency 'rspec-mocks', '~> 3.10'
spec.add_runtime_dependency 'terminal-table', '~> 3.0'
Expand Down
1 change: 1 addition & 0 deletions .circleci/gemspecs/latest
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ Gem::Specification.new do |spec|
spec.require_paths = %w[lib]

spec.add_runtime_dependency 'colorize', '>= 0.8.1'
spec.add_runtime_dependency 'rake', '~> 13.2', '>= 13.2.1'
spec.add_runtime_dependency 'rspec-core', '~> 3.10'
spec.add_runtime_dependency 'rspec-mocks', '~> 3.10'
spec.add_runtime_dependency 'terminal-table', '~> 3.0'
Expand Down
16 changes: 13 additions & 3 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,27 @@

The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).

## [0.4.0] - 2024-11-09

### Added

- Added rake task to analyze Flexmock usage and track migration progress to RSpec mocks

### Removed

- Removed CLI

## [0.3.1] - 2024-11-08

### Fixed

- Fixed CLI broken import.
- Fixed CLI broken import

## [0.3.0] - 2024-11-08

### Added

- Added CLI to analyze Flexmock usage and track migration progress to RSpec mocks.
- Added CLI to analyze Flexmock usage and track migration progress to RSpec mocks

## [0.2.0] - 2024-11-04

Expand All @@ -25,4 +35,4 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),

### Added

- First release of `RSpec::Mock`.
- First release of `RSpec::Mock`
29 changes: 7 additions & 22 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -130,25 +130,18 @@ end

### Migration Analytics

You can create a Rake task to analyze Flexmock usage and track migration progress to RSpec mocks. Or use the CLI directly.
This time implemented migration analytics for [Flexmock](https://github.com/doudou/flexmock) only. You can run Rake task to analyze Flexmock usage and track migration progress to RSpec mocks.

Example of the Rake task:
For non-Rails applications to use the task, you need to load it:

```ruby
namespace :rspec_mock do
namespace :migration_analytics do
desc 'Analyze Flexmock usage and track migration progress to RSpec mocks'
task :flexmock do
require 'rspec/mock/migration_analytics/cli'

path = ::ARGV[1] || 'spec'
puts("\n🔍 Analyzing Flexmock usage in: #{path}")
RSpec::Mock::MigrationAnalytics::Cli.verify_path(path)
end
end
end
require 'rspec/mock/task'

RSpec::Mock::Task.load
```

For Rails applications it will be automatically loaded, so just run:

```bash
# Analyze entire spec directory (default)
rake rspec_mock:migration_analytics:flexmock
Expand All @@ -160,14 +153,6 @@ rake rspec_mock:migration_analytics:flexmock spec/services
rake rspec_mock:migration_analytics:flexmock spec/services/sandbox_service_spec.rb
```

Example of the CLI usage:

```bash
ruby cli.rb spec
ruby cli.rb spec/services
ruby cli.rb spec/services/sandbox_service_spec.rb
```

## Contributing

Bug reports and pull requests are welcome on GitHub at <https://github.com/mocktools/ruby-rspec-mock>. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct. Please check the [open tickets](https://github.com/mocktools/ruby-rspec-mock/issues). Be sure to follow Contributor Code of Conduct below and our [Contributing Guidelines](CONTRIBUTING.md).
Expand Down
4 changes: 3 additions & 1 deletion lib/rspec/mock/core.rb
Original file line number Diff line number Diff line change
Expand Up @@ -13,13 +13,15 @@ module Tracker
end

require_relative 'migration_analytics/file_analyzer'
require_relative 'migration_analytics/cli'
require_relative 'migration_analytics/printer'
end

require_relative 'configuration'
require_relative 'context'
require_relative 'methods'
require_relative 'version'

require_relative(defined?(::Rails) ? 'railtie' : 'task')
end

module Core
Expand Down
34 changes: 4 additions & 30 deletions lib/rspec/mock/migration_analytics/cli.rb → ...rspec/mock/migration_analytics/printer.rb
100755 → 100644
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
#!/usr/bin/env ruby
# frozen_string_literal: true

require 'colorize'
Expand All @@ -12,39 +11,16 @@
module RSpec
module Mock
module MigrationAnalytics
class Cli
class Printer
class << self
def call
if ::ARGV.empty?
print_usage
exit 1
end

begin
verify_path(::ARGV[0])
rescue => error
puts("\n❌ Error: #{error.message}".red)
puts(error.backtrace) if ENV['DEBUG']
end
end
def call(path)
return verify_directory(path) if ::File.directory?(path)

def verify_path(path)
case
when ::File.directory?(path) then verify_directory(path)
else verify_file(path)
end
verify_file(path)
end

private

def print_usage
puts('Usage: ruby cli.rb <path_to_spec_file_or_directory>'.yellow)
puts("\nExamples:".blue)
puts(' ruby cli.rb spec/models/user_spec.rb')
puts(' ruby cli.rb spec/models/')
puts(' ruby cli.rb spec/')
end

def verify_directory(dir_path) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
results = []
stats = {
Expand Down Expand Up @@ -188,5 +164,3 @@ def create_file_row(result)
end
end
end

RSpec::Mock::MigrationAnalytics::Cli.call if __FILE__.eql?($PROGRAM_NAME)
11 changes: 11 additions & 0 deletions lib/rspec/mock/railtie.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# frozen_string_literal: true

module RSpec
module Mock
class Railtie < ::Rails::Railtie
rake_tasks do
load(::File.join(::File.dirname(__FILE__), 'tasks', 'rspec_mock.rake'))
end
end
end
end
13 changes: 13 additions & 0 deletions lib/rspec/mock/task.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
# frozen_string_literal: true

require 'rake'

module RSpec
module Mock
class Task
def self.load
::Kernel.load(::File.join(::File.dirname(__FILE__), 'tasks', 'rspec_mock.rake'))
end
end
end
end
16 changes: 16 additions & 0 deletions lib/rspec/mock/tasks/rspec_mock.rake
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
# frozen_string_literal: true

namespace :rspec_mock do
namespace :migration_analytics do
desc 'Analyze Flexmock usage and track migration progress to RSpec mocks'
# :nocov:
task :flexmock do
require 'rspec/mock/migration_analytics/printer'

path = ::ARGV[1] || 'spec'
puts("\n🔍 Analyzing Flexmock usage in: #{path}")
RSpec::Mock::MigrationAnalytics::Printer.call(path)
end
# :nocov:
end
end
2 changes: 1 addition & 1 deletion lib/rspec/mock/version.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,6 @@

module RSpec
module Mock
VERSION = '0.3.1'
VERSION = '0.4.0'
end
end
1 change: 1 addition & 0 deletions rspec-mock.gemspec
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ Gem::Specification.new do |spec|
spec.require_paths = %w[lib]

spec.add_runtime_dependency 'colorize', '>= 0.8.1'
spec.add_runtime_dependency 'rake', '~> 13.2', '>= 13.2.1'
spec.add_runtime_dependency 'rspec-core', '~> 3.10'
spec.add_runtime_dependency 'rspec-mocks', '~> 3.10'
spec.add_runtime_dependency 'terminal-table', '~> 3.0'
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# frozen_string_literal: true

RSpec.describe RSpec::Mock::MigrationAnalytics::Cli do
RSpec.describe RSpec::Mock::MigrationAnalytics::Printer do
let(:sample_file_path) { 'spec/models/user_spec.rb' }
let(:sample_dir_path) { 'spec/models' }

Expand All @@ -12,69 +12,21 @@
end

describe '.call' do
context 'when no arguments are provided' do
before { stub_const('ARGV', []) }

it 'prints usage and exits with status 1' do
expect(described_class).to receive(:print_usage)
expect { described_class.call }.to raise_error(SystemExit) do |error|
expect(error.status).to eq(1)
end
end
end

context 'when path is provided' do
before { stub_const('ARGV', [sample_file_path]) }

it 'verifies the provided path' do
expect(described_class).to receive(:verify_path).with(sample_file_path)
described_class.call
end

context 'when an error occurs' do
let(:error_message) { 'Something went wrong' }

before do
allow(described_class).to receive(:verify_path).and_raise(StandardError.new(error_message))
allow(ENV).to receive(:[]).and_return(nil)
allow(ENV).to receive(:[]).with('COLUMNS').and_return('80')
end

it 'prints the error message in red' do
expect { described_class.call }
.to output(/❌ Error: #{error_message}/).to_stdout
end

context 'when DEBUG env is set' do
before do
allow(ENV).to receive(:[]).with('DEBUG').and_return('true')
end

it 'prints the backtrace' do
expect { described_class.call }
.to output(%r{#{error_message}.*gems/rspec}m).to_stdout
end
end
end
end
end

describe '.verify_path' do
context 'when path is a directory' do
before { allow(File).to receive(:directory?).with(sample_dir_path).and_return(true) }
before { allow(::File).to receive(:directory?).with(sample_dir_path).and_return(true) }

it 'calls verify_directory' do
expect(described_class).to receive(:verify_directory).with(sample_dir_path)
described_class.verify_path(sample_dir_path)
described_class.call(sample_dir_path)
end
end

context 'when path is a file' do
before { allow(File).to receive(:directory?).with(sample_file_path).and_return(false) }
before { allow(::File).to receive(:directory?).with(sample_file_path).and_return(false) }

it 'calls verify_file' do
expect(described_class).to receive(:verify_file).with(sample_file_path)
described_class.verify_path(sample_file_path)
described_class.call(sample_file_path)
end
end
end
Expand All @@ -83,7 +35,7 @@
include_context 'with stubbed ENV'

context 'when file does not exist' do
before { allow(File).to receive(:exist?).with(sample_file_path).and_return(false) }
before { allow(::File).to receive(:exist?).with(sample_file_path).and_return(false) }

it 'prints error message' do
expect { described_class.send(:verify_file, sample_file_path) }
Expand All @@ -94,7 +46,7 @@
context 'when file is not a spec file' do
let(:non_spec_file) { 'app/models/user.rb' }

before { allow(File).to receive(:exist?).with(non_spec_file).and_return(true) }
before { allow(::File).to receive(:exist?).with(non_spec_file).and_return(true) }

it 'prints warning message' do
expect { described_class.send(:verify_file, non_spec_file) }
Expand All @@ -115,7 +67,7 @@
end

before do
allow(File).to receive(:exist?).with(sample_file_path).and_return(true)
allow(::File).to receive(:exist?).with(sample_file_path).and_return(true)
allow(RSpec::Mock::MigrationAnalytics::FileAnalyzer).to receive(:call)
.with(sample_file_path).and_return(analysis_result)
end
Expand All @@ -140,7 +92,7 @@
end

before do
allow(File).to receive(:exist?).with(sample_file_path).and_return(true)
allow(::File).to receive(:exist?).with(sample_file_path).and_return(true)
allow(RSpec::Mock::MigrationAnalytics::FileAnalyzer).to receive(:call)
.with(sample_file_path).and_return(analysis_result)
end
Expand Down Expand Up @@ -180,14 +132,16 @@
end

before do
allow(Dir).to receive(:glob).with("#{sample_dir_path}/**/*_spec.rb").and_return(spec_files)
allow(RSpec::Mock::MigrationAnalytics::FileAnalyzer).to receive(:call)
allow(::Dir).to receive(:glob).with("#{sample_dir_path}/**/*_spec.rb").and_return(spec_files)
allow(RSpec::Mock::MigrationAnalytics::FileAnalyzer)
.to receive(:call)
.and_return(*analysis_results)
end

it 'analyzes all spec files and prints summary' do
expect { described_class.send(:verify_directory, sample_dir_path) }
.to output(/=== Migration Status Report ===.*Files Requiring Migration/m).to_stdout
.to output(/=== Migration Status Report ===.*Files Requiring Migration/m)
.to_stdout
end
end

Expand Down Expand Up @@ -215,23 +169,4 @@
end
end
end

describe '.print_usage' do
include_context 'with stubbed ENV'

it 'prints usage instructions with examples' do
expected_lines = [
'Usage: ruby cli.rb <path_to_spec_file_or_directory>',
'',
'Examples:',
' ruby cli.rb spec/models/user_spec.rb',
' ruby cli.rb spec/models/',
' ruby cli.rb spec/'
]

expect { described_class.send(:print_usage) }
.to output(/#{expected_lines.join('.*')}/m)
.to_stdout
end
end
end
Loading

0 comments on commit 2484a51

Please sign in to comment.