rclone-api 1.1.45__py2.py3-none-any.whl → 1.1.47__py2.py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- rclone_api/mount.py +13 -0
- rclone_api/profile/mount_copy_bytes.py +25 -5
- rclone_api/rclone.py +45 -25
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/METADATA +1 -1
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/RECORD +9 -9
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/LICENSE +0 -0
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/WHEEL +0 -0
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/entry_points.txt +0 -0
- {rclone_api-1.1.45.dist-info → rclone_api-1.1.47.dist-info}/top_level.txt +0 -0
rclone_api/mount.py
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import os
|
|
2
2
|
import platform
|
|
3
|
+
import shutil
|
|
3
4
|
import subprocess
|
|
4
5
|
import time
|
|
5
6
|
import warnings
|
|
@@ -19,6 +20,8 @@ class Mount:
|
|
|
19
20
|
mount_path: Path
|
|
20
21
|
process: Process
|
|
21
22
|
read_only: bool
|
|
23
|
+
cache_dir: Path | None = None
|
|
24
|
+
cache_dir_delete_on_exit: bool | None = None
|
|
22
25
|
_closed: bool = False
|
|
23
26
|
|
|
24
27
|
def __post_init__(self):
|
|
@@ -32,6 +35,8 @@ class Mount:
|
|
|
32
35
|
return
|
|
33
36
|
self._closed = True
|
|
34
37
|
clean_mount(self, verbose=False, wait=wait)
|
|
38
|
+
if self.cache_dir and self.cache_dir_delete_on_exit:
|
|
39
|
+
_cache_dir_delete_on_exit(self.cache_dir)
|
|
35
40
|
|
|
36
41
|
def __enter__(self) -> "Mount":
|
|
37
42
|
return self
|
|
@@ -179,3 +184,11 @@ def clean_mount(mount: Mount | Path, verbose: bool = False, wait=True) -> None:
|
|
|
179
184
|
else:
|
|
180
185
|
if verbose:
|
|
181
186
|
print(f"{mount_path} successfully cleaned up.")
|
|
187
|
+
|
|
188
|
+
|
|
189
|
+
def _cache_dir_delete_on_exit(cache_dir: Path) -> None:
|
|
190
|
+
if cache_dir.exists():
|
|
191
|
+
try:
|
|
192
|
+
shutil.rmtree(cache_dir)
|
|
193
|
+
except Exception as e:
|
|
194
|
+
warnings.warn(f"Error removing cache directory {cache_dir}: {e}")
|
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
Unit test file.
|
|
3
3
|
"""
|
|
4
4
|
|
|
5
|
+
import argparse
|
|
5
6
|
import os
|
|
6
7
|
import shutil
|
|
7
8
|
import time
|
|
@@ -14,6 +15,11 @@ from dotenv import load_dotenv
|
|
|
14
15
|
from rclone_api import Config, Rclone, SizeSuffix
|
|
15
16
|
|
|
16
17
|
|
|
18
|
+
@dataclass
|
|
19
|
+
class Args:
|
|
20
|
+
direct_io: bool
|
|
21
|
+
|
|
22
|
+
|
|
17
23
|
@dataclass
|
|
18
24
|
class Credentials:
|
|
19
25
|
BUCKET_KEY_SECRET: str
|
|
@@ -105,7 +111,12 @@ def _init() -> None:
|
|
|
105
111
|
|
|
106
112
|
|
|
107
113
|
def _run_profile(
|
|
108
|
-
rclone: Rclone,
|
|
114
|
+
rclone: Rclone,
|
|
115
|
+
src_file: str,
|
|
116
|
+
transfers: int,
|
|
117
|
+
size: int,
|
|
118
|
+
log_dir: Path,
|
|
119
|
+
direct_io: bool = True,
|
|
109
120
|
) -> None:
|
|
110
121
|
|
|
111
122
|
mount_log = log_dir / f"mount_{SizeSuffix(size)}_threads_{transfers}.log"
|
|
@@ -119,7 +130,7 @@ def _run_profile(
|
|
|
119
130
|
src=src_file,
|
|
120
131
|
offset=0,
|
|
121
132
|
length=size,
|
|
122
|
-
direct_io=
|
|
133
|
+
direct_io=direct_io,
|
|
123
134
|
transfers=transfers,
|
|
124
135
|
mount_log=mount_log,
|
|
125
136
|
)
|
|
@@ -154,7 +165,7 @@ def _run_profile(
|
|
|
154
165
|
print(f"Time: {diff:.1f} seconds")
|
|
155
166
|
|
|
156
167
|
|
|
157
|
-
def test_profile_copy_bytes() -> None:
|
|
168
|
+
def test_profile_copy_bytes(args: Args) -> None:
|
|
158
169
|
print("Running test_profile_copy_bytes")
|
|
159
170
|
config, creds = _generate_rclone_config()
|
|
160
171
|
print("Config:")
|
|
@@ -179,7 +190,7 @@ def test_profile_copy_bytes() -> None:
|
|
|
179
190
|
# sftp mount
|
|
180
191
|
src_file = "src:aa_misc_data/aa_misc_data/world_lending_library_2024_11.tar.zst"
|
|
181
192
|
|
|
182
|
-
mount_root_path = Path("
|
|
193
|
+
mount_root_path = Path("rclone_logs") / "mount"
|
|
183
194
|
if mount_root_path.exists():
|
|
184
195
|
shutil.rmtree(mount_root_path)
|
|
185
196
|
|
|
@@ -190,15 +201,24 @@ def test_profile_copy_bytes() -> None:
|
|
|
190
201
|
src_file=src_file,
|
|
191
202
|
transfers=transfers,
|
|
192
203
|
size=size,
|
|
204
|
+
direct_io=args.direct_io,
|
|
193
205
|
log_dir=mount_root_path,
|
|
194
206
|
)
|
|
195
207
|
print("done")
|
|
196
208
|
|
|
197
209
|
|
|
210
|
+
def _parse_args() -> Args:
|
|
211
|
+
parser = argparse.ArgumentParser(description="Profile copy_bytes")
|
|
212
|
+
parser.add_argument("--direct-io", help="Use direct IO", action="store_true")
|
|
213
|
+
args = parser.parse_args()
|
|
214
|
+
return Args(direct_io=args.direct_io)
|
|
215
|
+
|
|
216
|
+
|
|
198
217
|
def main() -> None:
|
|
199
218
|
"""Main entry point."""
|
|
200
219
|
_init()
|
|
201
|
-
|
|
220
|
+
args = _parse_args()
|
|
221
|
+
test_profile_copy_bytes(args)
|
|
202
222
|
|
|
203
223
|
|
|
204
224
|
if __name__ == "__main__":
|
rclone_api/rclone.py
CHANGED
|
@@ -852,30 +852,35 @@ class Rclone:
|
|
|
852
852
|
if direct_io:
|
|
853
853
|
other_args += ["--direct-io"]
|
|
854
854
|
|
|
855
|
-
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
|
|
862
|
-
|
|
863
|
-
|
|
864
|
-
|
|
865
|
-
|
|
866
|
-
|
|
867
|
-
|
|
868
|
-
|
|
869
|
-
|
|
870
|
-
|
|
871
|
-
|
|
872
|
-
with open(
|
|
873
|
-
|
|
874
|
-
|
|
875
|
-
|
|
876
|
-
|
|
877
|
-
|
|
878
|
-
|
|
855
|
+
with TemporaryDirectory() as tmpdir:
|
|
856
|
+
cache_dir = Path(tmpdir) / "cache"
|
|
857
|
+
other_args += ["--cache-dir", str(cache_dir.absolute())]
|
|
858
|
+
try:
|
|
859
|
+
# use scoped mount to do the read, then write the bytes to the destination
|
|
860
|
+
with self.scoped_mount(
|
|
861
|
+
src_parent_path,
|
|
862
|
+
tmp_mnt,
|
|
863
|
+
use_links=True,
|
|
864
|
+
verbose=mount_log is not None,
|
|
865
|
+
vfs_cache_mode="minimal",
|
|
866
|
+
other_args=other_args,
|
|
867
|
+
log=mount_log,
|
|
868
|
+
cache_dir=cache_dir,
|
|
869
|
+
cache_dir_delete_on_exit=True,
|
|
870
|
+
):
|
|
871
|
+
src_file_mnt = tmp_mnt / src_file
|
|
872
|
+
with open(src_file_mnt, "rb") as f:
|
|
873
|
+
f.seek(offset)
|
|
874
|
+
data = f.read(length)
|
|
875
|
+
if outfile is None:
|
|
876
|
+
return data
|
|
877
|
+
with open(outfile, "wb") as out:
|
|
878
|
+
out.write(data)
|
|
879
|
+
del data
|
|
880
|
+
return bytes(0)
|
|
881
|
+
|
|
882
|
+
except Exception as e:
|
|
883
|
+
return e
|
|
879
884
|
|
|
880
885
|
def copy_dir(
|
|
881
886
|
self, src: str | Dir, dst: str | Dir, args: list[str] | None = None
|
|
@@ -909,6 +914,8 @@ class Rclone:
|
|
|
909
914
|
use_links: bool | None = None,
|
|
910
915
|
vfs_cache_mode: str | None = None,
|
|
911
916
|
verbose: bool | None = None,
|
|
917
|
+
cache_dir: Path | None = None,
|
|
918
|
+
cache_dir_delete_on_exit: bool | None = None,
|
|
912
919
|
log: Path | None = None,
|
|
913
920
|
other_args: list[str] | None = None,
|
|
914
921
|
) -> Mount:
|
|
@@ -941,6 +948,9 @@ class Rclone:
|
|
|
941
948
|
if vfs_cache_mode:
|
|
942
949
|
cmd_list.append("--vfs-cache-mode")
|
|
943
950
|
cmd_list.append(vfs_cache_mode)
|
|
951
|
+
if cache_dir:
|
|
952
|
+
cmd_list.append("--cache-dir")
|
|
953
|
+
cmd_list.append(str(cache_dir.absolute()))
|
|
944
954
|
if debug_fuse:
|
|
945
955
|
cmd_list.append("--debug-fuse")
|
|
946
956
|
if verbose:
|
|
@@ -949,7 +959,13 @@ class Rclone:
|
|
|
949
959
|
cmd_list += other_args
|
|
950
960
|
proc = self._launch_process(cmd_list, log=log)
|
|
951
961
|
mount_read_only = not allow_writes
|
|
952
|
-
mount: Mount = Mount(
|
|
962
|
+
mount: Mount = Mount(
|
|
963
|
+
mount_path=outdir,
|
|
964
|
+
process=proc,
|
|
965
|
+
read_only=mount_read_only,
|
|
966
|
+
cache_dir=cache_dir,
|
|
967
|
+
cache_dir_delete_on_exit=cache_dir_delete_on_exit,
|
|
968
|
+
)
|
|
953
969
|
return mount
|
|
954
970
|
|
|
955
971
|
@contextmanager
|
|
@@ -962,6 +978,8 @@ class Rclone:
|
|
|
962
978
|
vfs_cache_mode: str | None = None,
|
|
963
979
|
verbose: bool | None = None,
|
|
964
980
|
log: Path | None = None,
|
|
981
|
+
cache_dir: Path | None = None,
|
|
982
|
+
cache_dir_delete_on_exit: bool | None = None,
|
|
965
983
|
other_args: list[str] | None = None,
|
|
966
984
|
) -> Generator[Mount, None, None]:
|
|
967
985
|
"""Like mount, but can be used in a context manager."""
|
|
@@ -973,6 +991,8 @@ class Rclone:
|
|
|
973
991
|
use_links=use_links,
|
|
974
992
|
vfs_cache_mode=vfs_cache_mode,
|
|
975
993
|
verbose=verbose,
|
|
994
|
+
cache_dir=cache_dir,
|
|
995
|
+
cache_dir_delete_on_exit=cache_dir_delete_on_exit,
|
|
976
996
|
log=log,
|
|
977
997
|
other_args=other_args,
|
|
978
998
|
)
|
|
@@ -11,9 +11,9 @@ rclone_api/exec.py,sha256=Pd7pUBd8ib5MzqvMybG2DQISPRbDRu20VjVRL2mLAVY,1076
|
|
|
11
11
|
rclone_api/file.py,sha256=EP5yT2dZ0H2p7CY5n0y5k5pHhIliV25pm8KOwBklUTk,1863
|
|
12
12
|
rclone_api/filelist.py,sha256=xbiusvNgaB_b_kQOZoHMJJxn6TWGtPrWd2J042BI28o,767
|
|
13
13
|
rclone_api/group_files.py,sha256=H92xPW9lQnbNw5KbtZCl00bD6iRh9yRbCuxku4j_3dg,8036
|
|
14
|
-
rclone_api/mount.py,sha256=
|
|
14
|
+
rclone_api/mount.py,sha256=_19IXIm3PHoXc78dIvFwkVT7QKp7Z0ian_bbhqF5Uq0,6723
|
|
15
15
|
rclone_api/process.py,sha256=rBj_S86jC6nqCYop-jq8r9eMSteKeObxUrJMgH8LZvI,5084
|
|
16
|
-
rclone_api/rclone.py,sha256
|
|
16
|
+
rclone_api/rclone.py,sha256=ODHPq1OKKzOUgi7QPDc1UZHZLSoG9SkxnT1MXpoggx4,43823
|
|
17
17
|
rclone_api/remote.py,sha256=O9WDUFQy9f6oT1HdUbTixK2eg0xtBBm8k4Xl6aa6K00,431
|
|
18
18
|
rclone_api/rpath.py,sha256=8ZA_1wxWtskwcy0I8V2VbjKDmzPkiWd8Q2JQSvh-sYE,2586
|
|
19
19
|
rclone_api/scan_missing_folders.py,sha256=Kulca2Q6WZodt00ATFHkmqqInuoPvBkhTcS9703y6po,4740
|
|
@@ -25,7 +25,7 @@ rclone_api/cmd/copy_large_s3.py,sha256=-rfedi-ZzPUdCSP8ai9LRL0y1xVkvN-viQQlk8HVU
|
|
|
25
25
|
rclone_api/cmd/list_files.py,sha256=x8FHODEilwKqwdiU1jdkeJbLwOqUkUQuDWPo2u_zpf0,741
|
|
26
26
|
rclone_api/experimental/flags.py,sha256=qCVD--fSTmzlk9hloRLr0q9elzAOFzPsvVpKM3aB1Mk,2739
|
|
27
27
|
rclone_api/experimental/flags_base.py,sha256=ajU_czkTcAxXYU-SlmiCfHY7aCQGHvpCLqJ-Z8uZLk0,2102
|
|
28
|
-
rclone_api/profile/mount_copy_bytes.py,sha256=
|
|
28
|
+
rclone_api/profile/mount_copy_bytes.py,sha256=Jc7AKjX37W6S7ramFFKdcA0l6oeuK2mJi00w50rERew,6351
|
|
29
29
|
rclone_api/s3/api.py,sha256=qxtRDUpHYqJ7StJRtP8U_PbF_BvYRg705568SyvF-R0,3770
|
|
30
30
|
rclone_api/s3/basic_ops.py,sha256=hK3366xhVEzEcjz9Gk_8lFx6MRceAk72cax6mUrr6ko,2104
|
|
31
31
|
rclone_api/s3/chunk_file.py,sha256=D6wM9Nuu73LNGC8JCCfevqjF3qdZ21mQxYQClFLZLMU,3726
|
|
@@ -33,9 +33,9 @@ rclone_api/s3/chunk_types.py,sha256=LbXayXY1KgVU1LkdbASD_BQ7TpVpwVnzMjtz--8LBaE,
|
|
|
33
33
|
rclone_api/s3/create.py,sha256=wgfkapv_j904CfKuWyiBIWJVxfAx_ftemFSUV14aT68,3149
|
|
34
34
|
rclone_api/s3/types.py,sha256=yBnJ38Tjk6RlydJ-sqZ7DSfyFloy8KDYJ0mv3vlOzLE,1388
|
|
35
35
|
rclone_api/s3/upload_file_multipart.py,sha256=y9azNAU8QH5Ovwz33V2HZwNmJdlFjJg-jrXLZ1gtMds,10364
|
|
36
|
-
rclone_api-1.1.
|
|
37
|
-
rclone_api-1.1.
|
|
38
|
-
rclone_api-1.1.
|
|
39
|
-
rclone_api-1.1.
|
|
40
|
-
rclone_api-1.1.
|
|
41
|
-
rclone_api-1.1.
|
|
36
|
+
rclone_api-1.1.47.dist-info/LICENSE,sha256=b6pOoifSXiUaz_lDS84vWlG3fr4yUKwB8fzkrH9R8bQ,1064
|
|
37
|
+
rclone_api-1.1.47.dist-info/METADATA,sha256=M-oXvo69uIDQy-0571dGJTfxay9-gTq3x3Q8ZNS2rRo,4537
|
|
38
|
+
rclone_api-1.1.47.dist-info/WHEEL,sha256=rF4EZyR2XVS6irmOHQIJx2SUqXLZKRMUrjsg8UwN-XQ,109
|
|
39
|
+
rclone_api-1.1.47.dist-info/entry_points.txt,sha256=TV8kwP3FRzYwUEr0RLC7aJh0W03SAefIJNXTJ-FdMIQ,200
|
|
40
|
+
rclone_api-1.1.47.dist-info/top_level.txt,sha256=EvZ7uuruUpe9RiUyEp25d1Keq7PWYNT0O_-mr8FCG5g,11
|
|
41
|
+
rclone_api-1.1.47.dist-info/RECORD,,
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|