Why Gemfury? Push, build, and install  RubyGems npm packages Python packages Maven artifacts PHP packages Go Modules Debian packages RPM packages NuGet packages

Repository URL to install this package:

Details    
quickbooks-ruby / lib / quickbooks / model / base_model.rb
Size: Mime:
module Quickbooks
  module Model
    class BaseModel
      include ActiveModel::Validations
      include Validator
      include ROXML

      xml_convention :camelcase

      def initialize(attributes={})
        attributes.each {|key, value| public_send("#{key}=", value) }
      end

      # ROXML doesnt insert the namespaces into generated XML so we need to do it ourselves
      # insert the static namespaces in the first opening tag that matches the +model_name+
      def to_xml_inject_ns(model_name, options = {})
        s = StringIO.new
        xml = to_xml(options).write_to(s, :indent => 0, :indent_text => '')
        destination_name = options.fetch(:destination_name, nil)
        destination_name ||= model_name

        sparse = options.fetch(:sparse, false)
        sparse_string = %{sparse="#{sparse}"}
        step1 = s.string.sub("<#{model_name}>", "<#{destination_name} #{Quickbooks::Service::BaseService::XML_NS} #{sparse_string}>")
        step2 = step1.sub("</#{model_name}>", "</#{destination_name}>")
        step2
      end

      def to_xml_ns(options = {})
        to_xml_inject_ns(self.class::XML_NODE, options)
      end

      delegate :[], :fetch, :to => :attributes

      def attributes
        attributes = self.class.attribute_names.map do |name|
          value = public_send(name)
          value = value.attributes if value.respond_to?(:attributes)
          [name, value]
        end

        HashWithIndifferentAccess[attributes]
      end

      def ensure_line_items_initialization
        self.line_items ||= []
      end

      def inspect
        # it would be nice if we could inspect all the children,
        # but it's likely to blow the stack in some cases
        "#<#{self.class} " + 
        "#{attributes.map{|k,v| "#{k}: #{v.nil? ? 'nil' : v.to_s }"}.join ", "}>"
      end
      class << self
        def to_xml_big_decimal
          Proc.new { |val| val.nil? ? nil : val.to_f }
        end

        def attribute_names
          roxml_attrs.map(&:accessor)
        end

        # These can be over-ridden in each model object as needed
        def resource_for_collection
          self::REST_RESOURCE
        end

        def resource_for_singular
          self::REST_RESOURCE
        end

        # Automatically generate an ID setter.
        # Example:
        #   reference_setters :discount_ref
        # Would generate a method like:
        # def discount_id=(id)
        #    self.discount_ref = BaseReference.new(id)
        # end
        def reference_setters(*args)
          references = args.empty? ? reference_attrs : args

          references.each do |attribute|
            method_name = "#{attribute.to_s.gsub('_ref', '_id')}=".to_sym
            unless instance_methods(false).include?(method_name)
              method_definition = <<-METH
              def #{method_name}(id)
                self.#{attribute} = BaseReference.new(id)
              end
              METH
              class_eval(method_definition)
            end
          end
        end

        def reference_attrs
          matches = roxml_attrs.select{|attr| attr.sought_type == Quickbooks::Model::BaseReference}.map{|attr| attr.accessor}
        end

        def inspect
          "#{super}(#{attrs_with_types.join " "})"
        end

        def attrs_with_types
          roxml_attrs.map do |attr|
            "#{attr.accessor}:" +
              "#{attr.class.block_shorthands.invert[attr.blocks.last]}:#{attr.sought_type}"
          end
        end
      end
    end
  end
end