mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
80 lines
2.6 KiB
Ruby
80 lines
2.6 KiB
Ruby
![]() |
# typed: strict
|
||
|
# frozen_string_literal: true
|
||
|
|
||
|
require_relative "../../../global"
|
||
|
require "sorbet/tapioca/utils"
|
||
|
require "utils/ast"
|
||
|
|
||
|
module Tapioca
|
||
|
module Compilers
|
||
|
class Forwardables < Tapioca::Dsl::Compiler
|
||
|
FORWARDABLE_FILENAME = "forwardable.rb"
|
||
|
ARRAY_METHODS = T.let(["to_a", "to_ary"].freeze, T::Array[String])
|
||
|
HASH_METHODS = T.let(["to_h", "to_hash"].freeze, T::Array[String])
|
||
|
STRING_METHODS = T.let(["to_s", "to_str", "to_json"].freeze, T::Array[String])
|
||
|
|
||
|
ConstantType = type_member { { fixed: Module } }
|
||
|
|
||
|
sig { override.returns(T::Enumerable[Module]) }
|
||
|
def self.gather_constants
|
||
|
objects = T.cast(ObjectSpace.each_object(Forwardable), T::Enumerator[Module])
|
||
|
objects.map { |obj| named_object(obj) }.uniq.reject do |obj|
|
||
|
# Avoid duplicate stubs for forwardables that are defined in vendored gems
|
||
|
Object.const_source_location(T.must(obj.name))&.first&.include?("vendor/bundle/ruby")
|
||
|
end
|
||
|
end
|
||
|
|
||
|
sig { params(obj: Module).returns(Module) }
|
||
|
def self.named_object(obj)
|
||
|
if obj.is_a?(Class) && obj.name.nil?
|
||
|
T.cast(obj.attached_object, Module)
|
||
|
else
|
||
|
obj
|
||
|
end
|
||
|
end
|
||
|
|
||
|
sig { override.void }
|
||
|
def decorate
|
||
|
root.create_path(constant) do |klass|
|
||
|
Homebrew::Tapioca::Utils.methods_from_file(constant, FORWARDABLE_FILENAME)
|
||
|
.each { |method| compile_forwardable_method(klass, method) }
|
||
|
Homebrew::Tapioca::Utils.methods_from_file(constant, FORWARDABLE_FILENAME, class_methods: true)
|
||
|
.each { |method| compile_forwardable_method(klass, method, class_method: true) }
|
||
|
end
|
||
|
end
|
||
|
|
||
|
private
|
||
|
|
||
|
sig { params(klass: RBI::Scope, method: T.any(Method, UnboundMethod), class_method: T::Boolean).void }
|
||
|
def compile_forwardable_method(klass, method, class_method: false)
|
||
|
name = method.name.to_s
|
||
|
return_type = return_type(name)
|
||
|
klass.create_method(
|
||
|
name,
|
||
|
parameters: [
|
||
|
create_rest_param("args", type: "T.untyped"),
|
||
|
create_block_param("block", type: "T.untyped"),
|
||
|
],
|
||
|
return_type:,
|
||
|
class_method:,
|
||
|
)
|
||
|
end
|
||
|
|
||
|
sig { params(name: String).returns(String) }
|
||
|
def return_type(name)
|
||
|
if name.end_with?("?")
|
||
|
"T::Boolean"
|
||
|
elsif ARRAY_METHODS.include?(name)
|
||
|
"Array"
|
||
|
elsif HASH_METHODS.include?(name)
|
||
|
"Hash"
|
||
|
elsif STRING_METHODS.include?(name)
|
||
|
"String"
|
||
|
else
|
||
|
"T.untyped"
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|