test-recorder 0.1.0 → 0.1.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 +4 -4
- data/.github/workflows/ci.yml +29 -0
- data/Gemfile.lock +23 -54
- data/README.md +48 -2
- data/Rakefile +1 -1
- data/lib/test_recorder/cdp_recorder.rb +59 -0
- data/lib/test_recorder/rails/setup_and_teardown.rb +5 -14
- data/lib/test_recorder/rails.rb +1 -0
- data/lib/test_recorder/rspec/example_wrapper.rb +10 -0
- data/lib/test_recorder/rspec.rb +14 -18
- data/lib/test_recorder/version.rb +1 -1
- data/test-recorder.gemspec +3 -2
- metadata +22 -5
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: e868dca992d2695188debf897102cd76ed6374a25abcb0bbd8875414f9714f9e
|
4
|
+
data.tar.gz: ecad80fbed7bc759ef3d30043b1fcdeb15202f3caa3a6e095b29e9d5ccf77e9c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d13e0d7f4fe1056b99f367c08ea89424cfc86780a0d04754425098c7d9c9964f8abb675360a6fafd8c3e8db4fdb0e5f4b9d55890e2376dbf70391a2c3ddfac52
|
7
|
+
data.tar.gz: dccb98cfda3cf96ffd888f3cea87cc8cfd9d27631da0ce66408c410fbf1b74fc74c7ea45461c48ffe1c29cca1895d96c7716884189fe4e689ea7175c4dd6e480
|
@@ -0,0 +1,29 @@
|
|
1
|
+
name: CI
|
2
|
+
|
3
|
+
on: [push]
|
4
|
+
|
5
|
+
jobs:
|
6
|
+
build:
|
7
|
+
runs-on: ubuntu-latest
|
8
|
+
|
9
|
+
concurrency:
|
10
|
+
group: ${{ github.ref }}
|
11
|
+
cancel-in-progress: true
|
12
|
+
|
13
|
+
steps:
|
14
|
+
- uses: actions/checkout@v2
|
15
|
+
- uses: ruby/setup-ruby@v1
|
16
|
+
with:
|
17
|
+
ruby-version: '3.0.3'
|
18
|
+
- uses: actions/setup-node@v1
|
19
|
+
with:
|
20
|
+
node-version: '14'
|
21
|
+
- name: Install ffmpeg
|
22
|
+
run: |
|
23
|
+
sudo apt install ffmpeg
|
24
|
+
- name: Install dependencies
|
25
|
+
run: |
|
26
|
+
gem install bundler --no-document
|
27
|
+
bundle install
|
28
|
+
- name: Run test
|
29
|
+
run: bundle exec rake
|
data/Gemfile.lock
CHANGED
@@ -1,67 +1,36 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
test-recorder (0.1.
|
5
|
-
|
6
|
-
|
4
|
+
test-recorder (0.1.1)
|
5
|
+
activesupport
|
6
|
+
selenium-devtools
|
7
|
+
selenium-webdriver (>= 4.0)
|
7
8
|
|
8
9
|
GEM
|
9
10
|
remote: https://rubygems.org/
|
10
11
|
specs:
|
11
|
-
|
12
|
-
actionview (= 6.0.3.4)
|
13
|
-
activesupport (= 6.0.3.4)
|
14
|
-
rack (~> 2.0, >= 2.0.8)
|
15
|
-
rack-test (>= 0.6.3)
|
16
|
-
rails-dom-testing (~> 2.0)
|
17
|
-
rails-html-sanitizer (~> 1.0, >= 1.2.0)
|
18
|
-
actionview (6.0.3.4)
|
19
|
-
activesupport (= 6.0.3.4)
|
20
|
-
builder (~> 3.1)
|
21
|
-
erubi (~> 1.4)
|
22
|
-
rails-dom-testing (~> 2.0)
|
23
|
-
rails-html-sanitizer (~> 1.1, >= 1.2.0)
|
24
|
-
activesupport (6.0.3.4)
|
12
|
+
activesupport (7.0.0)
|
25
13
|
concurrent-ruby (~> 1.0, >= 1.0.2)
|
26
|
-
i18n (>=
|
27
|
-
minitest (
|
28
|
-
tzinfo (~>
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
crass (1.0.6)
|
33
|
-
erubi (1.9.0)
|
34
|
-
headless (2.3.1)
|
35
|
-
i18n (1.8.5)
|
14
|
+
i18n (>= 1.6, < 2)
|
15
|
+
minitest (>= 5.1)
|
16
|
+
tzinfo (~> 2.0)
|
17
|
+
childprocess (4.1.0)
|
18
|
+
concurrent-ruby (1.1.9)
|
19
|
+
i18n (1.8.11)
|
36
20
|
concurrent-ruby (~> 1.0)
|
37
|
-
loofah (2.7.0)
|
38
|
-
crass (~> 1.0.2)
|
39
|
-
nokogiri (>= 1.5.9)
|
40
|
-
method_source (1.0.0)
|
41
|
-
mini_portile2 (2.4.0)
|
42
21
|
minitest (5.14.2)
|
43
|
-
nokogiri (1.10.10)
|
44
|
-
mini_portile2 (~> 2.4.0)
|
45
|
-
rack (2.2.3)
|
46
|
-
rack-test (1.1.0)
|
47
|
-
rack (>= 1.0, < 3)
|
48
|
-
rails-dom-testing (2.0.3)
|
49
|
-
activesupport (>= 4.2.0)
|
50
|
-
nokogiri (>= 1.6)
|
51
|
-
rails-html-sanitizer (1.3.0)
|
52
|
-
loofah (~> 2.3)
|
53
|
-
railties (6.0.3.4)
|
54
|
-
actionpack (= 6.0.3.4)
|
55
|
-
activesupport (= 6.0.3.4)
|
56
|
-
method_source
|
57
|
-
rake (>= 0.8.7)
|
58
|
-
thor (>= 0.20.3, < 2.0)
|
59
22
|
rake (12.3.3)
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
23
|
+
rexml (3.2.5)
|
24
|
+
rubyzip (2.3.2)
|
25
|
+
selenium-devtools (0.96.0)
|
26
|
+
websocket (~> 1.0)
|
27
|
+
selenium-webdriver (4.1.0)
|
28
|
+
childprocess (>= 0.5, < 5.0)
|
29
|
+
rexml (~> 3.2, >= 3.2.5)
|
30
|
+
rubyzip (>= 1.2.2)
|
31
|
+
tzinfo (2.0.4)
|
32
|
+
concurrent-ruby (~> 1.0)
|
33
|
+
websocket (1.2.9)
|
65
34
|
|
66
35
|
PLATFORMS
|
67
36
|
ruby
|
@@ -72,4 +41,4 @@ DEPENDENCIES
|
|
72
41
|
test-recorder!
|
73
42
|
|
74
43
|
BUNDLED WITH
|
75
|
-
2.
|
44
|
+
2.2.32
|
data/README.md
CHANGED
@@ -1,3 +1,49 @@
|
|
1
|
-
#
|
1
|
+
# TestRecorder
|
2
2
|
|
3
|
-
|
3
|
+
Record a video automatically when tests failed. The videos are generated in `tmp/videos` directory.
|
4
|
+
|
5
|
+

|
6
|
+
[](http://badge.fury.io/rb/test-recorder)
|
7
|
+
|
8
|
+
|
9
|
+
## Requirements
|
10
|
+
|
11
|
+
This gem depends on FFmpeg. Please install that package.
|
12
|
+
|
13
|
+
On Debian/Ubuntu:
|
14
|
+
|
15
|
+
```bash
|
16
|
+
sudo apt-get install ffmpeg
|
17
|
+
```
|
18
|
+
|
19
|
+
## Supported libraries
|
20
|
+
|
21
|
+
Rails system tests and RSpec(System Spec and Feature Spec).
|
22
|
+
|
23
|
+
## Limitations
|
24
|
+
|
25
|
+
Currently, this gem only supports a Chrome headless.
|
26
|
+
|
27
|
+
## Usage
|
28
|
+
|
29
|
+
### 1: Install the gem
|
30
|
+
|
31
|
+
Using Bundler, add the following to your Gemfile:
|
32
|
+
|
33
|
+
```ruby
|
34
|
+
gem 'test-recorder', group: :test
|
35
|
+
```
|
36
|
+
|
37
|
+
### 2: Load library into your tests
|
38
|
+
|
39
|
+
#### Rails
|
40
|
+
|
41
|
+
```ruby
|
42
|
+
require 'test_recorder/rails'
|
43
|
+
```
|
44
|
+
|
45
|
+
#### RSpec
|
46
|
+
|
47
|
+
```ruby
|
48
|
+
require 'test_recorder/rspec'
|
49
|
+
```
|
data/Rakefile
CHANGED
@@ -0,0 +1,59 @@
|
|
1
|
+
require "fileutils"
|
2
|
+
|
3
|
+
module TestRecorder
|
4
|
+
class CdpRecorder
|
5
|
+
def initialize(enabled:)
|
6
|
+
@enabled = enabled
|
7
|
+
setup if @enabled
|
8
|
+
end
|
9
|
+
|
10
|
+
def setup
|
11
|
+
@video_dir = ::Rails.root.join("tmp", "videos")
|
12
|
+
FileUtils.mkdir_p(@video_dir)
|
13
|
+
end
|
14
|
+
|
15
|
+
def start(page:)
|
16
|
+
return unless @enabled
|
17
|
+
|
18
|
+
@tmpdir = Dir.mktmpdir("testrecorder")
|
19
|
+
@counter = 1
|
20
|
+
|
21
|
+
@page = page
|
22
|
+
@page.driver.browser.devtools.page.enable
|
23
|
+
|
24
|
+
@page.driver.browser.devtools.page.on(:screencast_frame) do |event|
|
25
|
+
decoded_data = Base64.decode64(event["data"])
|
26
|
+
filename = "%010d.jpeg" % @counter
|
27
|
+
if Dir.exist?(@tmpdir)
|
28
|
+
IO.binwrite("#{File.join(@tmpdir, filename)}", decoded_data)
|
29
|
+
@counter += 1
|
30
|
+
end
|
31
|
+
@page.driver.browser.devtools.page.screencast_frame_ack(session_id: event["sessionId"])
|
32
|
+
end
|
33
|
+
|
34
|
+
@page.driver.browser.devtools.page.start_screencast(format: "jpeg", quality: 90)
|
35
|
+
end
|
36
|
+
|
37
|
+
def stop_and_discard
|
38
|
+
FileUtils.rm_rf(@tmpdir)
|
39
|
+
end
|
40
|
+
|
41
|
+
def stop_and_save(filename)
|
42
|
+
return if !@enabled || @page.nil?
|
43
|
+
|
44
|
+
@page.driver.browser.devtools.page.stop_screencast
|
45
|
+
video_path = File.join(@video_dir, filename)
|
46
|
+
|
47
|
+
args = %W(-loglevel error -f image2 -avioflags direct -fpsprobesize 0
|
48
|
+
-probesize 32 -analyzeduration 0 -c:v mjpeg -i #{File.join(@tmpdir, "%010d.jpeg")}
|
49
|
+
-y -an -r 25 -qmin 0 -qmax 50 -crf 8 -deadline realtime -speed 8 -b:v 1M
|
50
|
+
-threads 1 #{video_path})
|
51
|
+
system("ffmpeg", *args, exception: true)
|
52
|
+
video_path
|
53
|
+
rescue => e
|
54
|
+
$stderr.puts("[TestRecorder] ffmpeg failed: #{e.message}")
|
55
|
+
ensure
|
56
|
+
FileUtils.rm_rf(@tmpdir)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
@@ -1,30 +1,21 @@
|
|
1
|
-
require "headless"
|
2
1
|
require "fileutils"
|
3
2
|
|
4
3
|
module TestRecorder
|
5
4
|
module Rails
|
6
5
|
module SetupAndTeardown
|
7
|
-
attr_reader :video_dir, :headless
|
8
|
-
|
9
6
|
def before_setup
|
10
|
-
@
|
11
|
-
|
12
|
-
|
13
|
-
# TODO: Allow configuring parameters.
|
14
|
-
@headless = Headless.new(video: { provider: :ffmpeg, codec: :libx264, extra: %w(-preset ultrafast) })
|
15
|
-
headless.start
|
16
|
-
headless.video.start_capture
|
7
|
+
@cdp_recorder = TestRecorder::CdpRecorder.new(enabled: true)
|
8
|
+
@cdp_recorder.start(page: page)
|
17
9
|
|
18
10
|
super
|
19
11
|
end
|
20
12
|
|
21
13
|
def before_teardown
|
22
14
|
if failures.empty?
|
23
|
-
|
15
|
+
@cdp_recorder.stop_and_discard
|
24
16
|
else
|
25
|
-
|
26
|
-
|
27
|
-
puts "[Video]: #{video}"
|
17
|
+
video_path = @cdp_recorder.stop_and_save("failures_#{self.name}.mp4")
|
18
|
+
puts "[Video]: #{video_path}" if File.exist?(video_path)
|
28
19
|
end
|
29
20
|
ensure
|
30
21
|
super
|
data/lib/test_recorder/rails.rb
CHANGED
data/lib/test_recorder/rspec.rb
CHANGED
@@ -1,20 +1,21 @@
|
|
1
|
-
require "
|
2
|
-
require "
|
1
|
+
require "test_recorder/cdp_recorder"
|
2
|
+
require "test_recorder/rspec/example_wrapper"
|
3
3
|
|
4
4
|
module TestRecorder
|
5
|
-
module
|
5
|
+
module RSpec
|
6
6
|
CHARS_TO_TRANSLATE = ['/', '.', ':', ',', "'", '"', " "].freeze
|
7
7
|
|
8
8
|
class << self
|
9
|
-
attr_accessor :
|
9
|
+
attr_accessor :cdp_recorder
|
10
10
|
|
11
11
|
def after_failed_example(example)
|
12
12
|
if example.exception
|
13
|
-
|
14
|
-
|
15
|
-
|
13
|
+
video_path = cdp_recorder.stop_and_save("failures_#{method_name(example)}.mp4").to_s
|
14
|
+
if File.exist?(video_path)
|
15
|
+
example.metadata[:extra_failure_lines] = [example.metadata[:extra_failure_lines], "[Video]: #{video_path}"].flatten
|
16
|
+
end
|
16
17
|
else
|
17
|
-
|
18
|
+
cdp_recorder.stop_and_discard
|
18
19
|
end
|
19
20
|
end
|
20
21
|
|
@@ -25,21 +26,16 @@ module TestRecorder
|
|
25
26
|
end
|
26
27
|
end
|
27
28
|
|
28
|
-
RSpec.
|
29
|
-
config.before do
|
30
|
-
TestRecorder::Rspec.video_dir = ::Rails.root.join("tmp", "videos")
|
31
|
-
FileUtils.mkdir_p(TestRecorder::Rspec.video_dir)
|
29
|
+
RSpec::Core::Example.prepend(TestRecorder::RSpec::ExampleWrapper)
|
32
30
|
|
33
|
-
|
34
|
-
|
35
|
-
TestRecorder::Rspec.headless.video.start_capture
|
36
|
-
end
|
31
|
+
RSpec.configure do |config|
|
32
|
+
TestRecorder::RSpec.cdp_recorder = TestRecorder::CdpRecorder.new(enabled: true)
|
37
33
|
|
38
34
|
config.after(type: :system) do |example|
|
39
|
-
TestRecorder::
|
35
|
+
TestRecorder::RSpec.after_failed_example(example)
|
40
36
|
end
|
41
37
|
|
42
38
|
config.after(type: :feature) do |example|
|
43
|
-
TestRecorder::
|
39
|
+
TestRecorder::RSpec.after_failed_example(example)
|
44
40
|
end
|
45
41
|
end
|
data/test-recorder.gemspec
CHANGED
@@ -19,6 +19,7 @@ Gem::Specification.new do |spec|
|
|
19
19
|
spec.bindir = "exe"
|
20
20
|
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
21
21
|
spec.require_paths = ["lib"]
|
22
|
-
spec.add_dependency "
|
23
|
-
spec.add_dependency "
|
22
|
+
spec.add_dependency "activesupport"
|
23
|
+
spec.add_dependency "selenium-webdriver", ">= 4.0"
|
24
|
+
spec.add_dependency "selenium-devtools"
|
24
25
|
end
|
metadata
CHANGED
@@ -1,17 +1,17 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: test-recorder
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.1.
|
4
|
+
version: 0.1.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Yuji Yaginuma
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2022-01-08 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
|
-
name:
|
14
|
+
name: activesupport
|
15
15
|
requirement: !ruby/object:Gem::Requirement
|
16
16
|
requirements:
|
17
17
|
- - ">="
|
@@ -25,7 +25,21 @@ dependencies:
|
|
25
25
|
- !ruby/object:Gem::Version
|
26
26
|
version: '0'
|
27
27
|
- !ruby/object:Gem::Dependency
|
28
|
-
name:
|
28
|
+
name: selenium-webdriver
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - ">="
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '4.0'
|
34
|
+
type: :runtime
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - ">="
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '4.0'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: selenium-devtools
|
29
43
|
requirement: !ruby/object:Gem::Requirement
|
30
44
|
requirements:
|
31
45
|
- - ">="
|
@@ -45,6 +59,7 @@ executables: []
|
|
45
59
|
extensions: []
|
46
60
|
extra_rdoc_files: []
|
47
61
|
files:
|
62
|
+
- ".github/workflows/ci.yml"
|
48
63
|
- ".gitignore"
|
49
64
|
- Gemfile
|
50
65
|
- Gemfile.lock
|
@@ -55,9 +70,11 @@ files:
|
|
55
70
|
- bin/setup
|
56
71
|
- lib/test-recorder.rb
|
57
72
|
- lib/test_recorder.rb
|
73
|
+
- lib/test_recorder/cdp_recorder.rb
|
58
74
|
- lib/test_recorder/rails.rb
|
59
75
|
- lib/test_recorder/rails/setup_and_teardown.rb
|
60
76
|
- lib/test_recorder/rspec.rb
|
77
|
+
- lib/test_recorder/rspec/example_wrapper.rb
|
61
78
|
- lib/test_recorder/version.rb
|
62
79
|
- test-recorder.gemspec
|
63
80
|
homepage: http://github.com/y-yagi/test-recorder
|
@@ -80,7 +97,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
80
97
|
- !ruby/object:Gem::Version
|
81
98
|
version: '0'
|
82
99
|
requirements: []
|
83
|
-
rubygems_version: 3.
|
100
|
+
rubygems_version: 3.2.32
|
84
101
|
signing_key:
|
85
102
|
specification_version: 4
|
86
103
|
summary: Automatically record videos when tests failed.
|