Source code for spack.cmd.develop

# Copyright Spack Project Developers. See COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
import argparse
import os
import shutil
from typing import Optional

import spack.cmd
import spack.config
import spack.environment
import spack.fetch_strategy
import spack.llnl.util.tty as tty
import spack.repo
import spack.spec
import spack.stage
import spack.util.path
import spack.version
from spack.cmd.common import arguments
from spack.error import SpackError

description = "add a spec to an environment's dev-build information"
section = "environments"
level = "long"


[docs] def setup_parser(subparser: argparse.ArgumentParser) -> None: subparser.add_argument("-p", "--path", help="source location of package") subparser.add_argument("-b", "--build-directory", help="build directory for the package") clone_group = subparser.add_mutually_exclusive_group() clone_group.add_argument( "--no-clone", action="store_false", dest="clone", help="do not clone, the package already exists at the source path", ) clone_group.add_argument( "--clone", action="store_true", dest="clone", default=True, help=( "(default) clone the package unless the path already exists, " "use ``--force`` to overwrite" ), ) subparser.add_argument( "--no-modify-concrete-specs", action="store_false", default=True, dest="apply_changes", help=( "do not mutate concrete specs to have dev_path provenance." " This requires a later `spack concretize --force` command to use develop specs" ), ) subparser.add_argument( "-f", "--force", action="store_true", default=False, help="remove any files or directories that block cloning source code", ) subparser.add_argument( "-r", "--recursive", action="store_true", help="traverse nodes of the graph to mark everything up to the root as a develop spec", ) arguments.add_common_arguments(subparser, ["spec"])
def _retrieve_develop_source(spec: spack.spec.Spec, abspath: str) -> None: # "steal" the source code via staging API. We ask for a stage # to be created, then copy it afterwards somewhere else. It would be # better if we can create the `source_path` directly into its final # destination. pkg_cls = spack.repo.PATH.get_pkg_class(spec.name) # We construct a package class ourselves, rather than asking for # Spec.package, since Spec only allows this when it is concrete package = pkg_cls(spec) source_stage: spack.stage.Stage = package.stage[0] if isinstance(source_stage.fetcher, spack.fetch_strategy.GitFetchStrategy): source_stage.fetcher.get_full_repo = True # If we retrieved this version before and cached it, we may have # done so without cloning the full git repo; likewise, any # mirror might store an instance with truncated history. source_stage.default_fetcher_only = True source_stage.fetcher.set_package(package) package.stage.steal_source(abspath)
[docs] def assure_concrete_spec(env: spack.environment.Environment, spec: spack.spec.Spec): version = spec.versions.concrete_range_as_version if not version: # first check environment for a matching concrete spec matching_specs = env.all_matching_specs(spec) if matching_specs: version = matching_specs[0].version test_spec = spack.spec.Spec(f"{spec}@{version}") for m_spec in matching_specs: if not m_spec.satisfies(test_spec): raise SpackError( f"{spec.name}: has multiple concrete instances in the graph that can't be" " satisfied by a single develop spec. To use `spack develop` ensure one" " of the following:" f"\n a) {spec.name} nodes can satisfy the same develop spec (minimally " "this means they all share the same version)" f"\n b) Provide a concrete develop spec ({spec.name}@[version]) to clearly" " indicate what should be developed" ) else: # look up the maximum version so infintiy versions are preferred for develop version = max(spack.repo.PATH.get_pkg_class(spec.fullname).versions.keys()) tty.msg(f"Defaulting to highest version: {spec.name}@{version}") spec.versions = spack.version.VersionList([version])
[docs] def setup_src_code(spec: spack.spec.Spec, src_path: str, clone: bool = True, force: bool = False): """ Handle checking, cloning or overwriting source code """ assert spec.versions if clone: _clone(spec, src_path, force) if not clone and not os.path.exists(src_path): raise SpackError(f"Provided path {src_path} does not exist") version = spec.versions.concrete_range_as_version if not version: # look up the maximum version so infintiy versions are preferred for develop version = max(spack.repo.PATH.get_pkg_class(spec.fullname).versions.keys()) tty.msg(f"Defaulting to highest version: {spec.name}@{version}") spec.versions = spack.version.VersionList([version])
def _update_config(spec, path): find_fn = lambda section: spec.name in section entry = {"spec": str(spec)} if path and path != spec.name: entry["path"] = path def change_fn(section): section[spec.name] = entry spack.config.change_or_add("develop", find_fn, change_fn)
[docs] def update_env( env: spack.environment.Environment, spec: spack.spec.Spec, specified_path: Optional[str] = None, build_dir: Optional[str] = None, apply_changes: bool = True, ): """ Update the spack.yaml file with additions or changes from a develop call """ tty.debug(f"Updating develop config for {env.name} transactionally") if not specified_path: dev_entry = env.dev_specs.get(spec.name) if dev_entry: specified_path = dev_entry.get("path", None) with env.write_transaction(): if build_dir is not None: spack.config.add( f"packages:{spec.name}:package_attributes:build_directory:{build_dir}", env.scope_name, ) # add develop spec and update path _update_config(spec, specified_path) # If we are automatically mutating the concrete specs for dev provenance, do so if apply_changes: env.apply_develop([spec], [_abs_code_path(env, spec, specified_path)])
def _clone(spec: spack.spec.Spec, abspath: str, force: bool = False): if os.path.exists(abspath): if force: shutil.rmtree(abspath) else: msg = f"Skipping developer download of {spec.name}" msg += f" because its path {abspath} already exists." tty.msg(msg) return # cloning can take a while and it's nice to get a message for the longer clones tty.msg(f"Cloning source code for {spec}") _retrieve_develop_source(spec, abspath) def _abs_code_path( env: spack.environment.Environment, spec: spack.spec.Spec, path: Optional[str] = None ): src_path = path if path else spec.name return spack.util.path.canonicalize_path(src_path, default_wd=env.path) def _dev_spec_generator(args, env): """ Generator function to loop over all the develop specs based on how the command is called If no specs are supplied then loop over the develop specs listed in the environment. """ if not args.spec: if args.clone is False: args.subparser.error("no spec provided") for name, entry in env.dev_specs.items(): path = entry.get("path", name) abspath = spack.util.path.canonicalize_path(path, default_wd=env.path) # Both old syntax `spack develop pkg@x` and new syntax `spack develop pkg@=x` # are currently supported. spec = spack.spec.parse_with_version_concrete(entry["spec"]) yield spec, abspath else: specs = spack.cmd.parse_specs(args.spec) if (args.path or args.build_directory) and len(specs) > 1: args.subparser.error( "requires at most one named spec when using the --path or --build-directory " "arguments" ) for spec in specs: if args.recursive: concrete_specs = env.all_matching_specs(spec) if not concrete_specs: tty.warn( f"{spec.name} has no matching concrete specs in the environment and " "will be skipped. `spack develop --recursive` requires a concretized" " environment" ) else: for s in concrete_specs: for node_spec in s.traverse(direction="parents", root=True): tty.debug(f"Recursive develop for {node_spec.name}") dev_spec = spack.spec.Spec(node_spec.format("{name}@{versions}")) yield dev_spec, _abs_code_path(env, node_spec, args.path) else: yield spec, _abs_code_path(env, spec, args.path)
[docs] def develop(parser, args): env = spack.cmd.require_active_env(args.subparser) for spec, abspath in _dev_spec_generator(args, env): assure_concrete_spec(env, spec) setup_src_code(spec, abspath, clone=args.clone, force=args.force) update_env(env, spec, args.path, args.build_directory, args.apply_changes)