retest 2.0.1 → 2.2.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: 4173047ece3fb83e7ffc7e9cf871266054fc3dc49974f55a7df1afa6957afd73
4
- data.tar.gz: c4c5a285c4d2f9195b289ca6746d1adba4f77f3fd76d5006c70b10c003fa8a2d
3
+ metadata.gz: af4a7f8046c06b50e0d59639373c1cd9fac41a99f3a68a7529b98095b2ef10ae
4
+ data.tar.gz: 93e64c8ac28fe1ae32bc79e5719a8a775c5524544a0c7bd96ede1a3668dd8c6c
5
5
  SHA512:
6
- metadata.gz: 9d66970c4c9f952d936cd34fc5291ea7bb8a0f2366d90b2c2f647991b4e286936c1c8e37a8dcca8151954c8bd353dbe79321b23ea22e6a2ecf8a39b1ae2dcaf2
7
- data.tar.gz: 3c172835d39561da42eee00fc99a6561f57854d6f9f6fb59c50613ac8619d8ee56a4349404d45228f87ba37bd95a6fec39b4b445cc81c0fee958c6aaa972fef8
6
+ metadata.gz: b7142bef6579a44f0534253113f4646823a83fbe50b9417cdb0009ba9d99d01fe800fbf93500657343a626cd60efac2848852b1527f355144d901f128d8ea45d
7
+ data.tar.gz: 48c33eb530ca3752430afd48b3968fb8359b6fb88007e304b7b47b14da2d7283a4401f46d034429623efd804165c26e0928734b541f3f223f2f9f8ba12db7e96
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- retest (2.0.1)
4
+ retest (2.2.0)
5
5
  listen (~> 3.9)
6
6
  observer (~> 0.1)
7
7
  string-similarity (~> 2.1)
data/README.md CHANGED
@@ -48,8 +48,10 @@ retest --diff origin/main # Test changes from a branch
48
48
  Stay in control with an interactive shell for test management. Start Retest and enter `h` to explore available commands.
49
49
 
50
50
  ```
51
- Setup identified: [RAKE]. Using command: 'bundle exec rake test TEST=<test>'
51
+ Setup: [RAKE]
52
+ Command: 'bundle exec rake test TEST=<test>'
52
53
  Watcher: [WATCHEXEC]
54
+
53
55
  Launching Retest...
54
56
  Ready to refactor! You can make file changes now
55
57
 
data/bin/test/bundler-app CHANGED
@@ -4,10 +4,10 @@ FOLDER="features/bundler-app"
4
4
 
5
5
  bundle install
6
6
  bundle exec rake build
7
- cp -R features/support features/bundler-app/retest
8
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} "$FOLDER/retest.gem"
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
9
 
10
- if [[ "$1" == "--no-build" ]]; then
10
+ if [[ "$1" == "--nobuild" ]]; then
11
11
  docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
12
  else
13
13
  docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
data/bin/test/git-ruby CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/git-ruby"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/git-ruby/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/git-ruby/retest.gem
7
- docker compose -f features/git-ruby/docker-compose.yml up --build --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/bin/test/hanami-app CHANGED
@@ -1,9 +1,17 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/hanami-app"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/hanami-app/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/hanami-app/retest.gem
7
- docker compose -f features/hanami-app/docker-compose.yml build
8
- docker compose -f features/hanami-app/docker-compose.yml run retest sh bin/test_setup
9
- docker compose -f features/hanami-app/docker-compose.yml up --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f $FOLDER/docker-compose.yml run retest sh bin/test_setup
12
+ docker compose -f $FOLDER/docker-compose.yml up --exit-code-from retest
13
+ else
14
+ docker compose -f $FOLDER/docker-compose.yml build
15
+ docker compose -f $FOLDER/docker-compose.yml run retest sh bin/test_setup
16
+ docker compose -f $FOLDER/docker-compose.yml up --exit-code-from retest
17
+ fi
data/bin/test/rails-app CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/rails-app"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/rails-app/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/rails-app/retest.gem
7
- docker compose -f features/rails-app/docker-compose.yml up --build --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/bin/test/rspec-rails CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/rspec-rails"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/rspec-rails/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/rspec-rails/retest.gem
7
- docker compose -f features/rspec-rails/docker-compose.yml up --build --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/bin/test/rspec-ruby CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/rspec-ruby"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/rspec-ruby/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/rspec-ruby/retest.gem
7
- docker compose -f features/rspec-ruby/docker-compose.yml up --build --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/bin/test/ruby-app CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/ruby-app"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/ruby-app/retest
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
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/bin/test/ruby-bare CHANGED
@@ -1,7 +1,14 @@
1
1
  #!/usr/bin/env bash
2
2
 
3
+ FOLDER="features/ruby-bare"
4
+
3
5
  bundle install
4
6
  bundle exec rake build
5
- cp -R features/support features/ruby-bare/retest
6
- ls -t pkg | head -n1 | xargs -I {} mv pkg/{} features/ruby-bare/retest.gem
7
- docker compose -f features/ruby-bare/docker-compose.yml up --build --exit-code-from retest
7
+ cp -R features/support $FOLDER/retest
8
+ ls -t pkg | head -n1 | xargs -I {} mv pkg/{} $FOLDER/retest.gem
9
+
10
+ if [[ "$1" == "--nobuild" ]]; then
11
+ docker compose -f "$FOLDER/docker-compose.yml" up --exit-code-from retest
12
+ else
13
+ docker compose -f "$FOLDER/docker-compose.yml" up --build --exit-code-from retest
14
+ fi
data/exe/retest CHANGED
@@ -4,12 +4,6 @@ require 'retest'
4
4
 
5
5
  $stdout.sync = true
6
6
  listen_rd, listen_wr = IO.pipe
7
- Signal.trap(:INT) do
8
- puts "Goodbye"
9
- listen_rd.close
10
- listen_wr.close
11
- exit
12
- end
13
7
 
14
8
  options = Retest::Options.new(ARGV)
15
9
 
@@ -39,25 +33,48 @@ program = Retest::Program.new(
39
33
  runner: runner
40
34
  )
41
35
 
36
+ # === LOGGING ===
37
+ case command
38
+ when Retest::Command::Rails then puts "Setup: [RAILS]"
39
+ when Retest::Command::Rspec then puts "Setup: [RSPEC]"
40
+ when Retest::Command::Rake then puts "Setup: [RAKE]"
41
+ when Retest::Command::Ruby then puts "Setup: [RUBY]"
42
+ else puts "Setup: [UNKNOWN]"
43
+ end
44
+
45
+ puts "Command: '#{command}'"
46
+
47
+ # === TRAP INTERRUPTION ===
48
+ Signal.trap(:INT) do
49
+ if !runner.interrupt_run
50
+ puts "Goodbye"
51
+ listen_rd.close
52
+ listen_wr.close
53
+ exit
54
+ end
55
+ end
56
+
57
+ # === DIFF ACTION ===
42
58
  if options.params[:diff]
43
59
  program.diff(options.params[:diff])
44
60
  return
45
61
  end
46
62
 
63
+ # === LOGGING ===
47
64
  if watcher == Retest::Watcher::Watchexec
48
65
  puts "Watcher: [WATCHEXEC]"
49
66
  else
50
67
  puts "Watcher: [LISTEN]"
51
68
  end
52
69
 
53
- launching_message = "Launching Retest..."
70
+ launching_message = "\nLaunching Retest..."
54
71
  if options.force_polling?
55
72
  launching_message = "Launching Retest with polling method..."
56
73
  end
57
74
 
58
- # Main action
59
-
60
75
  puts launching_message
76
+
77
+ # === MAIN ACTION ===
61
78
  Retest.listen(options, listener: watcher) do |modified, added, removed|
62
79
  begin
63
80
  repository.sync(added: added, removed: removed)
@@ -85,7 +102,7 @@ def run_command(input:, program:)
85
102
  Process.kill("INT", 0)
86
103
  when 'r', 'reset'
87
104
  program.reset_selection
88
- puts "command reset to '#{program.runner.command.to_s}'"
105
+ puts "command reset to '#{program.runner.command}'"
89
106
  when 'f', 'force'
90
107
  require 'tty-prompt'
91
108
  prompt = TTY::Prompt.new
@@ -1,16 +1,35 @@
1
1
  module Retest
2
2
  class Command
3
3
  class MultipleTestsNotSupported < StandardError; end
4
+ class AllTestsNotSupported < StandardError; end
4
5
 
5
6
  class Base
7
+ attr_reader :file_system, :command
8
+
6
9
  def initialize(all: false, file_system: FileSystem, command: nil)
7
10
  @file_system = file_system
8
- @all = all
9
- @command = command
11
+ @command = command || default_command(all: all)
10
12
  end
11
13
 
12
- def clone(params = {})
13
- self.class.new(**{ all: all, file_system: file_system, command: command }.merge(params))
14
+ def to_s
15
+ all ? all_command : batched_command
16
+ end
17
+
18
+ def eql?(other)
19
+ hash == other.hash
20
+ end
21
+ alias == eql?
22
+
23
+ def hash
24
+ [self.class, file_system, command].hash
25
+ end
26
+
27
+ def switch_to(type)
28
+ case type.to_s
29
+ when 'all' then clone(command: all_command)
30
+ when 'batched' then clone(command: batched_command)
31
+ else raise ArgumentError, "unknown type to switch to: #{type}"
32
+ end
14
33
  end
15
34
 
16
35
  def hardcoded?
@@ -18,24 +37,42 @@ module Retest
18
37
  end
19
38
 
20
39
  def has_changed?
21
- to_s.include?('<changed>')
40
+ command.include?('<changed>')
22
41
  end
23
42
 
24
43
  def has_test?
25
- to_s.include?('<test>')
26
- end
27
-
28
- def to_s
29
- @command
44
+ command.include?('<test>')
30
45
  end
31
46
 
32
47
  def format_batch(*files)
33
- raise MultipleTestsNotSupported, "Multiple test files run not supported for '#{to_s}'"
48
+ raise_multiple_test_not_supported
34
49
  end
35
50
 
36
51
  private
37
52
 
38
- attr_reader :all, :file_system, :command
53
+ def all
54
+ !has_test?
55
+ end
56
+
57
+ def batched_command
58
+ raise NotImplementedError, 'must define a BATCHED command'
59
+ end
60
+
61
+ def all_command
62
+ raise NotImplementedError, 'must define a ALL command'
63
+ end
64
+
65
+ def default_command(all: false)
66
+ raise NotImplementedError, 'must define a DEFAULT command'
67
+ end
68
+
69
+ def clone(params = {})
70
+ self.class.new(**{ all: all, file_system: file_system, command: command }.merge(params))
71
+ end
72
+
73
+ def raise_multiple_test_not_supported
74
+ raise MultipleTestsNotSupported, "Multiple test files run not supported for command: '#{to_s}'"
75
+ end
39
76
  end
40
77
  end
41
78
  end
@@ -1,5 +1,18 @@
1
1
  module Retest
2
2
  class Command
3
- class Hardcoded < Base; end
3
+ class Hardcoded < Base
4
+ private
5
+
6
+ def all_command
7
+ command
8
+ end
9
+
10
+ def batched_command
11
+ command
12
+ end
13
+
14
+ def default_command(all: false)
15
+ end
16
+ end
4
17
  end
5
18
  end
@@ -1,26 +1,17 @@
1
1
  module Retest
2
2
  class Command
3
- class Rails < Base
4
- def to_s
5
- if all
6
- root_command
7
- else
8
- "#{root_command} <test>"
9
- end
10
- end
11
-
12
- def format_batch(*files)
13
- files.join(' ')
14
- end
3
+ class Rails < Rspec
15
4
 
16
5
  private
17
6
 
18
- def root_command
19
- if file_system.exist? 'bin/rails'
7
+ def default_command(all:)
8
+ command = if file_system&.exist? 'bin/rails'
20
9
  'bin/rails test'
21
10
  else
22
11
  'bundle exec rails test'
23
12
  end
13
+
14
+ all ? command : "#{command} <test>"
24
15
  end
25
16
  end
26
17
  end
@@ -1,26 +1,32 @@
1
1
  module Retest
2
2
  class Command
3
3
  class Rake < Base
4
- def to_s
5
- if all
6
- root_command
7
- else
8
- "#{root_command} TEST=<test>"
9
- end
10
- end
11
-
12
4
  def format_batch(*files)
13
5
  files.size > 1 ? %Q{"{#{files.join(',')}}"} : files.first
14
6
  end
15
7
 
16
8
  private
17
9
 
18
- def root_command
19
- if file_system.exist? 'bin/rake'
10
+ def default_command(all: false)
11
+ command = if file_system&.exist? 'bin/rake'
20
12
  'bin/rake test'
21
13
  else
22
14
  'bundle exec rake test'
23
15
  end
16
+
17
+ all ? command : "#{command} TEST=<test>"
18
+ end
19
+
20
+ def all_command
21
+ return command if all
22
+
23
+ command.gsub('TEST=<test>', '').strip
24
+ end
25
+
26
+ def batched_command
27
+ return command unless all
28
+
29
+ "#{command} TEST=<test>"
24
30
  end
25
31
  end
26
32
  end
@@ -1,26 +1,32 @@
1
1
  module Retest
2
2
  class Command
3
3
  class Rspec < Base
4
- def to_s
5
- if all
6
- root_command
7
- else
8
- "#{root_command} <test>"
9
- end
10
- end
11
-
12
4
  def format_batch(*files)
13
5
  files.join(' ')
14
6
  end
15
7
 
16
8
  private
17
9
 
18
- def root_command
19
- if file_system.exist? 'bin/rspec'
10
+ def all_command
11
+ return command if all
12
+
13
+ command.gsub('<test>', '').strip
14
+ end
15
+
16
+ def batched_command
17
+ return command unless all
18
+
19
+ "#{command} <test>"
20
+ end
21
+
22
+ def default_command(all:)
23
+ command = if file_system&.exist? 'bin/rspec'
20
24
  'bin/rspec'
21
25
  else
22
26
  'bundle exec rspec'
23
27
  end
28
+
29
+ all ? command : "#{command} <test>"
24
30
  end
25
31
  end
26
32
  end
@@ -1,16 +1,26 @@
1
1
  module Retest
2
2
  class Command
3
3
  class Ruby < Base
4
- def to_s
5
- if file_system.exist? 'Gemfile.lock'
4
+ def format_batch(*files)
5
+ files.size > 1 ? %Q{-e "#{files.map { |file| "require './#{file}';" }.join}"} : files.first
6
+ end
7
+
8
+ private
9
+
10
+ def default_command(all: false)
11
+ if file_system&.exist? 'Gemfile.lock'
6
12
  'bundle exec ruby <test>'
7
13
  else
8
14
  'ruby <test>'
9
15
  end
10
16
  end
11
17
 
12
- def format_batch(*files)
13
- files.size > 1 ? %Q{-e "#{files.map { |file| "require './#{file}';" }.join}"} : files.first
18
+ def all_command
19
+ raise AllTestsNotSupported, "All tests run not supported for Ruby command: '#{to_s}'"
20
+ end
21
+
22
+ def batched_command
23
+ command
14
24
  end
15
25
  end
16
26
  end
@@ -1,50 +1,46 @@
1
1
  require_relative 'command/base'
2
2
  require_relative 'command/hardcoded'
3
+ require_relative 'command/rspec'
3
4
  require_relative 'command/rails'
4
5
  require_relative 'command/rake'
5
- require_relative 'command/rspec'
6
6
  require_relative 'command/ruby'
7
7
 
8
8
  module Retest
9
9
  class Command
10
10
  extend Forwardable
11
11
 
12
- def self.for_options(options, stdout: $stdout)
13
- new(options: options, stdout: stdout).command
12
+ def self.for_options(options)
13
+ new(options: options).command
14
14
  end
15
15
 
16
- def_delegator :setup, :type
17
16
  def_delegators :options, :params, :full_suite?, :auto?
18
17
 
19
- attr_accessor :options, :setup
20
- def initialize(options: Options.new, setup: Setup.new, stdout: $stdout)
18
+ attr_accessor :options, :setup, :setup_identified
19
+ def initialize(options: Options.new, setup: Setup.new)
21
20
  @options = options
22
21
  @setup = setup
23
- @stdout = stdout
22
+ @setup_identified = nil
24
23
  end
25
24
 
26
25
  def command
27
- options_command || default_command
26
+ options_command || setup_command
28
27
  end
29
28
 
30
29
  private
31
30
 
32
31
  def options_command
33
- if params[:command] then hardcoded_command(params[:command])
34
- elsif params[:rspec] then rspec_command
35
- elsif params[:rails] then rails_command
36
- elsif params[:ruby] then ruby_command
37
- elsif params[:rake] then rake_command
38
- end
39
- end
32
+ command_input = params[:command]
40
33
 
41
- def default_command
42
- log "Setup identified: [#{type.upcase}]. Using command: '#{setup_command}'"
43
- setup_command
34
+ if params[:rspec] then rspec_command(command: command_input)
35
+ elsif params[:rails] then rails_command(command: command_input)
36
+ elsif params[:ruby] then ruby_command(command: command_input)
37
+ elsif params[:rake] then rake_command(command: command_input)
38
+ elsif command_input then hardcoded_command(command: command_input)
39
+ end
44
40
  end
45
41
 
46
42
  def setup_command
47
- case type
43
+ case setup.type
48
44
  when :rake then rake_command
49
45
  when :rspec then rspec_command
50
46
  when :rails then rails_command
@@ -53,28 +49,24 @@ module Retest
53
49
  end
54
50
  end
55
51
 
56
- def log(message)
57
- @stdout&.puts(message)
58
- end
59
-
60
- def hardcoded_command(command)
52
+ def hardcoded_command(command:)
61
53
  Hardcoded.new(command: command)
62
54
  end
63
55
 
64
- def rspec_command
65
- Rspec.new(all: full_suite?)
56
+ def rspec_command(command: nil)
57
+ Rspec.new(all: full_suite?, command: command)
66
58
  end
67
59
 
68
- def rails_command
69
- Rails.new(all: full_suite?)
60
+ def rails_command(command: nil)
61
+ Rails.new(all: full_suite?, command: command)
70
62
  end
71
63
 
72
- def rake_command
73
- Rake.new(all: full_suite?)
64
+ def rake_command(command: nil)
65
+ Rake.new(all: full_suite?, command: command)
74
66
  end
75
67
 
76
- def ruby_command
77
- Ruby.new(all: full_suite?)
68
+ def ruby_command(command: nil)
69
+ Ruby.new(all: full_suite?, command: command)
78
70
  end
79
71
  end
80
72
  end
@@ -30,7 +30,6 @@ module Retest
30
30
  @stdout.puts <<~HINT
31
31
  Forced selection enabled.
32
32
  Reset to default settings by typing 'r' in the interactive console.
33
-
34
33
  HINT
35
34
 
36
35
  runner.run(test_files: selected_test_files)
@@ -48,17 +47,13 @@ module Retest
48
47
  raise "Git not installed" unless VersionControl::Git.installed?
49
48
 
50
49
  test_files = repository.find_tests VersionControl::Git.diff_files(branch)
51
- run_selected(test_files)
50
+ runner.run(test_files: test_files)
52
51
  end
53
52
 
54
53
  def run_all
55
54
  runner.run_all
56
55
  end
57
56
 
58
- def run_selected(test_files)
59
- runner.run(test_files: test_files)
60
- end
61
-
62
57
  def clear_terminal
63
58
  system('clear 2>/dev/null') || system('cls 2>/dev/null')
64
59
  end
data/lib/retest/runner.rb CHANGED
@@ -1,3 +1,4 @@
1
+ require 'forwardable'
1
2
  require_relative "runner/cached_test_file"
2
3
 
3
4
  module Retest
@@ -17,6 +18,14 @@ module Retest
17
18
  end
18
19
  end
19
20
 
21
+ def interrupt_run
22
+ return false unless @pid
23
+
24
+ Process.kill('INT', @pid)
25
+ rescue Errno::ESRCH
26
+ false
27
+ end
28
+
20
29
  def run_last_command
21
30
  unless last_command
22
31
  return log('Error - Not enough information to run a command. Please trigger a run first.')
@@ -35,17 +44,20 @@ module Retest
35
44
  end
36
45
 
37
46
  def run_all
38
- self.last_command = command.clone(all: true).to_s
47
+ self.last_command = command.switch_to(:all).to_s
39
48
  run_last_command
49
+ rescue Command::AllTestsNotSupported => e
50
+ log("Command::AllTestsNotSupported - #{e.message}")
40
51
  end
41
52
 
42
53
  def format_instruction(changed_files: [], test_files: [])
43
54
  if changed_files.empty? && test_files.size >= 1
44
- instruction = command.clone(all: false).to_s
45
- tests_string = command.format_batch(*test_files)
46
- log("Tests selected:")
55
+ new_command = command.switch_to(:batched)
56
+
57
+ log("\nTests selected:")
47
58
  test_files.each { |test_file| log(" - #{test_file}") }
48
- return instruction.gsub('<test>', tests_string)
59
+
60
+ return new_command.to_s.gsub('<test>', new_command.format_batch(*test_files))
49
61
  end
50
62
 
51
63
  instruction = command.to_s
@@ -86,7 +98,10 @@ module Retest
86
98
 
87
99
  def system_run(command)
88
100
  log("\n")
89
- result = system(command) ? :tests_pass : :tests_fail
101
+ @pid = spawn(command)
102
+ Process.wait
103
+ @pid = nil
104
+ result = $?.exitstatus&.zero? ? :tests_pass : :tests_fail
90
105
  changed
91
106
  notify_observers(result)
92
107
  end
@@ -1,3 +1,3 @@
1
1
  module Retest
2
- VERSION = "2.0.1"
2
+ VERSION = "2.2.0"
3
3
  end
@@ -21,7 +21,7 @@ module Retest
21
21
  end
22
22
 
23
23
  def diff_files(branch)
24
- `git diff #{branch} --name-only --diff-filter=ACMRT -z`.split("\x0")
24
+ `git diff #{branch}...HEAD --name-only --diff-filter=ACMRT -z`.split("\x0")
25
25
  end
26
26
 
27
27
  def untracked_files
@@ -1,3 +1,5 @@
1
+ require 'pathname'
2
+
1
3
  module Retest
2
4
  module Watcher
3
5
  def self.for(watcher)
@@ -24,15 +26,49 @@ module Retest
24
26
  true
25
27
  end
26
28
 
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
29
  def self.extensions_regex(extensions)
34
30
  Regexp.new("\\.(?:#{extensions.join("|")})$")
35
31
  end
32
+
33
+ def self.watch(dir:, extensions:, polling: false)
34
+ executable = File.expand_path("../../scripts/listen", __FILE__)
35
+ command = "#{executable} --exts #{extensions.join(',')} -w #{dir} --polling #{polling}"
36
+
37
+ watch_rd, watch_wr = IO.pipe
38
+ # Process needs its own process group otherwise the process gets killed on INT signal
39
+ # We need the process to still run when trying to stop the current test run
40
+ # Maybe there is another way to prevent killing these but for now a new process groups works
41
+ # Process group created with: pgroup: true
42
+ pid = Process.spawn(command, out: watch_wr, pgroup: true)
43
+
44
+ at_exit do
45
+ Process.kill("TERM", pid) if pid
46
+ watch_rd.close
47
+ watch_wr.close
48
+ end
49
+
50
+ Thread.new do
51
+ loop do
52
+ ready = IO.select([watch_rd])
53
+ readable_connections = ready[0]
54
+ readable_connections.each do |conn|
55
+ data = conn.readpartial(4096)
56
+ change = /^(?<action>create|remove|modify):(?<path>.*)/.match(data.strip)
57
+
58
+ next unless change
59
+
60
+ modified, added, removed = result = [[], [], []]
61
+ case change[:action]
62
+ when 'modify' then modified << change[:path]
63
+ when 'create' then added << change[:path]
64
+ when 'remove' then removed << change[:path]
65
+ end
66
+
67
+ yield result
68
+ end
69
+ end
70
+ end
71
+ end
36
72
  end
37
73
 
38
74
  module Watchexec
@@ -42,10 +78,14 @@ module Retest
42
78
 
43
79
  def self.watch(dir:, extensions:, polling: false)
44
80
  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
81
 
47
82
  watch_rd, watch_wr = IO.pipe
48
- pid = Process.spawn(command, out: watch_wr)
83
+ # Process needs its own process group otherwise the process gets killed on INT signal
84
+ # We need the process to still run when trying to stop the current test run
85
+ # Maybe there is another way to prevent killing these but for now a new process groups works
86
+ # Process group created with: pgroup: true
87
+ pid = Process.spawn(command, out: watch_wr, pgroup: true)
88
+
49
89
  at_exit do
50
90
  Process.kill("TERM", pid) if pid
51
91
  watch_rd.close
@@ -53,11 +93,15 @@ module Retest
53
93
  end
54
94
 
55
95
  Thread.new do
96
+ files = VersionControl.files(extensions: extensions).zip([]).to_h
97
+
56
98
  loop do
57
99
  ready = IO.select([watch_rd])
58
100
  readable_connections = ready[0]
59
101
  readable_connections.each do |conn|
60
102
  data = conn.readpartial(4096)
103
+ # Watchexec is not great at figuring out whether a file has been deleted and comes as an update.
104
+ # This is why we're not looking at the action like we do with Listen.
61
105
  change = /^(?:create|remove|rename|modify):(?<path>.*)/.match(data.strip)
62
106
 
63
107
  next unless change
@@ -81,41 +125,6 @@ module Retest
81
125
  end
82
126
  end
83
127
  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
128
  end
120
129
  end
121
130
  end
data/lib/retest.rb CHANGED
@@ -1,5 +1,3 @@
1
- require 'listen'
2
-
3
1
  require 'string/similarity'
4
2
  require 'observer'
5
3
 
@@ -17,8 +15,6 @@ require "retest/prompt"
17
15
  require "retest/sounds"
18
16
  require "retest/watcher"
19
17
 
20
- Listen.adapter_warn_behavior = :log
21
-
22
18
  module Retest
23
19
  class Error < StandardError; end
24
20
  class FileNotFound < StandardError; end
@@ -0,0 +1,57 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ $stdout.sync = true
4
+
5
+ require 'listen'
6
+ require 'optparse'
7
+
8
+ options = {}
9
+ OptionParser.new do |opts|
10
+ opts.banner = "Usage: scripts/listen.rb [options]"
11
+
12
+ opts.on("--exts rb,js,ts", Array, "Extensions to watch for") do |list|
13
+ options[:extensions] = list
14
+ end
15
+
16
+ opts.on("--polling BOOLEAN", "Force Listen to use polling") do |value|
17
+ options[:polling] = value
18
+ end
19
+
20
+ opts.on("-w", "--watch .", "Directory to listen to") do |value|
21
+ options[:dir] = value
22
+ end
23
+
24
+ opts.on("-h", "--help", "Prints help") do
25
+ puts opts
26
+ exit
27
+ end
28
+ end.parse!
29
+
30
+ unless options.key?(:extensions)
31
+ raise ArgumentError, 'must provide the files extensions to watch for'
32
+ end
33
+
34
+ unless options.key?(:polling)
35
+ raise ArgumentError, 'must provide the polling option'
36
+ end
37
+
38
+ unless options.key?(:dir)
39
+ raise ArgumentError, 'must provide the directory path to watch'
40
+ end
41
+
42
+ def extensions_regex(extensions)
43
+ Regexp.new("\\.(?:#{extensions.join("|")})$")
44
+ end
45
+
46
+ Listen.adapter_warn_behavior = :log
47
+
48
+ Listen.to(options[:dir], only: extensions_regex(options[:extensions]), relative: true, polling: options[:polling]) do |modified, added, removed|
49
+ if modified.any?
50
+ $stdout.puts "modify:#{modified.first}"
51
+ elsif added.any?
52
+ $stdout.puts "create:#{added.first}"
53
+ elsif removed.any?
54
+ $stdout.puts "remove:#{removed.first}"
55
+ end
56
+ end.start
57
+ sleep
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.1
4
+ version: 2.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Alexandre Barret
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2025-01-04 00:00:00.000000000 Z
11
+ date: 2025-03-07 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: string-similarity
@@ -138,6 +138,7 @@ files:
138
138
  - lib/retest/version_control/git.rb
139
139
  - lib/retest/version_control/no_version_control.rb
140
140
  - lib/retest/watcher.rb
141
+ - lib/scripts/listen
141
142
  - retest.gemspec
142
143
  homepage: https://github.com/AlexB52/retest
143
144
  licenses: