Skip to content

Commit 161e14a

Browse files
Improve annotation messages. (#6)
* Improve annotation messages. * Refactor formatting with delegator pattern * Change relative e2e test to run on real file * Do not require delegate Co-authored-by: Stef Schenkelaars <[email protected]>
1 parent c165736 commit 161e14a

File tree

6 files changed

+151
-70
lines changed

6 files changed

+151
-70
lines changed

.github/workflows/ci.yml

+6-3
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,6 @@ jobs:
3535
- name: Run RSpec
3636
run: bundle exec rspec spec/rspec
3737
e2e:
38-
continue-on-error: true
3938
runs-on: ubuntu-latest
4039
steps:
4140
- name: Check out code
@@ -56,5 +55,9 @@ jobs:
5655
gem install bundler:2.1.4 --no-doc
5756
bundle config path vendor/bundle
5857
bundle install --jobs 4 --retry 3
59-
- name: Run RSpec
60-
run: '! bundle exec rspec spec/integration'
58+
- name: Run RSpec in GITHUB_WORKSPACE
59+
run: '! bundle exec rspec spec/integration/failing_spec.rb'
60+
- name: Run RSpec in sub-directory of GITHUB_WORKSPACE
61+
run: |
62+
cd spec/integration
63+
bundle exec rspec relative_path/pending_spec.rb --require ../spec_helper --format RSpec::Github::Formatter

lib/rspec/github/formatter.rb

+7-11
Original file line numberDiff line numberDiff line change
@@ -2,27 +2,23 @@
22

33
require 'rspec/core'
44
require 'rspec/core/formatters/base_formatter'
5+
require 'rspec/github/notification_decorator'
56

67
module RSpec
78
module Github
89
class Formatter < RSpec::Core::Formatters::BaseFormatter
910
RSpec::Core::Formatters.register self, :example_failed, :example_pending
1011

11-
def self.relative_path(path)
12-
workspace = File.realpath(ENV.fetch('GITHUB_WORKSPACE', '.'))
13-
File.realpath(path).delete_prefix("#{workspace}#{File::SEPARATOR}")
14-
end
15-
1612
def example_failed(failure)
17-
file, line = failure.example.location.split(':')
18-
file = self.class.relative_path(file)
19-
output.puts "\n::error file=#{file},line=#{line}::#{failure.message_lines.join('%0A')}"
13+
notification = NotificationDecorator.new(failure)
14+
15+
output.puts "\n::error file=#{notification.path},line=#{notification.line}::#{notification.annotation}"
2016
end
2117

2218
def example_pending(pending)
23-
file, line = pending.example.location.split(':')
24-
file = self.class.relative_path(file)
25-
output.puts "\n::warning file=#{file},line=#{line}::#{pending.example.full_description}"
19+
notification = NotificationDecorator.new(pending)
20+
21+
output.puts "\n::warning file=#{notification.path},line=#{notification.line}::#{notification.annotation}"
2622
end
2723
end
2824
end
+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
# frozen_string_literal: true
2+
3+
module RSpec
4+
module Github
5+
class NotificationDecorator
6+
# See https://github.community/t/set-output-truncates-multiline-strings/16852/3.
7+
ESCAPE_MAP = {
8+
'%' => '%25',
9+
"\n" => '%0A',
10+
"\r" => '%0D'
11+
}.freeze
12+
13+
def initialize(notification)
14+
@notification = notification
15+
end
16+
17+
def line
18+
example.location.split(':')[1]
19+
end
20+
21+
def annotation
22+
"#{example.full_description}\n\n#{message}"
23+
.gsub(Regexp.union(ESCAPE_MAP.keys), ESCAPE_MAP)
24+
end
25+
26+
def path
27+
File.realpath(raw_path).delete_prefix("#{workspace}#{File::SEPARATOR}")
28+
end
29+
30+
private
31+
32+
def example
33+
@notification.example
34+
end
35+
36+
def message
37+
if @notification.respond_to?(:message_lines)
38+
@notification.message_lines.join("\n")
39+
else
40+
"#{example.skip ? 'Skipped' : 'Pending'}: #{example.execution_result.pending_message}"
41+
end
42+
end
43+
44+
def raw_path
45+
example.location.split(':')[0]
46+
end
47+
48+
def workspace
49+
File.realpath(ENV.fetch('GITHUB_WORKSPACE', '.'))
50+
end
51+
end
52+
end
53+
end

spec/integration/failing_spec.rb

+11-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,20 @@
11
# frozen_string_literal: true
22

3-
RSpec.describe RSpec::Github do
3+
RSpec.describe RSpec::Github::Formatter do
44
it 'creates an error annotation for failing specs' do
55
expect(true).to eq false
66
end
77

8-
it 'creates a warning annotation for pending specs'
8+
it 'creates a warning annotation for unimplemented specs'
9+
10+
it 'creates a warning annotation for pending specs' do
11+
pending 'because it is failing'
12+
raise
13+
end
14+
15+
it 'creates a warning annotation for skipped specs' do
16+
skip 'because reasons'
17+
end
918

1019
it 'does not create an annotiation for passing specs' do
1120
expect(true).to eq true
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# frozen_string_literal: true
2+
3+
RSpec.describe RSpec::Github::Formatter do
4+
it 'adds annotiations correctly when running in a subdirectory'
5+
end

spec/rspec/github/formatter_spec.rb

+69-54
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,16 @@
66
let(:output) { StringIO.new }
77
let(:formatter) { described_class.new(output) }
88
subject(:output_string) { output.string }
9+
let(:skip) { false }
10+
11+
let(:location) { './spec/models/user_spec.rb:12' }
12+
13+
let(:pending_message) { 'Not yet implemented' }
914

1015
let(:execution_result) do
1116
double(
1217
'RSpec::Core::Example::ExecutionResult',
13-
pending_message: 'Not yet implemented'
18+
pending_message: pending_message
1419
)
1520
end
1621

@@ -20,7 +25,8 @@
2025
execution_result: execution_result,
2126
full_description: 'User is expected to validate presence of name',
2227
description: 'is expected to validate presence of name',
23-
location: './spec/models/user_spec.rb:12'
28+
location: location,
29+
skip: skip
2430
)
2531
end
2632

@@ -30,53 +36,6 @@
3036
.and_return(File.join(Dir.pwd, 'spec/models/user_spec.rb'))
3137
end
3238

33-
describe '::relative_path' do
34-
around do |example|
35-
saved_github_workspace = ENV['GITHUB_WORKSPACE']
36-
ENV['GITHUB_WORKSPACE'] = github_workspace
37-
38-
FileUtils.mkpath File.dirname(absolute_path)
39-
FileUtils.touch absolute_path
40-
41-
Dir.chdir tmpdir do
42-
example.run
43-
end
44-
ensure
45-
FileUtils.rm_r tmpdir
46-
ENV['GITHUB_WORKSPACE'] = saved_github_workspace
47-
end
48-
49-
let(:tmpdir) { Dir.mktmpdir }
50-
let(:relative_path) { 'this/is/a/relative_path.rb' }
51-
let(:absolute_path) { File.join(tmpdir, relative_path) }
52-
53-
context 'if GITHUB_WORKSPACE is set' do
54-
let(:github_workspace) { tmpdir }
55-
56-
it 'returns the path relative to it when already inside it' do
57-
expect(described_class.relative_path('this/is/a/relative_path.rb')).to eq('this/is/a/relative_path.rb')
58-
end
59-
60-
it 'returns the path relative to it when in a subdirectory of it' do
61-
Dir.chdir 'this/is' do
62-
expect(described_class.relative_path('a/relative_path.rb')).to eq('this/is/a/relative_path.rb')
63-
end
64-
end
65-
end
66-
67-
context 'if GITHUB_WORKSPACE is unset' do
68-
let(:github_workspace) { nil }
69-
70-
it 'returns the unchanged relative path' do
71-
expect(described_class.relative_path('this/is/a/relative_path.rb')).to eq 'this/is/a/relative_path.rb'
72-
end
73-
74-
it 'returns the relative path without a ./ prefix' do
75-
expect(described_class.relative_path('./this/is/a/relative_path.rb')).to eq 'this/is/a/relative_path.rb'
76-
end
77-
end
78-
end
79-
8039
describe '#example_failed' do
8140
before { formatter.example_failed(notification) }
8241

@@ -98,9 +57,52 @@
9857
it 'outputs the GitHub annotation formatted error' do
9958
is_expected.to eq <<~MESSAGE
10059
101-
::error file=spec/models/user_spec.rb,line=12::#{notification.message_lines.join('%0A')}
60+
::error file=spec/models/user_spec.rb,line=12::#{example.full_description}%0A%0A#{notification.message_lines.join('%0A')}
10261
MESSAGE
10362
end
63+
64+
context 'relative_path to GITHUB_WORKSPACE' do
65+
around do |example|
66+
saved_github_workspace = ENV['GITHUB_WORKSPACE']
67+
ENV['GITHUB_WORKSPACE'] = tmpdir
68+
69+
FileUtils.mkpath File.dirname(absolute_path)
70+
FileUtils.touch absolute_path
71+
72+
Dir.chdir tmpdir do
73+
example.run
74+
end
75+
ensure
76+
FileUtils.rm_r tmpdir
77+
ENV['GITHUB_WORKSPACE'] = saved_github_workspace
78+
end
79+
80+
let(:tmpdir) { Dir.mktmpdir }
81+
let(:relative_path) { 'this/is/a/relative_path.rb' }
82+
let(:absolute_path) { File.join(tmpdir, relative_path) }
83+
84+
context 'inside root dir' do
85+
let(:github_workspace) { tmpdir }
86+
let(:location) { './this/is/a/relative_path.rb' }
87+
88+
it 'returns the relative path' do
89+
is_expected.to include 'this/is/a/relative_path.rb'
90+
end
91+
end
92+
context 'inside subdirectory dir' do
93+
let(:github_workspace) { tmpdir }
94+
let(:location) { './a/relative_path.rb' }
95+
around do |example|
96+
Dir.chdir 'this/is' do
97+
example.run
98+
end
99+
end
100+
101+
it 'returns the relative path' do
102+
is_expected.to include 'this/is/a/relative_path.rb'
103+
end
104+
end
105+
end
104106
end
105107

106108
describe '#example_pending' do
@@ -113,11 +115,24 @@
113115
)
114116
end
115117

116-
it 'outputs the GitHub annotation formatted error' do
117-
is_expected.to eq <<~MESSAGE
118+
context 'when pending' do
119+
it 'outputs the GitHub annotation formatted warning' do
120+
is_expected.to eq <<~MESSAGE
118121
119-
::warning file=spec/models/user_spec.rb,line=12::#{example.full_description}
120-
MESSAGE
122+
::warning file=spec/models/user_spec.rb,line=12::#{example.full_description}%0A%0APending: #{pending_message}
123+
MESSAGE
124+
end
125+
end
126+
127+
context 'when skipped' do
128+
let(:skip) { true }
129+
130+
it 'outputs the GitHub annotation formatted warning' do
131+
is_expected.to eq <<~MESSAGE
132+
133+
::warning file=spec/models/user_spec.rb,line=12::#{example.full_description}%0A%0ASkipped: #{pending_message}
134+
MESSAGE
135+
end
121136
end
122137
end
123138
end

0 commit comments

Comments
 (0)