# 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'
require 'ronin/core/class_registry'
require 'set'
module Ronin
module Repos
#
# Adds the ability to load classes from directories within installed
# repositories.
#
# ## Example
#
# `lib/ronin/exploits.rb`:
#
# require 'ronin/core/module_registry'
# require 'ronin/repos/class_dir'
#
# module Ronin
# module Exploits
# include Ronin::Core::ClassRegistry
# include Ronin::Repos::ClassDir
#
# class_dir "#{__dir__}/classes"
# repo_class_dir "exploits"
# end
# end
#
# `lib/ronin/exploits/exploit.rb`:
#
# module Ronin
# module Exploits
# class Exploit
#
# def self.register(name)
# Exploits.register(name,self)
# end
#
# end
# end
# end
#
# `~/.cache/ronin-repos/repo1/exploits/my_exploit.rb`:
#
# require 'ronin/exploits/exploit'
#
# module Ronin
# module Exploits
# class MyExploit < Exploit
#
# register 'my_exploit'
#
# end
# end
# end
#
# @api semipublic
#
module ClassDir
#
# Includes `Ronin::Core::ClassRegistry` and adds {ClassMethods}.
#
# @param [Module] namespace
# The module that is including {ClassDir}.
#
# @api private
#
def self.included(namespace)
namespace.send :include, Core::ClassRegistry
namespace.extend ClassMethods
end
#
# Class-methods.
#
module ClassMethods
#
# Gets or sets the repository module directory name.
#
# @param [String, nil] new_dir
# The new module directory path.
#
# @return [String]
# The repository module directory name.
#
# @raise [NotImplementedError]
# The `repo_class_dir` method was not defined in the module.
#
# @example
# repo_class_dir "exploits"
#
def repo_class_dir(new_dir=nil)
if new_dir
@repo_class_dir = new_dir
else
@repo_class_dir || raise(NotImplementedError,"#{self} did not define a repo_class_dir")
end
end
#
# List the module names within the `class_dir` and within
# {#repo_class_dir} across all installed repositories.
#
# @return [Array]
#
def list_files
paths = Set.new(super)
pattern = File.join(repo_class_dir,"{**/}*.rb")
# the String#slice range to remove the repo_class_dir/ and .rb ext
slice_range = (repo_class_dir.length + 1)...-3
Repos.list_files(pattern).each do |path|
# NOTE: String#slice is faster than .delete_prefix + delete.suffix
paths << path.slice(slice_range)
end
return paths.to_a
end
#
# Finds a module within `class_dir` or within `repo_class_dir`
# in one of the installed repositories.
#
# @param [String] name
# The module name to lookup.
#
# @return [String, nil]
# The path to the module or `nil` if the module could not be found
# in `class_dir` or any of the installed repositories.
#
def path_for(name)
super(name) ||
Repos.find_file(File.join(repo_class_dir,"#{name}.rb"))
end
end
end
end
end