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 +4 -4
- data/Gemfile.lock +1 -1
- data/README.md +3 -1
- data/bin/test/bundler-app +3 -3
- data/bin/test/git-ruby +10 -3
- data/bin/test/hanami-app +13 -5
- data/bin/test/rails-app +10 -3
- data/bin/test/rspec-rails +10 -3
- data/bin/test/rspec-ruby +10 -3
- data/bin/test/ruby-app +10 -3
- data/bin/test/ruby-bare +10 -3
- data/exe/retest +27 -10
- data/lib/retest/command/base.rb +49 -12
- data/lib/retest/command/hardcoded.rb +14 -1
- data/lib/retest/command/rails.rb +5 -14
- data/lib/retest/command/rake.rb +16 -10
- data/lib/retest/command/rspec.rb +16 -10
- data/lib/retest/command/ruby.rb +14 -4
- data/lib/retest/command.rb +24 -32
- data/lib/retest/program.rb +1 -6
- data/lib/retest/runner.rb +21 -6
- data/lib/retest/version.rb +1 -1
- data/lib/retest/version_control/git.rb +1 -1
- data/lib/retest/watcher.rb +52 -43
- data/lib/retest.rb +0 -4
- data/lib/scripts/listen +57 -0
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: af4a7f8046c06b50e0d59639373c1cd9fac41a99f3a68a7529b98095b2ef10ae
|
4
|
+
data.tar.gz: 93e64c8ac28fe1ae32bc79e5719a8a775c5524544a0c7bd96ede1a3668dd8c6c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: b7142bef6579a44f0534253113f4646823a83fbe50b9417cdb0009ba9d99d01fe800fbf93500657343a626cd60efac2848852b1527f355144d901f128d8ea45d
|
7
|
+
data.tar.gz: 48c33eb530ca3752430afd48b3968fb8359b6fb88007e304b7b47b14da2d7283a4401f46d034429623efd804165c26e0928734b541f3f223f2f9f8ba12db7e96
|
data/Gemfile.lock
CHANGED
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
|
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
|
8
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
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" == "--
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
8
|
-
|
9
|
-
docker compose -f
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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
|
6
|
-
ls -t pkg | head -n1 | xargs -I {} mv pkg/{}
|
7
|
-
|
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 = "
|
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
|
105
|
+
puts "command reset to '#{program.runner.command}'"
|
89
106
|
when 'f', 'force'
|
90
107
|
require 'tty-prompt'
|
91
108
|
prompt = TTY::Prompt.new
|
data/lib/retest/command/base.rb
CHANGED
@@ -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
|
-
@
|
9
|
-
@command = command
|
11
|
+
@command = command || default_command(all: all)
|
10
12
|
end
|
11
13
|
|
12
|
-
def
|
13
|
-
|
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
|
-
|
40
|
+
command.include?('<changed>')
|
22
41
|
end
|
23
42
|
|
24
43
|
def has_test?
|
25
|
-
|
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
|
-
|
48
|
+
raise_multiple_test_not_supported
|
34
49
|
end
|
35
50
|
|
36
51
|
private
|
37
52
|
|
38
|
-
|
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
|
data/lib/retest/command/rails.rb
CHANGED
@@ -1,26 +1,17 @@
|
|
1
1
|
module Retest
|
2
2
|
class Command
|
3
|
-
class Rails <
|
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
|
19
|
-
if file_system
|
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
|
data/lib/retest/command/rake.rb
CHANGED
@@ -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
|
19
|
-
if file_system
|
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
|
data/lib/retest/command/rspec.rb
CHANGED
@@ -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
|
19
|
-
if
|
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
|
data/lib/retest/command/ruby.rb
CHANGED
@@ -1,16 +1,26 @@
|
|
1
1
|
module Retest
|
2
2
|
class Command
|
3
3
|
class Ruby < Base
|
4
|
-
def
|
5
|
-
|
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
|
13
|
-
|
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
|
data/lib/retest/command.rb
CHANGED
@@ -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
|
13
|
-
new(options: options
|
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
|
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
|
-
@
|
22
|
+
@setup_identified = nil
|
24
23
|
end
|
25
24
|
|
26
25
|
def command
|
27
|
-
options_command ||
|
26
|
+
options_command || setup_command
|
28
27
|
end
|
29
28
|
|
30
29
|
private
|
31
30
|
|
32
31
|
def options_command
|
33
|
-
|
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
|
-
|
42
|
-
|
43
|
-
|
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
|
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
|
data/lib/retest/program.rb
CHANGED
@@ -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
|
-
|
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.
|
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
|
-
|
45
|
-
|
46
|
-
log("
|
55
|
+
new_command = command.switch_to(:batched)
|
56
|
+
|
57
|
+
log("\nTests selected:")
|
47
58
|
test_files.each { |test_file| log(" - #{test_file}") }
|
48
|
-
|
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
|
-
|
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
|
data/lib/retest/version.rb
CHANGED
data/lib/retest/watcher.rb
CHANGED
@@ -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
|
-
|
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
|
data/lib/scripts/listen
ADDED
@@ -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
|
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-
|
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:
|