module HasHandleFallback SUB_REGEXP = '\_\-a-zA-Z0-9' REGEXP = /\A[#{SUB_REGEXP}]+\z/ ANTI_REGEXP = /[^#{SUB_REGEXP}]+/ LENGTH_RANGE = 2..32 RECORD_ID_REGEXP = /\A\d+\z/ def self.str2handle(str) str = str.to_s.gsub ANTI_REGEXP, '' str << ('_' * (LENGTH_RANGE.min - str.length)) unless LENGTH_RANGE.include?(str.length) str end module ActiveRecordBaseMethods def has_handle_fallback(fallback_column, options = {}) include InstanceMethods extend ClassMethods class_eval do cattr_accessor :has_handle_fallback_options self.has_handle_fallback_options = {} has_handle_fallback_options[:fallback_column] = fallback_column.to_s has_handle_fallback_options[:handle_column] = options.delete(:handle_column) || 'handle' validate :handle_is_valid end end end module ClassMethods def find_by_id_or_handle(param) return if param.blank? param = param.to_s if param.to_s =~ HasHandleFallback::RECORD_ID_REGEXP find_by_id param else send "find_by_#{has_handle_fallback_options[:handle_column]}", param end end alias :[] :find_by_id_or_handle end module InstanceMethods def handle_is_valid raw = read_attribute self.class.has_handle_fallback_options[:handle_column] cooked = handle_fallback # inline check to make sure the handle_fallback method works unless cooked =~ HasHandleFallback::REGEXP and HasHandleFallback::LENGTH_RANGE.include?(cooked.length) raise "Dear Developer: your handle_fallback method is not generating valid handles (generated '#{handle_fallback}' for '#{raw}')" end # allow nils but not blanks if !raw.nil? and raw.blank? errors.add self.class.has_handle_fallback_options[:handle_column], "can't be blank if you're presented the opportunity to set it" end # trapdoor for nil handles return if raw.nil? # don't allow all integer handles, because it looks like a database record id if raw =~ HasHandleFallback::RECORD_ID_REGEXP errors.add self.class.has_handle_fallback_options[:handle_column], "can't be entirely composed of integers" end # validates_format_of :handle, :with => HasHandleFallback::REGEXP, :allow_nil => true unless raw =~ HasHandleFallback::REGEXP errors.add self.class.has_handle_fallback_options[:handle_column], "contains invalid characters" end # validates_length_of :handle, :in => HasHandleFallback::LENGTH_RANGE, :allow_nil => true unless HasHandleFallback::LENGTH_RANGE.include? raw.length errors.add self.class.has_handle_fallback_options[:handle_column], "must be #{HasHandleFallback::LENGTH_RANGE} characters in length" end # validates_uniqueness_of :handle, :case_sensitive => false, :allow_nil => true if self.class.exists? [ "#{self.class.quoted_table_name}.`#{self.class.primary_key}` <> ? AND LOWER(#{self.class.quoted_table_name}.`#{self.class.has_handle_fallback_options[:handle_column]}`) = ?", id, raw.downcase ] errors.add self.class.has_handle_fallback_options[:handle_column], "isn't unique" end end def handle raw = read_attribute self.class.has_handle_fallback_options[:handle_column] raw.present? ? raw : handle_fallback end def handle_fallback fallback = read_attribute self.class.has_handle_fallback_options[:fallback_column] fallback = fallback.split('@').first if fallback.to_s.include? '@' HasHandleFallback.str2handle fallback end def to_param raw = read_attribute self.class.has_handle_fallback_options[:handle_column] if new_record? '' elsif raw.blank? or changes.include?(self.class.has_handle_fallback_options[:handle_column]) id.to_s else raw end end end end ActiveRecord::Base.extend HasHandleFallback::ActiveRecordBaseMethods