mirror of
https://github.com/heartcombo/devise.git
synced 2022-11-09 12:18:31 -05:00
4837bb0a4e
Introspecting the scope of an object can make it difficult to use wrapper patterns. See issue plataformatec/devise#3307 for an example. Allow objects to specify their scope explicitly by implementing `devise_scope`.
176 lines
5.4 KiB
Ruby
176 lines
5.4 KiB
Ruby
module Devise
|
|
# Responsible for handling devise mappings and routes configuration. Each
|
|
# resource configured by devise_for in routes is actually creating a mapping
|
|
# object. You can refer to devise_for in routes for usage options.
|
|
#
|
|
# The required value in devise_for is actually not used internally, but it's
|
|
# inflected to find all other values.
|
|
#
|
|
# map.devise_for :users
|
|
# mapping = Devise.mappings[:user]
|
|
#
|
|
# mapping.name #=> :user
|
|
# # is the scope used in controllers and warden, given in the route as :singular.
|
|
#
|
|
# mapping.as #=> "users"
|
|
# # how the mapping should be search in the path, given in the route as :as.
|
|
#
|
|
# mapping.to #=> User
|
|
# # is the class to be loaded from routes, given in the route as :class_name.
|
|
#
|
|
# mapping.modules #=> [:authenticatable]
|
|
# # is the modules included in the class
|
|
#
|
|
class Mapping #:nodoc:
|
|
attr_reader :singular, :scoped_path, :path, :controllers, :path_names,
|
|
:class_name, :sign_out_via, :format, :used_routes, :used_helpers,
|
|
:failure_app, :router_name
|
|
|
|
alias :name :singular
|
|
|
|
# Receives an object and find a scope for it. If a scope cannot be found,
|
|
# raises an error. If a symbol is given, it's considered to be the scope.
|
|
def self.find_scope!(obj)
|
|
obj = obj.devise_scope if obj.respond_to?(:devise_scope)
|
|
case obj
|
|
when String, Symbol
|
|
return obj.to_sym
|
|
when Class
|
|
Devise.mappings.each_value { |m| return m.name if obj <= m.to }
|
|
else
|
|
Devise.mappings.each_value { |m| return m.name if obj.is_a?(m.to) }
|
|
end
|
|
|
|
raise "Could not find a valid mapping for #{obj.inspect}"
|
|
end
|
|
|
|
def self.find_by_path!(path, path_type=:fullpath)
|
|
Devise.mappings.each_value { |m| return m if path.include?(m.send(path_type)) }
|
|
raise "Could not find a valid mapping for path #{path.inspect}"
|
|
end
|
|
|
|
def initialize(name, options) #:nodoc:
|
|
@scoped_path = options[:as] ? "#{options[:as]}/#{name}" : name.to_s
|
|
@singular = (options[:singular] || @scoped_path.tr('/', '_').singularize).to_sym
|
|
|
|
@class_name = (options[:class_name] || name.to_s.classify).to_s
|
|
@klass = Devise.ref(@class_name)
|
|
|
|
@path = (options[:path] || name).to_s
|
|
@path_prefix = options[:path_prefix]
|
|
|
|
@sign_out_via = options[:sign_out_via] || Devise.sign_out_via
|
|
@format = options[:format]
|
|
|
|
@router_name = options[:router_name]
|
|
|
|
default_failure_app(options)
|
|
default_controllers(options)
|
|
default_path_names(options)
|
|
default_used_route(options)
|
|
default_used_helpers(options)
|
|
end
|
|
|
|
# Return modules for the mapping.
|
|
def modules
|
|
@modules ||= to.respond_to?(:devise_modules) ? to.devise_modules : []
|
|
end
|
|
|
|
# Gives the class the mapping points to.
|
|
def to
|
|
@klass.get
|
|
end
|
|
|
|
def strategies
|
|
@strategies ||= STRATEGIES.values_at(*self.modules).compact.uniq.reverse
|
|
end
|
|
|
|
def no_input_strategies
|
|
self.strategies & Devise::NO_INPUT
|
|
end
|
|
|
|
def routes
|
|
@routes ||= ROUTES.values_at(*self.modules).compact.uniq
|
|
end
|
|
|
|
def authenticatable?
|
|
@authenticatable ||= self.modules.any? { |m| m.to_s =~ /authenticatable/ }
|
|
end
|
|
|
|
def fullpath
|
|
"/#{@path_prefix}/#{@path}".squeeze("/")
|
|
end
|
|
|
|
# Create magic predicates for verifying what module is activated by this map.
|
|
# Example:
|
|
#
|
|
# def confirmable?
|
|
# self.modules.include?(:confirmable)
|
|
# end
|
|
#
|
|
def self.add_module(m)
|
|
class_eval <<-METHOD, __FILE__, __LINE__ + 1
|
|
def #{m}?
|
|
self.modules.include?(:#{m})
|
|
end
|
|
METHOD
|
|
end
|
|
|
|
private
|
|
|
|
def default_failure_app(options)
|
|
@failure_app = options[:failure_app] || Devise::FailureApp
|
|
if @failure_app.is_a?(String)
|
|
ref = Devise.ref(@failure_app)
|
|
@failure_app = lambda { |env| ref.get.call(env) }
|
|
end
|
|
end
|
|
|
|
def default_controllers(options)
|
|
mod = options[:module] || "devise"
|
|
@controllers = Hash.new { |h,k| h[k] = "#{mod}/#{k}" }
|
|
@controllers.merge!(options[:controllers]) if options[:controllers]
|
|
@controllers.each { |k,v| @controllers[k] = v.to_s }
|
|
end
|
|
|
|
def default_path_names(options)
|
|
@path_names = Hash.new { |h,k| h[k] = k.to_s }
|
|
@path_names[:registration] = ""
|
|
@path_names.merge!(options[:path_names]) if options[:path_names]
|
|
end
|
|
|
|
def default_constraints(options)
|
|
@constraints = Hash.new
|
|
@constraints.merge!(options[:constraints]) if options[:constraints]
|
|
end
|
|
|
|
def default_defaults(options)
|
|
@defaults = Hash.new
|
|
@defaults.merge!(options[:defaults]) if options[:defaults]
|
|
end
|
|
|
|
def default_used_route(options)
|
|
singularizer = lambda { |s| s.to_s.singularize.to_sym }
|
|
|
|
if options.has_key?(:only)
|
|
@used_routes = self.routes & Array(options[:only]).map(&singularizer)
|
|
elsif options[:skip] == :all
|
|
@used_routes = []
|
|
else
|
|
@used_routes = self.routes - Array(options[:skip]).map(&singularizer)
|
|
end
|
|
end
|
|
|
|
def default_used_helpers(options)
|
|
singularizer = lambda { |s| s.to_s.singularize.to_sym }
|
|
|
|
if options[:skip_helpers] == true
|
|
@used_helpers = @used_routes
|
|
elsif skip = options[:skip_helpers]
|
|
@used_helpers = self.routes - Array(skip).map(&singularizer)
|
|
else
|
|
@used_helpers = self.routes
|
|
end
|
|
end
|
|
end
|
|
end
|