module Vedeu module Repositories # Provides common methods for accessing the various repositories # Vedeu uses. # # @example # { 'models' => [Model, Model, Model] } # # { 'models' => [Model] } # class Repository include Vedeu::Common include Vedeu::Repositories::Registerable include Vedeu::Repositories::Store # @!attribute [r] model # @return [void] attr_reader :model # @!attribute [r] storage # @return [void] attr_reader :storage # Returns a new instance of Vedeu::Repositories::Repository. # # @param model [Class] # @param storage [Class|Hash] # @return [Vedeu::Repositories::Repository] def initialize(model = nil, storage = {}) @model = model @storage = storage end # Returns the repository class. # # @return [Class] def repository self.class end # Return all the registered models. # # @return [Array] An array containing each stored model. def all return storage.values if storage.is_a?(Hash) registered end # Return the named model or a null object if not registered. # # @example # # Fetch the cursor belonging to the interface of the same # # name. # Vedeu.cursors.by_name(:some_name) # # # Fetch the names of the interfaces belonging to this group. # Vedeu.groups.by_name(name) # # @param name [String|Symbol] The name of the stored model. # @return [void] def by_name(name = nil) name = present?(name) ? name : Vedeu.focus return find(name) if registered?(name) attrs = if null_attributes.any? null_attributes.merge!(name: name) else { name: name } end null_model.new(attrs) end # Return the model for the interface currently in focus. # # @return [String|NilClass] def current find_or_create(Vedeu.focus) if Vedeu.focus end # Find the model by name. # # @param name [String|Symbol] # @return [Hash Object>|NilClass] def find(name) storage[name] end # Find the model attributes by name, raises an exception when # the model cannot be found. # # @param name [String|Symbol] # @raise [Vedeu::Error::ModelNotFound] When the model cannot be # found with this name. # @return [Hash Object>] def find!(name) find(name) || fail(Vedeu::Error::ModelNotFound, "Cannot find model by name: '#{name}'".freeze) end # Find a model by name, registers the model by name when not # found. # # @param name [String|Symbol] # @return [void] def find_or_create(name) return find(name) if registered?(name) Vedeu.log(type: :store, message: "Model (#{model}) not found, " \ "registering: '#{name}'".freeze) model.new(name).store end # @return [String] def inspect "<#{self.class.name}>".freeze end # Returns a boolean indicating whether the named model is # registered. # # @param name [String|Symbol] # @return [Boolean] def registered?(name) return false if absent?(name) return false if empty? storage.include?(name) end # Returns the storage with the named model removed, or false # when the model does not exist. # # @param name [String|Symbol] # @return [Hash|FalseClass] def remove(name) return false if empty? return false unless registered?(name) storage.delete(name) storage unless storage.is_a?(Set) end alias_method :delete, :remove # Stores the model instance by name in the repository of the # model. # # @note If a block is given, store the model, return the model # after yielding. # @param model [void] A model instance. # @raise [Vedeu::Error::MissingRequired] When the name attribute # is not defined. # @return [void] The model instance which was stored. def store(model, &block) valid_model?(model) log_store(model) storage[model.name] = model yield if block_given? model end alias_method :register, :store alias_method :add, :store private # @param model [void] A model instance. # @return [String] def log_store(model) type = registered?(model.name) ? :update : :create Vedeu.log(type: type, message: "#{model.class.name}: '#{model.name}'".freeze) end # # @param model [void] A model instance. # @return [Boolean] def valid_model?(model) fail Vedeu::Error::MissingRequired, "Cannot store model '#{model.class}' without a name " \ 'attribute.'.freeze unless present?(model.name) end end # Repository end # Repositories end # Vedeu