diff options
-rw-r--r-- | lib/spack/spack/cmd/checksum.py | 31 | ||||
-rw-r--r-- | lib/spack/spack/package_base.py | 81 | ||||
-rw-r--r-- | lib/spack/spack/url.py | 46 | ||||
-rw-r--r-- | lib/spack/spack/util/web.py | 6 |
4 files changed, 79 insertions, 85 deletions
diff --git a/lib/spack/spack/cmd/checksum.py b/lib/spack/spack/cmd/checksum.py index 4eae8da7d4..e3e9c1a308 100644 --- a/lib/spack/spack/cmd/checksum.py +++ b/lib/spack/spack/cmd/checksum.py @@ -5,6 +5,7 @@ import re import sys +from typing import Dict, Optional import llnl.string import llnl.util.lang @@ -25,7 +26,7 @@ from spack.package_base import ( ) from spack.util.editor import editor from spack.util.format import get_version_lines -from spack.version import Version +from spack.version import StandardVersion, Version description = "checksum available versions of a package" section = "packaging" @@ -89,32 +90,30 @@ def checksum(parser, args): spec = spack.spec.Spec(args.package) # Get the package we're going to generate checksums for - pkg = spack.repo.PATH.get_pkg_class(spec.name)(spec) + pkg: PackageBase = spack.repo.PATH.get_pkg_class(spec.name)(spec) # Skip manually downloaded packages if pkg.manual_download: raise ManualDownloadRequiredError(pkg.download_instr) - versions = [Version(v) for v in args.versions] + versions = [StandardVersion.from_string(v) for v in args.versions] - # Define placeholder for remote versions. - # This'll help reduce redundant work if we need to check for the existance - # of remote versions more than once. - remote_versions = None + # Define placeholder for remote versions. This'll help reduce redundant work if we need to + # check for the existence of remote versions more than once. + remote_versions: Optional[Dict[StandardVersion, str]] = None # Add latest version if requested if args.latest: - remote_versions = pkg.fetch_remote_versions(args.jobs) + remote_versions = pkg.fetch_remote_versions(concurrency=args.jobs) if len(remote_versions) > 0: - latest_version = sorted(remote_versions.keys(), reverse=True)[0] - versions.append(latest_version) + versions.append(max(remote_versions.keys())) - # Add preferred version if requested + # Add preferred version if requested (todo: exclude git versions) if args.preferred: versions.append(preferred_version(pkg)) # Store a dict of the form version -> URL - url_dict = {} + url_dict: Dict[StandardVersion, str] = {} for version in versions: if deprecated_version(pkg, version): @@ -124,16 +123,16 @@ def checksum(parser, args): if url is not None: url_dict[version] = url continue - # if we get here, it's because no valid url was provided by the package - # do expensive fallback to try to recover + # If we get here, it's because no valid url was provided by the package. Do expensive + # fallback to try to recover if remote_versions is None: - remote_versions = pkg.fetch_remote_versions(args.jobs) + remote_versions = pkg.fetch_remote_versions(concurrency=args.jobs) if version in remote_versions: url_dict[version] = remote_versions[version] if len(versions) <= 0: if remote_versions is None: - remote_versions = pkg.fetch_remote_versions(args.jobs) + remote_versions = pkg.fetch_remote_versions(concurrency=args.jobs) url_dict = remote_versions # A spidered URL can differ from the package.py *computed* URL, pointing to different tarballs. diff --git a/lib/spack/spack/package_base.py b/lib/spack/spack/package_base.py index cc0af82ce5..8066c1b70f 100644 --- a/lib/spack/spack/package_base.py +++ b/lib/spack/spack/package_base.py @@ -67,7 +67,7 @@ from spack.installer import InstallError, PackageInstaller from spack.stage import DIYStage, ResourceStage, Stage, StageComposite, compute_stage_name from spack.util.executable import ProcessError, which from spack.util.package_hash import package_hash -from spack.version import GitVersion, StandardVersion, Version +from spack.version import GitVersion, StandardVersion FLAG_HANDLER_RETURN_TYPE = Tuple[ Optional[Iterable[str]], Optional[Iterable[str]], Optional[Iterable[str]] @@ -94,29 +94,26 @@ _spack_configure_argsfile = "spack-configure-args.txt" spack_times_log = "install_times.json" -def deprecated_version(pkg, version): - """Return True if the version is deprecated, False otherwise. +def deprecated_version(pkg: "PackageBase", version: Union[str, StandardVersion]) -> bool: + """Return True iff the version is deprecated. Arguments: - pkg (PackageBase): The package whose version is to be checked. - version (str or spack.version.StandardVersion): The version being checked + pkg: The package whose version is to be checked. + version: The version being checked """ if not isinstance(version, StandardVersion): - version = Version(version) + version = StandardVersion.from_string(version) - for k, v in pkg.versions.items(): - if version == k and v.get("deprecated", False): - return True + details = pkg.versions.get(version) + return details is not None and details.get("deprecated", False) - return False - -def preferred_version(pkg): +def preferred_version(pkg: "PackageBase"): """ Returns a sorted list of the preferred versions of the package. Arguments: - pkg (PackageBase): The package whose versions are to be assessed. + pkg: The package whose versions are to be assessed. """ # Here we sort first on the fact that a version is marked # as preferred in the package, then on the fact that the @@ -903,22 +900,16 @@ class PackageBase(WindowsRPath, PackageViewMixin, metaclass=PackageMeta): @classmethod @memoized - def version_urls(cls): - """OrderedDict of explicitly defined URLs for versions of this package. + def version_urls(cls) -> Dict[StandardVersion, str]: + """Dict of explicitly defined URLs for versions of this package. Return: - An OrderedDict (version -> URL) different versions of this - package, sorted by version. + An dict mapping version to url, ordered by version. - A version's URL only appears in the result if it has an an - explicitly defined ``url`` argument. So, this list may be empty - if a package only defines ``url`` at the top level. + A version's URL only appears in the result if it has an an explicitly defined ``url`` + argument. So, this list may be empty if a package only defines ``url`` at the top level. """ - version_urls = collections.OrderedDict() - for v, args in sorted(cls.versions.items()): - if "url" in args: - version_urls[v] = args["url"] - return version_urls + return {v: args["url"] for v, args in sorted(cls.versions.items()) if "url" in args} def nearest_url(self, version): """Finds the URL with the "closest" version to ``version``. @@ -961,36 +952,39 @@ class PackageBase(WindowsRPath, PackageViewMixin, metaclass=PackageMeta): """ pass - def all_urls_for_version(self, version): + def all_urls_for_version(self, version: StandardVersion) -> List[str]: """Return all URLs derived from version_urls(), url, urls, and list_url (if it contains a version) in a package in that order. Args: - version (spack.version.Version): the version for which a URL is sought + version: the version for which a URL is sought """ uf = None if type(self).url_for_version != PackageBase.url_for_version: uf = self.url_for_version return self._implement_all_urls_for_version(version, uf) - def _implement_all_urls_for_version(self, version, custom_url_for_version=None): - if not isinstance(version, StandardVersion): - version = Version(version) + def _implement_all_urls_for_version( + self, + version: Union[str, StandardVersion], + custom_url_for_version: Optional[Callable[[StandardVersion], Optional[str]]] = None, + ) -> List[str]: + version = StandardVersion.from_string(version) if isinstance(version, str) else version - urls = [] + urls: List[str] = [] # If we have a specific URL for this version, don't extrapolate. - version_urls = self.version_urls() - if version in version_urls: - urls.append(version_urls[version]) + url = self.version_urls().get(version) + if url: + urls.append(url) # if there is a custom url_for_version, use it if custom_url_for_version is not None: u = custom_url_for_version(version) - if u not in urls and u is not None: + if u is not None and u not in urls: urls.append(u) - def sub_and_add(u): + def sub_and_add(u: Optional[str]) -> None: if u is None: return # skip the url if there is no version to replace @@ -998,9 +992,7 @@ class PackageBase(WindowsRPath, PackageViewMixin, metaclass=PackageMeta): spack.url.parse_version(u) except spack.url.UndetectableVersionError: return - nu = spack.url.substitute_version(u, self.url_version(version)) - - urls.append(nu) + urls.append(spack.url.substitute_version(u, self.url_version(version))) # If no specific URL, use the default, class-level URL sub_and_add(getattr(self, "url", None)) @@ -2358,15 +2350,14 @@ class PackageBase(WindowsRPath, PackageViewMixin, metaclass=PackageMeta): return results.getvalue() @property - def all_urls(self): + def all_urls(self) -> List[str]: """A list of all URLs in a package. Check both class-level and version-specific URLs. - Returns: - list: a list of URLs + Returns a list of URLs """ - urls = [] + urls: List[str] = [] if hasattr(self, "url") and self.url: urls.append(self.url) @@ -2379,7 +2370,9 @@ class PackageBase(WindowsRPath, PackageViewMixin, metaclass=PackageMeta): urls.append(args["url"]) return urls - def fetch_remote_versions(self, concurrency=None): + def fetch_remote_versions( + self, concurrency: Optional[int] = None + ) -> Dict[StandardVersion, str]: """Find remote versions of this package. Uses ``list_url`` and any other URLs listed in the package file. diff --git a/lib/spack/spack/url.py b/lib/spack/spack/url.py index 6a84ddca2e..e9f89b02f1 100644 --- a/lib/spack/spack/url.py +++ b/lib/spack/spack/url.py @@ -29,6 +29,7 @@ import io import os import pathlib import re +from typing import Any, Dict, Optional, Sequence, Union import llnl.url from llnl.path import convert_to_posix_path @@ -284,20 +285,19 @@ def parse_version_offset(path): raise UndetectableVersionError(original_path) -def parse_version(path): +def parse_version(path: str) -> spack.version.StandardVersion: """Try to extract a version string from a filename or URL. Args: - path (str): The filename or URL for the package + path: The filename or URL for the package - Returns: - spack.version.Version: The version of the package + Returns: The version of the package Raises: UndetectableVersionError: If the URL does not match any regexes """ version, start, length, i, regex = parse_version_offset(path) - return spack.version.Version(version) + return spack.version.StandardVersion.from_string(version) def parse_name_offset(path, v=None): @@ -604,8 +604,12 @@ def color_url(path, **kwargs): def find_versions_of_archive( - archive_urls, list_url=None, list_depth=0, concurrency=32, reference_package=None -): + archive_urls: Union[str, Sequence[str]], + list_url: Optional[str] = None, + list_depth: int = 0, + concurrency: Optional[int] = 32, + reference_package: Optional[Any] = None, +) -> Dict[spack.version.StandardVersion, str]: """Scrape web pages for new versions of a tarball. This function prefers URLs in the following order: links found on the scraped page that match a url generated by the reference package, found and in the archive_urls list, found and derived from those @@ -613,22 +617,18 @@ def find_versions_of_archive( archive_urls list is included for the version. Args: - archive_urls (str or list or tuple): URL or sequence of URLs for - different versions of a package. Typically these are just the - tarballs from the package file itself. By default, this searches - the parent directories of archives. - list_url (str or None): URL for a listing of archives. - Spack will scrape these pages for download links that look - like the archive URL. - list_depth (int): max depth to follow links on list_url pages. - Defaults to 0. - concurrency (int): maximum number of concurrent requests - reference_package (spack.package_base.PackageBase or None): a spack package - used as a reference for url detection. Uses the url_for_version - method on the package to produce reference urls which, if found, - are preferred. + archive_urls: URL or sequence of URLs for different versions of a package. Typically these + are just the tarballs from the package file itself. By default, this searches the + parent directories of archives. + list_url: URL for a listing of archives. Spack will scrape these pages for download links + that look like the archive URL. + list_depth: max depth to follow links on list_url pages. Defaults to 0. + concurrency: maximum number of concurrent requests + reference_package: a spack package used as a reference for url detection. Uses the + url_for_version method on the package to produce reference urls which, if found, are + preferred. """ - if not isinstance(archive_urls, (list, tuple)): + if isinstance(archive_urls, str): archive_urls = [archive_urls] # Generate a list of list_urls based on archive urls and any @@ -689,7 +689,7 @@ def find_versions_of_archive( # Build a dict version -> URL from any links that match the wildcards. # Walk through archive_url links first. # Any conflicting versions will be overwritten by the list_url links. - versions = {} + versions: Dict[spack.version.StandardVersion, str] = {} matched = set() for url in sorted(links): url = convert_to_posix_path(url) diff --git a/lib/spack/spack/util/web.py b/lib/spack/spack/util/web.py index 6196cb66bd..d56078a28e 100644 --- a/lib/spack/spack/util/web.py +++ b/lib/spack/spack/util/web.py @@ -17,7 +17,7 @@ import traceback import urllib.parse from html.parser import HTMLParser from pathlib import Path, PurePosixPath -from typing import IO, Dict, List, Optional, Set, Union +from typing import IO, Dict, Iterable, List, Optional, Set, Union from urllib.error import HTTPError, URLError from urllib.request import HTTPSHandler, Request, build_opener @@ -554,7 +554,9 @@ def list_url(url, recursive=False): return gcs.get_all_blobs(recursive=recursive) -def spider(root_urls: Union[str, List[str]], depth: int = 0, concurrency: Optional[int] = None): +def spider( + root_urls: Union[str, Iterable[str]], depth: int = 0, concurrency: Optional[int] = None +): """Get web pages from root URLs. If depth is specified (e.g., depth=2), then this will also follow up to <depth> levels |