# frozen_string_literal: true module VersionFu def self.included(base) base.extend ClassMethods end module ClassMethods # rubocop:todo Metrics/PerceivedComplexity # rubocop:todo Metrics/MethodLength # rubocop:todo Metrics/AbcSize def version_fu(options = {}, &) # rubocop:todo Metrics/CyclomaticComplexity, Metrics/AbcSize, Metrics/MethodLength, Metrics/PerceivedComplexity return if included_modules.include? VersionFu::InstanceMethods __send__ :include, VersionFu::InstanceMethods cattr_accessor :versioned_class_name, :versioned_foreign_key, :versioned_table_name, :version_column, :versioned_columns self.versioned_class_name = options[:class_name] || 'Version' self.versioned_foreign_key = options[:foreign_key] || to_s.foreign_key # rubocop:todo Layout/LineLength self.versioned_table_name = options[:table_name] || "#{table_name_prefix}#{base_class.name.demodulize.underscore}_versions#{table_name_suffix}" # rubocop:enable Layout/LineLength self.version_column = options[:version_column] || 'version' # Setup versions association class_eval do has_many :versions, class_name: "#{self}::#{versioned_class_name}", foreign_key: versioned_foreign_key, dependent: :destroy do def latest order(version: :desc).first end end before_save :check_for_new_version end # Versioned Model const_set(versioned_class_name, Class.new(ActiveRecord::Base)).class_eval do # find first version before the given version def self.before(version) where("#{original_class.versioned_foreign_key} = ? and version < ?", version.send(original_class.versioned_foreign_key), version.version) .order(version: :desc).first end # find first version after the given version. def self.after(version) where("#{original_class.versioned_foreign_key} = ? and version > ?", version.send(original_class.versioned_foreign_key), version.version) .order(version: :asc).first end def previous self.class.before(self) end def next self.class.after(self) end end # Housekeeping on versioned class versioned_class.cattr_accessor :original_class versioned_class.original_class = self versioned_class.table_name = versioned_table_name # Version parent association versioned_class.belongs_to to_s.demodulize.underscore.to_sym, class_name: "::#{self}", foreign_key: versioned_foreign_key # Block extension versioned_class.class_eval(&) if block_given? if versioned_class.table_exists? # Finally setup which columns to version self.versioned_columns = versioned_class.new.attributes.keys - [versioned_class.primary_key, versioned_foreign_key, version_column, 'created_at', 'updated_at'] else ActiveRecord::Base.logger.warn 'Version Table not found' end end # rubocop:enable Metrics/AbcSize # rubocop:enable Metrics/MethodLength # rubocop:enable Metrics/PerceivedComplexity def versioned_class const_get versioned_class_name end end module InstanceMethods def find_version(number) versions.find_by(version: number) end def check_for_new_version instantiate_revision if create_new_version? true # Never halt save end # This the method to override if you want to have more control over when to version def create_new_version? # Any versioned column changed? self.class.versioned_columns.detect { |a| __send__ "#{a}_changed?" } end def instantiate_revision new_version = versions.build versioned_columns.each do |attribute| new_version.__send__ "#{attribute}=", __send__(attribute) end version_number = new_record? ? 1 : version + 1 new_version.version = version_number self.version = version_number end end end