# Copyright (c) 2022-2025, The Isaac Lab Project Developers.# All rights reserved.## SPDX-License-Identifier: BSD-3-Clause"""Sub-module with utility for importing all modules in a package recursively."""from__future__importannotationsimportimportlibimportpkgutilimportsys
[docs]defimport_packages(package_name:str,blacklist_pkgs:list[str]|None=None):"""Import all sub-packages in a package recursively. It is easier to use this function to import all sub-packages in a package recursively than to manually import each sub-package. It replaces the need of the following code snippet on the top of each package's ``__init__.py`` file: .. code-block:: python import .locomotion.velocity import .manipulation.reach import .manipulation.lift Args: package_name: The package name. blacklist_pkgs: The list of blacklisted packages to skip. Defaults to None, which means no packages are blacklisted. """# Default blacklistifblacklist_pkgsisNone:blacklist_pkgs=[]# Import the package itselfpackage=importlib.import_module(package_name)# Import all Python filesfor_in_walk_packages(package.__path__,package.__name__+".",blacklist_pkgs=blacklist_pkgs):pass
def_walk_packages(path:str|None=None,prefix:str="",onerror:callable|None=None,blacklist_pkgs:list[str]|None=None,):"""Yields ModuleInfo for all modules recursively on path, or, if path is None, all accessible modules. Note: This function is a modified version of the original ``pkgutil.walk_packages`` function. It adds the `blacklist_pkgs` argument to skip blacklisted packages. Please refer to the original ``pkgutil.walk_packages`` function for more details. """ifblacklist_pkgsisNone:blacklist_pkgs=[]defseen(p,m={}):ifpinm:returnTruem[p]=True# noqa: R503forinfoinpkgutil.iter_modules(path,prefix):# check blacklistedifany([black_pkg_nameininfo.nameforblack_pkg_nameinblacklist_pkgs]):continue# yield the module infoyieldinfoifinfo.ispkg:try:__import__(info.name)exceptException:ifonerrorisnotNone:onerror(info.name)else:raiseelse:path=getattr(sys.modules[info.name],"__path__",None)or[]# don't traverse path items we've seen beforepath=[pforpinpathifnotseen(p)]yield from_walk_packages(path,info.name+".",onerror,blacklist_pkgs)