require 'rbbt/annotations' module Entity class << self attr_accessor :formats, :entity_property_cache end self.entity_property_cache = "var/entity_property" self.formats = {} UNPERSISTED_PREFIX = "entity_unpersisted_property_" def self.extended(base) base.extend Annotation Entity.formats[base.to_s] = base base.module_eval do attr_accessor :_ary_property_cache attr_accessor :template, :list_template, :action_template, :list_action_template, :keep_id def _ary_property_cache @_ary_property_cache ||= {} end def self.format=(formats) formats = [formats] unless Array === formats formats.each do |format| Entity.formats[format] = self end end def base_entity self.annotation_types.select{|m| Entity === m}.last end def property(*args, &block) class << self; self; end.property(*args,&block) end def to_yaml(*args) self.clean_annotations.dup.to_yaml(*args) end def encode_with(coder) coder.scalar = clean_annotations end def marshal_dump clean_annotations end def consolidate self.inject(nil){|acc,e| if acc.nil? acc = e else acc.concat e end } end def self.property(name, &block) case when (Hash === name and name.size == 1) name, type = name.collect.first when (String === name or Symbol === name) type = :single else raise "Format of name ( => type) not understood: #{name.inspect}" end name = name.to_s unless String === name persisted_name = UNPERSISTED_PREFIX + name self.remove_method persisted_name if methods.include? persisted_name case type when :both define_method name, &block when :single, :single2array single_name = "_single_" << name define_method single_name, &block define_method name do |*args| if Array === self self.collect{|e| e.send(name, *args)} else self.send(single_name, *args) end end when :array, :array2single ary_name = "_ary_" << name define_method ary_name, &block define_method name do |*args| #ary_name = "_ary_" << __method__.to_s case when Array === self self.send(ary_name, *args) when (Array === self.container and not self.container_index.nil? and self.container.respond_to? ary_name) cache_code = Misc.hash2md5({:name => ary_name, :args => args}) res = (self.container._ary_property_cache[cache_code] ||= self.container.send(name, *args)) if Hash === res res[self] else res[self.container_index] end else res = self.make_list.send(ary_name, *args) Hash === res ? res[self] : res[0] end end end end def self.persist(method_name, type = nil, options = {}) type = :memory if type.nil? options ||= {} options = Misc.add_defaults options, :dir => Entity.entity_property_cache orig_name = UNPERSISTED_PREFIX + method_name.to_s alias_method orig_name, method_name unless self.instance_methods.include? orig_name.to_sym define_method method_name do |*args| id = self.id persist_name = __method__.to_s << ":" << (Array === id ? Misc.hash2md5(:id => id) : id) persist_name << ":" << Misc.hash2md5({:args => args}) unless args.nil? or args.empty? persist_options = options persist_options = persist_options.merge(:other => {:args => args}) if args.any? Persist.persist(persist_name, type, persist_options) do self.send(orig_name, *args) end end end def self.unpersist(method_name) return unless persisted? method_name orig_name = UNPERSISTED_PREFIX + method_name.to_s alias_method method_name, orig_name remove_method orig_name end def self.persisted?(method_name) orig_name = UNPERSISTED_PREFIX + method_name.to_s instance_methods.include? orig_name.to_sym end def self.with_persisted(method_name) persisted = persisted? method_name persist method_name unless persisted res = yield unpersist method_name unless persisted res end end end end