module Brief module Model extend ActiveSupport::Concern included do unless defined?(Virtus) require 'inflecto' require 'virtus' end include Virtus.model(finalize: false) include Initializers include AccessorMethods include Persistence include Serializers class_attribute :models, :after_initialization_hooks self.models = Array(models).to_set class << self include Enumerable end attribute :path, Pathname attribute :document, Brief::Document Brief::Model.classes << self end module AccessorMethods def data document.data end def content document.content end def extracted @extracted ||= Brief::Document::ContentExtractor.new(self.class.type_alias, document) end def method_missing(meth, *args, &block) if args.empty? if document.respond_to?(meth) document.send(meth) else document.data && document.data.key?(meth) ? data[meth] : extracted.send(meth) end else super end end end def self.classes @classes ||= Set.new end def self.table classes.inject({}.to_mash) do |memo, klass| memo.tap { memo[klass.type_alias] = klass } end end def self.lookup(type_alias) for_type(type_alias) || for_folder_name(type_alias) || for_type(type_alias.singularize) end def self.for_type(type_alias) table[type_alias] end def self.for_folder_name(folder_name=nil) folder_name = folder_name.to_s.downcase table[folder_name.singularize] || table[folder_name] end def self.lookup_class_from_args(args = []) args = Array(args) if model_class = for_type(args.first) model_class end end def self.finalize Virtus.finalize classes.each(&:finalize) end def ==(other) path == other.path end def extract_content(options = {}) document.extract_content(options) end module ClassMethods def to_schema { schema: { content: definition.content_schema, metadata: definition.metadata_schema, }, class_name: to_s, type_alias: type_alias, name: name, group: name.to_s.pluralize, actions: defined_actions, example: definition.example_body.to_s, template: definition.template_body.to_s, urls: { browse_url: "browse/#{ type_alias.to_s.pluralize }", schema_url: "schema/#{ type_alias }" } } end def has_actions? definition.has_actions? end def defined_actions definition.defined_actions ||= [] end def finalize klass = self klass.name ||= klass.to_s.split('::').last.humanize klass.type_alias ||= klass.name.parameterize.gsub(/-/, '_') klass.attribute_set.map(&:name).each do |attr| unless klass.method_defined?("find_by_#{ attr }") klass.define_singleton_method("find_by_#{ attr }") do |value| where(attr => value).first end end end klass.definition.apply_config Brief::Repository.define_document_finder_methods end def where(*args, &_block) Brief::DocumentMapper::Query.new(self).send(:where, *args) end def each(*args, &block) Array(models).send(:each, *args, &block) end def after_initialize(&block) (self.after_initialization_hooks ||= []).push(block) end attr_writer :name def name @name || to_s.split('::').last.underscore.gsub('_', ' ').titlecase end attr_writer :type_alias def type_alias @type_alias || name.parameterize.gsub(/-/, '_') end def definition @definition ||= Brief::Model::Definition.new(name, type_alias: type_alias, model_class: self) end def definition=(_value) @definition end def section_mapping(*args) definition.send(:section_mapping, *args) end def section_mappings(*args) definition.send(:section_mappings, *args) end def generate_template_content_from(object, include_frontmatter = true) @erb ||= ERB.new(template_body) content = @erb.result(binding) frontmatter = object.slice(*attribute_names) base = '' base += frontmatter.to_hash.to_yaml + "---\n" if include_frontmatter base += content base end def attribute_names attribute_set.map(&:name) end def template_body(*args) res = definition.send(:template_body, *args) res.to_s.length == 0 ? example_body : res.to_s.strip end def example_body(*args) definition.send(:example_body, *args).to_s.strip end def method_missing(meth, *args, &block) if %w(meta content template example actions helpers).include?(meth.to_s) definition.send(meth, *args, &block) finalize elsif %w(defined_helper_methods defined_actions).include?(meth.to_s) definition.send(meth) elsif meth.to_s.match(/^on_(.*)_change$/) create_change_handler(Regexp.last_match[1], *args, &block) else super end end def create_change_handler(_attribute, *_args, &block) block.call(self) end end module Initializers def set_default_attributes attribute_set.set_defaults(self) send(:after_initialize) if respond_to?(:after_initialize) self end def after_initialize Array(self.class.after_initialization_hooks).each do |hook| hook.call(self) end end def set_slug_from(column = :name) self.slug = send(column).to_s.downcase.parameterize if slug.to_s.length == 0 end end end end