Class: Chef::Node::ImmutableMash

Inherits:
Mash show all
Includes:
Immutablize
Defined in:
lib/chef/node/immutable_collections.rb

Overview

ImmutableMash

ImmutableMash implements Hash/Dict behavior for reading values from node attributes.

ImmutableMash acts like a Mash (Hash that is indifferent to String or Symbol keys), with some important exceptions:

  • Methods that mutate state are overridden to raise an error instead.

  • Methods that read from the collection are overriden so that they check if the Chef::Node::Attribute has been modified since an instance of this class was generated. An error is raised if the object detects that it is stale.

  • Values can be accessed in attr_reader-like fashion via method_missing.

Constant Summary collapse

DISALLOWED_MUTATOR_METHODS =
[
  :[]=,
  :clear,
  :collect!,
  :default=,
  :default_proc=,
  :delete,
  :delete_if,
  :keep_if,
  :map!,
  :merge!,
  :update,
  :reject!,
  :replace,
  :select!,
  :shift
]

Instance Method Summary collapse

Methods included from Immutablize

#immutablize

Methods inherited from Mash

#[]=, #default, #delete, #except, #fetch, from_hash, #initialize_copy, #key?, #merge, #regular_update, #regular_writer, #stringify_keys!, #symbolize_keys, #to_hash, #update, #values_at

Constructor Details

#initialize(mash_data) ⇒ ImmutableMash

Returns a new instance of ImmutableMash.



131
132
133
134
135
# File 'lib/chef/node/immutable_collections.rb', line 131

def initialize(mash_data)
  mash_data.each do |key, value|
    internal_set(key, immutablize(value))
  end
end

Dynamic Method Handling

This class handles dynamic methods through the method_missing method

#method_missing(symbol, *args) ⇒ Object



152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
# File 'lib/chef/node/immutable_collections.rb', line 152

def method_missing(symbol, *args)
  if args.empty?
    if key?(symbol)
      self[symbol]
    else
      raise NoMethodError, "Undefined method or attribute `#{symbol}' on `node'"
    end
  # This will raise a ImmutableAttributeModification error:
  elsif symbol.to_s =~ /=$/
    key_to_set = symbol.to_s[/^(.+)=$/, 1]
    self[key_to_set] = (args.length == 1 ? args[0] : args)
  else
    raise NoMethodError, "Undefined node attribute or method `#{symbol}' on `node'"
  end
end

Instance Method Details

#convert_value(value) ⇒ Object

Mash uses #convert_value to mashify values on input. Since we’re handling this ourselves, override it to be a no-op



170
171
172
# File 'lib/chef/node/immutable_collections.rb', line 170

def convert_value(value)
  value
end

#dupObject

NOTE: #default and #default= are likely to be pretty confusing. For a regular ruby Hash, they control what value is returned for, e.g.,

hash[:no_such_key] #=> hash.default

Of course, ‘default’ has a specific meaning in Chef-land



179
180
181
# File 'lib/chef/node/immutable_collections.rb', line 179

def dup
  Mash.new(self)
end