class RBS::TypeNameResolver
def self.from_env(env)
def self.from_env(env) new.add_names(env.class_decls.keys) .add_names(env.interface_decls.keys) .add_names(env.alias_decls.keys) end
def add_names(names)
def add_names(names) all_names.merge(names) self end
def has_name?(full_name)
def has_name?(full_name) if all_names.include?(full_name) full_name end end
def initialize()
def initialize() @all_names = Set[] @cache = {} end
def resolve(type_name, context:)
def resolve(type_name, context:) if type_name.absolute? return type_name end query = Query.new(type_name: type_name, context: context) try_cache(query) do path_head, *path_tail = type_name.to_namespace.path raise unless path_head name_head = TypeName.new(name: path_head, namespace: Namespace.empty) absolute_head = context.find do |namespace| # @type break: TypeName full_name = name_head.with_prefix(namespace) has_name?(full_name) and break full_name end case absolute_head when TypeName has_name?(Namespace.new(path: absolute_head.to_namespace.path.push(*path_tail), absolute: true).to_type_name) when Namespace # This cannot happen because the `context.find` doesn't return a Namespace. raise end end end
def try_cache(query)
def try_cache(query) cache.fetch(query) do result = yield cache[query] = result end end