# typed: true # frozen_string_literal: true module Tapioca module Runtime module Trackers # Registers a TracePoint immediately upon load to track points at which # classes and modules are opened for definition. This is used to track # correspondence between classes/modules and files, as this information isn't # available in the ruby runtime without extra accounting. module ConstantDefinition extend Reflection extend T::Sig class ConstantLocation < T::Struct const :lineno, Integer const :path, String end @class_files = {}.compare_by_identity # Immediately activated upon load. Observes class/module definition. Tapioca.register_trace(:class) do |tp| next if tp.self.singleton_class? key = tp.self path = tp.path if File.exist?(path) loc = build_constant_location(tp, caller_locations) else caller_location = T.must(caller_locations) .find { |loc| loc.path && File.exist?(loc.path) } next unless caller_location loc = ConstantLocation.new(path: caller_location.absolute_path || "", lineno: caller_location.lineno) end (@class_files[key] ||= Set.new) << loc end Tapioca.register_trace(:c_return) do |tp| next unless tp.method_id == :new next unless Module === tp.return_value key = tp.return_value loc = build_constant_location(tp, caller_locations) (@class_files[key] ||= Set.new) << loc end class << self def build_constant_location(tp, locations) file = resolve_loc(caller_locations) lineno = file == File.realpath(tp.path) ? tp.lineno : 0 ConstantLocation.new(path: file, lineno: lineno) end # Returns the files in which this class or module was opened. Doesn't know # about situations where the class was opened prior to +require+ing, # or where metaprogramming was used via +eval+, etc. def files_for(klass) locations_for(klass).map(&:path).to_set end def locations_for(klass) @class_files.fetch(klass, Set.new) end end end end end end