From 9e7dc0a51b996de449e13e00119e141c84e264a4 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:33:26 +0100 Subject: [PATCH 01/27] tests: add helper utils for marking override tests --- upath/tests/utils.py | 79 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 79 insertions(+) diff --git a/upath/tests/utils.py b/upath/tests/utils.py index f69fe2bd..5900ffbe 100644 --- a/upath/tests/utils.py +++ b/upath/tests/utils.py @@ -63,3 +63,82 @@ def temporary_register(protocol, cls): finally: m.clear() get_upath_class.cache_clear() + + +def extends_base(method): + """Decorator to ensure a method extends the base class and does NOT + override a method in base classes. + + Use this decorator in implementation-specific test classes to ensure that + test methods don't accidentally override methods defined in test base classes. + + Example: + class TestSpecificImpl(TestBaseClass, metaclass=OverrideMeta): + @extends_base + def test_something(self): # Raises TypeError if base has this method + ... + + @extends_base + def test_new_method(self): # This is fine - no override + ... + """ + method.__override_check__ = False + return method + + +def overrides_base(method): + """Decorator to ensure a method DOES override a method in base classes. + + Use this decorator in implementation-specific test classes to ensure that + test methods intentionally override methods defined in test base classes. + + Example: + class TestSpecificImpl(TestBaseClass, metaclass=OverrideMeta): + @overrides_base + def test_something(self): # Raises TypeError if base lacks this method + ... + + @overrides_base + def test_new_method(self): # Raises TypeError - no method to override + ... + """ + method.__override_check__ = True + return method + + +class OverrideMeta(type): + """Metaclass that enforces @extends_base and @overrides_base decorator constraints. + + When a class uses this metaclass: + - Methods decorated with @extends_base are checked to ensure they don't + override a method from any base class. + - Methods decorated with @overrides_base are checked to ensure they do + override a method from at least one base class. + """ + + def __new__(mcs, name, bases, namespace): + for attr_name, attr_value in namespace.items(): + if not callable(attr_value): + continue + + check = getattr(attr_value, "__override_check__", None) + if check is None: + continue + + has_in_base = any(hasattr(base, attr_name) for base in bases) + + if check is False and has_in_base: + base_name = next(b.__name__ for b in bases if hasattr(b, attr_name)) + raise TypeError( + f"Method '{attr_name}' in class '{name}' is decorated " + f"with @extends_base but overrides a method from base " + f"class '{base_name}'" + ) + elif check is True and not has_in_base: + raise TypeError( + f"Method '{attr_name}' in class '{name}' is decorated " + f"with @overrides_base but does not override any method from " + f"base classes" + ) + + return super().__new__(mcs, name, bases, namespace) From ca388e6857016f37cb6f1045f2e96dcfe2956ac7 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:42:22 +0100 Subject: [PATCH 02/27] tests: factor out cls test --- upath/tests/cases.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/upath/tests/cases.py b/upath/tests/cases.py index 309a8de2..0ac948cb 100644 --- a/upath/tests/cases.py +++ b/upath/tests/cases.py @@ -32,6 +32,9 @@ class JoinablePathTests: path: UPath + def test_is_correct_class(self): + raise NotImplementedError("must override") + def test_parser(self): parser = self.path.parser assert isinstance(parser, PathParser) From f0fda59fef7fd65fc2e61ab463b07f28c239f899 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:37:12 +0100 Subject: [PATCH 03/27] tests: cleanup azure tests --- upath/tests/implementations/test_azure.py | 25 +++++++++++++++-------- 1 file changed, 16 insertions(+), 9 deletions(-) diff --git a/upath/tests/implementations/test_azure.py b/upath/tests/implementations/test_azure.py index 30b683ac..f2af0aff 100644 --- a/upath/tests/implementations/test_azure.py +++ b/upath/tests/implementations/test_azure.py @@ -4,12 +4,14 @@ from upath.implementations.cloud import AzurePath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base from ..utils import skip_on_windows @skip_on_windows -@pytest.mark.usefixtures("path") -class TestAzurePath(BaseTests): +class TestAzurePath(BaseTests, metaclass=OverrideMeta): SUPPORTS_EMPTY_DIRS = False @pytest.fixture(autouse=True, scope="function") @@ -23,9 +25,18 @@ def path(self, azurite_credentials, azure_fixture): self.path = UPath(azure_fixture, **self.storage_options) self.prepare_file_system() - def test_is_AzurePath(self): - assert isinstance(self.path, AzurePath) + @overrides_base + def test_is_correct_class(self): + return isinstance(self.path, AzurePath) + @overrides_base + def test_protocol(self): + # test all valid protocols for azure... + protocol = self.path.protocol + protocols = ["abfs", "abfss", "adl", "az"] + assert protocol in protocols + + @extends_base def test_rmdir(self): new_dir = self.path / "new_dir_rmdir" new_dir.mkdir() @@ -38,11 +49,7 @@ def test_rmdir(self): with pytest.raises(NotADirectoryError): (self.path / "a" / "file.txt").rmdir() - def test_protocol(self): - # test all valid protocols for azure... - protocol = self.path.protocol - assert protocol in ["abfs", "abfss", "adl", "az"] - + @extends_base def test_broken_mkdir(self): path = UPath( "az://new-container/", From 5d45e4fd0f40489c49f477f2abd6f78786378ec7 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:48:08 +0100 Subject: [PATCH 04/27] tests: cleanup cached tests --- upath/tests/implementations/test_cached.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/upath/tests/implementations/test_cached.py b/upath/tests/implementations/test_cached.py index e7b757f2..e7708797 100644 --- a/upath/tests/implementations/test_cached.py +++ b/upath/tests/implementations/test_cached.py @@ -4,9 +4,11 @@ from upath.implementations.cached import SimpleCachePath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base -class TestSimpleCachePath(BaseTests): +class TestSimpleCachePath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): if not local_testdir.startswith("/"): @@ -15,5 +17,6 @@ def path(self, local_testdir): self.path = UPath(path) self.prepare_file_system() - def test_is_SimpleCachePath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, SimpleCachePath) From 41fe1ac8e283e541134e34d0def80022b67a4386 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:52:43 +0100 Subject: [PATCH 05/27] tests: cleanup ftp tests --- upath/tests/implementations/test_ftp.py | 21 +++++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/upath/tests/implementations/test_ftp.py b/upath/tests/implementations/test_ftp.py index d8f46758..b7edc7ce 100644 --- a/upath/tests/implementations/test_ftp.py +++ b/upath/tests/implementations/test_ftp.py @@ -1,21 +1,30 @@ import pytest from upath import UPath +from upath.implementations.ftp import FTPPath from upath.tests.cases import BaseTests from upath.tests.utils import skip_on_windows +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base + @skip_on_windows -class TestUPathFTP(BaseTests): +class TestUPathFTP(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, ftp_server): self.path = UPath("", protocol="ftp", **ftp_server) self.prepare_file_system() + @overrides_base + def test_is_correct_class(self): + assert isinstance(self.path, FTPPath) -def test_ftp_path_mtime(ftp_server): - path = UPath("file1.txt", protocol="ftp", **ftp_server) - path.touch() - mtime = path.stat().st_mtime - assert isinstance(mtime, float) + @extends_base + def test_ftp_path_mtime(self, ftp_server): + path = UPath("file1.txt", protocol="ftp", **ftp_server) + path.touch() + mtime = path.stat().st_mtime + assert isinstance(mtime, float) From 0f816a09c90d1d55e70ce7a72d2bb34b1df9bb2b Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:55:52 +0100 Subject: [PATCH 06/27] tests: cleanup gcs tests --- upath/tests/implementations/test_gcs.py | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/upath/tests/implementations/test_gcs.py b/upath/tests/implementations/test_gcs.py index 30db5645..41d16c63 100644 --- a/upath/tests/implementations/test_gcs.py +++ b/upath/tests/implementations/test_gcs.py @@ -5,12 +5,14 @@ from upath.implementations.cloud import GCSPath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base from ..utils import skip_on_windows @skip_on_windows -@pytest.mark.usefixtures("path") -class TestGCSPath(BaseTests): +class TestGCSPath(BaseTests, metaclass=OverrideMeta): SUPPORTS_EMPTY_DIRS = False @pytest.fixture(autouse=True, scope="function") @@ -18,9 +20,11 @@ def path(self, gcs_fixture): path, endpoint_url = gcs_fixture self.path = UPath(path, endpoint_url=endpoint_url, token="anon") - def test_is_GCSPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, GCSPath) + @extends_base def test_rmdir(self): dirname = "rmdir_test" mock_dir = self.path.joinpath(dirname) @@ -31,10 +35,6 @@ def test_rmdir(self): with pytest.raises(NotADirectoryError): self.path.joinpath("file1.txt").rmdir() - @pytest.mark.skip - def test_makedirs_exist_ok_false(self): - pass - @skip_on_windows def test_mkdir_in_empty_bucket(docker_gcs): From 67e45ec5f2e4b47da13a4dc4d9f0c8dde50ce73f Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 21:58:25 +0100 Subject: [PATCH 07/27] tests: cleanup hdfs tests --- upath/tests/implementations/test_hdfs.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/upath/tests/implementations/test_hdfs.py b/upath/tests/implementations/test_hdfs.py index 6ece8ad8..cc4fe2f9 100644 --- a/upath/tests/implementations/test_hdfs.py +++ b/upath/tests/implementations/test_hdfs.py @@ -6,15 +6,18 @@ from upath.implementations.hdfs import HDFSPath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base @pytest.mark.hdfs -class TestUPathHDFS(BaseTests): +class TestUPathHDFS(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir, hdfs): host, user, port = hdfs path = f"hdfs:{local_testdir}" self.path = UPath(path, host=host, user=user, port=port) - def test_is_HDFSPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, HDFSPath) From 16f680a48641e48d6d35281a2eefc81bc2f7366c Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 22:06:20 +0100 Subject: [PATCH 08/27] tests: cleanup local tests --- upath/tests/implementations/test_local.py | 36 ++++++++++------------- 1 file changed, 15 insertions(+), 21 deletions(-) diff --git a/upath/tests/implementations/test_local.py b/upath/tests/implementations/test_local.py index db1bfd56..aef91139 100644 --- a/upath/tests/implementations/test_local.py +++ b/upath/tests/implementations/test_local.py @@ -5,56 +5,50 @@ from upath import UPath from upath.implementations.local import LocalPath -from upath.tests.cases import BaseTests -from upath.tests.utils import xfail_if_version +from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base +from ..utils import xfail_if_version -class TestFSSpecLocal(BaseTests): + +class TestFSSpecLocal(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): path = f"file://{local_testdir}" self.path = UPath(path) - def test_is_LocalPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, LocalPath) + @overrides_base def test_cwd(self): + # .cwd() is implemented for local filesystems cwd = type(self.path).cwd() assert isinstance(cwd, LocalPath) assert cwd.path == Path.cwd().as_posix() + @overrides_base def test_home(self): + # .home() is implemented for local filesystems cwd = type(self.path).home() assert isinstance(cwd, LocalPath) assert cwd.path == Path.home().as_posix() + @overrides_base def test_chmod(self): + # .chmod() works for local filesystems self.path.joinpath("file1.txt").chmod(777) @xfail_if_version("fsspec", lt="2023.10.0", reason="requires fsspec>=2023.10.0") -class TestRayIOFSSpecLocal(BaseTests): +class TestRayIOFSSpecLocal(TestFSSpecLocal): @pytest.fixture(autouse=True) def path(self, local_testdir): path = f"local://{local_testdir}" self.path = UPath(path) - def test_is_LocalPath(self): - assert isinstance(self.path, LocalPath) - - def test_cwd(self): - cwd = type(self.path).cwd() - assert isinstance(cwd, LocalPath) - assert cwd.path == Path.cwd().as_posix() - - def test_home(self): - cwd = type(self.path).home() - assert isinstance(cwd, LocalPath) - assert cwd.path == Path.home().as_posix() - - def test_chmod(self): - self.path.joinpath("file1.txt").chmod(777) - @pytest.mark.parametrize( "protocol,path", From 5f44e92aad1f9118c459da7fadf800ecc393f556 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 22:07:52 +0100 Subject: [PATCH 09/27] tests: cleanup memory tests --- upath/tests/implementations/test_memory.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/upath/tests/implementations/test_memory.py b/upath/tests/implementations/test_memory.py index 7a0b9aea..c98222fa 100644 --- a/upath/tests/implementations/test_memory.py +++ b/upath/tests/implementations/test_memory.py @@ -4,9 +4,11 @@ from upath.implementations.memory import MemoryPath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base -class TestMemoryPath(BaseTests): +class TestMemoryPath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): if not local_testdir.startswith("/"): @@ -15,7 +17,8 @@ def path(self, local_testdir): self.path = UPath(path) self.prepare_file_system() - def test_is_MemoryPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, MemoryPath) From 3f1ff7f49717a41bde7eb7d658b531b61e2c5135 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 22:12:52 +0100 Subject: [PATCH 10/27] tests: cleanup s3 tests --- upath/tests/implementations/test_s3.py | 23 +++++++++++++++-------- 1 file changed, 15 insertions(+), 8 deletions(-) diff --git a/upath/tests/implementations/test_s3.py b/upath/tests/implementations/test_s3.py index 249ed172..d4027fdc 100644 --- a/upath/tests/implementations/test_s3.py +++ b/upath/tests/implementations/test_s3.py @@ -3,12 +3,15 @@ import sys import fsspec -import pytest # noqa: F401 +import pytest from upath import UPath from upath.implementations.cloud import S3Path from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base def silence_botocore_datetime_deprecation(cls): @@ -25,7 +28,7 @@ def silence_botocore_datetime_deprecation(cls): @silence_botocore_datetime_deprecation -class TestUPathS3(BaseTests): +class TestUPathS3(BaseTests, metaclass=OverrideMeta): SUPPORTS_EMPTY_DIRS = False @pytest.fixture(autouse=True) @@ -35,13 +38,11 @@ def path(self, s3_fixture): self.anon = anon self.s3so = s3so - def test_is_S3Path(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, S3Path) - def test_chmod(self): - # todo - pass - + @extends_base def test_rmdir(self): dirname = "rmdir_test" mock_dir = self.path.joinpath(dirname) @@ -51,11 +52,13 @@ def test_rmdir(self): with pytest.raises(NotADirectoryError): self.path.joinpath("file1.txt").rmdir() - def test_relative_to(self): + @extends_base + def test_relative_to_extra(self): assert "file.txt" == str( UPath("s3://test_bucket/file.txt").relative_to(UPath("s3://test_bucket")) ) + @extends_base def test_iterdir_root(self): client_kwargs = self.path.storage_options["client_kwargs"] bucket_path = UPath("s3://other_test_bucket", client_kwargs=client_kwargs) @@ -68,6 +71,7 @@ def test_iterdir_root(self): assert x.name != "" assert x.exists() + @extends_base @pytest.mark.parametrize( "joiner", [["bucket", "path", "file"], ["bucket/path/file"]] ) @@ -76,10 +80,12 @@ def test_no_bucket_joinpath(self, joiner): path = path.joinpath(*joiner) assert str(path) == "s3://bucket/path/file" + @extends_base def test_creating_s3path_with_bucket(self): path = UPath("s3://", bucket="bucket", anon=self.anon, **self.s3so) assert str(path) == "s3://bucket/" + @extends_base def test_iterdir_with_plus_in_name(self, s3_with_plus_chr_name): bucket, anon, s3so = s3_with_plus_chr_name p = UPath( @@ -93,6 +99,7 @@ def test_iterdir_with_plus_in_name(self, s3_with_plus_chr_name): (file,) = files assert file == p.joinpath("file.txt") + @extends_base @pytest.mark.xfail(reason="fsspec/universal_pathlib#144") def test_rglob_with_double_fwd_slash(self, s3_with_double_fwd_slash_files): import boto3 From d9d70ee60ec63c0fc958f7b57217043aa473608a Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 22:17:02 +0100 Subject: [PATCH 11/27] tests: cleanup sftp tests --- upath/tests/implementations/test_sftp.py | 39 ++++++------------------ 1 file changed, 10 insertions(+), 29 deletions(-) diff --git a/upath/tests/implementations/test_sftp.py b/upath/tests/implementations/test_sftp.py index 093d14b4..7ad10c06 100644 --- a/upath/tests/implementations/test_sftp.py +++ b/upath/tests/implementations/test_sftp.py @@ -1,43 +1,24 @@ import pytest from upath import UPath -from upath.tests.cases import BaseTests -from upath.tests.utils import skip_on_windows -from upath.tests.utils import xfail_if_version - -_xfail_old_fsspec = xfail_if_version( - "fsspec", - lt="2022.7.0", - reason="fsspec<2022.7.0 sftp does not support create_parents", -) +from upath.implementations.sftp import SFTPPath + +from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base +from ..utils import skip_on_windows @skip_on_windows -class TestUPathSFTP(BaseTests): +class TestUPathSFTP(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, ssh_fixture): self.path = UPath(ssh_fixture) - @_xfail_old_fsspec - def test_mkdir(self): - super().test_mkdir() - - @_xfail_old_fsspec - def test_mkdir_exists_ok_true(self): - super().test_mkdir_exists_ok_true() - - @_xfail_old_fsspec - def test_mkdir_exists_ok_false(self): - super().test_mkdir_exists_ok_false() - - @_xfail_old_fsspec - def test_mkdir_parents_true_exists_ok_false(self): - super().test_mkdir_parents_true_exists_ok_false() - - @_xfail_old_fsspec - def test_mkdir_parents_true_exists_ok_true(self): - super().test_mkdir_parents_true_exists_ok_true() + @overrides_base + def test_is_correct_class(self): + assert isinstance(self.path, SFTPPath) @pytest.mark.parametrize( From 0c21536f0e917e50ab9f3955def4427e09545706 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 22:22:27 +0100 Subject: [PATCH 12/27] tests: cleanup smb tests --- upath/tests/implementations/test_smb.py | 27 +++++++++++++------------ 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/upath/tests/implementations/test_smb.py b/upath/tests/implementations/test_smb.py index f4046137..5e16cd44 100644 --- a/upath/tests/implementations/test_smb.py +++ b/upath/tests/implementations/test_smb.py @@ -1,19 +1,27 @@ import pytest -from fsspec import __version__ as fsspec_version -from packaging.version import Version from upath import UPath -from upath.tests.cases import BaseTests -from upath.tests.utils import skip_on_windows + +from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import overrides_base +from ..utils import skip_on_windows @skip_on_windows -class TestUPathSMB(BaseTests): +class TestUPathSMB(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, smb_fixture): self.path = UPath(smb_fixture) + @overrides_base + def test_is_correct_class(self): + from upath.implementations.smb import SMBPath + + assert isinstance(self.path, SMBPath) + + @overrides_base @pytest.mark.parametrize( "pattern", ( @@ -24,14 +32,7 @@ def path(self, smb_fixture): reason="SMBFileSystem.info appends '/' to dirs" ), ), - pytest.param( - "**/*.txt", - marks=( - pytest.mark.xfail(reason="requires fsspec>=2023.9.0") - if Version(fsspec_version) < Version("2023.9.0") - else () - ), - ), + "**/*.txt", ), ) def test_glob(self, pathlib_base, pattern): From 61553e3aa3c29d185c8dd1c605c8acf3b02b388e Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 21 Dec 2025 23:00:00 +0100 Subject: [PATCH 13/27] tests: cleanup webdav tests --- upath/tests/implementations/test_webdav.py | 29 ++++++++++------------ 1 file changed, 13 insertions(+), 16 deletions(-) diff --git a/upath/tests/implementations/test_webdav.py b/upath/tests/implementations/test_webdav.py index 9ac9778c..36543edc 100644 --- a/upath/tests/implementations/test_webdav.py +++ b/upath/tests/implementations/test_webdav.py @@ -3,33 +3,30 @@ import pytest from upath import UPath +from upath.implementations.webdav import WebdavPath from ..cases import BaseTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base -class TestUPathWebdav(BaseTests): +class TestUPathWebdav(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True, scope="function") def path(self, webdav_fixture): self.path = UPath(webdav_fixture, auth=("USER", "PASSWORD")) - def test_fsspec_compat(self): - pass + @overrides_base + def test_is_correct_class(self): + assert isinstance(self.path, WebdavPath) - def test_storage_options(self): - # we need to add base_url to storage options for webdav filesystems, - # to be able to serialize the http protocol to string... - storage_options = self.path.storage_options - base_url = storage_options["base_url"] - assert storage_options == self.path.fs.storage_options + @extends_base + def test_storage_options_base_url(self): + # ensure that base_url is correct + base_url = self.path.storage_options["base_url"] assert base_url == self.path.fs.client.base_url - def test_read_with_fsspec(self): - # this test used to fail with fsspec<2022.5.0 because webdav was not - # registered in fsspec. But when UPath(webdav_fixture) is called, to - # run the BaseTests, the upath.implementations.webdav module is - # imported, which registers the webdav implementation in fsspec. - super().test_read_with_fsspec() - + @overrides_base @pytest.mark.parametrize( "target_factory", [ From 85b65ed9528ff98d7993dc096fad37193cf20ec6 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Mon, 22 Dec 2025 15:15:09 +0100 Subject: [PATCH 14/27] tests: cases prevent fs instantiation in readable path tests --- upath/tests/cases.py | 27 +++++++++++++++++---------- 1 file changed, 17 insertions(+), 10 deletions(-) diff --git a/upath/tests/cases.py b/upath/tests/cases.py index 0ac948cb..7fa72e98 100644 --- a/upath/tests/cases.py +++ b/upath/tests/cases.py @@ -8,6 +8,7 @@ import pytest from fsspec import __version__ as fsspec_version from fsspec import filesystem +from fsspec import get_filesystem_class from packaging.version import Version from pathlib_abc import PathParser from pathlib_abc import vfspath @@ -131,7 +132,7 @@ def test_copy_path(self): assert path.drive == copy_path.drive assert path.root == copy_path.root assert path.parts == copy_path.parts - assert path.fs.storage_options == copy_path.fs.storage_options + assert path.storage_options == copy_path.storage_options def test_pickling(self): path = self.path @@ -140,7 +141,7 @@ def test_pickling(self): assert type(path) is type(recovered_path) assert str(path) == str(recovered_path) - assert path.fs.storage_options == recovered_path.fs.storage_options + assert path.storage_options == recovered_path.storage_options def test_pickling_child_path(self): path = self.path / "subfolder" / "subsubfolder" @@ -152,7 +153,6 @@ def test_pickling_child_path(self): assert path.drive == recovered_path.drive assert path.root == recovered_path.root assert path.parts == recovered_path.parts - assert path.fs.storage_options == recovered_path.fs.storage_options assert path.storage_options == recovered_path.storage_options def test_as_uri(self): @@ -164,14 +164,11 @@ def test_as_uri(self): def test_protocol(self): protocol = self.path.protocol - protocols = [p] if isinstance((p := type(self.path.fs).protocol), str) else p + fs_cls = get_filesystem_class(protocol) + protocols = [p] if isinstance((p := fs_cls.protocol), str) else p print(protocol, protocols) assert protocol in protocols - def test_storage_options(self): - storage_options = self.path.storage_options - assert storage_options == self.path.fs.storage_options - def test_hashable(self): assert hash(self.path) @@ -273,6 +270,10 @@ class ReadablePathTests: path: UPath + def test_storage_options_match_fsspec(self): + storage_options = self.path.storage_options + assert storage_options == self.path.fs.storage_options + def test_stat(self): stat_ = self.path.stat() @@ -471,9 +472,12 @@ def test_rglob(self, pathlib_base): assert len(result) == len(expected) def test_walk(self, local_testdir): + def _raise(x): + raise x + # collect walk results from UPath upath_walk = [] - for dirpath, dirnames, filenames in self.path.walk(): + for dirpath, dirnames, filenames in self.path.walk(on_error=_raise): rel_dirpath = dirpath.relative_to(self.path) upath_walk.append((str(rel_dirpath), sorted(dirnames), sorted(filenames))) upath_walk.sort() @@ -488,9 +492,12 @@ def test_walk(self, local_testdir): assert upath_walk == os_walk def test_walk_top_down_false(self): + def _raise(x): + raise x + # test walk with top_down=False returns directories after their contents paths_seen = [] - for dirpath, _, _ in self.path.walk(top_down=False): + for dirpath, _, _ in self.path.walk(top_down=False, on_error=_raise): paths_seen.append(dirpath) # in bottom-up walk, parent directories should come after children From ddf53a54f685eec51170a05766ed0e353cacf935 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Fri, 26 Dec 2025 16:22:53 +0100 Subject: [PATCH 15/27] tests: cases disentanble joinable&readable path tests more --- upath/tests/cases.py | 45 ++++++++++++++++++++++---------------------- 1 file changed, 23 insertions(+), 22 deletions(-) diff --git a/upath/tests/cases.py b/upath/tests/cases.py index 7fa72e98..50b09eb3 100644 --- a/upath/tests/cases.py +++ b/upath/tests/cases.py @@ -270,6 +270,10 @@ class ReadablePathTests: path: UPath + @pytest.fixture(autouse=True) + def path_file(self, path): + self.path_file = self.path.joinpath("file1.txt") + def test_storage_options_match_fsspec(self): storage_options = self.path.storage_options assert storage_options == self.path.fs.storage_options @@ -297,11 +301,11 @@ def test_stat_dir_st_mode(self): assert stat.S_ISDIR(base.st_mode) def test_stat_file_st_mode(self): - file1 = self.path.joinpath("file1.txt").stat() + file1 = self.path_file.stat() assert stat.S_ISREG(file1.st_mode) def test_stat_st_size(self): - file1 = self.path.joinpath("file1.txt").stat() + file1 = self.path_file.stat() assert file1.st_size == 11 @pytest.mark.parametrize( @@ -424,44 +428,41 @@ def test_home(self): self.path.home() def test_open(self): - p = self.path.joinpath("file1.txt") + p = self.path_file with p.open(mode="r") as f: assert f.read() == "hello world" with p.open(mode="rb") as f: assert f.read() == b"hello world" def test_open_buffering(self): - p = self.path.joinpath("file1.txt") + p = self.path_file p.open(buffering=-1) def test_open_block_size(self): - p = self.path.joinpath("file1.txt") + p = self.path_file with p.open(mode="r", block_size=8192) as f: assert f.read() == "hello world" def test_open_errors(self): - p = self.path.joinpath("file1.txt") + p = self.path_file with p.open(mode="r", encoding="ascii", errors="strict") as f: assert f.read() == "hello world" - def test_read_bytes(self, pathlib_base): + def test_read_bytes(self): mock = self.path.joinpath("file2.txt") - pl = pathlib_base.joinpath("file2.txt") - assert mock.read_bytes() == pl.read_bytes() + assert mock.read_bytes() == b"hello world" - def test_read_text(self, local_testdir): + def test_read_text(self): upath = self.path.joinpath("file1.txt") - assert ( - upath.read_text() == Path(local_testdir).joinpath("file1.txt").read_text() - ) + assert upath.read_text() == "hello world" def test_read_text_encoding(self): - upath = self.path.joinpath("file1.txt") + upath = self.path_file content = upath.read_text(encoding="utf-8") assert content == "hello world" def test_read_text_errors(self): - upath = self.path.joinpath("file1.txt") + upath = self.path_file content = upath.read_text(encoding="ascii", errors="strict") assert content == "hello world" @@ -532,7 +533,7 @@ def test_info(self): def test_copy_local(self, tmp_path: Path): target = UPath(tmp_path) / "target-file1.txt" - source = self.path / "file1.txt" + source = self.path_file content = source.read_text() source.copy(target) assert target.exists() @@ -542,16 +543,16 @@ def test_copy_into_local(self, tmp_path: Path): target_dir = UPath(tmp_path) / "target-dir" target_dir.mkdir() - source = self.path / "file1.txt" + source = self.path_file content = source.read_text() source.copy_into(target_dir) - target = target_dir / "file1.txt" + target = target_dir / source.name assert target.exists() assert target.read_text() == content def test_copy_memory(self, clear_fsspec_memory_cache): target = UPath("memory:///target-file1.txt") - source = self.path / "file1.txt" + source = self.path_file content = source.read_text() source.copy(target) assert target.exists() @@ -561,15 +562,15 @@ def test_copy_into_memory(self, clear_fsspec_memory_cache): target_dir = UPath("memory:///target-dir") target_dir.mkdir() - source = self.path / "file1.txt" + source = self.path_file content = source.read_text() source.copy_into(target_dir) - target = target_dir / "file1.txt" + target = target_dir / source.name assert target.exists() assert target.read_text() == content def test_read_with_fsspec(self): - p = self.path.joinpath("file2.txt") + p = self.path_file protocol = p.protocol storage_options = p.storage_options From c454cd6541f354bd8426671b1a49f4a41d5838ce Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Fri, 26 Dec 2025 16:53:34 +0100 Subject: [PATCH 16/27] tests: implement non-writable test cases --- upath/tests/cases.py | 72 +++++++++++++++++++++++++++++++------------- 1 file changed, 51 insertions(+), 21 deletions(-) diff --git a/upath/tests/cases.py b/upath/tests/cases.py index 50b09eb3..848ec2b3 100644 --- a/upath/tests/cases.py +++ b/upath/tests/cases.py @@ -598,7 +598,57 @@ def test_owner(self): # ============================================================================= -class WritablePathTests: +class _CommonWritablePathTests: + + SUPPORTS_EMPTY_DIRS = True + + path: UPath + + def test_chmod(self): + with pytest.raises(NotImplementedError): + self.path_file.chmod(777) + + def test_lchmod(self): + with pytest.raises(UnsupportedOperation): + self.path.lchmod(mode=0o777) + + def test_symlink_to(self): + with pytest.raises(UnsupportedOperation): + self.path_file.symlink_to("target") + with pytest.raises(UnsupportedOperation): + self.path.joinpath("link").symlink_to("target") + + def test_hardlink_to(self): + with pytest.raises(UnsupportedOperation): + self.path_file.symlink_to("target") + with pytest.raises(UnsupportedOperation): + self.path.joinpath("link").hardlink_to("target") + + +class NonWritablePathTests(_CommonWritablePathTests): + + def test_mkdir_raises(self): + with pytest.raises(UnsupportedOperation): + self.path.mkdir() + + def test_touch_raises(self): + with pytest.raises(UnsupportedOperation): + self.path.touch() + + def test_unlink(self): + with pytest.raises(UnsupportedOperation): + self.path.unlink() + + def test_write_bytes(self): + with pytest.raises(UnsupportedOperation): + self.path_file.write_bytes(b"abc") + + def test_write_text(self): + with pytest.raises(UnsupportedOperation): + self.path_file.write_text("abc") + + +class WritablePathTests(_CommonWritablePathTests): """Tests for WritablePath interface. These tests verify operations that write to the filesystem: @@ -608,10 +658,6 @@ class WritablePathTests: - Removing files/directories (unlink, rmdir) """ - SUPPORTS_EMPTY_DIRS = True - - path: UPath - def test_mkdir(self): new_dir = self.path.joinpath("new_dir") new_dir.mkdir() @@ -710,22 +756,6 @@ def test_write_text_errors(self): path.write_text(s, encoding="ascii", errors="strict") assert path.read_text(encoding="ascii") == s - def test_chmod(self): - with pytest.raises(NotImplementedError): - self.path.joinpath("file1.txt").chmod(777) - - def test_lchmod(self): - with pytest.raises(UnsupportedOperation): - self.path.lchmod(mode=0o777) - - def test_symlink_to(self): - with pytest.raises(UnsupportedOperation): - self.path.joinpath("link").symlink_to("target") - - def test_hardlink_to(self): - with pytest.raises(UnsupportedOperation): - self.path.joinpath("link").hardlink_to("target") - class ReadWritePathTests: """Tests requiring both ReadablePath and WritablePath interfaces. From 891a63ca5560e3ac734038311b81844a65f1dd2e Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 16:59:28 +0100 Subject: [PATCH 17/27] tests: cleanup data tests --- upath/tests/implementations/test_data.py | 450 +++++++++-------------- 1 file changed, 173 insertions(+), 277 deletions(-) diff --git a/upath/tests/implementations/test_data.py b/upath/tests/implementations/test_data.py index b869c268..3a6760ff 100644 --- a/upath/tests/implementations/test_data.py +++ b/upath/tests/implementations/test_data.py @@ -1,240 +1,219 @@ import stat -import fsspec import pytest from upath import UnsupportedOperation from upath import UPath from upath.implementations.data import DataPath -from upath.tests.cases import BaseTests -from ..utils import xfail_if_version +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import overrides_base -pytestmark = xfail_if_version( - "fsspec", lt="2023.12.2", reason="fsspec<2023.12.2 does not support data" -) - -class TestUPathDataPath(BaseTests): +class TestUPathDataPath( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): """ Unit-tests for the DataPath implementation of UPath. """ @pytest.fixture(autouse=True) def path(self): - """ - Fixture for the UPath instance to be tested. - """ - path = "" # noqa: E501 + path = "data:text/plain;base64,aGVsbG8gd29ybGQ=" self.path = UPath(path) - def test_is_DataPath(self): - """ - Test that the path is a GitHubPath instance. - """ + @pytest.fixture(autouse=True) + def path_file(self, path): + self.path_file = self.path + + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, DataPath) + @overrides_base def test_with_segments(self): + # DataPath does not support joins, so in all usual cases it'll raise with pytest.raises(UnsupportedOperation): - super().test_with_segments() - - def test_is_relative_to(self): - with pytest.raises(UnsupportedOperation): - super().test_is_relative_to() - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_stat_dir_st_mode(self): - super().test_stat_dir_st_mode() - - def test_stat_file_st_mode(self): - assert self.path.is_file() - assert stat.S_ISREG(self.path.stat().st_mode) - - def test_stat_st_size(self): - assert self.path.stat().st_size == 69 - - def test_exists(self): - # datapath exists is always true... - path = self.path - assert path.exists() - - @pytest.mark.skip(reason="DataPath does support joins or globs") - def test_glob(self, pathlib_base): - with pytest.raises(NotImplementedError): - pathlib_base.glob("*") - - def test_is_dir(self): - assert not self.path.is_dir() - - def test_is_file(self): - assert self.path.is_file() - - def test_iterdir(self): - with pytest.raises(NotADirectoryError): - list(self.path.iterdir()) - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_iterdir2(self): - pass - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_iterdir_trailing_slash(self): - pass - - def test_mkdir(self): - with pytest.raises(FileExistsError): - self.path.mkdir() - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_mkdir_exists_ok_true(self): - pass - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_mkdir_exists_ok_false(self): - pass - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_mkdir_parents_true_exists_ok_true(self): - pass - - @pytest.mark.skip(reason="DataPath does not have directories") - def test_mkdir_parents_true_exists_ok_false(self): - pass - - def test_open(self): - p = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - with p.open(mode="r") as f: - assert f.read() == "hello world" - with p.open(mode="rb") as f: - assert f.read() == b"hello world" - - def test_open_buffering(self): - self.path.open(buffering=-1) + self.path.with_segments("data:text/plain;base64,", "aGVsbG8K") + # but you can instantiate with a single full url + self.path.with_segments("data:text/plain;base64,aGVsbG8K") - def test_open_block_size(self): - p = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - with p.open(mode="r", block_size=8192) as f: - assert f.read() == "hello world" - - def test_open_errors(self): - p = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - with p.open(mode="r", encoding="ascii", errors="strict") as f: - assert f.read() == "hello world" + @overrides_base + def test_parents(self): + # DataPath is always a absolute path with no parents + assert self.path.parents == [] - def test_read_bytes(self, pathlib_base): - assert len(self.path.read_bytes()) == 69 + @overrides_base + def test_with_name(self): + # DataPath does not support name changes + with pytest.raises(UnsupportedOperation): + self.path.with_name("newname") - def test_read_text(self, local_testdir): - assert UPath("data:base64,SGVsbG8gV29ybGQ=").read_text() == "Hello World" + @overrides_base + def test_with_suffix(self): + # DataPath does not support suffix changes + with pytest.raises(UnsupportedOperation): + self.path.with_suffix(".new") - def test_parents(self): - with pytest.raises(NotImplementedError): - self.path.parents[0] + @overrides_base + def test_with_stem(self): + # DataPath does not support stem changes + with pytest.raises(UnsupportedOperation): + self.path.with_stem("newname") - def test_rename(self): - with pytest.raises(NotImplementedError): - self.path.rename("newname") + @overrides_base + def test_suffix(self): + # DataPath does not have suffixes + assert self.path.suffix == "" - @pytest.mark.skip("DataPath does not support rename") - def test_rename_with_target_relative(self): - pass + @overrides_base + def test_suffixes(self): + # DataPath does not have suffixes + assert self.path.suffixes == [] - @pytest.mark.skip("DataPath does not support rename") - def test_rename_with_target_absolute(self): - pass + @overrides_base + def test_repr_after_with_name(self): + with pytest.raises(UnsupportedOperation): + repr(self.path.with_name("data:,ABC")) - def test_rglob(self, pathlib_base): - with pytest.raises(NotImplementedError): - list(self.path.rglob("*")) + @overrides_base + def test_repr_after_with_suffix(self): + with pytest.raises(UnsupportedOperation): + repr(self.path.with_suffix("")) - def test_touch(self): - self.path.touch() + @overrides_base + def test_child_path(self): + # DataPath does not support joins, so child paths are unsupported + with pytest.raises(UnsupportedOperation): + super().test_child_path() - def test_touch_exists_ok_false(self): - with pytest.raises(FileExistsError): - self.path.touch(exist_ok=False) + @overrides_base + def test_pickling_child_path(self): + # DataPath does not support joins, so child paths are unsupported + with pytest.raises(UnsupportedOperation): + super().test_pickling_child_path() - def test_touch_exists_ok_true(self): - self.path.touch() + @overrides_base + def test_relative_to(self): + # DataPath only relative_to with itself + with pytest.raises(ValueError): + self.path.relative_to("data:,ABC") + self.path.relative_to(self.path) - def test_touch_unlink(self): - self.path.touch() - with pytest.raises(NotImplementedError): - self.path.unlink() + @overrides_base + def test_is_relative_to(self): + # DataPath only relative_to with itself + assert not self.path.is_relative_to("data:,ABC") + assert self.path.is_relative_to(self.path) - def test_write_bytes(self, pathlib_base): - with pytest.raises(NotImplementedError): - self.path.write_bytes(b"test") + @overrides_base + def test_full_match(self): + assert self.path.full_match("*") + assert not self.path.full_match("xxx") - def test_write_text(self, pathlib_base): - with pytest.raises(NotImplementedError): - self.path.write_text("test") + @overrides_base + def test_trailing_slash_joinpath_is_identical(self): + # DataPath has no slashes, and is not joinable + with pytest.raises(UnsupportedOperation): + super().test_trailing_slash_joinpath_is_identical() - def test_read_with_fsspec(self): - pth = self.path - fs = fsspec.filesystem(pth.protocol, **pth.storage_options) - assert fs.cat_file(pth.path) == pth.read_bytes() + @overrides_base + def test_trailing_slash_is_stripped(self): + # DataPath has no slashes, and is not joinable + with pytest.raises(UnsupportedOperation): + super().test_trailing_slash_is_stripped() - @pytest.mark.skip(reason="DataPath does not support joins") - def test_pickling_child_path(self): - pass + @overrides_base + def test_private_url_attr_in_sync(self): + # DataPath does not support joins, so we check on self.path + assert self.path._url - @pytest.mark.skip(reason="DataPath does not support joins") - def test_child_path(self): - pass + @overrides_base + def test_stat_dir_st_mode(self): + # DataPath does not have directories + assert not stat.S_ISDIR(self.path.stat().st_mode) - def test_with_name(self): - with pytest.raises(NotImplementedError): - self.path.with_name("newname") + @overrides_base + def test_exists(self): + # A valid DataPath always exists + assert self.path.exists() - def test_with_suffix(self): - with pytest.raises(NotImplementedError): - self.path.with_suffix(".new") + @overrides_base + def test_glob(self): + # DataPath does not have dirs, joins or globs + assert list(self.path.glob("*")) == [] - def test_suffix(self): - assert self.path.suffix == "" + @overrides_base + def test_rglob(self): + # DataPath does not have dirs, joins or globs + assert list(self.path.rglob("*")) == [] - def test_suffixes(self): - assert self.path.suffixes == [] + @overrides_base + def test_is_dir(self): + # DataPath does not have directories + assert not self.path.is_dir() - def test_with_stem(self): - with pytest.raises(NotImplementedError): - self.path.with_stem("newname") + @overrides_base + def test_is_file(self): + # DataPath is always a file + assert self.path.is_file() - @pytest.mark.skip(reason="DataPath does not support joins") - def test_repr_after_with_suffix(self): - pass + @overrides_base + def test_iterdir(self): + # DataPath does not have directories + with pytest.raises(NotADirectoryError): + self.path.iterdir() - @pytest.mark.skip(reason="DataPath does not support joins") - def test_repr_after_with_name(self): - pass + @overrides_base + def test_iterdir2(self): + # DataPath does not have directories, or joins + with pytest.raises(NotADirectoryError): + self.path_file.iterdir() - @pytest.mark.skip(reason="DataPath does not support directories") - def test_rmdir_no_dir(self): - pass + @overrides_base + def test_iterdir_trailing_slash(self): + # DataPath does not have directories, or joins + with pytest.raises(UnsupportedOperation): + super().test_iterdir_trailing_slash() - @pytest.mark.skip(reason="DataPath does not support directories") - def test_iterdir_no_dir(self): - pass + @overrides_base + def test_read_bytes(self): + assert self.path.read_bytes() == b"hello world" - @pytest.mark.skip(reason="DataPath does not support joins") - def test_private_url_attr_in_sync(self): - pass + @overrides_base + def test_read_text(self): + assert self.path.read_text() == "hello world" - @pytest.mark.skip(reason="DataPath does not support joins") - def test_fsspec_compat(self): - pass + @overrides_base + def test_walk(self): + # DataPath does not have directories + assert list(self.path.walk()) == [] - def test_rmdir_not_empty(self): - with pytest.raises(NotADirectoryError): - self.path.rmdir() + @overrides_base + def test_walk_top_down_false(self): + # DataPath does not have directories + assert list(self.path.walk(top_down=False)) == [] + @overrides_base def test_samefile(self): - f1 = self.path + # DataPath doesn't have joins, so only identical paths are samefile + f1 = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") + f2 = UPath("data:text/plain;base64,SGVsbG8gd29ybGQ=") + assert f1.samefile(f2) is False + assert f1.samefile(f2.path) is False assert f1.samefile(f1) is True + assert f1.samefile(f1.path) is True + @overrides_base def test_info(self): + # DataPath info checks p0 = self.path assert p0.info.exists() is True @@ -242,102 +221,19 @@ def test_info(self): assert p0.info.is_dir() is False assert p0.info.is_symlink() is False - def test_copy_local(self, tmp_path): - target = UPath(tmp_path) / "target-file1.txt" - - source = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - content = source.read_text() - source.copy(target) - assert target.exists() - assert target.read_text() == content - - def test_copy_into_local(self, tmp_path): - target_dir = UPath(tmp_path) / "target-dir" - target_dir.mkdir() - - source = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - content = source.read_text() - source.copy_into(target_dir) - target = target_dir / source.name - assert target.exists() - assert target.read_text() == content - - def test_copy_memory(self, clear_fsspec_memory_cache): - target = UPath("memory:///target-file1.txt") - - source = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - content = source.read_text() - source.copy(target) - assert target.exists() - assert target.read_text() == content - - def test_copy_into_memory(self, clear_fsspec_memory_cache): - target_dir = UPath("memory:///target-dir") - target_dir.mkdir() - - source = UPath("data:text/plain;base64,aGVsbG8gd29ybGQ=") - content = source.read_text() - source.copy_into(target_dir) - target = target_dir / source.name - assert target.exists() - assert target.read_text() == content - - @pytest.mark.skip(reason="DataPath does not support unlink") - def test_move_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="DataPath does not support unlink") - def test_move_into_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="DataPath does not support unlink") - def test_move_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="DataPath does not support unlink") - def test_move_into_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="DataPath does not support relative_to") - def test_relative_to(self): - pass - - @pytest.mark.skip(reason="DataPath does not support joins") - def test_trailing_slash_joinpath_is_identical(self): - pass - - @pytest.mark.skip(reason="DataPath does not support joins") - def test_trailing_slash_is_stripped(self): - pass - - @pytest.mark.skip(reason="DataPath does not support joins") - def test_parents_are_absolute(self): - pass - - @pytest.mark.skip(reason="DataPath does not support write_text") - def test_write_text_encoding(self): - pass - - @pytest.mark.skip(reason="DataPath does not support write_text") - def test_write_text_errors(self): - pass - - @pytest.mark.skip(reason="base test incompatible with DataPath") - def test_read_text_encoding(self): - pass - - @pytest.mark.skip(reason="base test incompatible with DataPath") - def test_read_text_errors(self): - pass + @overrides_base + def test_mkdir_raises(self): + # DataPaths always exist and are files + with pytest.raises(FileExistsError): + self.path_file.mkdir() - @pytest.mark.skip(reason="DataPath does not support walk") - def test_walk(self, local_testdir): - pass + @overrides_base + def test_touch_raises(self): + # DataPaths always exist, so touch is a noop + self.path_file.touch() - @pytest.mark.skip(reason="DataPath does not support walk") - def test_walk_top_down_false(self): - pass - - @pytest.mark.skip(reason="DataPath does not support full_match") - def test_full_match(self): - pass + @overrides_base + def test_unlink(self): + # DataPaths can't be deleted + with pytest.raises(UnsupportedOperation): + self.path_file.unlink() From d84aef7eb499898dc038d8ac1fff73bd19cff7c7 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Fri, 26 Dec 2025 20:03:01 +0100 Subject: [PATCH 18/27] tests: cleanup http tests --- upath/tests/implementations/test_http.py | 141 ++++------------------- 1 file changed, 22 insertions(+), 119 deletions(-) diff --git a/upath/tests/implementations/test_http.py b/upath/tests/implementations/test_http.py index f3b22d43..66d76e62 100644 --- a/upath/tests/implementations/test_http.py +++ b/upath/tests/implementations/test_http.py @@ -1,15 +1,17 @@ import pytest # noqa: F401 -from fsspec import __version__ as fsspec_version from fsspec import get_filesystem_class -from packaging.version import Version from upath import UPath from upath.implementations.http import HTTPPath -from ..cases import BaseTests +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base from ..utils import skip_on_windows from ..utils import xfail_if_no_ssl_connection -from ..utils import xfail_if_version try: get_filesystem_class("http") @@ -36,113 +38,42 @@ def test_httppath(internet_connection): @xfail_if_no_ssl_connection -def test_httpspath(): +def test_httpspath(internet_connection): path = UPath("https://example.com") assert isinstance(path, HTTPPath) assert path.exists() @skip_on_windows -class TestUPathHttp(BaseTests): +class TestUPathHttp( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): @pytest.fixture(autouse=True, scope="function") def path(self, http_fixture): self.path = UPath(http_fixture) + @overrides_base + def test_is_correct_class(self): + assert isinstance(self.path, HTTPPath) + + @extends_base def test_work_at_root(self): assert "folder" in (f.name for f in self.path.parent.iterdir()) - @pytest.mark.skip - def test_mkdir(self): - pass - - @pytest.mark.parametrize( - "pattern", - ( - "*.txt", - pytest.param( - "*", - marks=xfail_if_version( - "fsspec", - gt="2023.10.0", - lt="2024.5.0", - reason="requires fsspec>=2024.5.0", - ), - ), - pytest.param( - "**/*.txt", - marks=( - pytest.mark.xfail(reason="requires fsspec>=2023.9.0") - if Version(fsspec_version) < Version("2023.9.0") - else () - ), - ), - ), - ) - def test_glob(self, pathlib_base, pattern): - super().test_glob(pathlib_base, pattern) - - @pytest.mark.skip - def test_mkdir_exists_ok_false(self): - pass - - @pytest.mark.skip - def test_mkdir_exists_ok_true(self): - pass - - @pytest.mark.skip - def test_mkdir_parents_true_exists_ok_true(self): - pass - - @pytest.mark.skip - def test_mkdir_parents_true_exists_ok_false(self): - pass - - @pytest.mark.skip - def test_makedirs_exist_ok_true(self): - pass - - @pytest.mark.skip - def test_makedirs_exist_ok_false(self): - pass - - @pytest.mark.skip - def test_touch(self): - pass - - @pytest.mark.skip - def test_touch_unlink(self): - pass - - @pytest.mark.skip - def test_write_bytes(self, pathlib_base): - pass - - @pytest.mark.skip - def test_write_text(self, pathlib_base): - pass - - def test_fsspec_compat(self): - pass - + @extends_base def test_resolve(self): # Also tests following redirects, because the test server issues a # 301 redirect for `http://127.0.0.1:8080/folder` to # `http://127.0.0.1:8080/folder/` assert str(self.path.resolve()).endswith("/") - def test_rename(self): - with pytest.raises(NotImplementedError): - return super().test_rename() - - def test_rename2(self): - with pytest.raises(NotImplementedError): - return super().test_rename() - - @xfail_if_version("fsspec", lt="2024.2.0", reason="requires fsspec>=2024.2.0") - def test_stat_dir_st_mode(self): - super().test_stat_dir_st_mode() - + @overrides_base def test_info(self): + # HTTPPath folders are files too + p0 = self.path.joinpath("file1.txt") p1 = self.path.joinpath("folder1") @@ -157,34 +88,6 @@ def test_info(self): assert p1.info.is_dir() is True assert p1.info.is_symlink() is False - @pytest.mark.skip(reason="HttpPath does not support unlink") - def test_move_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="HttpPath does not support unlink") - def test_move_into_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="HttpPath does not support unlink") - def test_move_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="HttpPath does not support unlink") - def test_move_into_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="Only testing read on HttpPath") - def test_rename_with_target_absolute(self, target_factory): - return super().test_rename_with_target_absolute(target_factory) - - @pytest.mark.skip(reason="Only testing read on HttpPath") - def test_write_text_encoding(self): - return super().test_write_text_encoding() - - @pytest.mark.skip(reason="Only testing read on HttpPath") - def test_write_text_errors(self): - return super().test_write_text_errors() - @pytest.mark.parametrize( "args,parts", From 4ec197caadda203dfe041eb0b66e390f93b4e5b9 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 11:10:15 +0100 Subject: [PATCH 19/27] tests: cleanup tar tests --- upath/tests/implementations/test_tar.py | 84 ++++--------------------- 1 file changed, 13 insertions(+), 71 deletions(-) diff --git a/upath/tests/implementations/test_tar.py b/upath/tests/implementations/test_tar.py index c528a7e2..6430b252 100644 --- a/upath/tests/implementations/test_tar.py +++ b/upath/tests/implementations/test_tar.py @@ -5,7 +5,11 @@ from upath import UPath from upath.implementations.tar import TarPath -from ..cases import BaseTests +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import overrides_base @pytest.fixture(scope="function") @@ -17,84 +21,22 @@ def tarred_testdir_file(local_testdir, tmp_path_factory): return str(tar_path) -class TestTarPath(BaseTests): +class TestTarPath( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): @pytest.fixture(autouse=True) def path(self, tarred_testdir_file): self.path = UPath("tar://", fo=tarred_testdir_file) # self.prepare_file_system() done outside of UPath - def test_is_TarPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, TarPath) - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_mkdir(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_mkdir_exists_ok_false(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_mkdir_parents_true_exists_ok_false(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_rename(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_rename2(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_touch(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_touch_unlink(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_write_bytes(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_write_text(self): - pass - - @pytest.mark.skip(reason="Tar filesystem is read-only") - def test_fsspec_compat(self): - pass - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_move_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_move_into_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_move_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_move_into_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_rename_with_target_absolute(self, target_factory): - return super().test_rename_with_target_str_absolute(target_factory) - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_write_text_encoding(self): - return super().test_write_text_encoding() - - @pytest.mark.skip(reason="Only testing read on TarPath") - def test_write_text_errors(self): - return super().test_write_text_errors() - @pytest.fixture(scope="function") def tarred_testdir_file_in_memory(tarred_testdir_file, clear_fsspec_memory_cache): From f712934ec78af97ce990a60f33c7b0666ff9ddad Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 13:35:09 +0100 Subject: [PATCH 20/27] tests: cleanup zip tests --- upath/tests/implementations/test_zip.py | 133 ++++-------------------- 1 file changed, 23 insertions(+), 110 deletions(-) diff --git a/upath/tests/implementations/test_zip.py b/upath/tests/implementations/test_zip.py index 28779e32..61a60be5 100644 --- a/upath/tests/implementations/test_zip.py +++ b/upath/tests/implementations/test_zip.py @@ -3,10 +3,16 @@ import pytest +from upath import UnsupportedOperation from upath import UPath from upath.implementations.zip import ZipPath -from ..cases import BaseTests +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import extends_base +from ..utils import overrides_base @pytest.fixture(scope="function") @@ -33,8 +39,12 @@ def empty_zipped_testdir_file(tmp_path): return str(zip_path) -class TestZipPath(BaseTests): - +class TestZipPath( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): @pytest.fixture(autouse=True) def path(self, zipped_testdir_file, request): try: @@ -47,115 +57,18 @@ def path(self, zipped_testdir_file, request): finally: self.path.fs.clear_instance_cache() - def test_is_ZipPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, ZipPath) - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_mkdir(self): - super().test_mkdir() - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_mkdir_exists_ok_true(self): - super().test_mkdir_exists_ok_true() - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_mkdir_exists_ok_false(self): - super().test_mkdir_exists_ok_false() - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_mkdir_parents_true_exists_ok_true(self): - super().test_mkdir_parents_true_exists_ok_true() - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_mkdir_parents_true_exists_ok_false(self): - super().test_mkdir_parents_true_exists_ok_false() - - def test_rename(self): - with pytest.raises(NotImplementedError): - super().test_rename() # delete is not implemented in fsspec - - def test_move_local(self, tmp_path): - with pytest.raises(NotImplementedError): - super().test_move_local(tmp_path) # delete is not implemented in fsspec - - def test_move_into_local(self, tmp_path): - with pytest.raises(NotImplementedError): - super().test_move_into_local( - tmp_path - ) # delete is not implemented in fsspec - - def test_move_memory(self, clear_fsspec_memory_cache): - with pytest.raises(NotImplementedError): - super().test_move_memory(clear_fsspec_memory_cache) - - def test_move_into_memory(self, clear_fsspec_memory_cache): - with pytest.raises(NotImplementedError): - super().test_move_into_memory(clear_fsspec_memory_cache) - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_touch(self): - super().test_touch() - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_touch_unlink(self): - with pytest.raises(NotImplementedError): - super().test_touch_unlink() # delete is not implemented in fsspec - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_write_bytes(self): - fn = "test_write_bytes.txt" - s = b"hello_world" - path = self.path.joinpath(fn) - path.write_bytes(s) - so = {**path.storage_options, "mode": "r"} - urlpath = str(path) - path.fs.close() - assert UPath(urlpath, **so).read_bytes() == s - - @pytest.mark.parametrize( - "path", [("w",)], ids=["zipfile_mode_write"], indirect=True - ) - def test_write_text(self): - fn = "test_write_text.txt" - s = "hello_world" - path = self.path.joinpath(fn) - path.write_text(s) - so = {**path.storage_options, "mode": "r"} - urlpath = str(path) - path.fs.close() - assert UPath(urlpath, **so).read_text() == s - - @pytest.mark.skip(reason="fsspec zipfile filesystem is either read xor write mode") - def test_fsspec_compat(self): - pass - - @pytest.mark.skip(reason="fsspec zipfile filesystem is either read xor write mode") - def test_rename_with_target_absolute(self, target_factory): - return super().test_rename_with_target_absolute(target_factory) - - @pytest.mark.skip(reason="fsspec zipfile filesystem is either read xor write mode") - def test_write_text_encoding(self): - return super().test_write_text_encoding() - - @pytest.mark.skip(reason="fsspec zipfile filesystem is either read xor write mode") - def test_write_text_errors(self): - return super().test_write_text_errors() + @extends_base + def test_write_mode_is_disabled(self, tmp_path): + with pytest.raises(UnsupportedOperation): + UPath("zip://", fo=tmp_path.joinpath("myzip.zip"), mode="a") + with pytest.raises(UnsupportedOperation): + UPath("zip://", fo=tmp_path.joinpath("myzip.zip"), mode="x") + with pytest.raises(UnsupportedOperation): + UPath("zip://", fo=tmp_path.joinpath("myzip.zip"), mode="w") @pytest.fixture(scope="function") From c8be415bfd8e2dabcb74bd9e2e6f8b78534f8c19 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:09:44 +0100 Subject: [PATCH 21/27] tests: split iterdir test cases --- upath/tests/cases.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/upath/tests/cases.py b/upath/tests/cases.py index 848ec2b3..24485e69 100644 --- a/upath/tests/cases.py +++ b/upath/tests/cases.py @@ -394,6 +394,8 @@ def test_iterdir(self, local_testdir): assert len(up_iter) == len(pl_iter) assert {p.name for p in pl_iter} == {u.name for u in up_iter} + + def test_iterdir_parent_iteration(self): assert next(self.path.parent.iterdir()).exists() def test_iterdir2(self, local_testdir): @@ -407,7 +409,6 @@ def test_iterdir2(self, local_testdir): assert len(up_iter) == len(pl_iter) assert {p.name for p in pl_iter} == {u.name for u in up_iter} - assert next(self.path.parent.iterdir()).exists() def test_iterdir_trailing_slash(self): files_noslash = list(self.path.joinpath("folder1").iterdir()) From 52ca3feab975ead1fc17989826aa6dbf4e1cd0ea Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:10:44 +0100 Subject: [PATCH 22/27] tests: cleanup huggingface tests --- upath/tests/conftest.py | 15 +++- upath/tests/implementations/test_hf.py | 97 ++++++-------------------- 2 files changed, 32 insertions(+), 80 deletions(-) diff --git a/upath/tests/conftest.py b/upath/tests/conftest.py index 2f5e7b1c..c9333c6f 100644 --- a/upath/tests/conftest.py +++ b/upath/tests/conftest.py @@ -585,12 +585,16 @@ def mock_hf_api(pathlib_base, monkeypatch, hf_test_repo): # noqa: C901 hf_file_system = pytest.importorskip( "huggingface_hub.hf_file_system", reason="hf tests require huggingface_hub" ) + httpx = pytest.importorskip("httpx") class MockedHfApi(huggingface_hub.HfApi): def repo_info(self, repo_id, *args, repo_type=None, **kwargs): if repo_id != hf_test_repo: - raise huggingface_hub.errors.RepositoryNotFoundError(repo_id) + raise huggingface_hub.errors.RepositoryNotFoundError( + repo_id, + response=httpx.Response(404, request=...), + ) elif repo_type is None or repo_type == "model": return huggingface_hub.hf_api.ModelInfo(id=repo_id) elif repo_type == "dataset": @@ -602,7 +606,10 @@ def repo_info(self, repo_id, *args, repo_type=None, **kwargs): def get_paths_info(self, repo_id, paths, *args, **kwargs): if repo_id != hf_test_repo: - raise huggingface_hub.errors.RepositoryNotFoundError(repo_id) + raise huggingface_hub.errors.RepositoryNotFoundError( + repo_id, + response=httpx.Response(404, request=...), + ) paths_info = [] for path in paths: if path: @@ -628,7 +635,9 @@ def list_repo_tree( self, repo_id, path_in_repo, *args, recursive=False, **kwargs ): if repo_id != hf_test_repo: - raise huggingface_hub.errors.RepositoryNotFoundError(repo_id) + raise huggingface_hub.errors.RepositoryNotFoundError( + repo_id, response=httpx.Response(404, request=...) + ) pathlib_dir = pathlib_base / path_in_repo if path_in_repo else pathlib_base for path in pathlib_dir.rglob("*") if recursive else pathlib_dir.glob("*"): if path.is_file(): diff --git a/upath/tests/implementations/test_hf.py b/upath/tests/implementations/test_hf.py index d5727d89..4b30dbfc 100644 --- a/upath/tests/implementations/test_hf.py +++ b/upath/tests/implementations/test_hf.py @@ -1,10 +1,15 @@ import pytest from fsspec import get_filesystem_class +from upath import UnsupportedOperation from upath import UPath from upath.implementations.cloud import HfPath -from ..cases import BaseTests +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import overrides_base try: get_filesystem_class("hf") @@ -31,84 +36,22 @@ def test_hfpath(): raise -class TestUPathHf(BaseTests): +class TestUPathHf( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): @pytest.fixture(autouse=True, scope="function") def path(self, hf_fixture_with_readonly_mocked_hf_api): self.path = UPath(hf_fixture_with_readonly_mocked_hf_api) - @pytest.mark.skip - def test_mkdir(self): - pass + @overrides_base + def test_is_correct_class(self): + assert isinstance(self.path, HfPath) - @pytest.mark.skip - def test_mkdir_exists_ok_false(self): - pass - - @pytest.mark.skip - def test_mkdir_exists_ok_true(self): - pass - - @pytest.mark.skip - def test_mkdir_parents_true_exists_ok_true(self): - pass - - @pytest.mark.skip - def test_mkdir_parents_true_exists_ok_false(self): - pass - - @pytest.mark.skip - def test_makedirs_exist_ok_true(self): - pass - - @pytest.mark.skip - def test_makedirs_exist_ok_false(self): - pass - - @pytest.mark.skip - def test_touch(self): - pass - - @pytest.mark.skip - def test_touch_unlink(self): - pass - - @pytest.mark.skip - def test_write_bytes(self, pathlib_base): - pass - - @pytest.mark.skip - def test_write_text(self, pathlib_base): - pass - - def test_fsspec_compat(self): - pass - - def test_rename(self): - pass - - def test_rename2(self): - pass - - def test_move_local(self, tmp_path): - pass - - def test_move_into_local(self, tmp_path): - pass - - def test_move_memory(self, clear_fsspec_memory_cache): - pass - - def test_move_into_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="HfPath does not support listing repositories") - def test_iterdir(self, local_testdir): - pass - - @pytest.mark.skip(reason="HfPath does not support listing repositories") - def test_iterdir2(self, local_testdir): - pass - - @pytest.mark.skip(reason="HfPath does not currently test write") - def test_rename_with_target_absolute(self, target_factory): - return super().test_rename_with_target_absolute(target_factory) + @overrides_base + def test_iterdir_parent_iteration(self): + # HfPath does not support listing all available Repositories + with pytest.raises(UnsupportedOperation): + super().test_iterdir_parent_iteration() From 120598c0236225f7409bb62a7d1160acee458290 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:19:56 +0100 Subject: [PATCH 23/27] tests: cleanup upath Mock tests --- upath/tests/test_core.py | 27 ++++++++++----------------- 1 file changed, 10 insertions(+), 17 deletions(-) diff --git a/upath/tests/test_core.py b/upath/tests/test_core.py index ef8138de..822cc8c7 100644 --- a/upath/tests/test_core.py +++ b/upath/tests/test_core.py @@ -15,7 +15,9 @@ from upath.types import WritablePath from .cases import BaseTests +from .utils import OverrideMeta from .utils import only_on_windows +from .utils import overrides_base from .utils import skip_on_windows @@ -43,7 +45,7 @@ def test_UPath_file_protocol_no_warning(): assert len(w) == 0 -class TestUpath(BaseTests): +class TestUpath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): with warnings.catch_warnings(): @@ -55,29 +57,20 @@ def path(self, local_testdir): root = "/" if sys.platform.startswith("win") else "" self.path = UPath(f"mock:{root}{local_testdir}") - def test_fsspec_compat(self): - pass - - def test_cwd(self): - with pytest.raises( - NotImplementedError, - match=r".+Path[.]cwd\(\) is unsupported", - ): - type(self.path).cwd() + @overrides_base + def test_is_correct_class(self): + # testing dynamically created UPath classes + from upath.implementations._experimental import _MockPath - def test_home(self): - with pytest.raises( - NotImplementedError, - match=r".+Path[.]home\(\) is unsupported", - ): - type(self.path).home() + assert isinstance(self.path, _MockPath) + @overrides_base @pytest.mark.skipif( sys.platform.startswith("win"), reason="mock fs is not well defined on windows", ) def test_parents_are_absolute(self): - return super().test_parents_are_absolute() + super().test_parents_are_absolute() def test_multiple_backend_paths(local_testdir): From ea1f8d8c70b54731815f6fd5758f8fe2e8b08152 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:28:28 +0100 Subject: [PATCH 24/27] tests: cleanup extensions tests --- upath/tests/test_extensions.py | 48 +++++++++++++++++++++++++++------- 1 file changed, 38 insertions(+), 10 deletions(-) diff --git a/upath/tests/test_extensions.py b/upath/tests/test_extensions.py index fdd802a5..7a3bc8c8 100644 --- a/upath/tests/test_extensions.py +++ b/upath/tests/test_extensions.py @@ -11,10 +11,14 @@ from upath.implementations.local import PosixUPath from upath.implementations.local import WindowsUPath from upath.implementations.memory import MemoryPath -from upath.tests.cases import BaseTests +from .cases import BaseTests +from .utils import OverrideMeta +from .utils import extends_base +from .utils import overrides_base -class TestProxyMemoryPath(BaseTests): + +class TestProxyMemoryPath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): if not local_testdir.startswith("/"): @@ -22,49 +26,60 @@ def path(self, local_testdir): self.path = ProxyUPath(f"memory:{local_testdir}") self.prepare_file_system() - def test_is_ProxyUPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, ProxyUPath) - def test_is_not_MemoryPath(self): + @extends_base + def test_is_not_wrapped_class(self): assert not isinstance(self.path, MemoryPath) -class TestProxyFilePath(BaseTests): +class TestProxyFilePath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): self.path = ProxyUPath(f"file://{local_testdir}") self.prepare_file_system() - def test_is_ProxyUPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, ProxyUPath) - def test_is_not_FilePath(self): + @extends_base + def test_is_not_wrapped_class(self): assert not isinstance(self.path, FilePath) + @overrides_base def test_chmod(self): self.path.joinpath("file1.txt").chmod(777) + @overrides_base def test_cwd(self): + # ProxyUPath.cwd() works differently on the instance self.path.cwd() with pytest.raises(UnsupportedOperation): type(self.path).cwd() -class TestProxyPathlibPath(BaseTests): +class TestProxyPathlibPath(BaseTests, metaclass=OverrideMeta): @pytest.fixture(autouse=True) def path(self, local_testdir): self.path = ProxyUPath(f"{local_testdir}") self.prepare_file_system() - def test_is_ProxyUPath(self): + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, ProxyUPath) - def test_is_not_PosixUPath_WindowsUPath(self): + @extends_base + def test_is_not_wrapped_class(self): assert not isinstance(self.path, (PosixUPath, WindowsUPath)) + @overrides_base def test_chmod(self): self.path.joinpath("file1.txt").chmod(777) + @overrides_base @pytest.mark.skipif( sys.version_info < (3, 12), reason="storage options only handled in 3.12+" ) @@ -73,6 +88,7 @@ def test_eq(self): if sys.version_info < (3, 12): + @overrides_base def test_storage_options_dont_affect_hash(self): # On Python < 3.12, storage_options trigger warnings for LocalPath with pytest.warns( @@ -81,14 +97,17 @@ def test_storage_options_dont_affect_hash(self): ): super().test_storage_options_dont_affect_hash() + @overrides_base def test_group(self): pytest.importorskip("grp") self.path.group() + @overrides_base def test_owner(self): pytest.importorskip("pwd") self.path.owner() + @overrides_base def test_readlink(self): try: os.readlink @@ -97,14 +116,17 @@ def test_readlink(self): with pytest.raises((OSError, UnsupportedOperation)): self.path.readlink() + @overrides_base def test_protocol(self): assert self.path.protocol == "" + @overrides_base def test_as_uri(self): assert self.path.as_uri().startswith("file://") if sys.version_info < (3, 10): + @overrides_base def test_lstat(self): # On Python < 3.10, stat(follow_symlinks=False) triggers warnings with pytest.warns( @@ -116,21 +138,25 @@ def test_lstat(self): else: + @overrides_base def test_lstat(self): st = self.path.lstat() assert st is not None + @overrides_base def test_relative_to(self): base = self.path child = self.path / "folder1" / "file1.txt" relative = child.relative_to(base) assert str(relative) == f"folder1{os.sep}file1.txt" + @overrides_base def test_cwd(self): self.path.cwd() with pytest.raises(UnsupportedOperation): type(self.path).cwd() + @overrides_base def test_lchmod(self): # setup a = self.path.joinpath("a") @@ -146,9 +172,11 @@ def test_lchmod(self): with cm: b.lchmod(mode=0o777) + @overrides_base def test_symlink_to(self): self.path.joinpath("link").symlink_to(self.path) + @overrides_base def test_hardlink_to(self): try: self.path.joinpath("link").hardlink_to(self.path) From 2c3236e79cce4fddfc074e6442bcc77ea22c98cf Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:35:46 +0100 Subject: [PATCH 25/27] tests: fix azure test --- upath/tests/implementations/test_azure.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/upath/tests/implementations/test_azure.py b/upath/tests/implementations/test_azure.py index f2af0aff..eb82e930 100644 --- a/upath/tests/implementations/test_azure.py +++ b/upath/tests/implementations/test_azure.py @@ -27,7 +27,7 @@ def path(self, azurite_credentials, azure_fixture): @overrides_base def test_is_correct_class(self): - return isinstance(self.path, AzurePath) + assert isinstance(self.path, AzurePath) @overrides_base def test_protocol(self): From cb1ce0d30837dfaf90231bdd58b1019d1c63fc77 Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:36:06 +0100 Subject: [PATCH 26/27] tests: fix datapath test --- upath/tests/implementations/test_data.py | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/upath/tests/implementations/test_data.py b/upath/tests/implementations/test_data.py index 3a6760ff..b539e30e 100644 --- a/upath/tests/implementations/test_data.py +++ b/upath/tests/implementations/test_data.py @@ -170,6 +170,11 @@ def test_iterdir(self): with pytest.raises(NotADirectoryError): self.path.iterdir() + @overrides_base + def test_iterdir_parent_iteration(self): + with pytest.raises(NotADirectoryError): + super().test_iterdir_parent_iteration() + @overrides_base def test_iterdir2(self): # DataPath does not have directories, or joins From 11997fc366d38e8d4d7f2f739a1b2acda45c8b3e Mon Sep 17 00:00:00 2001 From: Andreas Poehlmann Date: Sun, 28 Dec 2025 15:38:18 +0100 Subject: [PATCH 27/27] tests: cleanup github tests --- upath/tests/implementations/test_github.py | 90 ++++------------------ 1 file changed, 15 insertions(+), 75 deletions(-) diff --git a/upath/tests/implementations/test_github.py b/upath/tests/implementations/test_github.py index d132702c..c0120232 100644 --- a/upath/tests/implementations/test_github.py +++ b/upath/tests/implementations/test_github.py @@ -7,10 +7,15 @@ from upath import UPath from upath.implementations.github import GitHubPath -from upath.tests.cases import BaseTests + +from ..cases import JoinablePathTests +from ..cases import NonWritablePathTests +from ..cases import ReadablePathTests +from ..utils import OverrideMeta +from ..utils import overrides_base pytestmark = pytest.mark.skipif( - os.environ.get("CI") + os.environ.get("CI", False) and not ( platform.system() == "Linux" and sys.version_info[:2] in {(3, 9), (3, 13)} ), @@ -54,7 +59,12 @@ def class_decorator(cls): @wrap_all_tests(xfail_on_github_connection_error) -class TestUPathGitHubPath(BaseTests): +class TestUPathGitHubPath( + JoinablePathTests, + ReadablePathTests, + NonWritablePathTests, + metaclass=OverrideMeta, +): """ Unit-tests for the GitHubPath implementation of UPath. """ @@ -67,76 +77,6 @@ def path(self): path = "github://ap--:universal_pathlib@test_data/data" self.path = UPath(path) - def test_is_GitHubPath(self): - """ - Test that the path is a GitHubPath instance. - """ + @overrides_base + def test_is_correct_class(self): assert isinstance(self.path, GitHubPath) - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_mkdir(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_mkdir_exists_ok_false(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_mkdir_parents_true_exists_ok_false(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_rename(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_rename2(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_touch(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_touch_unlink(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_write_bytes(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_write_text(self): - pass - - @pytest.mark.skip(reason="GitHub filesystem is read-only") - def test_fsspec_compat(self): - pass - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_move_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_move_into_local(self, tmp_path): - pass - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_move_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_move_into_memory(self, clear_fsspec_memory_cache): - pass - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_rename_with_target_absolute(self, target_factory): - return super().test_rename_with_target_str_absolute(target_factory) - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_write_text_encoding(self): - return super().test_write_text_encoding() - - @pytest.mark.skip(reason="Only testing read on GithubPath") - def test_write_text_errors(self): - return super().test_write_text_errors()