Source code for pykern.pkresource
"""Where external resources are stored
:copyright: Copyright (c) 2015 RadiaSoft LLC. All Rights Reserved.
:license: http://www.apache.org/licenses/LICENSE-2.0.html
"""
# Root module: avoid importing modules which import pkconfig
from pykern import pkconst
from pykern import pkinspect
from pykern import pkio
import errno
import glob
import importlib
import os.path
import sys
from pykern.pkdebug import pkdp
if sys.version_info < (3, 10):
# There's a bug resources.files() in 3.9.15 so need this.
# Also works with older versions than 3.9
import pkg_resources
def _resource_filename(package, path):
return pkg_resources.resource_filename(
package,
os.path.join(pkconst.PACKAGE_DATA, path),
)
else:
import importlib.resources
def _resource_filename(package, path):
return str(
importlib.resources.files(package).joinpath(pkconst.PACKAGE_DATA, path)
)
[docs]
def file_path(relative_filename, caller_context=None, packages=None):
"""Return the path to the resource
Args:
relative_filename (str): file name relative to package_data directory.
caller_context (object): Any object from which to get the `root_package`
packages (List[str]): Packages to search.
Returns:
py.path: absolute path of the resource file
"""
return pkio.py_path(filename(relative_filename, caller_context, packages))
[docs]
def filename(relative_filename, caller_context=None, packages=None):
"""Return the filename to the resource
Args:
relative_filename (str): file name relative to package_data directory.
caller_context (object): Any object from which to get the `root_package`
packages (List[str]): Packages to search.
Returns:
str: absolute path of the resource file
"""
assert not os.path.isabs(
relative_filename
), "must not be an absolute file name={}".format(relative_filename)
a = []
for f, p in _files(relative_filename, caller_context, packages):
a.append(p)
if os.path.exists(f):
return f
_raise_no_file_found(a, relative_filename)
[docs]
def glob_paths(relative_path, caller_context=None, packages=None):
"""Find all paths that match the relative path in all packages
Args:
relative_path(str): Path relative to package_data directory.
caller_context (object): Any object from which to get the `root_package`.
packages (List[str]): Packages to search.
Returns:
py.path: absolute paths of the matched files
"""
r = []
for f, p in _files(relative_path, caller_context, packages):
r.extend(glob.glob(f))
return [pkio.py_path(f) for f in r]
def _files(path, caller_context, packages):
if caller_context and packages:
raise ValueError(
f"Use only one of caller_context={caller_context} and packages={packages}",
)
for p in list(
map(
lambda m: pkinspect.root_package(importlib.import_module(m)),
packages
or [
pkinspect.root_package(
caller_context if caller_context else pkinspect.caller_module()
)
],
)
):
yield (
_resource_filename(p, path),
p,
)
def _raise_no_file_found(packages, path):
msg = f"unable to locate in packages={packages}"
if "__main__" in packages:
msg += "; do not call module as a program"
raise IOError(errno.ENOENT, msg, path)