Skip to content

Commit c9d26e5

Browse files
committed
Rewrite loadjson to use the modern function API
This also resolves a bug where JSON.parse returned a StringIO. To properly catch this, the tests are rewritten to avoid mocking where possible. Exceptions are with external URLs and where a failure is expected.
1 parent 7e7ded4 commit c9d26e5

File tree

3 files changed

+89
-185
lines changed

3 files changed

+89
-185
lines changed

lib/puppet/functions/loadjson.rb

+58
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
# frozen_string_literal: true
2+
3+
# @summary
4+
# Load a JSON file containing an array, string, or hash, and return the data
5+
# in the corresponding native data type.
6+
#
7+
# @example Example Usage:
8+
# $myhash = loadjson('/etc/puppet/data/myhash.json')
9+
# $myhash = loadjson('https://example.local/my_hash.json')
10+
# $myhash = loadjson('https://username:[email protected]/my_hash.json')
11+
# $myhash = loadjson('no-file.json', {'default' => 'value'})
12+
Puppet::Functions.create_function(:loadjson) do
13+
# @param path
14+
# A file path or a URL.
15+
# @param default
16+
# The default value to be returned if the file was not found or could not
17+
# be parsed.
18+
#
19+
# @return
20+
# The data stored in the JSON file, the type depending on the type of data
21+
# that was stored.
22+
dispatch :loadjson do
23+
param 'String[1]', :path
24+
optional_param 'Data', :default
25+
return_type 'Data'
26+
end
27+
28+
def loadjson(path, default = nil)
29+
if path.start_with?('http://', 'https://')
30+
require 'open-uri'
31+
begin
32+
content = URI.open(path) { |f| load_json_source(f) }
33+
rescue OpenURI::HTTPError => e
34+
Puppet.warn("Can't load '#{url}' HTTP Error Code: '#{e.io.status[0]}'")
35+
return default
36+
end
37+
elsif File.exist?(path)
38+
content = File.open(path) { |f| load_json_source(f) }
39+
else
40+
Puppet.warn("Can't load '#{path}' File does not exist!")
41+
return default
42+
end
43+
44+
content || default
45+
rescue StandardError => e
46+
raise e unless default
47+
48+
default
49+
end
50+
51+
def load_json_source(source)
52+
if Puppet::Util::Package.versioncmp(Puppet.version, '8.0.0').negative?
53+
PSON.load(source)
54+
else
55+
JSON.load(source)
56+
end
57+
end
58+
end

lib/puppet/parser/functions/loadjson.rb

-75
This file was deleted.

spec/functions/loadjson_spec.rb

+31-110
Original file line numberDiff line numberDiff line change
@@ -1,178 +1,99 @@
11
# frozen_string_literal: true
22

33
require 'spec_helper'
4+
require 'open-uri'
45

56
describe 'loadjson' do
67
it { is_expected.not_to be_nil }
7-
it { is_expected.to run.with_params.and_raise_error(ArgumentError, %r{wrong number of arguments}i) }
8+
it { is_expected.to run.with_params.and_raise_error(ArgumentError, "'loadjson' expects between 1 and 2 arguments, got none") }
89

910
describe 'when calling with valid arguments' do
10-
before :each do
11-
# In Puppet 7, there are two prior calls to File.read prior to the responses we want to mock
12-
allow(File).to receive(:read).with(anything, anything).and_call_original
13-
allow(File).to receive(:read).with(%r{/(stdlib|test)/metadata.json}, encoding: 'utf-8').and_return('{"name": "puppetlabs-stdlib"}')
14-
allow(File).to receive(:read).with(%r{/(stdlib|test)/metadata.json}).and_return('{"name": "puppetlabs-stdlib"}')
15-
# Additional modules used by litmus which are identified while running these dues to being in fixtures
16-
allow(File).to receive(:read).with(%r{/(provision|puppet_agent|facts)/metadata.json}, encoding: 'utf-8')
17-
end
18-
1911
context 'when a non-existing file is specified' do
2012
let(:filename) do
21-
if Puppet::Util::Platform.windows?
22-
'C:/tmp/doesnotexist'
23-
else
24-
'/tmp/doesnotexist'
25-
end
13+
file = Tempfile.create
14+
file.close
15+
File.unlink(file.path)
16+
file.path
2617
end
2718

2819
before(:each) do
29-
allow(File).to receive(:exist?).and_call_original
30-
allow(File).to receive(:exist?).with(filename).and_return(false).once
3120
if Puppet::PUPPETVERSION[0].to_i < 8
3221
allow(PSON).to receive(:load).never # rubocop:disable RSpec/ReceiveNever Switching to not_to receive breaks testing in this case
3322
else
3423
allow(JSON).to receive(:parse).never # rubocop:disable RSpec/ReceiveNever
3524
end
3625
end
3726

38-
it { is_expected.to run.with_params(filename, 'default' => 'value').and_return('default' => 'value') }
39-
it { is_expected.to run.with_params(filename, 'đẽƒằưļŧ' => '٧ẵłựέ').and_return('đẽƒằưļŧ' => '٧ẵłựέ') }
40-
it { is_expected.to run.with_params(filename, 'デフォルト' => '値').and_return('デフォルト' => '値') }
27+
it { is_expected.to run.with_params(filename, {'default' => 'value'}).and_return({'default' => 'value'}) }
28+
it { is_expected.to run.with_params(filename, {'đẽƒằưļŧ' => '٧ẵłựέ'}).and_return({'đẽƒằưļŧ' => '٧ẵłựέ'}) }
29+
it { is_expected.to run.with_params(filename, {'デフォルト' => '値'}).and_return({'デフォルト' => '値'}) }
4130
end
4231

4332
context 'when an existing file is specified' do
44-
let(:filename) do
45-
if Puppet::Util::Platform.windows?
46-
'C:/tmp/doesexist'
47-
else
48-
'/tmp/doesexist'
49-
end
50-
end
5133
let(:data) { { 'key' => 'value', 'ķęŷ' => 'νậŀųề', 'キー' => '値' } }
5234
let(:json) { '{"key":"value", {"ķęŷ":"νậŀųề" }, {"キー":"値" }' }
5335

54-
before(:each) do
55-
allow(File).to receive(:exist?).and_call_original
56-
allow(File).to receive(:exist?).with(filename).and_return(true).once
57-
allow(File).to receive(:read).with(filename).and_return(json).once
58-
allow(File).to receive(:read).with(filename).and_return(json).once
59-
if Puppet::PUPPETVERSION[0].to_i < 8
60-
allow(PSON).to receive(:load).with(json).and_return(data).once
61-
else
62-
allow(JSON).to receive(:parse).with(json).and_return(data).once
36+
it do
37+
Tempfile.new do |file|
38+
file.write(json)
39+
file.flush
40+
41+
is_expected.to run.with_params(file.path).and_return(data)
6342
end
6443
end
65-
66-
it { is_expected.to run.with_params(filename).and_return(data) }
6744
end
6845

6946
context 'when the file could not be parsed' do
70-
let(:filename) do
71-
if Puppet::Util::Platform.windows?
72-
'C:/tmp/doesexist'
73-
else
74-
'/tmp/doesexist'
75-
end
76-
end
7747
let(:json) { '{"key":"value"}' }
7848

79-
before(:each) do
80-
allow(File).to receive(:exist?).and_call_original
81-
allow(File).to receive(:exist?).with(filename).and_return(true).once
82-
allow(File).to receive(:read).with(filename).and_return(json).once
83-
if Puppet::PUPPETVERSION[0].to_i < 8
84-
allow(PSON).to receive(:load).with(json).once.and_raise StandardError, 'Something terrible have happened!'
85-
else
86-
allow(JSON).to receive(:parse).with(json).once.and_raise StandardError, 'Something terrible have happened!'
87-
end
88-
end
89-
90-
it { is_expected.to run.with_params(filename, 'default' => 'value').and_return('default' => 'value') }
91-
end
92-
93-
context 'when an existing URL is specified' do
94-
let(:filename) do
95-
'https://example.local/myhash.json'
96-
end
97-
let(:data) { { 'key' => 'value', 'ķęŷ' => 'νậŀųề', 'キー' => '値' } }
98-
let(:json) { '{"key":"value", {"ķęŷ":"νậŀųề" }, {"キー":"値" }' }
49+
it do
50+
Tempfile.new do |file|
51+
file.write(json)
52+
file.flush
9953

100-
it {
101-
expect(OpenURI).to receive(:open_uri).with(filename, {}).and_return(json)
102-
if Puppet::PUPPETVERSION[0].to_i < 8
103-
expect(PSON).to receive(:load).with(json).and_return(data).once
104-
else
105-
expect(JSON).to receive(:parse).with(json).and_return(data).once
54+
is_expected.to run.with_params(filename, {'default' => 'value'}).and_return({'default' => 'value'})
10655
end
107-
expect(subject).to run.with_params(filename).and_return(data)
108-
}
109-
end
110-
111-
context 'when an existing URL (with username and password) is specified' do
112-
let(:filename) do
113-
'https://user1:[email protected]/myhash.json'
11456
end
115-
let(:url_no_auth) { 'https://example.local/myhash.json' }
116-
let(:basic_auth) { { http_basic_authentication: ['user1', 'pass1'] } }
117-
let(:data) { { 'key' => 'value', 'ķęŷ' => 'νậŀųề', 'キー' => '値' } }
118-
let(:json) { '{"key":"value", {"ķęŷ":"νậŀųề" }, {"キー":"値" }' }
119-
120-
it {
121-
expect(OpenURI).to receive(:open_uri).with(url_no_auth, basic_auth).and_return(json)
122-
if Puppet::PUPPETVERSION[0].to_i < 8
123-
expect(PSON).to receive(:load).with(json).and_return(data).once
124-
else
125-
expect(JSON).to receive(:parse).with(json).and_return(data).once
126-
end
127-
expect(subject).to run.with_params(filename).and_return(data)
128-
}
12957
end
13058

131-
context 'when an existing URL (with username) is specified' do
59+
context 'when an existing URL is specified' do
13260
let(:filename) do
133-
'https://user1@example.local/myhash.json'
61+
'https://example.com/myhash.json'
13462
end
135-
let(:url_no_auth) { 'https://example.local/myhash.json' }
136-
let(:basic_auth) { { http_basic_authentication: ['user1', ''] } }
13763
let(:data) { { 'key' => 'value', 'ķęŷ' => 'νậŀųề', 'キー' => '値' } }
138-
let(:json) { '{"key":"value", {"ķęŷ":"νậŀųề" }, {"キー":"値" }' }
64+
let(:json) { '{"key":"value", "ķęŷ":"νậŀųề", "キー":"値" }' }
13965

14066
it {
141-
expect(OpenURI).to receive(:open_uri).with(url_no_auth, basic_auth).and_return(json)
142-
if Puppet::PUPPETVERSION[0].to_i < 8
143-
expect(PSON).to receive(:load).with(json).and_return(data).once
144-
else
145-
expect(JSON).to receive(:parse).with(json).and_return(data).once
146-
end
67+
expect(URI).to receive(:open).with(filename).and_yield(json)
14768
expect(subject).to run.with_params(filename).and_return(data)
14869
}
14970
end
15071

15172
context 'when the URL output could not be parsed, with default specified' do
15273
let(:filename) do
153-
'https://example.local/myhash.json'
74+
'https://example.com/myhash.json'
15475
end
15576
let(:json) { ',;{"key":"value"}' }
15677

15778
it {
158-
expect(OpenURI).to receive(:open_uri).with(filename, {}).and_return(json)
79+
expect(URI).to receive(:open).with(filename).and_yield(json)
15980
if Puppet::PUPPETVERSION[0].to_i < 8
16081
expect(PSON).to receive(:load).with(json).once.and_raise StandardError, 'Something terrible have happened!'
16182
else
162-
expect(JSON).to receive(:parse).with(json).once.and_raise StandardError, 'Something terrible have happened!'
83+
expect(JSON).to receive(:load).with(json).once.and_raise StandardError, 'Something terrible have happened!'
16384
end
164-
expect(subject).to run.with_params(filename, 'default' => 'value').and_return('default' => 'value')
85+
expect(subject).to run.with_params(filename, {'default' => 'value'}).and_return({'default' => 'value'})
16586
}
16687
end
16788

16889
context 'when the URL does not exist, with default specified' do
16990
let(:filename) do
170-
'https://example.local/myhash.json'
91+
'https://example.com/myhash.json'
17192
end
17293

17394
it {
174-
expect(OpenURI).to receive(:open_uri).with(filename, {}).and_raise OpenURI::HTTPError, '404 File not Found'
175-
expect(subject).to run.with_params(filename, 'default' => 'value').and_return('default' => 'value')
95+
expect(URI).to receive(:open).with(filename).and_raise OpenURI::HTTPError, '404 File not Found'
96+
expect(subject).to run.with_params(filename, {'default' => 'value'}).and_return({'default' => 'value'})
17697
}
17798
end
17899
end

0 commit comments

Comments
 (0)