1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
|
#! /usr/bin/env ruby
require 'spec_helper'
require 'puppet/indirector/yaml'
describe Puppet::Indirector::Yaml do
include PuppetSpec::Files
class TestSubject
attr_accessor :name
end
before :all do
@indirection = stub 'indirection', :name => :my_yaml, :register_terminus_type => nil
Puppet::Indirector::Indirection.expects(:instance).with(:my_yaml).returns(@indirection)
module MyYaml; end
@store_class = class MyYaml::MyType < Puppet::Indirector::Yaml
self
end
end
before :each do
@store = @store_class.new
@subject = TestSubject.new
@subject.name = :me
@dir = tmpdir("yaml_indirector")
Puppet[:clientyamldir] = @dir
Puppet.run_mode.stubs(:master?).returns false
@request = stub 'request', :key => :me, :instance => @subject
end
let(:serverdir) { File.expand_path("/server/yaml/dir") }
let(:clientdir) { File.expand_path("/client/yaml/dir") }
describe "when choosing file location" do
it "should use the server_datadir if the run_mode is master" do
Puppet.run_mode.stubs(:master?).returns true
Puppet[:yamldir] = serverdir
expect(@store.path(:me)).to match(/^#{serverdir}/)
end
it "should use the client yamldir if the run_mode is not master" do
Puppet.run_mode.stubs(:master?).returns false
Puppet[:clientyamldir] = clientdir
expect(@store.path(:me)).to match(/^#{clientdir}/)
end
it "should use the extension if one is specified" do
Puppet.run_mode.stubs(:master?).returns true
Puppet[:yamldir] = serverdir
expect(@store.path(:me,'.farfignewton')).to match(%r{\.farfignewton$})
end
it "should assume an extension of .yaml if none is specified" do
Puppet.run_mode.stubs(:master?).returns true
Puppet[:yamldir] = serverdir
expect(@store.path(:me)).to match(%r{\.yaml$})
end
it "should store all files in a single file root set in the Puppet defaults" do
expect(@store.path(:me)).to match(%r{^#{@dir}})
end
it "should use the terminus name for choosing the subdirectory" do
expect(@store.path(:me)).to match(%r{^#{@dir}/my_yaml})
end
it "should use the object's name to determine the file name" do
expect(@store.path(:me)).to match(%r{me.yaml$})
end
['../foo', '..\\foo', './../foo', '.\\..\\foo',
'/foo', '//foo', '\\foo', '\\\\goo',
"test\0/../bar", "test\0\\..\\bar",
"..\\/bar", "/tmp/bar", "/tmp\\bar", "tmp\\bar",
" / bar", " /../ bar", " \\..\\ bar",
"c:\\foo", "c:/foo", "\\\\?\\UNC\\bar", "\\\\foo\\bar",
"\\\\?\\c:\\foo", "//?/UNC/bar", "//foo/bar",
"//?/c:/foo",
].each do |input|
it "should resist directory traversal attacks (#{input.inspect})" do
expect { @store.path(input) }.to raise_error(ArgumentError)
end
end
end
describe "when storing objects as YAML" do
it "should only store objects that respond to :name" do
@request.stubs(:instance).returns Object.new
expect { @store.save(@request) }.to raise_error(ArgumentError)
end
end
describe "when retrieving YAML" do
it "should read YAML in from disk and convert it to Ruby objects" do
@store.save(Puppet::Indirector::Request.new(:my_yaml, :save, "testing", @subject))
expect(@store.find(Puppet::Indirector::Request.new(:my_yaml, :find, "testing", nil)).name).to eq(:me)
end
it "should fail coherently when the stored YAML is invalid" do
saved_structure = Struct.new(:name).new("testing")
@store.save(Puppet::Indirector::Request.new(:my_yaml, :save, "testing", saved_structure))
File.open(@store.path(saved_structure.name), "w") do |file|
file.puts "{ invalid"
end
expect {
@store.find(Puppet::Indirector::Request.new(:my_yaml, :find, "testing", nil))
}.to raise_error(Puppet::Error, /Could not parse YAML data/)
end
end
describe "when searching" do
it "should return an array of fact instances with one instance for each file when globbing *" do
@request = stub 'request', :key => "*", :instance => @subject
@one = mock 'one'
@two = mock 'two'
@store.expects(:path).with(@request.key,'').returns :glob
Dir.expects(:glob).with(:glob).returns(%w{one.yaml two.yaml})
YAML.expects(:load_file).with("one.yaml").returns @one;
YAML.expects(:load_file).with("two.yaml").returns @two;
expect(@store.search(@request)).to eq([@one, @two])
end
it "should return an array containing a single instance of fact when globbing 'one*'" do
@request = stub 'request', :key => "one*", :instance => @subject
@one = mock 'one'
@store.expects(:path).with(@request.key,'').returns :glob
Dir.expects(:glob).with(:glob).returns(%w{one.yaml})
YAML.expects(:load_file).with("one.yaml").returns @one;
expect(@store.search(@request)).to eq([@one])
end
it "should return an empty array when the glob doesn't match anything" do
@request = stub 'request', :key => "f*ilglobcanfail*", :instance => @subject
@store.expects(:path).with(@request.key,'').returns :glob
Dir.expects(:glob).with(:glob).returns []
expect(@store.search(@request)).to eq([])
end
describe "when destroying" do
let(:path) do
File.join(@dir, @store.class.indirection_name.to_s, @request.key.to_s + ".yaml")
end
it "should unlink the right yaml file if it exists" do
Puppet::FileSystem.expects(:exist?).with(path).returns true
Puppet::FileSystem.expects(:unlink).with(path)
@store.destroy(@request)
end
it "should not unlink the yaml file if it does not exists" do
Puppet::FileSystem.expects(:exist?).with(path).returns false
Puppet::FileSystem.expects(:unlink).with(path).never
@store.destroy(@request)
end
end
end
end
|