retest 2.0.0.pre3 → 2.0.0.pre5

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: 2a4544ea33941c52a493aab49a02f293cf6539e08db2482a858840f49e1ea1e6
4
- data.tar.gz: fe2125582bec55b07b0e224ec196f37ed9f9c78c7f21dfd91e5e518ccb92c3d4
3
+ metadata.gz: 754751292557705217a2e6010b5fda60d10fffbdf5a1b66ffe48976841407e09
4
+ data.tar.gz: 2156346b0cd5caa0d5dcf329391a0b84a85fc88342a97fe2e3150ba196fdd7c5
5
5
  SHA512:
6
- metadata.gz: 7971f9c3dc66d7e0389d859f8ddb6d817474c000da3314f4fd4a916fd35c4ea34bd0d0663884769e149739384785a7bd702c72df2b0e898f0e963ba9edd4425c
7
- data.tar.gz: 7c3e6a114f9693c2d47fb9ed59e5c5d8337eeb9b1df02bd0cb1f7519128a63b9618ec5873a66264d712d009532153c4917f9f81e1207e561a0e729e31ab8e082
6
+ metadata.gz: 328df6d6fe4153956230bb4029337f68f20ef8a796337b7c423ce2cd07ddf01ce3c69d531e3daac5babe17dae6c266053919246f6f2e93e69ecf56704cf52cc1
7
+ data.tar.gz: ea86b8e02ddced2b484c780ca8ec673098d23a8e86cde6f0f11b15d903e7206bcb88840b61438aa947cbac4b72c34ad8831bc2d2c76f205c6bcf2f73ff75a97f
@@ -20,13 +20,13 @@ jobs:
20
20
  matrix:
21
21
  os: [ubuntu-latest]
22
22
  ruby:
23
- - 2.5
24
- - 2.6
25
- - 2.7
23
+ - '2.5'
24
+ - '2.6'
25
+ - '2.7'
26
26
  - '3.0'
27
- - 3.1
28
- - 3.2
29
- - 3.3
27
+ - '3.1'
28
+ - '3.2'
29
+ - '3.3'
30
30
  include:
31
31
  - os: macos-13
32
32
  ruby: 2.5
data/Gemfile.lock CHANGED
@@ -1,11 +1,12 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- retest (2.0.0.pre3)
4
+ retest (2.0.0.pre5)
5
5
  listen (~> 3.9)
6
6
  observer (~> 0.1)
7
7
  string-similarity (~> 2.1)
8
8
  tty-option (~> 0.1)
9
+ tty-prompt (~> 0.1)
9
10
 
10
11
  GEM
11
12
  remote: https://rubygems.org/
@@ -17,12 +18,25 @@ GEM
17
18
  rb-inotify (~> 0.9, >= 0.9.10)
18
19
  minitest (5.15.0)
19
20
  observer (0.1.2)
21
+ pastel (0.8.0)
22
+ tty-color (~> 0.5)
20
23
  rake (13.0.6)
21
24
  rb-fsevent (0.11.2)
22
25
  rb-inotify (0.11.1)
23
26
  ffi (~> 1.0)
24
27
  string-similarity (2.1.0)
28
+ tty-color (0.6.0)
29
+ tty-cursor (0.7.1)
25
30
  tty-option (0.3.0)
31
+ tty-prompt (0.23.1)
32
+ pastel (~> 0.8)
33
+ tty-reader (~> 0.8)
34
+ tty-reader (0.9.0)
35
+ tty-cursor (~> 0.7)
36
+ tty-screen (~> 0.8)
37
+ wisper (~> 2.0)
38
+ tty-screen (0.8.2)
39
+ wisper (2.0.1)
26
40
 
27
41
  PLATFORMS
28
42
  ruby
@@ -34,4 +48,4 @@ DEPENDENCIES
34
48
  retest!
35
49
 
36
50
  BUNDLED WITH
37
- 2.3.22
51
+ 2.3.27
@@ -0,0 +1,7 @@
1
+ #!/usr/bin/env bash
2
+
3
+ docker buildx build \
4
+ --platform linux/amd64,linux/arm64 \
5
+ -t ghcr.io/alexb52/slim-bullseye-watchexec:latest \
6
+ -f builds/dockerfiles/WatchexecSlimBullseye \
7
+ --push .
data/bin/test/ruby-app CHANGED
@@ -2,6 +2,6 @@
2
2
 
3
3
  bundle install
4
4
  bundle exec rake build
5
- cp -R features/support features/ruby-app/retest
5
+ # cp -R features/support features/ruby-app/retest
6
6
  ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/ruby-app/retest.gem
7
- docker compose -f features/ruby-app/docker-compose.yml up --build --exit-code-from retest
7
+ docker compose -f features/ruby-app/docker-compose.yml up --build --exit-code-from retest
@@ -0,0 +1,12 @@
1
+ # Stage 1: Build watchexec with Rust
2
+ FROM rust:1.83.0-slim-bullseye AS rust-builder
3
+
4
+ # Install necessary dependencies for Rust
5
+ RUN apt-get update -qq && \
6
+ apt-get install --no-install-recommends -y build-essential git
7
+
8
+ # Install watchexec
9
+ RUN cargo install watchexec-cli
10
+
11
+ # Verify installation
12
+ RUN watchexec --version
data/exe/retest CHANGED
@@ -5,7 +5,7 @@ require 'retest'
5
5
  $stdout.sync = true
6
6
  listen_rd, listen_wr = IO.pipe
7
7
  Signal.trap(:INT) do
8
- $stdout.puts "Goodbye"
8
+ puts "Goodbye"
9
9
  listen_rd.close
10
10
  listen_wr.close
11
11
  exit
@@ -14,25 +14,21 @@ end
14
14
  options = Retest::Options.new(ARGV)
15
15
 
16
16
  if options.help?
17
- $stdout.puts options.help
17
+ puts options.help
18
18
  return
19
19
  end
20
20
 
21
21
  if options.version?
22
- $stdout.puts Retest::VERSION
22
+ puts Retest::VERSION
23
23
  return
24
24
  end
25
25
 
26
26
  prompt = Retest::Prompt.new
27
27
  repository = Retest::Repository.new(files: Retest::VersionControl.files, prompt: prompt)
28
28
  command = Retest::Command.for_options(options)
29
- runner = Retest::Runners.runner_for(command.to_s)
29
+ runner = Retest::Runner.new(command)
30
30
  sounds = Retest::Sounds.for(options)
31
-
32
- # All test runner
33
- all_test_command = Retest::Command.for_options(options.merge(%w[--all]), stdout: nil)
34
- all_test_runner = Retest::Runners.runner_for(all_test_command.to_s)
35
- all_test_runner.add_observer(sounds)
31
+ watcher = Retest::Watcher.for(options.watcher)
36
32
 
37
33
  sounds.play(:start)
38
34
  runner.add_observer(sounds)
@@ -40,7 +36,6 @@ prompt.add_observer(sounds)
40
36
 
41
37
  program = Retest::Program.new(
42
38
  repository: repository,
43
- command: command,
44
39
  runner: runner
45
40
  )
46
41
 
@@ -49,6 +44,12 @@ if options.params[:diff]
49
44
  return
50
45
  end
51
46
 
47
+ if watcher == Retest::Watcher::Watchexec
48
+ puts "Watcher: [WATCHEXEC]"
49
+ else Retest::Watcher::Default
50
+ puts "Watcher: [LISTEN]"
51
+ end
52
+
52
53
  launching_message = "Launching Retest..."
53
54
  if options.force_polling?
54
55
  launching_message = "Launching Retest with polling method..."
@@ -56,21 +57,22 @@ end
56
57
 
57
58
  # Main action
58
59
 
59
- $stdout.puts launching_message
60
- Retest.listen(options) do |modified, added, removed|
60
+ puts launching_message
61
+ Retest.listen(options, listener: watcher) do |modified, added, removed|
61
62
  begin
62
63
  repository.sync(added: added, removed: removed)
63
64
  runner.sync(added: added, removed: removed)
64
65
 
65
66
  listen_wr.puts "file changed: #{(modified + added).first}"
66
67
  rescue => e
67
- $stdout.puts "Something went wrong: #{e.message}"
68
+ puts "Something went wrong: #{e.message}"
68
69
  end
69
70
  end
70
- $stdout.puts "Ready to refactor! You can make file changes now"
71
+ puts "Ready to refactor! You can make file changes now"
71
72
 
72
- def run_command(input:, program:, all_test_runner:)
73
- case input
73
+ def run_command(input:, program:)
74
+ program.clear_terminal
75
+ case input.strip
74
76
  when /^file changed:\s(.*)$/
75
77
  puts "File changed: #{$1}"
76
78
  program.run($1)
@@ -82,24 +84,38 @@ def run_command(input:, program:, all_test_runner:)
82
84
  puts "Program has been resumed\n"
83
85
  when 'e', 'exit'
84
86
  Process.kill("INT", 0)
85
- when ''
86
- puts "Running last command\n"
87
+ when 'r', 'reset'
88
+ program.reset_selection
89
+ puts "command reset to '#{program.runner.command.to_s}'"
90
+ when 'f', 'force'
91
+ require 'tty-prompt'
92
+ prompt = TTY::Prompt.new
93
+ program.force_selection prompt.multi_select(
94
+ "What test files do you want to run when saving a file?",
95
+ program.repository.test_files,
96
+ filter: true, min: 1
97
+ )
87
98
  program.run(nil, force_run: true)
99
+ when ''
100
+ puts "Running last command: '#{program.last_command}'\n"
101
+ program.run_last_command
88
102
  when 'ra', 'run all'
89
103
  puts "Running all tests\n"
90
- all_test_runner.run
104
+ program.run_all
91
105
  when /^di?f?f?\s(.*)$/
92
106
  program.diff($1)
93
107
  when 'h', 'help'
94
108
  puts <<~HELP
95
109
 
96
- * 'h', 'help' # Prints help
97
- * 'p', 'pause' # Pauses Retest. Tests aren't run until unpaused.
98
- * 'u', 'unpause' # Unpauses Retest
99
- * <ENTER> # Runs last changed triggered command
100
- * 'ra, 'run all' # Runs all tests
101
- * 'd', 'diff' [GIT BRANCH] # Run matching specs that changed from a target branch
102
- * 'e', 'exit' # Exits Retest
110
+ * 'h', 'help' # Prints help.
111
+ * 'p', 'pause' # Pauses Retest. Tests aren't run on file change events until unpaused.
112
+ * 'u', 'unpause' # Unpauses Retest.
113
+ * <ENTER> # Runs last changed triggered command.
114
+ * 'ra, 'run all' # Runs all tests.
115
+ * 'f', 'force' # Forces a selection of test to run on every file change.
116
+ * 'r', 'reset' # Disables forced selection.
117
+ * 'd', 'diff' [GIT BRANCH] # Runs matching specs that changed from a target branch.
118
+ * 'e', 'exit' # Exits Retest.
103
119
  HELP
104
120
  else
105
121
  puts "Unknown interactive command #{input}\n"
@@ -108,13 +124,13 @@ end
108
124
 
109
125
  connections = [$stdin, listen_rd]
110
126
  loop do
111
- puts "\nType interactive command and press enter"
127
+ puts "\nType interactive command and press enter. Enter 'h' for help."
112
128
  print(">\s")
113
129
 
114
130
  ready = IO.select(connections)
115
131
  readable_connections = ready[0]
116
132
  readable_connections.each do |conn|
117
133
  data = conn.readpartial(4096)
118
- run_command(input: data.to_s.chomp, program: program, all_test_runner: all_test_runner)
134
+ run_command(input: data.to_s.chomp, program: program)
119
135
  end
120
136
  end
@@ -0,0 +1,53 @@
1
+ module Retest
2
+ class Command
3
+ class MultipleTestsNotSupported < StandardError; end
4
+
5
+ class Base
6
+ def initialize(all: false, file_system: FileSystem, command: nil)
7
+ @file_system = file_system
8
+ @all = all
9
+ @command = command
10
+ end
11
+
12
+ def clone(params = {})
13
+ self.class.new(**{ all: all, file_system: file_system, command: command }.merge(params))
14
+ end
15
+
16
+ def has_changed?
17
+ to_s.include?('<changed>')
18
+ end
19
+
20
+ def has_test?
21
+ to_s.include?('<test>')
22
+ end
23
+
24
+ def changed_type?
25
+ !has_test? && has_changed?
26
+ end
27
+
28
+ def test_type?
29
+ has_test? && !has_changed?
30
+ end
31
+
32
+ def variable_type?
33
+ has_test? && has_changed?
34
+ end
35
+
36
+ def hardcoded_type?
37
+ !has_test? && !has_changed?
38
+ end
39
+
40
+ def to_s
41
+ @command
42
+ end
43
+
44
+ def format_batch(*files)
45
+ raise MultipleTestsNotSupported, "Multiple test files run not supported for '#{to_s}'"
46
+ end
47
+
48
+ private
49
+
50
+ attr_reader :all, :file_system, :command
51
+ end
52
+ end
53
+ end
@@ -0,0 +1,5 @@
1
+ module Retest
2
+ class Command
3
+ class Hardcoded < Base; end
4
+ end
5
+ end
@@ -1,16 +1,12 @@
1
1
  module Retest
2
2
  class Command
3
- class Rails
4
- attr_reader :all, :file_system
5
-
6
- def initialize(all:, file_system: FileSystem)
7
- @file_system = file_system
8
- @all = all
9
- end
10
-
3
+ class Rails < Base
11
4
  def to_s
12
- return "#{root_command} <test>" unless all
13
- root_command
5
+ if all
6
+ root_command
7
+ else
8
+ "#{root_command} <test>"
9
+ end
14
10
  end
15
11
 
16
12
  def format_batch(*files)
@@ -20,9 +16,11 @@ module Retest
20
16
  private
21
17
 
22
18
  def root_command
23
- return 'bin/rails test' if file_system.exist? 'bin/rails'
24
-
25
- 'bundle exec rails test'
19
+ if file_system.exist? 'bin/rails'
20
+ 'bin/rails test'
21
+ else
22
+ 'bundle exec rails test'
23
+ end
26
24
  end
27
25
  end
28
26
  end
@@ -1,28 +1,26 @@
1
1
  module Retest
2
2
  class Command
3
- class Rake
4
- attr_reader :all, :file_system
5
-
6
- def initialize(all:, file_system: FileSystem)
7
- @file_system = file_system
8
- @all = all
9
- end
10
-
3
+ class Rake < Base
11
4
  def to_s
12
- return "#{root_command} TEST=<test>" unless all
13
- root_command
5
+ if all
6
+ root_command
7
+ else
8
+ "#{root_command} TEST=<test>"
9
+ end
14
10
  end
15
11
 
16
12
  def format_batch(*files)
17
- files.size > 1 ? "\"{#{files.join(',')}}\"" : files.first
13
+ files.size > 1 ? %Q{"{#{files.join(',')}}"} : files.first
18
14
  end
19
15
 
20
16
  private
21
17
 
22
18
  def root_command
23
- return 'bin/rake test' if file_system.exist? 'bin/rake'
24
-
25
- 'bundle exec rake test'
19
+ if file_system.exist? 'bin/rake'
20
+ 'bin/rake test'
21
+ else
22
+ 'bundle exec rake test'
23
+ end
26
24
  end
27
25
  end
28
26
  end
@@ -1,16 +1,12 @@
1
1
  module Retest
2
2
  class Command
3
- class Rspec
4
- attr_reader :all, :file_system
5
-
6
- def initialize(all:, file_system: FileSystem)
7
- @file_system = file_system
8
- @all = all
9
- end
10
-
3
+ class Rspec < Base
11
4
  def to_s
12
- return "#{root_command} <test>" unless all
13
- root_command
5
+ if all
6
+ root_command
7
+ else
8
+ "#{root_command} <test>"
9
+ end
14
10
  end
15
11
 
16
12
  def format_batch(*files)
@@ -20,9 +16,11 @@ module Retest
20
16
  private
21
17
 
22
18
  def root_command
23
- return 'bin/rspec' if file_system.exist? 'bin/rspec'
24
-
25
- 'bundle exec rspec'
19
+ if file_system.exist? 'bin/rspec'
20
+ 'bin/rspec'
21
+ else
22
+ 'bundle exec rspec'
23
+ end
26
24
  end
27
25
  end
28
26
  end
@@ -1,17 +1,6 @@
1
1
  module Retest
2
2
  class Command
3
- class Ruby
4
- attr_reader :all, :file_system
5
-
6
- def initialize(all:, file_system: FileSystem)
7
- @file_system = file_system
8
- @all = all
9
- end
10
-
11
- def format_batch(*files)
12
- %Q{-e "#{files.map { |file| "require './#{file}';" }.join}"}
13
- end
14
-
3
+ class Ruby < Base
15
4
  def to_s
16
5
  if file_system.exist? 'Gemfile.lock'
17
6
  'bundle exec ruby <test>'
@@ -19,6 +8,10 @@ module Retest
19
8
  'ruby <test>'
20
9
  end
21
10
  end
11
+
12
+ def format_batch(*files)
13
+ files.size > 1 ? %Q{-e "#{files.map { |file| "require './#{file}';" }.join}"} : files.first
14
+ end
22
15
  end
23
16
  end
24
17
  end
@@ -1,3 +1,5 @@
1
+ require_relative 'command/base'
2
+ require_relative 'command/hardcoded'
1
3
  require_relative 'command/rails'
2
4
  require_relative 'command/rake'
3
5
  require_relative 'command/rspec'
@@ -26,7 +28,9 @@ module Retest
26
28
  end
27
29
 
28
30
  def options_command
29
- return params[:command] if params[:command]
31
+ if params[:command]
32
+ return hardcoded_command(params[:command])
33
+ end
30
34
 
31
35
  if params[:rspec] then rspec_command
32
36
  elsif params[:rails] then rails_command
@@ -57,6 +61,10 @@ module Retest
57
61
  @stdout&.puts(message)
58
62
  end
59
63
 
64
+ def hardcoded_command(command)
65
+ Hardcoded.new(command: command)
66
+ end
67
+
60
68
  def rspec_command
61
69
  Rspec.new(all: full_suite?)
62
70
  end
@@ -63,10 +63,19 @@ module Retest
63
63
  long "--diff=git-branch"
64
64
  end
65
65
 
66
- option :ext do
67
- desc "Regex of file extensions to listen to"
68
- long "--ext=regex"
69
- default "\\.rb$"
66
+ option :exts do
67
+ desc "Comma separated of filenames extensions to filter to"
68
+ long "--exts=<EXTENSIONS>"
69
+ default "rb"
70
+ convert :list
71
+ end
72
+
73
+ option :watcher do
74
+ desc "Tool used to watch file events"
75
+ permit %i[listen watchexec]
76
+ long "--watcher=<WATCHER>"
77
+ short "-w"
78
+ convert :sym
70
79
  end
71
80
 
72
81
  flag :all do
@@ -155,8 +164,12 @@ module Retest
155
164
  params[:polling]
156
165
  end
157
166
 
158
- def extension
159
- Regexp.new(params[:ext])
167
+ def extensions
168
+ params[:exts]
169
+ end
170
+
171
+ def watcher
172
+ params[:watcher] || :installed
160
173
  end
161
174
 
162
175
  def merge(options = [])
@@ -0,0 +1,19 @@
1
+ module ForcedSelection
2
+ attr_reader :selected_test_files
3
+
4
+ def initialize_forced_selection(value = [])
5
+ @selected_test_files = value
6
+ end
7
+
8
+ def forced_selection?
9
+ !@selected_test_files.empty?
10
+ end
11
+
12
+ def reset_selection
13
+ @selected_test_files = []
14
+ end
15
+
16
+ def force_selection(test_files)
17
+ @selected_test_files = Array(test_files)
18
+ end
19
+ end
@@ -1,17 +1,24 @@
1
1
  require_relative 'program/pausable'
2
+ require_relative 'program/forced_selection'
2
3
 
3
4
  module Retest
4
5
  class Program
6
+ extend Forwardable
5
7
  include Pausable
8
+ include ForcedSelection
6
9
 
7
- attr_accessor :runner, :repository, :command, :stdout
8
- def initialize(runner: nil, repository: nil, command: nil, clear_window: true, stdout: $stdout)
10
+ attr_accessor :runner, :repository, :stdout
11
+
12
+ def_delegators :runner,
13
+ :run_last_command, :last_command
14
+
15
+ def initialize(runner: nil, repository: nil, clear_window: true, stdout: $stdout)
9
16
  @runner = runner
10
17
  @repository = repository
11
- @command = command
12
18
  @clear_window = clear_window
13
19
  @stdout = stdout
14
20
  initialize_pause(false)
21
+ initialize_forced_selection([])
15
22
  end
16
23
 
17
24
  def run(file, force_run: false)
@@ -20,19 +27,37 @@ module Retest
20
27
  return
21
28
  end
22
29
 
23
- clear_terminal
24
- runner.run file, repository: repository
30
+ if forced_selection?
31
+ @stdout.puts <<~HINT
32
+ Forced selection enabled.
33
+ Reset to default settings by typing 'r' in the interactive console.
34
+
35
+ HINT
36
+
37
+ runner.run(test_files: selected_test_files)
38
+ return
39
+ end
40
+
41
+ test_file = if runner.has_test?
42
+ repository.find_test(file)
43
+ end
44
+
45
+ runner.run changed_files: [file], test_files: [test_file]
25
46
  end
26
47
 
27
48
  def diff(branch)
28
49
  raise "Git not installed" unless VersionControl::Git.installed?
50
+
29
51
  test_files = repository.find_tests VersionControl::Git.diff_files(branch)
52
+ run_selected(test_files)
53
+ end
30
54
 
31
- @stdout.puts "Tests found:"
32
- test_files.each { |test_file| @stdout.puts " - #{test_file}" }
55
+ def run_all
56
+ runner.run_all
57
+ end
33
58
 
34
- @stdout.puts "Running tests..."
35
- runner.run_all_tests command.format_batch(*test_files)
59
+ def run_selected(test_files)
60
+ runner.run(test_files: test_files)
36
61
  end
37
62
 
38
63
  def clear_terminal
@@ -31,6 +31,10 @@ module Retest
31
31
  .sort
32
32
  end
33
33
 
34
+ def test_files
35
+ files.select { |file| MatchingOptions::Path.new(file).test? }
36
+ end
37
+
34
38
  def sync(added:, removed:)
35
39
  add(added)
36
40
  remove(removed)
@@ -0,0 +1,19 @@
1
+ module Retest
2
+ module CachedTestFile
3
+ attr_reader :cached_test_file
4
+
5
+ def cached_test_file=(value)
6
+ @cached_test_file = value || cached_test_file
7
+ end
8
+
9
+ def purge_test_file(purged)
10
+ return if purged.empty?
11
+
12
+ if purged.is_a?(Array) && purged.include?(cached_test_file)
13
+ @cached_test_file = nil
14
+ elsif purged.is_a?(String) && purged == cached_test_file
15
+ @cached_test_file = nil
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,98 @@
1
+ require_relative "runner/cached_test_file"
2
+
3
+ module Retest
4
+ class Runner
5
+ extend Forwardable
6
+ include Observable
7
+ include CachedTestFile
8
+
9
+ def_delegators :command,
10
+ :has_changed?, :has_test?,
11
+ :changed_type?, :test_type?, :variable_type?, :harcoded_type?
12
+
13
+ attr_accessor :command, :stdout, :last_command
14
+ def initialize(command, stdout: $stdout)
15
+ @stdout = stdout
16
+ @command = command
17
+ end
18
+
19
+ def run_last_command
20
+ system_run last_command
21
+ end
22
+
23
+ def run(changed_files: [], test_files: [])
24
+ self.last_command = format_instruction(changed_files: changed_files, test_files: test_files)
25
+ system_run last_command
26
+ rescue FileNotFound => e
27
+ log("FileNotFound - #{e.message}")
28
+ rescue Command::MultipleTestsNotSupported => e
29
+ log("Command::MultipleTestsNotSupported - #{e.message}")
30
+ end
31
+
32
+ def run_all
33
+ system_run command.clone(all: true).to_s
34
+ end
35
+
36
+ def format_instruction(changed_files: [], test_files: [])
37
+ if changed_files.empty? && test_files.size >= 1
38
+ instruction = command.clone(all: false).to_s
39
+ tests_string = command.format_batch(*test_files)
40
+ log("Tests selected:")
41
+ test_files.each { |test_file| log(" - #{test_file}") }
42
+ return instruction.gsub('<test>', tests_string)
43
+ end
44
+
45
+ instruction = command.to_s
46
+ instruction = format_changed_files(instruction: instruction, files: changed_files)
47
+ instruction = format_test_files(instruction: instruction, files: test_files)
48
+ end
49
+
50
+ def format_test_files(instruction:, files:)
51
+ return instruction unless has_test?
52
+
53
+ self.cached_test_file = files.first
54
+
55
+ if cached_test_file.nil?
56
+ raise FileNotFound, "Retest could not find a matching test file to run."
57
+ end
58
+
59
+ log("Test file: #{cached_test_file}")
60
+ instruction.gsub('<test>', cached_test_file)
61
+ end
62
+
63
+ def format_changed_files(instruction:, files:)
64
+ return instruction unless has_changed?
65
+ changed_file = files.first
66
+
67
+ if changed_file.nil?
68
+ raise FileNotFound, "Retest could not find a changed file to run."
69
+ end
70
+
71
+ log("Changed file: #{changed_file}")
72
+ instruction.gsub('<changed>', changed_file)
73
+ end
74
+
75
+ def sync(added:, removed:)
76
+ purge_test_file(removed)
77
+ end
78
+
79
+ private
80
+
81
+ def print_test_file_not_found
82
+ log(<<~ERROR)
83
+ FileNotFound - Retest could not find a matching test file to run.
84
+ ERROR
85
+ end
86
+
87
+ def system_run(command)
88
+ log("\n")
89
+ result = system(command) ? :tests_pass : :tests_fail
90
+ changed
91
+ notify_observers(result)
92
+ end
93
+
94
+ def log(message)
95
+ stdout.puts(message)
96
+ end
97
+ end
98
+ end
@@ -1,3 +1,3 @@
1
1
  module Retest
2
- VERSION = "2.0.0.pre3"
2
+ VERSION = "2.0.0.pre5"
3
3
  end
@@ -12,8 +12,12 @@ module Retest
12
12
  'git'
13
13
  end
14
14
 
15
- def files
16
- (untracked_files + tracked_files).sort
15
+ def files(extensions: [])
16
+ result = (untracked_files + tracked_files).sort
17
+ unless extensions.empty?
18
+ result.select! { |file| /\.(?:#{extensions.join('|')})$/.match?(file) }
19
+ end
20
+ result
17
21
  end
18
22
 
19
23
  def diff_files(branch)
@@ -12,8 +12,14 @@ module Retest
12
12
  'default'
13
13
  end
14
14
 
15
- def files
16
- Dir.glob('**/*') - Dir.glob('{tmp,node_modules}/**/*')
15
+ def files(extensions: [])
16
+ result = if extensions.empty?
17
+ Dir.glob('**/*')
18
+ else
19
+ Dir.glob("**/*.{#{extensions.join(',')}}")
20
+ end
21
+
22
+ result - Dir.glob('{tmp,node_modules}/**/*')
17
23
  end
18
24
  end
19
25
  end
@@ -6,8 +6,8 @@ module Retest
6
6
 
7
7
  module_function
8
8
 
9
- def files
10
- [Git, NoVersionControl].find(&:installed?).files
9
+ def files(extensions: [])
10
+ [Git, NoVersionControl].find(&:installed?).files(extensions: extensions)
11
11
  end
12
12
  end
13
13
  end
@@ -0,0 +1,122 @@
1
+ module Retest
2
+ module Watcher
3
+ def self.for(watcher)
4
+ tool = case watcher.to_s
5
+ when 'listen' then Default
6
+ when 'watchexec' then Watchexec
7
+ when '', 'installed' then installed
8
+ else raise ArgumentError, "Unknown #{watcher}"
9
+ end
10
+
11
+ unless tool.installed?
12
+ raise ArgumentError, "#{watcher} not installed on machine"
13
+ end
14
+
15
+ tool
16
+ end
17
+
18
+ def self.installed
19
+ [Watchexec, Default].find(&:installed?)
20
+ end
21
+
22
+ module Default
23
+ def self.installed?
24
+ true
25
+ end
26
+
27
+ def self.watch(dir:, extensions:, polling: false)
28
+ Listen.to(dir, only: extensions_regex(extensions), relative: true, polling: polling) do |modified, added, removed|
29
+ yield modified, added, removed
30
+ end.start
31
+ end
32
+
33
+ def self.extensions_regex(extensions)
34
+ Regexp.new("\\.(?:#{extensions.join("|")})$")
35
+ end
36
+ end
37
+
38
+ module Watchexec
39
+ def self.installed?
40
+ system "watchexec --version > /dev/null 2>&1"
41
+ end
42
+
43
+ def self.watch(dir:, extensions:, polling: false)
44
+ command = "watchexec --exts #{extensions.join(',')} -w #{dir} --emit-events-to stdio --no-meta --only-emit-events"
45
+ files = VersionControl.files(extensions: extensions).zip([]).to_h
46
+
47
+ watch_rd, watch_wr = IO.pipe
48
+ pid = Process.spawn(command, out: watch_wr)
49
+ at_exit do
50
+ Process.kill("TERM", pid) if pid
51
+ watch_rd.close
52
+ watch_wr.close
53
+ end
54
+
55
+ Thread.new do
56
+ loop do
57
+ ready = IO.select([watch_rd])
58
+ readable_connections = ready[0]
59
+ readable_connections.each do |conn|
60
+ data = conn.readpartial(4096)
61
+ change = /^(?:create|remove|rename|modify):(?<path>.*)/.match(data.strip)
62
+
63
+ next unless change
64
+
65
+ path = Pathname(change[:path]).relative_path_from(Dir.pwd).to_s
66
+ file_exist = File.exist?(path)
67
+ file_cached = files.key?(path)
68
+
69
+ modified, added, removed = result = [[], [], []]
70
+ if file_exist && file_cached
71
+ modified << path
72
+ elsif file_exist && !file_cached
73
+ added << path
74
+ files[path] = nil
75
+ elsif !file_exist && file_cached
76
+ removed << path
77
+ files.delete(path)
78
+ end
79
+
80
+ yield result
81
+ end
82
+ end
83
+ end
84
+
85
+ # require 'open3'
86
+ # Thread.new do
87
+ # files = VersionControl.files(extensions: extensions).zip([]).to_h
88
+
89
+ # Open3.popen3(command) do |stdin, stdout, stderr, wait_thr|
90
+ # loop do
91
+ # ready = IO.select([stdout])
92
+ # readable_connections = ready[0]
93
+ # readable_connections.each do |conn|
94
+ # data = conn.readpartial(4096)
95
+ # change = /^(?:create|remove|rename|modify):(?<path>.*)/.match(data.strip)
96
+
97
+ # next unless change
98
+
99
+ # path = Pathname(change[:path]).relative_path_from(Dir.pwd).to_s
100
+ # file_exist = File.exist?(path)
101
+ # file_cached = files.key?(path)
102
+
103
+ # modified, added, removed = result = [[], [], []]
104
+ # if file_exist && file_cached
105
+ # modified << path
106
+ # elsif file_exist && !file_cached
107
+ # added << path
108
+ # files[path] = nil
109
+ # elsif !file_exist && file_cached
110
+ # removed << path
111
+ # files.delete(path)
112
+ # end
113
+
114
+ # yield result
115
+ # end
116
+ # end
117
+ # end
118
+ # end
119
+ end
120
+ end
121
+ end
122
+ end
data/lib/retest.rb CHANGED
@@ -4,7 +4,7 @@ require 'string/similarity'
4
4
  require 'observer'
5
5
 
6
6
  require "retest/version"
7
- require "retest/runners"
7
+ require "retest/runner"
8
8
  require "retest/repository"
9
9
  require "retest/matching_options"
10
10
  require "retest/options"
@@ -15,15 +15,17 @@ require "retest/file_system"
15
15
  require "retest/program"
16
16
  require "retest/prompt"
17
17
  require "retest/sounds"
18
+ require "retest/watcher"
18
19
 
19
20
  Listen.adapter_warn_behavior = :log
20
21
 
21
22
  module Retest
22
23
  class Error < StandardError; end
24
+ class FileNotFound < StandardError; end
23
25
 
24
- def self.listen(options, listener: Listen)
25
- listener.to('.', only: options.extension, relative: true, force_polling: options.force_polling?) do |modified, added, removed|
26
+ def self.listen(options, listener: Watcher::Default)
27
+ listener.watch(dir: '.', extensions: options.extensions, polling: options.force_polling?) do |modified, added, removed|
26
28
  yield modified, added, removed
27
- end.start
29
+ end
28
30
  end
29
31
  end
data/retest.gemspec CHANGED
@@ -28,5 +28,6 @@ Gem::Specification.new do |spec|
28
28
  spec.add_runtime_dependency "string-similarity", ["~> 2.1"]
29
29
  spec.add_runtime_dependency "listen", ["~> 3.9"]
30
30
  spec.add_runtime_dependency "tty-option", ["~> 0.1"]
31
+ spec.add_runtime_dependency "tty-prompt", ["~> 0.1"]
31
32
  spec.add_runtime_dependency "observer", ["~> 0.1"]
32
33
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: retest
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.0.0.pre3
4
+ version: 2.0.0.pre5
5
5
  platform: ruby
6
6
  authors:
7
7
  - Alexandre Barret
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2024-10-28 00:00:00.000000000 Z
11
+ date: 2024-12-05 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: string-similarity
@@ -52,6 +52,20 @@ dependencies:
52
52
  - - "~>"
53
53
  - !ruby/object:Gem::Version
54
54
  version: '0.1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: tty-prompt
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '0.1'
62
+ type: :runtime
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '0.1'
55
69
  - !ruby/object:Gem::Dependency
56
70
  name: observer
57
71
  requirement: !ruby/object:Gem::Requirement
@@ -83,6 +97,7 @@ files:
83
97
  - README.md
84
98
  - README/demo.gif
85
99
  - Rakefile
100
+ - bin/build/watchexec
86
101
  - bin/console
87
102
  - bin/debug
88
103
  - bin/setup
@@ -95,9 +110,12 @@ files:
95
110
  - bin/test/rspec-ruby
96
111
  - bin/test/ruby-app
97
112
  - bin/test/ruby-bare
113
+ - builds/dockerfiles/WatchexecSlimBullseye
98
114
  - exe/retest
99
115
  - lib/retest.rb
100
116
  - lib/retest/command.rb
117
+ - lib/retest/command/base.rb
118
+ - lib/retest/command/hardcoded.rb
101
119
  - lib/retest/command/rails.rb
102
120
  - lib/retest/command/rake.rb
103
121
  - lib/retest/command/rspec.rb
@@ -107,21 +125,19 @@ files:
107
125
  - lib/retest/matching_options/path.rb
108
126
  - lib/retest/options.rb
109
127
  - lib/retest/program.rb
128
+ - lib/retest/program/forced_selection.rb
110
129
  - lib/retest/program/pausable.rb
111
130
  - lib/retest/prompt.rb
112
131
  - lib/retest/repository.rb
113
- - lib/retest/runners.rb
114
- - lib/retest/runners/cached_test_file.rb
115
- - lib/retest/runners/change_runner.rb
116
- - lib/retest/runners/runner.rb
117
- - lib/retest/runners/test_runner.rb
118
- - lib/retest/runners/variable_runner.rb
132
+ - lib/retest/runner.rb
133
+ - lib/retest/runner/cached_test_file.rb
119
134
  - lib/retest/setup.rb
120
135
  - lib/retest/sounds.rb
121
136
  - lib/retest/version.rb
122
137
  - lib/retest/version_control.rb
123
138
  - lib/retest/version_control/git.rb
124
139
  - lib/retest/version_control/no_version_control.rb
140
+ - lib/retest/watcher.rb
125
141
  - retest.gemspec
126
142
  homepage: https://github.com/AlexB52/retest
127
143
  licenses:
@@ -144,7 +160,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
144
160
  - !ruby/object:Gem::Version
145
161
  version: 1.3.1
146
162
  requirements: []
147
- rubygems_version: 3.1.6
163
+ rubygems_version: 3.0.3.1
148
164
  signing_key:
149
165
  specification_version: 4
150
166
  summary: A simple command line tool to watch file change and run its matching spec.
@@ -1,23 +0,0 @@
1
- module Retest
2
- module Runners
3
- module CachedTestFile
4
- def cached_test_file
5
- @cached_test_file
6
- end
7
-
8
- def cached_test_file=(value)
9
- @cached_test_file = value || @cached_test_file
10
- end
11
-
12
- def purge_test_file(purged)
13
- return if purged.empty?
14
-
15
- if purged.is_a?(Array) && purged.include?(cached_test_file)
16
- @cached_test_file = nil
17
- elsif purged.is_a?(String) && purged == cached_test_file
18
- @cached_test_file = nil
19
- end
20
- end
21
- end
22
- end
23
- end
@@ -1,21 +0,0 @@
1
- module Retest
2
- module Runners
3
- class ChangeRunner < Runner
4
- def run(changed_file = nil, repository: nil)
5
- return print_file_not_found unless changed_file
6
-
7
- log("Changed File Selected: #{changed_file}")
8
- system_run command.gsub('<changed>', changed_file)
9
- end
10
-
11
- private
12
-
13
- def print_file_not_found
14
- log(<<~ERROR)
15
- 404 - File Not Found
16
- Retest could not find a changed file to run.
17
- ERROR
18
- end
19
- end
20
- end
21
- end
@@ -1,46 +0,0 @@
1
- module Retest
2
- module Runners
3
- class Runner
4
- include Observable
5
-
6
- attr_accessor :command, :stdout
7
- def initialize(command, stdout: $stdout)
8
- @stdout = stdout
9
- @command = command
10
- end
11
-
12
- def ==(obj)
13
- command == obj.command && obj.class == self.class
14
- end
15
-
16
- def run(changed_file = nil, repository: nil)
17
- system_run command
18
- end
19
-
20
- def run_all_tests(tests_string)
21
- raise NotSupportedError, 'cannot run multiple test files against this command'
22
- end
23
-
24
- def sync(added:, removed:)
25
- end
26
-
27
- def running?
28
- @running
29
- end
30
-
31
- private
32
-
33
- def system_run(command)
34
- @running = true
35
- result = system(command) ? :tests_pass : :tests_fail
36
- changed
37
- notify_observers(result)
38
- @running = false
39
- end
40
-
41
- def log(message)
42
- stdout.puts(message)
43
- end
44
- end
45
- end
46
- end
@@ -1,36 +0,0 @@
1
- require_relative "cached_test_file"
2
-
3
- module Retest
4
- module Runners
5
- class TestRunner < Runner
6
- include CachedTestFile
7
-
8
- def run(changed_file, repository:)
9
- self.cached_test_file = repository.find_test(changed_file)
10
-
11
- return print_file_not_found unless cached_test_file
12
-
13
- log("Test File Selected: #{cached_test_file}")
14
- system_run command.gsub('<test>', cached_test_file)
15
- end
16
-
17
- def run_all_tests(tests_string)
18
- log("Test File Selected: #{tests_string}")
19
- system_run command.gsub('<test>', tests_string)
20
- end
21
-
22
- def sync(added:, removed:)
23
- purge_test_file(removed)
24
- end
25
-
26
- private
27
-
28
- def print_file_not_found
29
- log(<<~ERROR)
30
- 404 - Test File Not Found
31
- Retest could not find a matching test file to run.
32
- ERROR
33
- end
34
- end
35
- end
36
- end
@@ -1,39 +0,0 @@
1
- require_relative "cached_test_file"
2
-
3
- module Retest
4
- module Runners
5
- class VariableRunner < Runner
6
- include CachedTestFile
7
-
8
- def run(changed_file, repository:)
9
- self.cached_test_file = repository.find_test(changed_file)
10
-
11
- return print_file_not_found unless cached_test_file
12
-
13
- log(<<~FILES)
14
- Files Selected:
15
- - changed: #{changed_file}
16
- - test: #{cached_test_file}
17
-
18
- FILES
19
-
20
- system_run command
21
- .gsub('<test>', cached_test_file)
22
- .gsub('<changed>', changed_file)
23
- end
24
-
25
- def sync(added:, removed:)
26
- purge_test_file(removed)
27
- end
28
-
29
- private
30
-
31
- def print_file_not_found
32
- log(<<~ERROR)
33
- 404 - Test File Not Found
34
- Retest could not find a matching test file to run.
35
- ERROR
36
- end
37
- end
38
- end
39
- end
@@ -1,23 +0,0 @@
1
- require_relative 'runners/runner'
2
- require_relative 'runners/test_runner'
3
- require_relative 'runners/change_runner'
4
- require_relative 'runners/variable_runner'
5
-
6
- module Retest
7
- module Runners
8
- class NotSupportedError < StandardError; end
9
-
10
- module_function
11
-
12
- def runner_for(command, **opts)
13
- for_test = command.include?('<test>')
14
- for_change = command.include?('<changed>')
15
-
16
- if for_test && for_change then VariableRunner
17
- elsif for_test then TestRunner
18
- elsif for_change then ChangeRunner
19
- else Runner
20
- end.new command, **opts
21
- end
22
- end
23
- end