Skip to content

Commit 66c1452

Browse files
authoredAug 2, 2024··
Fix frozen string literal issue (#27)
1 parent 292cb22 commit 66c1452

File tree

5 files changed

+170
-10
lines changed

5 files changed

+170
-10
lines changed
 

‎.github/workflows/ci.yml

+18-2
Original file line numberDiff line numberDiff line change
@@ -12,5 +12,21 @@ permissions:
1212
contents: read
1313

1414
jobs:
15-
Shared:
16-
uses: fog/.github/.github/workflows/ci.yml@v1.4.0
15+
test:
16+
17+
runs-on: ubuntu-latest
18+
19+
strategy:
20+
matrix:
21+
ruby-version: ['3.0', '3.1', '3.2', '3.3', 'head']
22+
continue-on-error: ${{ matrix.ruby-version == 'head' }}
23+
24+
steps:
25+
- uses: actions/checkout@v4
26+
- name: Set up Ruby
27+
uses: ruby/setup-ruby@v1
28+
with:
29+
ruby-version: ${{ matrix.ruby-version }}
30+
bundler-cache: true # runs 'bundle install' and caches installed gems automatically
31+
- name: Run tests
32+
run: bundle exec rake RUBYOPT="--enable-frozen-string-literal"

‎lib/fog/parsers/base.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ def reset
1818
end
1919

2020
def characters(string)
21-
@value ||= ''
21+
@value ||= +''
2222
@value << string
2323
end
2424

‎lib/fog/to_hash_document.rb

+7-5
Original file line numberDiff line numberDiff line change
@@ -5,20 +5,22 @@ def initialize
55
end
66

77
def characters(string)
8-
@value ||= ""
8+
@value ||= +''
99
@value << string.strip
1010
end
1111

1212
def end_element(name)
1313
last = @stack.pop
14-
if last.empty? && @value.empty?
15-
@stack.last[name.to_sym] = ""
14+
@stack.push({}) if @stack.empty?
15+
16+
if last&.empty? && @value.empty?
17+
@stack.last[name.to_sym] = +''
1618
elsif last == { :i_nil => "true" }
1719
@stack.last[name.to_sym] = nil
1820
elsif !@value.empty?
1921
@stack.last[name.to_sym] = @value
2022
end
21-
@value = ""
23+
@value = +''
2224
end
2325

2426
def body
@@ -30,7 +32,7 @@ def response
3032
end
3133

3234
def start_element(name, attributes = [])
33-
@value = ""
35+
@value = +''
3436
parsed_attributes = {}
3537
until attributes.empty?
3638
if attributes.first.is_a?(Array)

‎lib/fog/xml/response.rb

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ class Response
44
def initialize(parser)
55
@parser = parser
66
@data_stream = Nokogiri::XML::SAX::PushParser.new(parser)
7-
@response_string = ""
7+
@response_string = +''
88
end
99

1010
def call(chunk, _remaining, _total)
@@ -14,7 +14,7 @@ def call(chunk, _remaining, _total)
1414

1515
def rewind
1616
@parser.reset
17-
@response_string = ""
17+
@response_string = +''
1818
end
1919

2020
def finish

‎spec/fog/to_hash_document_spec.rb

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
# frozen_string_literal: true
2+
3+
require 'minitest_helper'
4+
require 'fog/xml'
5+
6+
# We expose accessors just for testing purposes
7+
Fog::ToHashDocument.attr_accessor(:value, :stack)
8+
9+
describe Fog::ToHashDocument do
10+
before do
11+
@document = Fog::ToHashDocument.new
12+
end
13+
14+
describe '#characters' do
15+
it 'appends characters to @value' do
16+
@document.characters('some text')
17+
_(@document.value).must_equal 'some text'
18+
end
19+
20+
it 'strips whitespace from characters' do
21+
@document.characters(' some text ')
22+
_(@document.value).must_equal 'some text'
23+
end
24+
end
25+
26+
describe '#end_element' do
27+
before do
28+
@document.stack << {}
29+
@document.characters('some text')
30+
end
31+
32+
it 'adds element with text content to the stack' do
33+
@document.end_element('element')
34+
35+
expected = { element: 'some text' }
36+
_(@document.stack.last).must_equal(expected)
37+
end
38+
39+
it 'can mutate the new empty value' do
40+
@document.end_element('element')
41+
42+
_(@document.value).must_equal('')
43+
44+
# Mutate the new empty value even when frozen string literals are enabled
45+
_(@document.characters('one'))
46+
end
47+
48+
it 'adds empty string if element is empty and value is empty' do
49+
@document.value = ''
50+
51+
@document.end_element('element')
52+
53+
expected = { element: '' }
54+
_(@document.stack.last).must_equal(expected)
55+
end
56+
57+
it 'adds nil if element has :i_nil attribute' do
58+
@document.stack.last[:i_nil] = 'true'
59+
@document.value = ''
60+
61+
@document.end_element('element')
62+
63+
expected = { element: nil }
64+
_(@document.stack.last).must_equal(expected)
65+
end
66+
end
67+
68+
describe '#body' do
69+
it 'returns the first element of the stack' do
70+
@document.stack << { key: 'value' }
71+
72+
expected = { key: 'value' }
73+
_(@document.body).must_equal(expected)
74+
end
75+
end
76+
77+
describe '#response' do
78+
it 'returns the body' do
79+
@document.stack << { key: 'value' }
80+
81+
expected = { key: 'value' }
82+
_(@document.response).must_equal(expected)
83+
end
84+
end
85+
86+
describe '#start_element' do
87+
it 'parses attributes correctly' do
88+
@document.start_element('element', [%w[key value]])
89+
90+
expected = { key: 'value' }
91+
_(@document.stack.last).must_equal(expected)
92+
end
93+
94+
it 'handles elements without attributes' do
95+
@document.start_element('element')
96+
97+
_(@document.stack.last).must_equal({})
98+
end
99+
100+
it 'adds nested elements to the stack' do
101+
@document.start_element('parent')
102+
@document.start_element('child')
103+
104+
_(@document.stack).must_equal([{ child: {} }, { child: {} }, {}])
105+
end
106+
107+
it 'adds nested elements with attributes to the stack' do
108+
@document.start_element('parent')
109+
@document.start_element('child', [%w[key value]])
110+
expected = [
111+
{ child: { key: 'value' } },
112+
{ child: { key: 'value' } },
113+
{ key: 'value' }
114+
]
115+
116+
_(@document.stack).must_equal(expected)
117+
end
118+
119+
it 'handles multiple children elements correctly' do
120+
@document.start_element('parent')
121+
@document.start_element('child1')
122+
@document.end_element('child1')
123+
@document.start_element('child2', [%w[key value]])
124+
@document.end_element('child2')
125+
expected = {
126+
child1: '',
127+
child2: { key: 'value' }
128+
}
129+
130+
_(@document.stack.first).must_equal(expected)
131+
end
132+
133+
it 'handles text content within elements' do
134+
@document.start_element('parent')
135+
@document.characters('some text')
136+
@document.end_element('parent')
137+
138+
expected = { parent: 'some text' }
139+
_(@document.stack.first).must_equal(expected)
140+
end
141+
end
142+
end

0 commit comments

Comments
 (0)
Please sign in to comment.