hiera-consul 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 73f393c35a128be6bfaae874febd674059dba594
4
+ data.tar.gz: c1e33bcd1a674e40faaf21a64c76070a1cea1014
5
+ SHA512:
6
+ metadata.gz: ac2dab1523653d471e01dca3f5ebbb366ec039fda5f0d92cce61435732e67d6889f3cc404975eb1b9347c215e1f53111acf7e996e7173f1e33cbc16c7a7c00de
7
+ data.tar.gz: 86944abef51a03957ac77321dd493f2903fccd1b5404490c2782de554c21e6ffd596e38f6131b4a7714709b6d616858793043191d5090833683aa605ba9e5b4f
@@ -0,0 +1,123 @@
1
+ # Hiera backend for Consul
2
+ class Hiera
3
+ module Backend
4
+ class Consul_backend
5
+
6
+ def initialize()
7
+ require 'base64'
8
+ require 'net/http'
9
+ require 'json'
10
+ require 'uri'
11
+
12
+ @config = Config[:consul]
13
+ if ENV['CONSUL_HTTP_ADDR']
14
+ # By convention the ENV var does not contain a scheme, but URI
15
+ # requires one. Net::HTTP will switch to https if configured later.
16
+ uri = URI('http://' + ENV['CONSUL_HTTP_ADDR'])
17
+ @consul = Net::HTTP.new(uri.host, uri.port)
18
+ elsif (@config[:host] && @config[:port])
19
+ @consul = Net::HTTP.new(@config[:host], @config[:port])
20
+ else
21
+ raise "[hiera-consul] Missing minimum configuration, please check hiera.yaml"
22
+ end
23
+
24
+ Hiera.debug("[hiera-consul] Client configured to connect to #{@consul.address}:#{@consul.port}")
25
+
26
+ @consul.read_timeout = @config[:http_read_timeout] || 10
27
+ @consul.open_timeout = @config[:http_connect_timeout] || 10
28
+ end
29
+
30
+ def lookup(key, scope, order_override, resolution_type)
31
+ answer = nil
32
+
33
+ paths = @config[:paths].map { |p| Backend.parse_string(p, scope, { 'key' => key }) }
34
+ paths.insert(0, order_override) if order_override
35
+
36
+ paths.each do |path|
37
+ Hiera.debug("[hiera-consul] Looking up #{path}/#{key} in consul backend")
38
+
39
+ # Check that we are not looking somewhere that will make hiera crash subsequent lookups
40
+ if "#{path}/#{key}".match("//")
41
+ Hiera.debug("[hiera-consul] The specified path #{path}/#{key} is malformed, skipping")
42
+ next
43
+ end
44
+
45
+ query_path = "#{path}/#{key}"
46
+ recurse = resolution_type == :hash
47
+
48
+ result = wrapquery(query_path, recurse)
49
+ next unless result
50
+
51
+ api_prefix = /^\/v\d\/kv\//
52
+ prefix = query_path.sub(api_prefix, '')
53
+ answer = parse_result(result, prefix)
54
+ next unless answer
55
+
56
+ Hiera.debug("[hiera-consul] Read key #{key} from path #{path}")
57
+ break
58
+ end
59
+
60
+ answer
61
+ end
62
+
63
+ private
64
+
65
+ def wrapquery(path, recurse = false)
66
+ path += "?recurse" if recurse
67
+ httpreq = Net::HTTP::Get.new("#{path}")
68
+
69
+ data = nil
70
+ begin
71
+ response = @consul.request(httpreq)
72
+ case response
73
+ when Net::HTTPSuccess
74
+ data = response.body
75
+ else
76
+ Hiera.debug("[hiera-consul] Could not read key: #{path}")
77
+ end
78
+ rescue Exception => e
79
+ Hiera.warn("[hiera-consul] Error occurred read value #{path}: #{e}")
80
+ end
81
+
82
+ data
83
+ end
84
+
85
+ def parse_result(res, prefix)
86
+ if res == "null"
87
+ Hiera.debug("[hiera-consul] Skipped null result")
88
+ return nil
89
+ end
90
+
91
+ res_array = JSON.parse(res)
92
+ case res_array.length
93
+ when 0
94
+ Hiera.debug("[hiera-consul] Skipped empty result")
95
+ answer = nil
96
+ when 1
97
+ answer = Base64.decode64(res_array.first['Value'])
98
+ else
99
+ # Interpret the results as a nested Hash
100
+ answer = res_array.each_with_object({}) do |entry, memo|
101
+ # Strip the mount prefix and leading slash
102
+ k = entry['Key'][(prefix.length+1)..-1]
103
+ v = entry['Value'].nil? ? {} : Base64.decode64(entry['Value'])
104
+ deep_merge!(memo, k.split('/').reverse.inject(v) { |a, n| { n => a } })
105
+ end
106
+ end
107
+
108
+ answer
109
+ end
110
+
111
+ def deep_merge!(tgt_hash, src_hash)
112
+ tgt_hash.merge!(src_hash) do |key, oldval, newval|
113
+ if oldval.kind_of?(Hash) && newval.kind_of?(Hash)
114
+ deep_merge!(oldval, newval)
115
+ else
116
+ newval
117
+ end
118
+ end
119
+ end
120
+
121
+ end
122
+ end
123
+ end
@@ -0,0 +1,60 @@
1
+ # Vault backend for Hiera
2
+ class Hiera
3
+ module Backend
4
+ class Vault_backend
5
+
6
+ def initialize()
7
+ require 'json'
8
+ require 'vault'
9
+
10
+ @config = Config[:vault]
11
+ @config[:mounts] ||= {}
12
+ @config[:mounts][:generic] ||= ['secret']
13
+
14
+ begin
15
+ @vault = Vault::Client.new(address: @config[:addr], token: @config[:token])
16
+ fail if @vault.sys.seal_status.sealed?
17
+ Hiera.debug("[hiera-vault] Client configured to connect to #{@vault.address}")
18
+ rescue Exception => e
19
+ @vault = nil
20
+ Hiera.warn("[hiera-vault] Skipping backend. Configuration error: #{e}")
21
+ end
22
+ end
23
+
24
+ def lookup(key, scope, order_override, resolution_type)
25
+ return nil if @vault.nil?
26
+
27
+ answer = nil
28
+
29
+ # Only generic mounts supported so far
30
+ @config[:mounts][:generic].each do |mount|
31
+ path = Backend.parse_string(mount, scope, { 'key' => key })
32
+ answer = lookup_generic("#{path}/#{key}", scope)
33
+
34
+ break if answer.kind_of? Hash
35
+ end
36
+
37
+ answer
38
+ end
39
+
40
+ def lookup_generic(key, scope)
41
+ begin
42
+ secret = @vault.logical.read(key)
43
+ rescue Vault::HTTPConnectionError
44
+ Hiera.debug("[hiera-vault] Could not connect to read secret: #{key}")
45
+ rescue Vault::HTTPError => e
46
+ Hiera.warn("[hiera-vault] Could not read secret #{key}: #{e.errors.join("\n").rstrip}")
47
+ end
48
+
49
+ return nil if secret.nil?
50
+
51
+ Hiera.debug("[hiera-vault] Read secret: #{key}")
52
+ # Turn secret's hash keys into strings
53
+ data = secret.data.inject({}) { |h, (k, v)| h[k.to_s] = v; h }
54
+
55
+ return Backend.parse_answer(data, scope)
56
+ end
57
+
58
+ end
59
+ end
60
+ end
metadata ADDED
@@ -0,0 +1,59 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: hiera-consul
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - Jonathan Sokolowski
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2015-06-19 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: json
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ description: Hiera backend for looking up KV data stored in Vault
28
+ email: jonathan.sokolowski@gmail.com
29
+ executables: []
30
+ extensions: []
31
+ extra_rdoc_files: []
32
+ files:
33
+ - lib/hiera/backend/consul_backend.rb
34
+ - lib/hiera/backend/vault_backend.rb
35
+ homepage: http://github.com/jsok/hiera-consul
36
+ licenses:
37
+ - Apache-2.0
38
+ metadata: {}
39
+ post_install_message:
40
+ rdoc_options: []
41
+ require_paths:
42
+ - lib
43
+ required_ruby_version: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ required_rubygems_version: !ruby/object:Gem::Requirement
49
+ requirements:
50
+ - - ">="
51
+ - !ruby/object:Gem::Version
52
+ version: '0'
53
+ requirements: []
54
+ rubyforge_project:
55
+ rubygems_version: 2.4.5
56
+ signing_key:
57
+ specification_version: 4
58
+ summary: Module for using consul as a hiera backend
59
+ test_files: []