module Stringex
module ActsAsUrl
module Adapter
class Base
attr_accessor :base_url, :callback_options, :configuration, :instance, :klass, :settings
def initialize(configuration)
ensure_loadable
self.configuration = configuration
self.settings = configuration.settings
end
def create_callbacks!(klass)
self.klass = klass
self.callback_options = {}
create_method_to_callback
create_callback
end
def ensure_unique_url!(instance)
@url_owners = nil
self.instance = instance
handle_url!
handle_blacklisted_url!
handle_duplicate_url! unless settings.allow_duplicates
end
def initialize_urls!(klass)
self.klass = klass
klass_previous_instances do |instance|
ensure_unique_url_for! instance
end
end
def url_attribute(instance)
# Retrieve from database record if there are errors on attribute_to_urlify
if !is_new?(instance) && is_present?(instance.errors[settings.attribute_to_urlify])
self.instance = instance
read_attribute instance_from_db, settings.url_attribute
else
read_attribute instance, settings.url_attribute
end
end
def self.ensure_loadable
raise "The #{self} adapter cannot be loaded" unless loadable?
Stringex::ActsAsUrl::Adapter.add_loaded_adapter self
end
def self.loadable?
orm_class
rescue NameError
false
end
private
def add_new_record_url_owner_conditions
return if is_new?(instance)
@url_owner_conditions.first << " and #{primary_key} != ?"
@url_owner_conditions << instance.id
end
def add_scoped_url_owner_conditions
[settings.scope_for_url].flatten.compact.each do |scope|
@url_owner_conditions.first << " and #{scope} = ?"
@url_owner_conditions << instance.send(scope)
end
end
def create_callback
klass.send klass_callback_method, :ensure_unique_url, callback_options
end
def klass_callback_method
settings.sync_url ? klass_sync_url_callback_method : klass_non_sync_url_callback_method
end
def klass_sync_url_callback_method
configuration.settings.callback_method
end
def klass_non_sync_url_callback_method
case configuration.settings.callback_method
when :before_save
:before_create
else # :before_validation
callback_options[:on] = :create
configuration.settings.callback_method
end
end
def create_method_to_callback
klass.class_eval <<-"END"
def #{settings.url_attribute}
acts_as_url_configuration.adapter.url_attribute self
end
END
end
def duplicate_for_base_url(n)
"#{base_url}#{settings.duplicate_count_separator}#{n}"
end
def ensure_loadable
self.class.ensure_loadable
end
# NOTE: The instance here is not the cached instance but a block variable
# passed from klass_previous_instances, just to be clear
def ensure_unique_url_for!(instance)
instance.send :ensure_unique_url
instance.save
end
def get_base_url_owner_conditions
@url_owner_conditions = ["#{settings.url_attribute} LIKE ?", base_url + '%']
end
def handle_duplicate_url!
return if !url_taken?(base_url)
n = nil
sequence = duplicate_url_sequence.tap(&:rewind)
loop do
n = sequence.next
break unless url_taken?(duplicate_for_base_url(n))
end
write_url_attribute duplicate_for_base_url(n)
end
def duplicate_url_sequence
settings.duplicate_sequence ||
Enumerator.new do |enum|
n = 1
loop do
enum.yield n
n += 1
end
end
end
def url_taken?(url)
if settings.url_taken_method
instance.send(settings.url_taken_method, url)
else
url_owners.any?{|owner| url_attribute_for(owner) == url}
end
end
def handle_url!
self.base_url = instance.send(settings.url_attribute)
modify_base_url if is_blank?(base_url) || !settings.only_when_blank
write_url_attribute base_url
end
def handle_blacklisted_url!
return unless settings.blacklist.to_set.include?(base_url)
self.base_url = settings.blacklist_policy.call(instance, base_url)
write_url_attribute base_url
end
def instance_from_db
instance.class.find(instance.id)
end
def is_blank?(object)
object.blank?
end
def is_new?(object)
object.new_record?
end
def is_present?(object)
object.present?
end
def loadable?
self.class.loadable?
end
def modify_base_url
root = instance.send(settings.attribute_to_urlify).to_s
self.base_url = root.to_url(configuration.string_extensions_settings)
end
def orm_class
self.class.orm_class
end
def primary_key
instance.class.primary_key
end
def read_attribute(instance, attribute)
instance.read_attribute attribute
end
def url_attribute_for(object)
object.send settings.url_attribute
end
def url_owner_conditions
get_base_url_owner_conditions
add_new_record_url_owner_conditions
add_scoped_url_owner_conditions
@url_owner_conditions
end
def url_owners
@url_owners ||= url_owners_class.unscoped.where(url_owner_conditions).to_a
end
def url_owners_class
return instance.class unless settings.enforce_uniqueness_on_sti_base_class
klass = instance.class
while klass.superclass < orm_class
klass = klass.superclass
end
klass
end
def write_attribute(instance, attribute, value)
instance.send :write_attribute, attribute, value
end
def write_url_attribute(value)
write_attribute instance, settings.url_attribute, value
end
end
end
end
end