lib/bootsnap/load_path_cache/path_scanner.rb
require_relative '../explicit_require' module Bootsnap module LoadPathCache module PathScanner # Glob pattern to find requirable files and subdirectories in given path. # It expands to: # # * `/*{.rb,.so,/}` - It matches requirable files, directories and # symlinks to directories at given path. # * `/*/**/*{.rb,.so,/}` - It matches requirable files and # subdirectories in any (even symlinked) directory at given path at # any directory tree depth. # REQUIRABLES_AND_DIRS = "/{,*/**/}*{#{DOT_RB},#{DL_EXTENSIONS.join(',')},/}" NORMALIZE_NATIVE_EXTENSIONS = !DL_EXTENSIONS.include?(LoadPathCache::DOT_SO) ALTERNATIVE_NATIVE_EXTENSIONS_PATTERN = /\.(o|bundle|dylib)\z/ BUNDLE_PATH = Bootsnap.bundler? ? (Bundler.bundle_path.cleanpath.to_s << LoadPathCache::SLASH).freeze : ''.freeze def self.call(path) path = path.to_s relative_slice = (path.size + 1)..-1 # If the bundle path is a descendent of this path, we do additional # checks to prevent recursing into the bundle path as we recurse # through this path. We don't want to scan the bundle path because # anything useful in it will be present on other load path items. # # This can happen if, for example, the user adds '.' to the load path, # and the bundle path is '.bundle'. contains_bundle_path = BUNDLE_PATH.start_with?(path) dirs = [] requirables = [] Dir.glob(path + REQUIRABLES_AND_DIRS).each do |absolute_path| next if contains_bundle_path && absolute_path.start_with?(BUNDLE_PATH) relative_path = absolute_path.slice!(relative_slice) if relative_path.end_with?('/') dirs << relative_path[0..-2] else requirables << relative_path end end [requirables, dirs] end end end end