retest 2.1.0 → 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/exe/retest +11 -7
- data/lib/retest/runner.rb +13 -1
- 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 +4 -3
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/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
|
|
@@ -50,6 +44,16 @@ end
|
|
50
44
|
|
51
45
|
puts "Command: '#{command}'"
|
52
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
|
+
|
53
57
|
# === DIFF ACTION ===
|
54
58
|
if options.params[:diff]
|
55
59
|
program.diff(options.params[:diff])
|
@@ -98,7 +102,7 @@ def run_command(input:, program:)
|
|
98
102
|
Process.kill("INT", 0)
|
99
103
|
when 'r', 'reset'
|
100
104
|
program.reset_selection
|
101
|
-
puts "command reset to '#{program.runner.command
|
105
|
+
puts "command reset to '#{program.runner.command}'"
|
102
106
|
when 'f', 'force'
|
103
107
|
require 'tty-prompt'
|
104
108
|
prompt = TTY::Prompt.new
|
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.')
|
@@ -89,7 +98,10 @@ module Retest
|
|
89
98
|
|
90
99
|
def system_run(command)
|
91
100
|
log("\n")
|
92
|
-
|
101
|
+
@pid = spawn(command)
|
102
|
+
Process.wait
|
103
|
+
@pid = nil
|
104
|
+
result = $?.exitstatus&.zero? ? :tests_pass : :tests_fail
|
93
105
|
changed
|
94
106
|
notify_observers(result)
|
95
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.
|
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:
|
@@ -160,7 +161,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
160
161
|
- !ruby/object:Gem::Version
|
161
162
|
version: '0'
|
162
163
|
requirements: []
|
163
|
-
rubygems_version: 3.
|
164
|
+
rubygems_version: 3.1.6
|
164
165
|
signing_key:
|
165
166
|
specification_version: 4
|
166
167
|
summary: A simple command line tool to watch file change and run its matching spec.
|