class VersionRecorder EVENT_TYPES = ['create', 'update', 'discard', 'restore'].freeze def initialize record:, event_type:, created_by_user: @record = record @event_type = event_type.to_s @created_by_user = created_by_user validate_event_type! end def record! record, event_type:, created_by_user: @record.with_lock do latest = latest_version if !(latest) && @event_type != 'create' raise "#{ version_class.name } first event must be create" end if @event_type == 'create' && latest raise "#{ version_class.name } create event already exists" end attrs = snapshot_attributes return latest if @event_type == 'update' && latest && same_snapshot?(latest, attrs) version_class.create!(base_attributes(latest).merge(record_key => @record).merge(attrs)) end end private def latest_version = versions.order(version_no: :desc).first def versions = @record.public_send(version_association) def base_attributes latest { version_no: (latest&.version_no || 0) + 1, event_type: @event_type, created_at: Time.current, created_by_user: @created_by_user } end def same_snapshot?(version, attrs) = attrs.all? { |k, v| version.public_send(k) == v } def validate_event_type! return if EVENT_TYPES.include?(@event_type) raise ArgumentError, "Invalid event_type: #{ @event_type }" end def version_class = raise NotImplementedError def version_association = raise NotImplementedError def record_key = raise NotImplementedError def snapshot_attributes = raise NotImplementedError end