# frozen_string_literal: true class TableSync::Publisher < TableSync::BasePublisher DEBOUNCE_TIME = 1.minute # 'original_attributes' are not published, they are used to resolve the routing key def initialize(object_class, original_attributes, **opts) @object_class = object_class.constantize @original_attributes = filter_safe_for_serialization(original_attributes.deep_symbolize_keys) @confirm = opts.fetch(:confirm, true) @debounce_time = opts[:debounce_time]&.seconds || DEBOUNCE_TIME if opts[:destroyed].nil? @state = opts.fetch(:state, :updated).to_sym validate_state else # TODO Legacy job support, remove @state = opts[:destroyed] ? :destroyed : :updated end end def publish return enqueue_job if destroyed? || debounce_time.zero? sync_time = Rails.cache.read(cache_key) || current_time - debounce_time - 1.second return if sync_time > current_time next_sync_time = sync_time + debounce_time next_sync_time <= current_time ? enqueue_job : enqueue_job(next_sync_time) end def publish_now # Update request and object does not exist -> skip publishing return if !object && !destroyed? Rabbit.publish(params) TableSync::Instrument.notify table: TableSync.orm.table_name(object_class), event: event, direction: :publish end private attr_reader :original_attributes attr_reader :state attr_reader :debounce_time def attrs_for_callables original_attributes end def attrs_for_routing_key return object.attrs_for_routing_key if attrs_for_routing_key_defined? attrs_for_callables end def attrs_for_metadata return object.attrs_for_metadata if attrs_for_metadata_defined? attrs_for_callables end def job_callable TableSync.publishing_job_class_callable end def job_callable_error_message "Can't publish, set TableSync.publishing_job_class_callable" end def enqueue_job(perform_at = current_time) job = job_class.set(wait_until: perform_at) job.perform_later(object_class.name, original_attributes, state: state.to_s, confirm: confirm?) Rails.cache.write(cache_key, perform_at) end def routing_key resolve_routing_key end def publishing_data { **super, event: event, metadata: { created: created? }, } end def attributes_for_sync if destroyed? if object_class.respond_to?(:table_sync_destroy_attributes) object_class.table_sync_destroy_attributes(original_attributes) else needle end elsif attributes_for_sync_defined? object.attributes_for_sync else TableSync.orm.attributes(object) end end memoize def object TableSync.orm.find(object_class, needle) end def event destroyed? ? :destroy : :update end def needle original_attributes.slice(*primary_keys) end def cache_key "#{object_class}/#{needle}_table_sync_time".delete(" ") end def destroyed? state == :destroyed end def created? state == :created end def validate_state raise "Unknown state: #{state.inspect}" unless %i[created updated destroyed].include?(state) end end