mirror of https://github.com/python/cpython
174 lines
4.8 KiB
Python
174 lines
4.8 KiB
Python
import os
|
|
import pathlib
|
|
import py_compile
|
|
import shutil
|
|
import textwrap
|
|
import unittest
|
|
import warnings
|
|
import importlib
|
|
import contextlib
|
|
|
|
from importlib import resources
|
|
from importlib.resources.abc import Traversable
|
|
from . import util
|
|
from test.support import os_helper, import_helper
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def suppress_known_deprecation():
|
|
with warnings.catch_warnings(record=True) as ctx:
|
|
warnings.simplefilter('default', category=DeprecationWarning)
|
|
yield ctx
|
|
|
|
|
|
class FilesTests:
|
|
def test_read_bytes(self):
|
|
files = resources.files(self.data)
|
|
actual = files.joinpath('utf-8.file').read_bytes()
|
|
assert actual == b'Hello, UTF-8 world!\n'
|
|
|
|
def test_read_text(self):
|
|
files = resources.files(self.data)
|
|
actual = files.joinpath('utf-8.file').read_text(encoding='utf-8')
|
|
assert actual == 'Hello, UTF-8 world!\n'
|
|
|
|
def test_traversable(self):
|
|
assert isinstance(resources.files(self.data), Traversable)
|
|
|
|
def test_joinpath_with_multiple_args(self):
|
|
files = resources.files(self.data)
|
|
binfile = files.joinpath('subdirectory', 'binary.file')
|
|
self.assertTrue(binfile.is_file())
|
|
|
|
def test_old_parameter(self):
|
|
"""
|
|
Files used to take a 'package' parameter. Make sure anyone
|
|
passing by name is still supported.
|
|
"""
|
|
with suppress_known_deprecation():
|
|
resources.files(package=self.data)
|
|
|
|
|
|
class OpenDiskTests(FilesTests, util.DiskSetup, unittest.TestCase):
|
|
pass
|
|
|
|
|
|
class OpenZipTests(FilesTests, util.ZipSetup, unittest.TestCase):
|
|
pass
|
|
|
|
|
|
class OpenNamespaceTests(FilesTests, util.DiskSetup, unittest.TestCase):
|
|
MODULE = 'namespacedata01'
|
|
|
|
|
|
class OpenNamespaceZipTests(FilesTests, util.ZipSetup, unittest.TestCase):
|
|
ZIP_MODULE = 'namespacedata01'
|
|
|
|
|
|
class DirectSpec:
|
|
"""
|
|
Override behavior of ModuleSetup to write a full spec directly.
|
|
"""
|
|
|
|
MODULE = 'unused'
|
|
|
|
def load_fixture(self, name):
|
|
self.tree_on_path(self.spec)
|
|
|
|
|
|
class ModulesFiles:
|
|
spec = {
|
|
'mod.py': '',
|
|
'res.txt': 'resources are the best',
|
|
}
|
|
|
|
def test_module_resources(self):
|
|
"""
|
|
A module can have resources found adjacent to the module.
|
|
"""
|
|
import mod
|
|
|
|
actual = resources.files(mod).joinpath('res.txt').read_text(encoding='utf-8')
|
|
assert actual == self.spec['res.txt']
|
|
|
|
|
|
class ModuleFilesDiskTests(DirectSpec, util.DiskSetup, ModulesFiles, unittest.TestCase):
|
|
pass
|
|
|
|
|
|
class ModuleFilesZipTests(DirectSpec, util.ZipSetup, ModulesFiles, unittest.TestCase):
|
|
pass
|
|
|
|
|
|
class ImplicitContextFiles:
|
|
set_val = textwrap.dedent(
|
|
f"""
|
|
import {resources.__name__} as res
|
|
val = res.files().joinpath('res.txt').read_text(encoding='utf-8')
|
|
"""
|
|
)
|
|
spec = {
|
|
'somepkg': {
|
|
'__init__.py': set_val,
|
|
'submod.py': set_val,
|
|
'res.txt': 'resources are the best',
|
|
},
|
|
'frozenpkg': {
|
|
'__init__.py': set_val.replace(resources.__name__, 'c_resources'),
|
|
'res.txt': 'resources are the best',
|
|
},
|
|
}
|
|
|
|
def test_implicit_files_package(self):
|
|
"""
|
|
Without any parameter, files() will infer the location as the caller.
|
|
"""
|
|
assert importlib.import_module('somepkg').val == 'resources are the best'
|
|
|
|
def test_implicit_files_submodule(self):
|
|
"""
|
|
Without any parameter, files() will infer the location as the caller.
|
|
"""
|
|
assert importlib.import_module('somepkg.submod').val == 'resources are the best'
|
|
|
|
def _compile_importlib(self):
|
|
"""
|
|
Make a compiled-only copy of the importlib resources package.
|
|
|
|
Currently only code is copied, as importlib resources doesn't itself
|
|
have any resources.
|
|
"""
|
|
bin_site = self.fixtures.enter_context(os_helper.temp_dir())
|
|
c_resources = pathlib.Path(bin_site, 'c_resources')
|
|
sources = pathlib.Path(resources.__file__).parent
|
|
|
|
for source_path in sources.glob('**/*.py'):
|
|
c_path = c_resources.joinpath(source_path.relative_to(sources)).with_suffix('.pyc')
|
|
py_compile.compile(source_path, c_path)
|
|
self.fixtures.enter_context(import_helper.DirsOnSysPath(bin_site))
|
|
|
|
def test_implicit_files_with_compiled_importlib(self):
|
|
"""
|
|
Caller detection works for compiled-only resources module.
|
|
|
|
python/cpython#123085
|
|
"""
|
|
self._compile_importlib()
|
|
assert importlib.import_module('frozenpkg').val == 'resources are the best'
|
|
|
|
|
|
class ImplicitContextFilesDiskTests(
|
|
DirectSpec, util.DiskSetup, ImplicitContextFiles, unittest.TestCase
|
|
):
|
|
pass
|
|
|
|
|
|
class ImplicitContextFilesZipTests(
|
|
DirectSpec, util.ZipSetup, ImplicitContextFiles, unittest.TestCase
|
|
):
|
|
pass
|
|
|
|
|
|
if __name__ == '__main__':
|
|
unittest.main()
|