Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow remote .conda artifact listing #254

Merged
merged 13 commits into from
Jul 18, 2024
1 change: 1 addition & 0 deletions conda.recipe/meta.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ requirements:
- python
- zstandard >=0.15
- conda-package-streaming >=0.9.0
- requests

test:
source_files:
Expand Down
19 changes: 19 additions & 0 deletions news/254-list-remote
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
### Enhancements

* Allow `cph list` on remote `.conda` artifact URLs. (#252 via #254)

### Bug fixes

* <news item>

### Deprecations

* <news item>

### Docs

* <news item>

### Other

* <news item>
46 changes: 44 additions & 2 deletions src/conda_package_handling/conda_fmt.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,16 @@

import json
import os
import stat
import tarfile
import time
from contextlib import closing
from typing import Callable
from zipfile import ZIP_STORED, ZipFile

import zstandard
from conda_package_streaming.package_streaming import stream_conda_component
from conda_package_streaming.url import conda_reader_for_url

from . import utils
from .interface import AbstractBaseFormat
Expand Down Expand Up @@ -145,9 +150,46 @@ def get_pkg_details(in_file):
md5, sha256 = utils.checksums(in_file, ("md5", "sha256"))
return {"size": size, "md5": md5, "sha256": sha256}

@staticmethod
def list_contents(fn, verbose=False, **kw):
@classmethod
def list_contents(cls, fn, verbose=False, **kw):
components = utils.ensure_list(kw.get("components")) or ("info", "pkg")
if "://" in fn:
return cls._list_remote_contents(fn, components=components, verbose=verbose)
# local resource
if not os.path.isabs(fn):
fn = os.path.abspath(fn)
_list(fn, components=components, verbose=verbose)

@staticmethod
def _list_remote_contents(url, verbose=False, components=("info", "pkg")):
"""
List contents of a remote .conda artifact (by URL). It only fetches the 'info' component
and uses the metadata to infer details of the 'pkg' component. Some fields like
modification time or permissions will be missing in verbose mode.
"""
components = utils.ensure_list(components or ("info", "pkg"))
lines = {}
filename, conda = conda_reader_for_url(url)
with closing(conda):
for tar, member in stream_conda_component(filename, conda, component="info"):
path = member.name + ("/" if member.isdir() else "")
if "info" in components:
line = ""
if verbose:
line = (
f"{stat.filemode(member.mode)} "
f"{member.uname or member.uid}/{member.gname or member.gid} "
f"{member.size:10d} "
)
line += "%d-%02d-%02d %02d:%02d:%02d " % time.localtime(member.mtime)[:6]
lines[path] = line + path
if "pkg" in components and member.name == "info/paths.json":
data = json.loads(tar.extractfile(member).read().decode())
assert data.get("paths_version", 1) == 1, data
for path in data.get("paths", ()):
line = ""
if verbose:
size = path["size_in_bytes"]
line = f"?????????? ?/? {size:10d} ????-??-?? ??:??:?? "
lines[path["_path"]] = line + path["_path"]
print(*[line for _, line in sorted(lines.items())], sep="\n")
2 changes: 2 additions & 0 deletions src/conda_package_handling/tarball.py
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,8 @@ def get_pkg_details(in_file):

@staticmethod
def list_contents(fn, verbose=False, **kw):
if "://" in fn:
raise ValueError("Remote .tar.bz2 artifact listing is not supported.")
if not os.path.isabs(fn):
fn = os.path.abspath(fn)
streaming._list(str(fn), components=["pkg"], verbose=verbose)
24 changes: 24 additions & 0 deletions tests/test_cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -82,3 +82,27 @@ def test_list(artifact, n_files, capsys):

with pytest.raises(ValueError):
cli.main(["list", "setup.py"])


@pytest.mark.parametrize(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Several other conda projects including conda, conda-index, conda-package-streaming, use local web servers as a CI fixture for this sort of thing.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added a bottle fixture similar to what CPS is doing (but supported by pytest-xprocess for simplicity).

"url,n_files",
[
("https://conda.anaconda.org/conda-forge/win-64/7zip-23.01-h91493d7_2.conda", 27),
("https://conda.anaconda.org/conda-forge/win-64/7zip-19.00-h74a9793_2.tar.bz2", -1),
],
)
def test_list_remote(url, n_files, capsys):
"Integration test to ensure `cph list <URL>` works correctly."
if url.endswith(".tar.bz2"):
# This is not supported in streaming mode
with pytest.raises(ValueError):
cli.main(["list", url])
return

cli.main(["list", url])
stdout, stderr = capsys.readouterr()
assert n_files == sum(bool(line.strip()) for line in stdout.splitlines())

cli.main(["list", url, "-v"])
stdout, stderr = capsys.readouterr()
assert n_files == sum(bool(line.strip()) for line in stdout.splitlines())
Loading