retest 2.1.0 → 2.2.1

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: d136e02d0fe9258511865619e208809547e4ab021ecd735755b3b76b0893ac6d
4
+ data.tar.gz: 5e93ba6847b556359e921cb8b4807676937708e8d8f55987b7cf6795082fe649
5
5
  SHA512:
6
- metadata.gz: 69b59be1bf2c414d66fa9b80eb656e430d32ede6fd88734169096d7b0b7c1f64d48edfcceabc4b48676a5dcf36cf307379926030b75e6fffa95525a504c9e279
7
- data.tar.gz: c063a98b171140f21e66567a2c37d1014f8a3aed5c85f935e6a2e99f8d0845db84c046ccd47f7e739cc115e894111afc07d08b3564f02bdce09698c1a6674f03
6
+ metadata.gz: 1a88d8f871d30a0d1bf74cf04dd19eed678290a74e2ccfa5454f8a8f70d3dafe5f83e76ec6e7177995be57527f54700137ae63995d48957e899e702aaf7607fe
7
+ data.tar.gz: c7527faa79a247c04de260cab878b40619de1a546980e8d2371c53c26c5056a5d9a8ad0dd489d4ebad63c6e3df2703c3fc6ed52748f40bf54699b2c4fdceb4a7
@@ -20,23 +20,24 @@ jobs:
20
20
  matrix:
21
21
  os: [ubuntu-latest]
22
22
  ruby:
23
- - '2.5'
24
- - '2.6'
25
- - '2.7'
26
- - '3.0'
27
- - '3.1'
28
- - '3.2'
29
- - '3.3'
30
- include:
31
- - os: macos-13
32
- ruby: 2.5
33
- name: Ruby ${{ matrix.ruby }} test (${{ matrix.os }})
23
+ - version: '2.5'
24
+ - version: '2.6'
25
+ - version: '2.7'
26
+ - version: '3.0'
27
+ - version: '3.1'
28
+ - version: '3.2'
29
+ - version: '3.3'
30
+ - version: '3.4'
31
+ gemfile: Gemfile-3.4
32
+ name: Ruby ${{ matrix.ruby.version }} test (${{ matrix.os }})
33
+ env:
34
+ BUNDLE_GEMFILE: ${{ matrix.ruby.gemfile || 'Gemfile' }}
34
35
  steps:
35
36
  - uses: actions/checkout@v4
36
37
  - name: Set up Ruby
37
38
  uses: ruby/setup-ruby@v1
38
39
  with:
39
- ruby-version: ${{ matrix.ruby }}
40
+ ruby-version: ${{ matrix.ruby.version }}
40
41
  bundler-cache: true
41
42
  - run: bundle install
42
43
  - run: bundle exec rake
data/Gemfile-3.4 ADDED
@@ -0,0 +1,8 @@
1
+ source "https://rubygems.org"
2
+
3
+ # Specify your gem's dependencies in retest.gemspec
4
+ gemspec
5
+
6
+ gem "rake", "~> 13.0"
7
+ gem "minitest", "~> 5.0"
8
+ gem "byebug", require: false
data/Gemfile-3.4.lock ADDED
@@ -0,0 +1,71 @@
1
+ PATH
2
+ remote: .
3
+ specs:
4
+ retest (2.2.1)
5
+ listen (~> 3.9)
6
+ observer (~> 0.1)
7
+ string-similarity (~> 2.1)
8
+ tty-option (~> 0.1)
9
+ tty-prompt (~> 0.1)
10
+
11
+ GEM
12
+ remote: https://rubygems.org/
13
+ specs:
14
+ byebug (11.1.3)
15
+ ffi (1.17.1)
16
+ ffi (1.17.1-aarch64-linux-gnu)
17
+ ffi (1.17.1-aarch64-linux-musl)
18
+ ffi (1.17.1-arm-linux-gnu)
19
+ ffi (1.17.1-arm-linux-musl)
20
+ ffi (1.17.1-arm64-darwin)
21
+ ffi (1.17.1-x86-linux-gnu)
22
+ ffi (1.17.1-x86-linux-musl)
23
+ ffi (1.17.1-x86_64-darwin)
24
+ ffi (1.17.1-x86_64-linux-gnu)
25
+ ffi (1.17.1-x86_64-linux-musl)
26
+ listen (3.9.0)
27
+ rb-fsevent (~> 0.10, >= 0.10.3)
28
+ rb-inotify (~> 0.9, >= 0.9.10)
29
+ minitest (5.25.5)
30
+ observer (0.1.2)
31
+ pastel (0.8.0)
32
+ tty-color (~> 0.5)
33
+ rake (13.2.1)
34
+ rb-fsevent (0.11.2)
35
+ rb-inotify (0.11.1)
36
+ ffi (~> 1.0)
37
+ string-similarity (2.1.0)
38
+ tty-color (0.6.0)
39
+ tty-cursor (0.7.1)
40
+ tty-option (0.3.0)
41
+ tty-prompt (0.23.1)
42
+ pastel (~> 0.8)
43
+ tty-reader (~> 0.8)
44
+ tty-reader (0.9.0)
45
+ tty-cursor (~> 0.7)
46
+ tty-screen (~> 0.8)
47
+ wisper (~> 2.0)
48
+ tty-screen (0.8.2)
49
+ wisper (2.0.1)
50
+
51
+ PLATFORMS
52
+ aarch64-linux-gnu
53
+ aarch64-linux-musl
54
+ arm-linux-gnu
55
+ arm-linux-musl
56
+ arm64-darwin
57
+ ruby
58
+ x86-linux-gnu
59
+ x86-linux-musl
60
+ x86_64-darwin
61
+ x86_64-linux-gnu
62
+ x86_64-linux-musl
63
+
64
+ DEPENDENCIES
65
+ byebug
66
+ minitest (~> 5.0)
67
+ rake (~> 13.0)
68
+ retest!
69
+
70
+ BUNDLED WITH
71
+ 2.6.6
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.1)
5
5
  listen (~> 3.9)
6
6
  observer (~> 0.1)
7
7
  string-similarity (~> 2.1)
@@ -0,0 +1,15 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "rubygems"
4
+
5
+ `brew upgrade ruby-build`
6
+
7
+ ruby_list = `rbenv install --list-all`
8
+
9
+ puts ruby_list
10
+ .split("\n")
11
+ .select { |version| version =~ /^\d+\.\d+\.\d+$/ }
12
+ .select { |version| Gem::Version.new(version) >= Gem::Version.new("2.5") }
13
+ .group_by { |version| version[0..2] }
14
+ .transform_values { |version| version.max_by { |number| Gem::Version.new(number) } }
15
+ .values
data/bin/start ADDED
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env bash
2
+
3
+ docker run --rm -it -v "$(pwd):/app" -w /app ruby:$1 bash
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.1"
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,50 @@ 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
+ thread = Thread.new do
45
+ loop do
46
+ ready = IO.select([watch_rd])
47
+ readable_connections = ready[0]
48
+ readable_connections.each do |conn|
49
+ data = conn.readpartial(4096)
50
+ change = /^(?<action>create|remove|modify):(?<path>.*)/.match(data.strip)
51
+
52
+ next unless change
53
+
54
+ modified, added, removed = result = [[], [], []]
55
+ case change[:action]
56
+ when 'modify' then modified << change[:path]
57
+ when 'create' then added << change[:path]
58
+ when 'remove' then removed << change[:path]
59
+ end
60
+
61
+ yield result
62
+ end
63
+ end
64
+ end
65
+
66
+ at_exit do
67
+ thread.exit
68
+ Process.kill("TERM", pid) if pid
69
+ watch_rd.close
70
+ watch_wr.close
71
+ end
72
+ end
36
73
  end
37
74
 
38
75
  module Watchexec
@@ -42,22 +79,24 @@ module Retest
42
79
 
43
80
  def self.watch(dir:, extensions:, polling: false)
44
81
  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
82
 
47
83
  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
84
+ # Process needs its own process group otherwise the process gets killed on INT signal
85
+ # We need the process to still run when trying to stop the current test run
86
+ # Maybe there is another way to prevent killing these but for now a new process groups works
87
+ # Process group created with: pgroup: true
88
+ pid = Process.spawn(command, out: watch_wr, pgroup: true)
89
+
90
+ thread = Thread.new do
91
+ files = VersionControl.files(extensions: extensions).zip([]).to_h
54
92
 
55
- Thread.new do
56
93
  loop do
57
94
  ready = IO.select([watch_rd])
58
95
  readable_connections = ready[0]
59
96
  readable_connections.each do |conn|
60
97
  data = conn.readpartial(4096)
98
+ # Watchexec is not great at figuring out whether a file has been deleted and comes as an update.
99
+ # This is why we're not looking at the action like we do with Listen.
61
100
  change = /^(?:create|remove|rename|modify):(?<path>.*)/.match(data.strip)
62
101
 
63
102
  next unless change
@@ -82,40 +121,12 @@ module Retest
82
121
  end
83
122
  end
84
123
 
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
124
+ at_exit do
125
+ thread.exit
126
+ Process.kill("TERM", pid) if pid
127
+ watch_rd.close
128
+ watch_wr.close
129
+ end
119
130
  end
120
131
  end
121
132
  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,13 @@
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.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Alexandre Barret
8
- autorequire:
9
8
  bindir: exe
10
9
  cert_chain: []
11
- date: 2025-01-27 00:00:00.000000000 Z
10
+ date: 2025-03-17 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: string-similarity
@@ -80,7 +79,6 @@ dependencies:
80
79
  - - "~>"
81
80
  - !ruby/object:Gem::Version
82
81
  version: '0.1'
83
- description:
84
82
  email:
85
83
  - alex@abletech.nz
86
84
  executables:
@@ -92,6 +90,8 @@ files:
92
90
  - ".github/workflows/ci.yml"
93
91
  - ".gitignore"
94
92
  - Gemfile
93
+ - Gemfile-3.4
94
+ - Gemfile-3.4.lock
95
95
  - Gemfile.lock
96
96
  - LICENSE.txt
97
97
  - README.md
@@ -100,7 +100,9 @@ files:
100
100
  - bin/build/watchexec
101
101
  - bin/console
102
102
  - bin/debug
103
+ - bin/list-ruby-versions
103
104
  - bin/setup
105
+ - bin/start
104
106
  - bin/test/bundler-app
105
107
  - bin/test/git-ruby
106
108
  - bin/test/hanami-app
@@ -138,6 +140,7 @@ files:
138
140
  - lib/retest/version_control/git.rb
139
141
  - lib/retest/version_control/no_version_control.rb
140
142
  - lib/retest/watcher.rb
143
+ - lib/scripts/listen
141
144
  - retest.gemspec
142
145
  homepage: https://github.com/AlexB52/retest
143
146
  licenses:
@@ -145,7 +148,6 @@ licenses:
145
148
  metadata:
146
149
  homepage_uri: https://github.com/AlexB52/retest
147
150
  source_code_uri: https://github.com/AlexB52/retest
148
- post_install_message:
149
151
  rdoc_options: []
150
152
  require_paths:
151
153
  - lib
@@ -160,8 +162,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
160
162
  - !ruby/object:Gem::Version
161
163
  version: '0'
162
164
  requirements: []
163
- rubygems_version: 3.0.3.1
164
- signing_key:
165
+ rubygems_version: 3.6.2
165
166
  specification_version: 4
166
167
  summary: A simple command line tool to watch file change and run its matching spec.
167
168
  test_files: []