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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 7eb4992f6d7149b443a2833f6481694b551fcf3ca98355acbe014f9209ed68c0
4
- data.tar.gz: f1c50bb7340803ae2abc65178c204b983fe0a99471ac502ad0726d94d7219638
3
+ metadata.gz: af4a7f8046c06b50e0d59639373c1cd9fac41a99f3a68a7529b98095b2ef10ae
4
+ data.tar.gz: 93e64c8ac28fe1ae32bc79e5719a8a775c5524544a0c7bd96ede1a3668dd8c6c
5
5
  SHA512:
6
- metadata.gz: 69b59be1bf2c414d66fa9b80eb656e430d32ede6fd88734169096d7b0b7c1f64d48edfcceabc4b48676a5dcf36cf307379926030b75e6fffa95525a504c9e279
7
- data.tar.gz: c063a98b171140f21e66567a2c37d1014f8a3aed5c85f935e6a2e99f8d0845db84c046ccd47f7e739cc115e894111afc07d08b3564f02bdce09698c1a6674f03
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.1.0)
4
+ retest (2.2.0)
5
5
  listen (~> 3.9)
6
6
  observer (~> 0.1)
7
7
  string-similarity (~> 2.1)
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.to_s}'"
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
- 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
93
105
  changed
94
106
  notify_observers(result)
95
107
  end
@@ -1,3 +1,3 @@
1
1
  module Retest
2
- VERSION = "2.1.0"
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.1.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-01-27 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:
@@ -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.0.3.1
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.