Skip to content

Commit ff0337e

Browse files
committed
Fix Ensure check methods can't modify tokens array
The functionality originally released in rodjek#296 was probably broken at the time, possibly made worse by later rubocop 'fixes' and also incompatible with ruby 3.4. We now look at the stack trace only from the `tokens` wrapper method in `checkplugin`. Fixes puppetlabs#225
1 parent 002f275 commit ff0337e

File tree

5 files changed

+46
-28
lines changed

5 files changed

+46
-28
lines changed

.github/workflows/ci.yml

+2
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ jobs:
2222
- '2.7'
2323
- '3.2'
2424
- '3.3'
25+
- '3.4'
2526
name: "spec (ruby ${{ matrix.ruby_version }})"
2627
uses: "puppetlabs/cat-github-actions/.github/workflows/gem_ci.yml@main"
2728
secrets: "inherit"
@@ -37,6 +38,7 @@ jobs:
3738
- '2.7'
3839
- '3.2'
3940
- '3.3'
41+
- '3.4'
4042
name: "acceptance (ruby ${{ matrix.ruby_version }})"
4143
needs: "spec"
4244
uses: "puppetlabs/cat-github-actions/.github/workflows/gem_acceptance.yml@main"

.rubocop_todo.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -129,14 +129,15 @@ RSpec/RepeatedExampleGroupDescription:
129129
- 'spec/unit/puppet-lint/plugins/check_resources/unquoted_file_mode_spec.rb'
130130
- 'spec/unit/puppet-lint/plugins/legacy_facts/legacy_facts_spec.rb'
131131

132-
# Offense count: 8
132+
# Offense count: 9
133133
# Configuration parameters: Include, CustomTransform, IgnoreMethods, IgnoreMetadata.
134134
# Include: **/*_spec.rb
135135
RSpec/SpecFilePathFormat:
136136
Exclude:
137137
- '**/spec/routing/**/*'
138138
- 'spec/unit/puppet-lint/bin_spec.rb'
139139
- 'spec/unit/puppet-lint/checks_spec.rb'
140+
- 'spec/unit/puppet-lint/checkplugin_spec.rb'
140141
- 'spec/unit/puppet-lint/configuration_spec.rb'
141142
- 'spec/unit/puppet-lint/data_spec.rb'
142143
- 'spec/unit/puppet-lint/lexer/string_slurper_spec.rb'

lib/puppet-lint/checkplugin.rb

+3-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,9 @@ def fix_problems
5555
#
5656
# Returns an Array of PuppetLint::Lexer::Token objects.
5757
def tokens
58-
PuppetLint::Data.tokens
58+
# When called from a plugins `check` method, the tokens array returned should be a (shallow) copy
59+
called_from_check = (caller_locations(1..1).first.base_label == 'check')
60+
PuppetLint::Data.tokens(duplicate: called_from_check)
5961
end
6062

6163
def add_token(index, token)

lib/puppet-lint/data.rb

+3-26
Original file line numberDiff line numberDiff line change
@@ -38,37 +38,14 @@ def tokens=(tokens)
3838
@defaults_indexes = nil
3939
end
4040

41-
# @api private
42-
def ruby1?
43-
@ruby1 = RbConfig::CONFIG['MAJOR'] == '1' if @ruby1.nil?
44-
@ruby1
45-
end
46-
4741
# Get the tokenised manifest.
4842
#
43+
# @param duplicate [Boolean] if true, returns a duplicate of the token array.
4944
# @return [Array[PuppetLint::Lexer::Token]]
5045
#
5146
# @api public
52-
def tokens
53-
calling_method = if ruby1?
54-
begin
55-
caller[0][%r{`.*'}][1..-2] # rubocop:disable Performance/Caller
56-
rescue NoMethodError
57-
caller[1][%r{`.*'}][1..-2] # rubocop:disable Performance/Caller
58-
end
59-
else
60-
begin
61-
caller(0..0).first[%r{`.*'}][1..-2]
62-
rescue NoMethodError
63-
caller(1..1).first[%r{`.*'}][1..-2]
64-
end
65-
end
66-
67-
if calling_method == 'check'
68-
@tokens.dup
69-
else
70-
@tokens
71-
end
47+
def tokens(duplicate: false)
48+
duplicate ? @tokens.dup : @tokens
7249
end
7350

7451
# Add new token.
+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
require 'spec_helper'
2+
3+
class DummyCheckPlugin < PuppetLint::CheckPlugin
4+
def check
5+
# Since we're calling `tokens` from a `check` method, we should get our own Array object.
6+
# If we add an extra token to it, PuppetLint::Data.tokens should remain unchanged.
7+
tokens << :extra_token
8+
end
9+
10+
def fix
11+
tokens << :fix_token
12+
end
13+
end
14+
15+
describe PuppetLint::CheckPlugin do
16+
before(:each) do
17+
PuppetLint::Data.tokens = [:token1, :token2, :token3]
18+
end
19+
20+
it 'returns a duplicate of the token array when called from check' do
21+
plugin = DummyCheckPlugin.new
22+
23+
plugin.check
24+
25+
# Verify that the global token array remains unchanged.
26+
expect(PuppetLint::Data.tokens).to eq([:token1, :token2, :token3])
27+
end
28+
29+
it 'other methods can modify the tokens array' do
30+
plugin = DummyCheckPlugin.new
31+
32+
plugin.fix
33+
34+
expect(PuppetLint::Data.tokens).to eq([:token1, :token2, :token3, :fix_token])
35+
end
36+
end

0 commit comments

Comments
 (0)