# frozen_string_literal: true require 'intercom/lib/flat_store' require 'intercom/lib/dynamic_accessors' require 'intercom/lib/dynamic_accessors_on_method_missing' require 'intercom/traits/dirty_tracking' require 'intercom/lib/typed_json_deserializer' module Intercom module Traits module ApiResource include DirtyTracking attr_accessor :id, :client def initialize(attributes = {}) from_hash(attributes) end def ==(other) self.class == other.class && to_json == other.to_json end def from_response(response) from_hash(response) reset_changed_fields! self end def from_hash(hash) hash.each do |attribute, value| initialize_property(attribute, value) end initialize_missing_flat_store_attributes if respond_to? :flat_store_attributes self end def to_hash instance_variables_excluding_dirty_tracking_field.each_with_object({}) do |variable, hash| hash[variable.to_s.delete('@')] = instance_variable_get(variable) end end def to_json(*args) instance_variables_excluding_dirty_tracking_field.each_with_object({}) do |variable, hash| next if variable == :@client value = instance_variable_get(variable) hash[variable.to_s.delete('@')] = value.respond_to?(:to_json) ? value.to_json(*args) : value end end def to_submittable_hash submittable_hash = {} to_hash.each do |attribute, value| next unless submittable_attribute?(attribute, value) submittable_hash[attribute] = value.respond_to?(:to_submittable_hash) ? value.to_submittable_hash : value end submittable_hash end def method_missing(method_sym, *arguments, &block) Lib::DynamicAccessorsOnMethodMissing.new(method_sym, *arguments, self) .define_accessors_or_call { super } end def flat_store_attribute?(attribute) respond_to?(:flat_store_attributes) && flat_store_attributes.map(&:to_s).include?(attribute.to_s) end private def initialize_property(attribute, value) return if addressable_list?(attribute, value) Lib::DynamicAccessors.define_accessors(attribute, value, self) unless accessors_already_defined?(attribute) set_property(attribute, value) end def addressable_list?(attribute, value) return false unless typed_property?(attribute, value) value['type'] == 'list' && value['url'] end def accessors_already_defined?(attribute) respond_to?(attribute) && respond_to?("#{attribute}=") end def set_property(attribute, value) value_to_set = parsed_value_for_attribute(attribute, value) call_setter_for_attribute(attribute, value_to_set) end def parsed_value_for_attribute(attribute, value) if typed_property?(attribute, value) Intercom::Lib::TypedJsonDeserializer.new(value, client).deserialize elsif flat_store_attribute?(attribute) Intercom::Lib::FlatStore.new(value) else value end end def custom_attribute_field?(attribute) attribute.to_s == 'custom_attributes' end def message_from_field?(attribute, value) attribute.to_s == 'from' && value.is_a?(Hash) && value['type'] end def message_to_field?(attribute, value) attribute.to_s == 'to' && value.is_a?(Hash) && value['type'] end def typed_property?(attribute, value) typed_value?(value) && !custom_attribute_field?(attribute) && !message_from_field?(attribute, value) && !message_to_field?(attribute, value) && attribute.to_s != 'metadata' end def typed_value?(value) value.is_a?(Hash) && !!value['type'] end def call_setter_for_attribute(attribute, value) setter_method = "#{attribute}=" send(setter_method, value) end def initialize_missing_flat_store_attributes flat_store_attributes.each do |attribute| unless instance_variables_excluding_dirty_tracking_field.map(&:to_s).include? "@#{attribute}" initialize_property(attribute, {}) end end end def submittable_attribute?(attribute, value) # FlatStores always submitted, even if not changed, as we don't track their dirtyness value.is_a?(Intercom::Lib::FlatStore) || field_changed?(attribute) end module ClassMethods def from_api(api_response) object = new object.from_response(api_response) object end end def self.included(base) base.extend(ClassMethods) end end end end