Skip to content

Commit 1eb7b66

Browse files
authored
Code cleanup for bundle.py (#586)
* Clarify logic for flattened_data() and flattened_buffer() * Streamline logic for Bundle.to_file() * Use Mapping instead of dict * Fixes from code review
1 parent a2c8f5a commit 1eb7b66

File tree

4 files changed

+81
-62
lines changed

4 files changed

+81
-62
lines changed

rsconnect/api.py

+5-4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
Callable,
2424
List,
2525
Literal,
26+
Mapping,
2627
Optional,
2728
TypeVar,
2829
Union,
@@ -298,7 +299,7 @@ def python_settings(self) -> PyInfo:
298299
response = self._server.handle_bad_response(response)
299300
return response
300301

301-
def app_search(self, filters: Optional[dict[str, JsonData]]) -> AppSearchResults:
302+
def app_search(self, filters: Optional[Mapping[str, JsonData]]) -> AppSearchResults:
302303
response = cast(Union[AppSearchResults, HTTPResponse], self.get("applications", query_params=filters))
303304
response = self._server.handle_bad_response(response)
304305
return response
@@ -318,7 +319,7 @@ def app_upload(self, app_id: str, tarball: typing.IO[bytes]) -> ContentItemV0:
318319
response = self._server.handle_bad_response(response)
319320
return response
320321

321-
def app_update(self, app_id: str, updates: dict[str, str | None]) -> ContentItemV0:
322+
def app_update(self, app_id: str, updates: Mapping[str, str | None]) -> ContentItemV0:
322323
response = cast(Union[ContentItemV0, HTTPResponse], self.post("applications/%s" % app_id, body=updates))
323324
response = self._server.handle_bad_response(response)
324325
return response
@@ -1419,7 +1420,7 @@ def create_revision(self, content_id: str) -> PositClientCloudOutputRevision:
14191420
response = self._server.handle_bad_response(response)
14201421
return response
14211422

1422-
def update_output(self, output_id: int, output_data: dict[str, str]):
1423+
def update_output(self, output_id: int, output_data: Mapping[str, str]):
14231424
return self.patch("/v1/outputs/{}".format(output_id), body=output_data)
14241425

14251426
def get_accounts(self) -> PositClientAccountSearchResults:
@@ -1866,7 +1867,7 @@ def retrieve_matching_apps(
18661867
"""
18671868
page_size = 100
18681869
result: list[ContentItemV0 | AbbreviatedAppItem] = []
1869-
search_filters = filters.copy() if filters else {}
1870+
search_filters: dict[str, str | int] = filters.copy() if filters else {}
18701871
search_filters["count"] = min(limit, page_size) if limit else page_size
18711872
total_returned = 0
18721873
maximum = limit

rsconnect/bundle.py

+56-38
Original file line numberDiff line numberDiff line change
@@ -264,77 +264,89 @@ def discard_from_buffer(self, key: str):
264264
del self.data["files"][key]
265265
return self
266266

267-
def raise_on_empty_entrypoint(self):
267+
def require_entrypoint(self) -> str:
268+
"""
269+
If self.entrypoint is a string, return it; if it is None, raise an exception.
270+
"""
268271
if self.entrypoint is None:
269272
raise RSConnectException("A valid entrypoint must be provided.")
270-
return self
273+
return self.entrypoint
274+
275+
def get_manifest_files(self) -> dict[str, ManifestDataFile]:
276+
new_data_files: dict[str, ManifestDataFile] = {}
277+
deploy_dir: str
278+
279+
entrypoint = self.require_entrypoint()
280+
if self.deploy_dir is not None:
281+
deploy_dir = self.deploy_dir
282+
elif entrypoint is not None and isfile(entrypoint):
283+
deploy_dir = dirname(entrypoint)
284+
else:
285+
# TODO: This branch might be an error case. Need to investigate.
286+
deploy_dir = entrypoint
271287

272-
@property
273-
def flattened_data(self):
274-
self.raise_on_empty_entrypoint()
275-
new_data_files = {}
276-
deploy_dir = dirname(self.entrypoint) if isfile(self.entrypoint) else self.entrypoint
277-
deploy_dir = self.deploy_dir or deploy_dir
278288
for path in self.data["files"]:
279289
rel_path = relpath(path, deploy_dir)
280290
manifestPath = Path(rel_path).as_posix()
281291
new_data_files[manifestPath] = self.data["files"][path]
282292
return new_data_files
283293

284-
@property
285-
def flattened_buffer(self) -> dict[str, str]:
286-
self.raise_on_empty_entrypoint()
294+
def get_manifest_files_from_buffer(self) -> dict[str, str]:
287295
new_buffer: dict[str, str] = {}
288-
deploy_dir = dirname(self.entrypoint) if isfile(self.entrypoint) else self.entrypoint
289-
deploy_dir = self.deploy_dir or deploy_dir
296+
deploy_dir: str
297+
298+
entrypoint = self.require_entrypoint()
299+
if self.deploy_dir is not None:
300+
deploy_dir = self.deploy_dir
301+
elif entrypoint is not None and isfile(entrypoint):
302+
deploy_dir = dirname(entrypoint)
303+
else:
304+
# TODO: This branch might be an error case. Need to investigate.
305+
deploy_dir = entrypoint
306+
290307
for k, v in self.buffer.items():
291308
rel_path = relpath(k, deploy_dir)
292309
manifestPath = Path(rel_path).as_posix()
293310
new_buffer[manifestPath] = v
294311
return new_buffer
295312

296-
@property
297-
def flattened_entrypoint(self):
298-
self.raise_on_empty_entrypoint()
299-
return relpath(self.entrypoint, dirname(self.entrypoint))
313+
def get_relative_entrypoint(self) -> str:
314+
entrypoint = self.require_entrypoint()
315+
return basename(entrypoint)
300316

301-
@property
302-
def flattened_primary_html(self):
317+
def get_flattened_primary_html(self):
303318
if self.primary_html is None:
304319
raise RSConnectException("A valid primary_html must be provided.")
305320
return relpath(self.primary_html, dirname(self.primary_html))
306321

307-
@property
308-
def flattened_copy(self):
309-
self.raise_on_empty_entrypoint()
322+
def get_flattened_copy(self):
310323
new_manifest = deepcopy(self)
311-
new_manifest.data["files"] = self.flattened_data
312-
new_manifest.buffer = self.flattened_buffer
313-
new_manifest.entrypoint = self.flattened_entrypoint
324+
new_manifest.data["files"] = self.get_manifest_files()
325+
new_manifest.buffer = self.get_manifest_files_from_buffer()
326+
new_manifest.entrypoint = self.get_relative_entrypoint()
314327
if self.primary_html:
315-
new_manifest.primary_html = self.flattened_primary_html
328+
new_manifest.primary_html = self.get_flattened_primary_html()
316329
return new_manifest
317330

318331

319332
class Bundle:
320333
def __init__(self) -> None:
321334
self.file_paths: set[str] = set()
322335
self.buffer: dict[str, str] = {}
323-
self.deploy_dir: str | None = None
324336

325337
def add_file(self, filepath: str) -> None:
326338
self.file_paths.add(filepath)
327339

328340
def discard_file(self, filepath: str) -> None:
329341
self.file_paths.discard(filepath)
330342

331-
def to_file(self, flatten_to_deploy_dir: bool = True) -> typing.IO[bytes]:
343+
def to_file(self, deploy_dir: str) -> typing.IO[bytes]:
332344
bundle_file = tempfile.TemporaryFile(prefix="rsc_bundle")
333345
with tarfile.open(mode="w:gz", fileobj=bundle_file) as bundle:
334346
for fp in self.file_paths:
335347
if Path(fp).name in self.buffer:
336348
continue
337-
rel_path = Path(fp).relative_to(self.deploy_dir) if flatten_to_deploy_dir else None
349+
rel_path = Path(fp).relative_to(deploy_dir)
338350
logger.log(VERBOSE, "Adding file: %s", fp)
339351
bundle.add(fp, arcname=rel_path)
340352
for k, v in self.buffer.items():
@@ -1062,20 +1074,21 @@ def make_html_bundle(
10621074

10631075
if manifest.data.get("files") is None:
10641076
raise RSConnectException("No valid files were found for the manifest.")
1077+
if manifest.deploy_dir is None:
1078+
raise RSConnectException("deploy_dir was not set for the manifest.")
10651079

10661080
bundle = Bundle()
10671081
for f in manifest.data["files"]:
10681082
if f in manifest.buffer:
10691083
continue
10701084
bundle.add_file(f)
1071-
for k, v in manifest.flattened_buffer.items():
1085+
for k, v in manifest.get_manifest_files_from_buffer().items():
10721086
bundle.add_to_buffer(k, v)
10731087

1074-
manifest_flattened_copy_data = manifest.flattened_copy.data
1088+
manifest_flattened_copy_data = manifest.get_flattened_copy().data
10751089
bundle.add_to_buffer("manifest.json", json.dumps(manifest_flattened_copy_data, indent=2))
1076-
bundle.deploy_dir = manifest.deploy_dir
10771090

1078-
return bundle.to_file()
1091+
return bundle.to_file(manifest.deploy_dir)
10791092

10801093

10811094
def create_file_list(
@@ -1255,22 +1268,23 @@ def make_voila_bundle(
12551268

12561269
if manifest.data.get("files") is None:
12571270
raise RSConnectException("No valid files were found for the manifest.")
1271+
if manifest.deploy_dir is None:
1272+
raise RSConnectException("deploy_dir was not set for the manifest.")
12581273

12591274
bundle = Bundle()
12601275
for f in manifest.data["files"]:
12611276
if f in manifest.buffer:
12621277
continue
12631278
bundle.add_file(f)
1264-
for k, v in manifest.flattened_buffer.items():
1279+
for k, v in manifest.get_manifest_files_from_buffer().items():
12651280
bundle.add_to_buffer(k, v)
12661281

1267-
manifest_flattened_copy_data = manifest.flattened_copy.data
1282+
manifest_flattened_copy_data = manifest.get_flattened_copy().data
12681283
if multi_notebook and "metadata" in manifest_flattened_copy_data:
12691284
manifest_flattened_copy_data["metadata"]["entrypoint"] = ""
12701285
bundle.add_to_buffer("manifest.json", json.dumps(manifest_flattened_copy_data, indent=2))
1271-
bundle.deploy_dir = manifest.deploy_dir
12721286

1273-
return bundle.to_file()
1287+
return bundle.to_file(manifest.deploy_dir)
12741288

12751289

12761290
def make_api_bundle(
@@ -1969,8 +1983,12 @@ def write_voila_manifest_json(
19691983
env_management_r=env_management_r,
19701984
multi_notebook=multi_notebook,
19711985
)
1986+
1987+
if manifest.entrypoint is None:
1988+
raise RSConnectException("Voila manifest requires an entrypoint.")
1989+
19721990
deploy_dir = dirname(manifest.entrypoint) if isfile(manifest.entrypoint) else manifest.entrypoint
1973-
manifest_flattened_copy_data = manifest.flattened_copy.data
1991+
manifest_flattened_copy_data = manifest.get_flattened_copy().data
19741992
if multi_notebook and "metadata" in manifest_flattened_copy_data:
19751993
manifest_flattened_copy_data["metadata"]["entrypoint"] = ""
19761994
manifest_path = join(deploy_dir, "manifest.json")

tests/test_Manifest.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -81,7 +81,7 @@ def test_Manifest_flattened_copy():
8181
"test_folder1/testfoldertext1.txt": {"checksum": "0a576fd324b6985bac6aa934131d2f5c"},
8282
},
8383
}
84-
assert m.flattened_copy.data == html_manifest_dict
84+
assert m.get_flattened_copy().data == html_manifest_dict
8585

8686

8787
def test_Manifest_empty_init():
@@ -97,4 +97,4 @@ def test_Manifest_empty_init():
9797
def test_Manifest_empty_exceptions():
9898
m = Manifest()
9999
with pytest.raises(RSConnectException) as _:
100-
m.raise_on_empty_entrypoint()
100+
m.require_entrypoint()

0 commit comments

Comments
 (0)