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 +7 -0
- data/lib/hiera/backend/consul_backend.rb +123 -0
- data/lib/hiera/backend/vault_backend.rb +60 -0
- metadata +59 -0
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: []
|