# frozen_string_literal: true # # Copyright (c) 2021-2024 Hal Brodigan (postmodern.mod3 at gmail.com) # # ronin-repos is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published # by the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # ronin-repos is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with ronin-repos. If not, see . # require 'ronin/repos/exceptions' require 'ronin/repos/repository' require 'ronin/core/home' require 'set' module Ronin module Repos # # Manages the `~/.cache/ronin-repos/` directory and the repositories # contained within. # # @api private # class CacheDir include Enumerable # The `~/.cache/ronin-repos/` directory where all repos are stored. PATH = Core::Home.cache_dir('ronin-repos') # The path to the cache directory. # # @return [String] attr_reader :path # # Initializes the repository cache. # # @param [String] path # The path to the repository cache directory. # def initialize(path=PATH) @path = path end # # Accesses a repository from the cache directory. # # @param [String] name # The name of the repository. # # @return [Repository] # The repository from the cache. # # @raise [RepositoryNotFound] # No repository exists with the given name in the cache directory. # def [](name) path = File.join(@path,name.to_s) unless File.directory?(path) raise(RepositoryNotFound,"repository not found: #{name.inspect}") end return Repository.new(path) end # # Enumerates through every repository in the cache directory. # # @yield [repo] # The given block will be passed each repository. # # @yieldparam [Repository] repo # A repository from the cache directory. # # @return [Enumerator] # If no block is given, an Enumerator will be returned. # def each return enum_for unless block_given? each_child_directory do |path| yield Repository.new(path) end end # # Clones and installs a repository into the cache directory. # # @param [String, URI::HTTPS] uri # The URI to clone the repository from. # # @param [String, nil] name # The explicit name of the repository to use. Defaults to the base-name # of the URI's path, sans any `.git` extension. # # @return [Repository] # The newly installed repository. # # @raise [CommandFailed] # One of the `git` commands failed. # # @raise [CommandNotInstalled] # The `git` command is not installed. # def install(uri,name=nil) uri = uri.to_s name ||= File.basename(uri,File.extname(uri)) path = File.join(@path,name) return Repository.install(uri,path) end # # Updates all repositories in the cache directory. # # @raise [CommandNotInstalled] # The `git` command is not installed. # def update each do |repo| repo.update rescue CommandFailed # ignore any `git` errors when updating end end # # Removes a repository from the cache directory. # # @param [String] name # The repository name to find and delete. # # @raise [RepositoryNotFound] # The repository with the given name does not exist in the cache # directory. # def remove(name) self[name].delete end # # Deletes all repositories in the cache directory. # def purge each(&:delete) end # # Finds the first matching file. # # @param [String] path # The relative path of the file. # # @return [String, nil] # The absolute path of the matching file or `nil` if no matching file # could be found. # # @example # repos.find_file("wordlists/wordlist.txt") # # => "/home/user/.cache/ronin-repos/foo-repo/wordlists/wordlist.txt" # def find_file(path) each do |repo| if (file = repo.find_file(path)) return file end end end # # Finds all files in all repos that matches the glob pattern. # # @param [String] pattern # The file glob pattern to search for. # # @return [Array] # The absolute paths to the files that match the glob pattern. # # @example # repos.glob("wordlists/*.txt") # # => ["/home/user/.cache/ronin-repos/foo-repo/wordlists/cities.txt", # # "/home/user/.cache/ronin-repos/foo-repo/wordlists/states.txt", # # "/home/user/.cache/ronin-repos/bar-repo/wordlists/bands.txt", # # "/home/user/.cache/ronin-repos/bar-repo/wordlists/beers.txt"] # def glob(pattern,&block) return enum_for(__method__,pattern).to_a unless block_given? each do |repo| repo.glob(pattern,&block) end end # # Lists all files across all repos installed in the cache directory. # # @param [String] pattern # The optional glob pattern to use to list specific files. # # @return [Set] # The matching paths within the repository. # # @example # repos.list_files('exploits/{**/}*.rb') # # => # # def list_files(pattern='{**/}*.*') each_with_object(Set.new) do |repo,files| files.merge(repo.list_files(pattern)) end end # # Converts the cache directory to a String. # # @return [String] # The path to the cache directory. # def to_s @path end private # # Enumerates over each directory in the cache directory. # # @yield [dir] # The given block will be passed each repository's directory path. # # @yieldparam [String] dir # A path to a repository directory within the cache directory. # # @return [Enumerator] # If no block is given, an Enumerator will be returned. # def each_child_directory return enum_for(__method__) unless block_given? if File.directory?(@path) Dir.children(@path).sort.each do |name| path = File.join(@path,name) if File.directory?(path) yield path end end end return nil end end end end