# A ConfigBuilder model implements a logic-less interface to a component of # Vagrant. # # A model should implement the following methods: # # ## `self.new_from_hash` # # This method takes an arbitrarily nested data structure of basic data types # (Arrays, Hashes, Numerics, Strings, etc) and instantiates a new object # with attributes set based on that data structure. # # ## `#to_proc` # # This method takes the object attributes and generates a lambda that will # create a Vagrant config with the state specified by the attributes. The # lambda should have an arity of one and should be passed a `config` object. # The generated block will generate the Vagrant config that implements the # behavior specified by the object attributes. # # If the Model delegates certain configuration to other models, the generated # lambda should be able to evaluate lambdas from the delegated models. # # Implementing classes do not need to inherit from ConfigBuilder::Model::Base, # but it makes life easier. class ConfigBuilder::Model::Base # Deserialize a hash into a configbuilder model # # @param attributes [Hash] The model attributes as represented in a hash. # @return [Object < ConfigBuilder::Model] def self.new_from_hash(attributes) obj = new() obj.attrs = attributes obj end # @api private def attrs=(config) hash = config.inject({}) { |hash, (key, value)| hash[key.to_sym] = value; hash } if @defaults @attrs = @defaults.merge(hash) else @attrs = hash end end # Generate a block based on configuration specified by the attributes # # @abstract # @return [Proc] def to_proc raise NotImplementedError end # Generate a block based on the attribute configuration and call it with # the given config. # # @param config [Vagrant.plugin('2', :config)] # @return [void] def call(config) to_proc.call(config) end # @param identifier [Symbol] # # @return [Object] The value of the model attribute specified by `identifier` # # @todo validate identifier def attr(identifier) @attrs[identifier] end private :attr # Conditionally evaluate a block with a model attribute if it's defined # # @since 0.6.0 # # @param identifier [Symbol] The attribute identifier # # @return [void] def with_attr(identifier) val = @attrs[identifier] unless val.nil? yield val end end private :with_attr class << self # @param identifier [Symbol] def def_model_attribute(identifier) model_attributes << identifier end def model_attributes (@identifiers ||= []) end end end