summaryrefslogtreecommitdiff
path: root/lib/chef/dsl/chef_vault.rb
blob: dfae1ca49d4e20139f9b7e2711f2e1e7271b8678 (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
#
# Author: Joshua Timberman <joshua@chef.io>
#
# Copyright:: 2013-2020, Chef Software, Inc.
# Copyright:: 2014-2015 Bloomberg Finance L.P.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#

require "chef-vault"
require_relative "data_query"

class Chef
  module DSL
    module ChefVault
      include Chef::DSL::DataQuery

      # Helper method which provides a Recipe/Resource DSL for wrapping
      # creation of {ChefVault::Item}.
      # @note
      # Falls back to normal data bag item loading if the item is not
      # actually a Chef Vault item. This is controlled via
      # +node['chef-vault']['databag_fallback']+.
      # @example
      # item = chef_vault_item('secrets', 'bacon')
      # log 'Yeah buddy!' if item['_default']['type']
      # @param [String] bag Name of the data bag to load from.
      # @param [String] id Identifier of the data bag item to load.
      def chef_vault_item(bag, id)
        if ChefVault::Item.vault?(bag, id)
          ChefVault::Item.load(bag, id)
        elsif node["chef-vault"]["databag_fallback"]
          data_bag_item(bag, id)
        else
          raise "Trying to load a regular data bag item #{id} from #{bag}, and databag_fallback is disabled"
        end
      end

      # Helper method that allows for listing the ids of a vault in a recipe.
      # This method is needed because data_bag() returns the keys along with
      # the items, so this method strips out the keys for users so that they
      # don't have to do it in their recipes.
      # @example
      # ids = chef_vault('secrets')
      # log 'Yeah buddy!' if ids[0] == 'bacon'
      # @param [String] bag Name of the data bag to load from.
      # @return [Array]
      def chef_vault(bag)
        raise "'#{bag}' is not a vault" unless Chef::DataBag.list.include? bag

        pattern = Regexp.new(/_keys$/).freeze
        data_bag(bag).each_with_object([]) do |id, acc|
          acc << id unless pattern.match?(id)
        end
      end

      # Helper method which provides an environment wrapper for a data bag.
      # This allows for easy access to current environment secrets inside
      # of an item.
      # @example
      # item = chef_vault_item_for_environment('secrets', 'bacon')
      # log 'Yeah buddy!' if item['type'] == 'applewood_smoked'
      # @param [String] bag Name of the data bag to load from.
      # @param [String] id Identifier of the data bag item to load.
      # @return [Hash]
      def chef_vault_item_for_environment(bag, id)
        item = chef_vault_item(bag, id)
        return {} unless item[node.chef_environment]

        item[node.chef_environment]
      end
    end
  end
end