# frozen_string_literal: truemoduleActiveSupport# Error generated by +delegate+ when a method is called on +nil+ and +allow_nil+# option is not used.classDelegationError<NoMethodErrorclass<<selfdefnil_target(method_name,target)# :nodoc:new("#{method_name} delegated to #{target}, but #{target} is nil")endendendmoduleDelegation# :nodoc:RUBY_RESERVED_KEYWORDS=%w(__ENCODING__ __LINE__ __FILE__ alias and BEGIN begin break
case class def defined? do else elsif END end ensure false for if in module next nil
not or redo rescue retry return self super then true undef unless until when while yield)RESERVED_METHOD_NAMES=(RUBY_RESERVED_KEYWORDS+%w(_ arg args block)).to_set.freezeclass<<selfdefgenerate(owner,methods,location: nil,to: nil,prefix: nil,allow_nil: nil,nilable: true,private: nil,as: nil,signature: nil)unlesstoraiseArgumentError,"Delegation needs a target. Supply a keyword argument 'to' (e.g. delegate :hello, to: :greeter)."endifprefix==true&&/^[^a-z_]/.match?(to)raiseArgumentError,"Can only automatically set the delegation prefix when delegating to a method."endmethod_prefix=\ifprefix"#{prefix==true?to:prefix}_"else""endlocation||=caller_locations(1,1).firstfile,line=location.path,location.linenoreceiver=ifto.is_a?(Module)ifto.name.nil?raiseArgumentError,"Can't delegate to anonymous class or module: #{to}"endunlessInflector.safe_constantize(to.name).equal?(to)raiseArgumentError,"Can't delegate to detached class or module: #{to.name}"end"::#{to.name}"elseto.to_sendreceiver="self.#{receiver}"ifRESERVED_METHOD_NAMES.include?(receiver)explicit_receiver=falsereceiver_class=ifasexplicit_receiver=trueaselsifto.is_a?(Module)to.singleton_classelsifreceiver=="self.class"nilable=false# self.class can't possibly be nilowner.singleton_classendmethod_def=[]method_names=[]method_def<<"self.private"ifprivatemethods.eachdo|method|method_name=prefix?"#{method_prefix}#{method}":methodmethod_names<<method_name.to_sym# Attribute writer methods only accept one argument. Makes sure []=# methods still accept two arguments.definition=\ifsignaturesignatureelsif/[^\]]=\z/.match?(method)"arg"elsemethod_object=ifreceiver_classbeginreceiver_class.public_instance_method(method)rescueNameErrorraiseifexplicit_receiver# Do nothing. Fall back to `"..."`endendifmethod_objectparameters=method_object.parametersifparameters.map(&:first).intersect?([:opt,:rest,:keyreq,:key,:keyrest])"..."elsedefn=parameters.filter_map{|type,arg|argiftype==:req}defn<<"&"defn.join(", ")endelse"..."endend# The following generated method calls the target exactly once, storing# the returned value in a dummy variable.## Reason is twofold: On one hand doing less calls is in general better.# On the other hand it could be that the target has side-effects,# whereas conceptually, from the user point of view, the delegator should# be doing one call.ifnilable==falsemethod_def<<"def #{method_name}(#{definition})"<<" (#{receiver}).#{method}(#{definition})"<<"end"elsifallow_nilmethod=method.to_smethod_def<<"def #{method_name}(#{definition})"<<" _ = #{receiver}"<<" if !_.nil? || nil.respond_to?(:#{method})"<<" _.#{method}(#{definition})"<<" end"<<"end"elsemethod=method.to_smethod_name=method_name.to_smethod_def<<"def #{method_name}(#{definition})"<<" _ = #{receiver}"<<" _.#{method}(#{definition})"<<"rescue NoMethodError => e"<<" if _.nil? && e.name == :#{method}"<<" raise ::ActiveSupport::DelegationError.nil_target(:#{method_name}, :'#{receiver}')"<<" else"<<" raise"<<" end"<<"end"endendowner.module_eval(method_def.join(";"),file,line)method_namesenddefgenerate_method_missing(owner,target,allow_nil: nil)target=target.to_starget="self.#{target}"ifRESERVED_METHOD_NAMES.include?(target)||target=="__target"ifallow_nilowner.module_eval<<~RUBY,__FILE__,__LINE__+1
def respond_to_missing?(name, include_private = false)
# It may look like an oversight, but we deliberately do not pass
# +include_private+, because they do not get delegated.
return false if name == :marshal_dump || name == :_dump
#{target}.respond_to?(name) || super
end
def method_missing(method, ...)
__target = #{target}
if __target.nil? && !nil.respond_to?(method)
nil
elsif __target.respond_to?(method)
__target.public_send(method, ...)
else
super
end
end
RUBYelseowner.module_eval<<~RUBY,__FILE__,__LINE__+1
def respond_to_missing?(name, include_private = false)
# It may look like an oversight, but we deliberately do not pass
# +include_private+, because they do not get delegated.
return false if name == :marshal_dump || name == :_dump
#{target}.respond_to?(name) || super
end
def method_missing(method, ...)
__target = #{target}
if __target.nil? && !nil.respond_to?(method)
raise ::ActiveSupport::DelegationError.nil_target(method, :'#{target}')
elsif __target.respond_to?(method)
__target.public_send(method, ...)
else
super
end
end
RUBYendendendendend