diff --git a/lib/puppet/provider/ini_section/ruby.rb b/lib/puppet/provider/ini_section/ruby.rb new file mode 100644 index 00000000..14083f8c --- /dev/null +++ b/lib/puppet/provider/ini_section/ruby.rb @@ -0,0 +1,89 @@ +# frozen_string_literal: true + +require File.expand_path('../../util/ini_file', __dir__) + +Puppet::Type.type(:ini_section).provide(:ruby) do + def self.instances + desc ' + Creates new ini_section file, a specific config file with a provider that uses + this as its parent and implements the method + self.file_path, and that will provide the value for the path to the + ini file.' + raise(Puppet::Error, 'Ini_section only support collecting instances when a file path is hard coded') unless respond_to?(:file_path) + + # figure out what to do about the seperator + ini_file = Puppet::Util::IniFile.new(file_path, '=') + resources = [] + ini_file.section_names.each do |section_name| + next if section_name.empty? + resources.push( + new( + name: namevar(section_name), + ensure: :present, + ), + ) + end + resources + end + + def self.namevar(section_name) + section_name + end + + def exists? + ini_file.section_names.include?(section) + end + + def create + ini_file.set_value(section) + ini_file.save + @ini_file = nil + end + + def destroy + ini_file.remove_section(section) + ini_file.save + @ini_file = nil + end + + def file_path + # this method is here to support purging and sub-classing. + # if a user creates a type and subclasses our provider and provides a + # 'file_path' method, then they don't have to specify the + # path as a parameter for every ini_section declaration. + # This implementation allows us to support that while still + # falling back to the parameter value when necessary. + if self.class.respond_to?(:file_path) + self.class.file_path + else + resource[:path] + end + end + + def section + # this method is here so that it can be overridden by a child provider + resource[:section] + end + + def section_prefix + if resource.class.validattr?(:section_prefix) + resource[:section_prefix] || '[' + else + '[' + end + end + + def section_suffix + if resource.class.validattr?(:section_suffix) + resource[:section_suffix] || ']' + else + ']' + end + end + + private + + def ini_file + @ini_file ||= Puppet::Util::IniFile.new(file_path, '=', section_prefix, section_suffix, ' ', 2) + end +end diff --git a/lib/puppet/type/ini_section.rb b/lib/puppet/type/ini_section.rb new file mode 100644 index 00000000..9f562aae --- /dev/null +++ b/lib/puppet/type/ini_section.rb @@ -0,0 +1,52 @@ +# frozen_string_literal: true + +Puppet::Type.newtype(:ini_section) do + desc 'ini_section is used to manage a single section in an INI file' + ensurable do + desc 'Ensurable method handles modeling creation. It creates an ensure property' + newvalue(:present) do + provider.create + end + newvalue(:absent) do + provider.destroy + end + def insync?(current) + if @resource[:refreshonly] + true + else + current == should + end + end + defaultto :present + end + + newparam(:name, namevar: true) do + desc 'An arbitrary name used as the identity of the resource.' + end + + newparam(:section) do + desc 'The name of the section in the ini file which should be managed.' + defaultto('') + end + + newparam(:path) do + desc 'The ini file Puppet will ensure contains the specified section.' + validate do |value| + raise(Puppet::Error, _("File paths must be fully qualified, not '%{value}'") % { value: value }) unless Puppet::Util.absolute_path?(value) + end + end + + newparam(:section_prefix) do + desc 'The prefix to the section name\'s header.' + defaultto('[') + end + + newparam(:section_suffix) do + desc 'The suffix to the section name\'s header.' + defaultto(']') + end + + autorequire(:file) do + Pathname.new(self[:path]).parent.to_s + end +end diff --git a/lib/puppet/util/ini_file.rb b/lib/puppet/util/ini_file.rb index 3acbf7a4..c337e65c 100644 --- a/lib/puppet/util/ini_file.rb +++ b/lib/puppet/util/ini_file.rb @@ -122,12 +122,17 @@ def remove_setting(section_name, setting) # was modified. section_index = @section_names.index(section_name) decrement_section_line_numbers(section_index + 1) + end - return unless section.empty? + def remove_section(section_name) + section = @sections_hash[section_name] + return unless section + + lines.replace(lines[0..(section.start_line - 1)] + lines[(section.end_line + 1)..-1]) + + section_index = @section_names.index(section.name) + decrement_section_line_numbers(section_index + 1, amount: section.length) - # By convention, it's time to remove this newly emptied out section - lines.delete_at(section.start_line) - decrement_section_line_numbers(section_index + 1) @section_names.delete_at(section_index) @sections_hash.delete(section.name) end @@ -222,7 +227,6 @@ def read_section(name, start_line, line_iter) end_line_num = start_line min_indentation = nil empty = true - empty_line_count = 0 loop do line, line_num = line_iter.peek if line_num.nil? || @section_regex.match(line) @@ -230,7 +234,7 @@ def read_section(name, start_line, line_iter) # when it's empty, we must be sure it's thought of as new, # which is signalled with a nil ending line end_line_num = nil if name == '' && empty - return Section.new(name, start_line, end_line_num, settings, min_indentation, empty_line_count) + return Section.new(name, start_line, end_line_num, settings, min_indentation) end if (match = @setting_regex.match(line)) settings[match[2]] = match[4] @@ -238,8 +242,6 @@ def read_section(name, start_line, line_iter) min_indentation = [indentation, min_indentation || indentation].min end end_line_num = line_num - empty_line_count += 1 if line == "\n" - empty = false line_iter.next end @@ -311,10 +313,10 @@ def insert_inline_setting_line(result, section, complete_setting) # Utility method; given a section index (index into the @section_names # array), decrement the start/end line numbers for that section and all # all of the other sections that appear *after* the specified section. - def decrement_section_line_numbers(section_index) + def decrement_section_line_numbers(section_index, amount: 1) @section_names[section_index..(@section_names.length - 1)].each do |name| section = @sections_hash[name] - section.decrement_line_nums + section.decrement_line_nums(amount) end end diff --git a/lib/puppet/util/ini_file/section.rb b/lib/puppet/util/ini_file/section.rb index 4e29b5fd..8dc73fdf 100644 --- a/lib/puppet/util/ini_file/section.rb +++ b/lib/puppet/util/ini_file/section.rb @@ -14,14 +14,13 @@ class Section # `end_line` of `nil`. # * `start_line` and `end_line` will be set to `nil` for a new non-global # section. - def initialize(name, start_line, end_line, settings, indentation, empty_line_count = 0) + def initialize(name, start_line, end_line, settings, indentation) @name = name @start_line = start_line @end_line = end_line @existing_settings = settings.nil? ? {} : settings @additional_settings = {} @indentation = indentation - @empty_line_count = empty_line_count end attr_reader :name, :start_line, :end_line, :additional_settings, :indentation @@ -51,7 +50,11 @@ def existing_setting?(setting_name) # the global section is empty whenever it's new; # other sections are empty when they have no lines def empty? - global? ? new_section? : (start_line == end_line || (end_line && (end_line - @empty_line_count)) == start_line) + global? ? new_section? : start_line == end_line + end + + def length + end_line - start_line + 1 end def update_existing_setting(setting_name, value) @@ -80,9 +83,9 @@ def set_additional_setting(setting_name, value) # Decrement the start and end line numbers for the section (if they are # defined); this is intended to be called when a setting is removed # from a section that comes before this section in the ini file. - def decrement_line_nums - @start_line -= 1 if @start_line - @end_line -= 1 if @end_line + def decrement_line_nums(amount) + @start_line -= amount if @start_line + @end_line -= amount if @end_line end # Increment the start and end line numbers for the section (if they are diff --git a/spec/acceptance/ini_setting_spec.rb b/spec/acceptance/ini_setting_spec.rb index 140a35e5..93c75cc4 100644 --- a/spec/acceptance/ini_setting_spec.rb +++ b/spec/acceptance/ini_setting_spec.rb @@ -97,7 +97,7 @@ subject { super().content } it { is_expected.to match %r{four = five} } - it { is_expected.not_to match %r{\[one\]} } + it { is_expected.to match %r{\[one\]} } it { is_expected.not_to match %r{two = three} } end end @@ -296,7 +296,8 @@ describe '#content' do subject { super().content } - it { is_expected.to be_empty } + it { is_expected.to match %r{\[section1\]} } + it { is_expected.not_to match %r{valueinsection1 = newValue} } end end end diff --git a/spec/acceptance/ini_subsetting_spec.rb b/spec/acceptance/ini_subsetting_spec.rb index 297ce4c7..af16b12d 100644 --- a/spec/acceptance/ini_subsetting_spec.rb +++ b/spec/acceptance/ini_subsetting_spec.rb @@ -130,7 +130,7 @@ describe '#content' do subject { super().content } - it { is_expected.not_to match %r{\[one\]} } + it { is_expected.to match %r{\[one\]} } it { is_expected.not_to match %r{key =} } it { is_expected.not_to match %r{alphabet} } it { is_expected.not_to match %r{betatrons} } diff --git a/spec/fixtures/inherit_ini_section/lib/puppet/provider/inherit_ini_section/ini_section.rb b/spec/fixtures/inherit_ini_section/lib/puppet/provider/inherit_ini_section/ini_section.rb new file mode 100644 index 00000000..a4b9b4c9 --- /dev/null +++ b/spec/fixtures/inherit_ini_section/lib/puppet/provider/inherit_ini_section/ini_section.rb @@ -0,0 +1,12 @@ +Puppet::Type.type(:inherit_ini_section).provide( + :ini_section, + parent: Puppet::Type.type(:ini_section).provider(:ruby), +) do + def self.namevar(section) + section + end + + def self.file_path + File.expand_path(File.dirname(__FILE__) + '/../../../../../../tmp/inherit_inifile.cfg') + end +end diff --git a/spec/fixtures/inherit_ini_section/lib/puppet/type/inherit_ini_section.rb b/spec/fixtures/inherit_ini_section/lib/puppet/type/inherit_ini_section.rb new file mode 100644 index 00000000..b15a76cb --- /dev/null +++ b/spec/fixtures/inherit_ini_section/lib/puppet/type/inherit_ini_section.rb @@ -0,0 +1,5 @@ +Puppet::Type.newtype(:inherit_ini_section) do + ensurable + newparam(:section, namevar: true) + newproperty(:value) +end diff --git a/spec/unit/puppet/provider/ini_section/inheritance_spec.rb b/spec/unit/puppet/provider/ini_section/inheritance_spec.rb new file mode 100644 index 00000000..b013c788 --- /dev/null +++ b/spec/unit/puppet/provider/ini_section/inheritance_spec.rb @@ -0,0 +1,63 @@ +# frozen_string_literal: true + +require 'spec_helper' + +# This is a reduced version of ruby_spec.rb just to ensure we can subclass as +# documented +$LOAD_PATH << './spec/fixtures/inherit_ini_section/lib' + +describe Puppet::Type.type(:inherit_ini_section).provider(:ini_section) do + include PuppetlabsSpec::Files + + let(:tmpfile) { tmpfilename('inherit_ini_section_test') } + + def validate_file(expected_content, tmpfile) + expect(File.read(tmpfile)).to eq(expected_content) + end + + before :each do + File.write(tmpfile, orig_content) + end + + context 'when calling instances' do + let(:orig_content) { '' } + + it 'parses nothing when the file is empty' do + allow(described_class).to receive(:file_path).and_return(tmpfile) + expect(described_class.instances).to eq([]) + end + + context 'when the file has contents' do + let(:orig_content) do + <<-INIFILE + [red] + # A comment + red = blue + [green] + green = purple + INIFILE + end + + it 'parses the results' do + allow(described_class).to receive(:file_path).and_return(tmpfile) + instances = described_class.instances + expect(instances.size).to eq(2) + # inherited version of namevar flattens the names + names = instances.map do |instance| instance.instance_variable_get(:@property_hash)[:name] end # rubocop:disable Style/BlockDelimiters + expect(names.sort).to eq(['green', 'red']) + end + end + end + + context 'when ensuring that a setting is present' do + let(:orig_content) { '' } + + it 'adds a value to the file' do + allow(described_class).to receive(:file_path).and_return(tmpfile) + resource = Puppet::Type::Inherit_ini_section.new(section: 'set_this') + provider = described_class.new(resource) + provider.create + expect(validate_file("[set_this]\n", tmpfile)).to be_truthy + end + end +end diff --git a/spec/unit/puppet/provider/ini_section/ruby_spec.rb b/spec/unit/puppet/provider/ini_section/ruby_spec.rb new file mode 100644 index 00000000..872ada08 --- /dev/null +++ b/spec/unit/puppet/provider/ini_section/ruby_spec.rb @@ -0,0 +1,290 @@ +# frozen_string_literal: true + +require 'spec_helper' +require 'puppet' + +provider_class = Puppet::Type.type(:ini_section).provider(:ruby) +describe provider_class do + include PuppetlabsSpec::Files + + let(:tmpfile) { tmpfilename('ini_section_test') } + let(:emptyfile) { tmpfilename('ini_section_test_empty') } + + let(:common_params) do + { + title: 'ini_section_ensure_present_test', + path: tmpfile, + } + end + + def validate_file(expected_content, tmpfile) + expect(File.read(tmpfile)).to eq(expected_content) + end + + before :each do + File.write(tmpfile, orig_content) + File.write(emptyfile, '') + end + + context 'when calling instances' do + let :orig_content do + '' + end + + it 'fails when file path is not set' do + expect { + provider_class.instances + }.to raise_error(Puppet::Error, 'Ini_section only support collecting instances when a file path is hard coded') + end + + context 'when file path is set by a child class' do + child_one = Class.new(provider_class) do + def self.file_path + emptyfile + end + end + it 'returns [] when file is empty' do + allow(child_one).to receive(:file_path).and_return(emptyfile) + expect(child_one.instances).to eq([]) + end + + child_two = Class.new(provider_class) do + def self.file_path + '/some/file/path' + end + end + it 'overrides the provider instances file_path' do + resource = Puppet::Type::Ini_section.new(common_params) + provider = child_two.new(resource) + expect(provider.file_path).to eq('/some/file/path') + end + end + + context 'when file has contents' do + let(:orig_content) do + <<-INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + INIFILE + end + + it 'is able to parse the results' do + child_three = Class.new(provider_class) do + def self.file_path + '/some/file/path' + end + end + expect(child_three).to receive(:file_path).twice.and_return(tmpfile) + expect(child_three.instances.size).to eq(3) + expected_array = [ + { name: 'section1' }, + { name: 'section2' }, + { name: 'section:sub' }, + ] + real_array = [] + ensure_array = [] + child_three.instances.each do |x| + prop_hash = x.instance_variable_get(:@property_hash) + ensure_value = prop_hash.delete(:ensure) + ensure_array.push(ensure_value) + real_array.push(prop_hash) + end + expect(ensure_array.uniq).to eq([:present]) + expect((real_array - expected_array) && (expected_array - real_array)).to eq([]) + end + end + end + + context 'when ensuring that a section is present' do + let(:orig_content) do + <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + + -nonstandard- + shoes = purple + INIFILE + end + + expected_content_one = <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + + -nonstandard- + shoes = purple + + [yippee] + INIFILE + it 'adds a missing section' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'yippee')) + provider = described_class.new(resource) + expect(provider.exists?).to be false + provider.create + validate_file(expected_content_one, tmpfile) + end + + expected_content_two = <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + + -nonstandard- + shoes = purple + INIFILE + it 'does nothing when the section is already there' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'section2')) + provider = described_class.new(resource) + expect(provider.exists?).to be true + provider.create + validate_file(expected_content_two, tmpfile) + end + + it 'does nothing when the section with custom prefix/suffix is already there' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'nonstandard', section_prefix: '-', section_suffix: '-')) + provider = described_class.new(resource) + expect(provider.exists?).to be true + provider.create + validate_file(expected_content_two, tmpfile) + end + end + + context 'when ensuring that a section is absent' do + let(:orig_content) do + <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + + -nonstandard- + shoes = purple + INIFILE + end + + expected_content_one = <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section:sub] + subby=bar + #another comment + ; yet another comment + + -nonstandard- + shoes = purple + INIFILE + it 'removes an existing section' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'section2')) + provider = described_class.new(resource) + expect(provider.exists?).to be true + provider.destroy + validate_file(expected_content_one, tmpfile) + end + + expected_content_two = <<~INIFILE + # This is a comment + [section1] + ; This is also a comment + foo=foovalue + + bar = barvalue + main = true + [section2] + + foo= foovalue2 + baz=bazvalue + url = http://192.168.1.1:8080 + [section:sub] + subby=bar + #another comment + ; yet another comment + + INIFILE + it 'removes an existing section with custom prefix/suffix' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'nonstandard', section_prefix: '-', section_suffix: '-')) + provider = described_class.new(resource) + expect(provider.exists?).to be true + provider.destroy + validate_file(expected_content_two, tmpfile) + end + + it 'does nothing when the section does not exist' do + resource = Puppet::Type::Ini_section.new(common_params.merge(section: 'yippee')) + provider = described_class.new(resource) + expect(provider.exists?).to be false + provider.destroy + validate_file(orig_content, tmpfile) + end + end +end diff --git a/spec/unit/puppet/provider/ini_setting/ruby_spec.rb b/spec/unit/puppet/provider/ini_setting/ruby_spec.rb index ef7da0eb..30c5e693 100644 --- a/spec/unit/puppet/provider/ini_setting/ruby_spec.rb +++ b/spec/unit/puppet/provider/ini_setting/ruby_spec.rb @@ -1002,6 +1002,7 @@ def self.file_path #another comment ; yet another comment + -nonstandard- INIFILE it 'removes a setting with pre/suffix that exists' do resource = Puppet::Type::Ini_setting.new(common_params.merge(section: 'nonstandard', setting: 'shoes', ensure: 'absent', section_prefix: '-', section_suffix: '-')) @@ -1108,69 +1109,6 @@ def self.file_path provider.destroy validate_file(expected_content_five, tmpfile) end - - expected_content_six = <<~INIFILE - [section1] - ; This is also a comment - foo=foovalue - - bar = barvalue - main = true - [section2] - - foo= foovalue2 - baz=bazvalue - url = http://192.168.1.1:8080 - [section3] - # com = ment - uncom = ment - [section:sub] - subby=bar - #another comment - ; yet another comment - - -nonstandard- - shoes = purple - INIFILE - it 'removes the section when removing the last line in the section' do - resource = Puppet::Type::Ini_setting.new(common_params.merge(section: 'section4', setting: 'uncom', ensure: 'absent')) - provider = described_class.new(resource) - expect(provider.exists?).to be true - provider.destroy - validate_file(expected_content_six, tmpfile) - end - end - - context 'when section has only empty line' do - let(:orig_content) do - <<~INIFILE - [section1] - foo=foovalue - - - [section2] - - foo= foovalue2 - baz=bazvalue - url = http:// - INIFILE - end - - expected_content = <<~INIFILE - [section2] - - foo= foovalue2 - baz=bazvalue - url = http:// - INIFILE - - it 'remove empty section' do - resource = Puppet::Type::Ini_setting.new(common_params.merge(section: 'section1', setting: 'foo', ensure: 'absent')) - provider = described_class.new(resource) - expect(provider.exists?).to be true - provider.destroy - validate_file(expected_content, tmpfile) - end end context 'when dealing with indentation in sections' do diff --git a/spec/unit/puppet/provider/ini_subsetting/ruby_spec.rb b/spec/unit/puppet/provider/ini_subsetting/ruby_spec.rb index 9236fb68..49a186a2 100644 --- a/spec/unit/puppet/provider/ini_subsetting/ruby_spec.rb +++ b/spec/unit/puppet/provider/ini_subsetting/ruby_spec.rb @@ -345,7 +345,9 @@ def validate_file(expected_content, tmpfile) something = else INIFILE - expected_content_two = '' + expected_content_two = <<-INIFILE + [main] + INIFILE it 'removes the subsetting when the it is empty' do resource = Puppet::Type::Ini_subsetting.new(common_params.merge(setting: 'reports', subsetting: 'http', subsetting_separator: ',')) diff --git a/spec/unit/puppet/type/ini_section_spec.rb b/spec/unit/puppet/type/ini_section_spec.rb new file mode 100644 index 00000000..7a2f1b5c --- /dev/null +++ b/spec/unit/puppet/type/ini_section_spec.rb @@ -0,0 +1,104 @@ +# frozen_string_literal: true + +require 'spec_helper' + +ini_section = Puppet::Type.type(:ini_section) + +describe ini_section do + describe 'path validation' do + subject(:ini_section_path) { described_class.new(name: 'foo', path: path) } + + context 'when on posix platforms' do + before(:each) do + Puppet.features.stub(:posix?) { true } + Puppet.features.stub(:microsoft_windows?) { false } + Puppet::Util::Platform.stub(:windows?) { false } + end + + context 'with an absolute path' do + let(:path) { '/absolute/path' } + + it { expect { ini_section_path }.not_to raise_exception } + end + + context 'with a relative path' do + let(:path) { 'relative/path' } + + it { expect { ini_section_path }.to raise_exception(Puppet::ResourceError) } + end + end + + context 'when on windows platforms' do + before(:each) do + Puppet.features.stub(:posix?) { false } + Puppet.features.stub(:microsoft_windows?) { true } + Puppet::Util::Platform.stub(:windows?) { true } + end + + context 'with an absolute path with front slashes' do + let(:path) { 'c:/absolute/path' } + + it { expect { ini_section_path }.not_to raise_exception } + end + + context 'with an absolute path with backslashes' do + let(:path) { 'c:\absolute\path' } + + it { expect { ini_section_path }.not_to raise_exception } + end + + context 'with an absolute path with mixed slashes' do + let(:path) { 'c:/absolute\path' } + + it { expect { ini_section_path }.not_to raise_exception } + end + + context 'with a relative path with front slashes' do + let(:path) { 'relative/path' } + + it { expect { ini_section_path }.to raise_exception(Puppet::ResourceError) } + end + + context 'with a relative path with back slashes' do + let(:path) { 'relative\path' } + + it { expect { ini_section_path }.to raise_exception(Puppet::ResourceError) } + end + end + end + + describe 'when parent of :path is in the catalog' do + ['posix', 'windows'].each do |platform| + context "when on #{platform} platforms" do + before(:each) do + Puppet.features.stub(:posix?) { platform == 'posix' } + Puppet.features.stub(:microsoft_windows?) { platform == 'windows' } + Puppet::Util::Platform.stub(:windows?) { platform == 'windows' } + end + + let(:file_path) { (platform == 'posix') ? '/tmp' : 'c:/tmp' } + let(:file_resource) { Puppet::Type.type(:file).new(name: file_path) } + let(:ini_section_resource) { described_class.new(name: 'foo', path: "#{file_path}/foo.ini") } + let(:auto_req) do + catalog = Puppet::Resource::Catalog.new + catalog.add_resource(file_resource) + catalog.add_resource(ini_section_resource) + + ini_section_resource.autorequire + end + + it 'creates relationship' do + expect(auto_req.size).to be 1 + end + + it 'links to ini_section resource' do + expect(auto_req[0].target).to eq(ini_section_resource) + end + + it 'autorequires parent directory' do + expect(auto_req[0].source).to eq(file_resource) + end + end + end + end +end