summaryrefslogtreecommitdiffstats
path: root/lib/puppet/file_serving/mount/file.rb
blob: d934d1d106e8d4ed2a383d2105662d8f7721056e (plain)
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
require 'puppet/util/cacher'

require 'puppet/file_serving/mount'

class Puppet::FileServing::Mount::File < Puppet::FileServing::Mount
    class << self
        include Puppet::Util::Cacher

        cached_attr(:localmap) do
            {   "h" =>  Facter.value("hostname"),
                "H" => [Facter.value("hostname"),
                        Facter.value("domain")].join("."),
                "d" =>  Facter.value("domain")
            }
        end
    end

    def complete_path(relative_path, node)
        full_path = path(node)

        raise ArgumentError.new("Mounts without paths are not usable") unless full_path

        # If there's no relative path name, then we're serving the mount itself.
        return full_path unless relative_path

        file = ::File.join(full_path, relative_path)

        if !(FileTest.exist?(file) or FileTest.symlink?(file))
            Puppet.info("File does not exist or is not accessible: #{file}")
            return nil
        end

        file
    end

    # Return an instance of the appropriate class.
    def find(short_file, request)
        complete_path(short_file, request.node)
    end

    # Return the path as appropriate, expanding as necessary.
    def path(node = nil)
        if expandable?
            return expand(@path, node)
        else
            return @path
        end
    end

    # Set the path.
    def path=(path)
        # FIXME: For now, just don't validate paths with replacement
        # patterns in them.
        if path =~ /%./
            # Mark that we're expandable.
            @expandable = true
        else
            raise ArgumentError, "#{path} does not exist or is not a directory" unless FileTest.directory?(path)
            raise ArgumentError, "#{path} is not readable" unless FileTest.readable?(path)
            @expandable = false
        end
        @path = path
    end

    def search(path, request)
        return nil unless path = complete_path(path, request.node)
        [path]
    end

    # Verify our configuration is valid.  This should really check to
    # make sure at least someone will be allowed, but, eh.
    def validate
        raise ArgumentError.new("Mounts without paths are not usable") if @path.nil?
    end

    private

    # Create a map for a specific node.
    def clientmap(node)
        {
            "h" => node.sub(/\..*$/, ""),
            "H" => node,
            "d" => node.sub(/[^.]+\./, "") # domain name
        }
    end

    # Replace % patterns as appropriate.
    def expand(path, node = nil)
        # This map should probably be moved into a method.
        map = nil

        if node
            map = clientmap(node)
        else
            Puppet.notice "No client; expanding '#{path}' with local host"
            # Else, use the local information
            map = localmap()
        end

        path.gsub(/%(.)/) do |v|
            key = $1
            if key == "%"
                "%"
            else
                map[key] || v
            end
        end
    end

    # Do we have any patterns in our path, yo?
    def expandable?
        if defined?(@expandable)
            @expandable
        else
            false
        end
    end

    # Cache this manufactured map, since if it's used it's likely
    # to get used a lot.
    def localmap
        self.class.localmap
    end
end