Source code for spack.test.package_class

# Copyright 2013-2022 Lawrence Livermore National Security, LLC and other
# Spack Project Developers. See the top-level COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)

"""Test class methods on Package objects.

This doesn't include methods on package *instances* (like do_install(),
etc.).  Only methods like ``possible_dependencies()`` that deal with the
static DSL metadata for packages.
"""

import os
import shutil

import pytest

import llnl.util.filesystem as fs

import spack.package
import spack.repo


[docs]@pytest.fixture(scope="module") def mpi_names(mock_repo_path): return [spec.name for spec in mock_repo_path.providers_for('mpi')]
[docs]@pytest.fixture() def mpileaks_possible_deps(mock_packages, mpi_names): possible = { 'callpath': set(['dyninst'] + mpi_names), 'low-priority-provider': set(), 'dyninst': set(['libdwarf', 'libelf']), 'fake': set(), 'libdwarf': set(['libelf']), 'libelf': set(), 'mpich': set(), 'mpich2': set(), 'mpileaks': set(['callpath'] + mpi_names), 'multi-provider-mpi': set(), 'zmpi': set(['fake']), } return possible
[docs]def test_possible_dependencies(mock_packages, mpileaks_possible_deps): mpileaks = spack.repo.get('mpileaks') assert mpileaks_possible_deps == ( mpileaks.possible_dependencies(expand_virtuals=True)) assert { 'callpath': set(['dyninst', 'mpi']), 'dyninst': set(['libdwarf', 'libelf']), 'libdwarf': set(['libelf']), 'libelf': set(), 'mpi': set(), 'mpileaks': set(['callpath', 'mpi']), } == mpileaks.possible_dependencies(expand_virtuals=False)
[docs]def test_possible_direct_dependencies(mock_packages, mpileaks_possible_deps): mpileaks = spack.repo.get('mpileaks') deps = mpileaks.possible_dependencies(transitive=False, expand_virtuals=False) assert { 'callpath': set(), 'mpi': set(), 'mpileaks': set(['callpath', 'mpi']), } == deps
[docs]def test_possible_dependencies_virtual(mock_packages, mpi_names): expected = dict( (name, set(spack.repo.get(name).dependencies)) for name in mpi_names ) # only one mock MPI has a dependency expected['fake'] = set() assert expected == spack.package.possible_dependencies( "mpi", transitive=False)
[docs]def test_possible_dependencies_missing(mock_packages): md = spack.repo.get("missing-dependency") missing = {} md.possible_dependencies(transitive=True, missing=missing) assert set([ "this-is-a-missing-dependency" ]) == missing["missing-dependency"]
[docs]def test_possible_dependencies_with_deptypes(mock_packages): dtbuild1 = spack.repo.get('dtbuild1') assert { 'dtbuild1': set(['dtrun2', 'dtlink2']), 'dtlink2': set(), 'dtrun2': set(), } == dtbuild1.possible_dependencies(deptype=('link', 'run')) assert { 'dtbuild1': set(['dtbuild2', 'dtlink2']), 'dtbuild2': set(), 'dtlink2': set(), } == dtbuild1.possible_dependencies(deptype=('build')) assert { 'dtbuild1': set(['dtlink2']), 'dtlink2': set(), } == dtbuild1.possible_dependencies(deptype=('link'))
[docs]def test_possible_dependencies_with_multiple_classes( mock_packages, mpileaks_possible_deps): pkgs = ['dt-diamond', 'mpileaks'] expected = mpileaks_possible_deps.copy() expected.update({ 'dt-diamond': set(['dt-diamond-left', 'dt-diamond-right']), 'dt-diamond-left': set(['dt-diamond-bottom']), 'dt-diamond-right': set(['dt-diamond-bottom']), 'dt-diamond-bottom': set(), }) assert expected == spack.package.possible_dependencies(*pkgs)
[docs]def setup_install_test(source_paths, install_test_root): """ Set up the install test by creating sources and install test roots. The convention used here is to create an empty file if the path name ends with an extension otherwise, a directory is created. """ fs.mkdirp(install_test_root) for path in source_paths: if os.path.splitext(path)[1]: fs.touchp(path) else: fs.mkdirp(path)
[docs]@pytest.mark.parametrize('spec,sources,extras,expect', [ ('a', ['example/a.c'], # Source(s) ['example/a.c'], # Extra test source ['example/a.c']), # Test install dir source(s) ('b', ['test/b.cpp', 'test/b.hpp', 'example/b.txt'], # Source(s) ['test'], # Extra test source ['test/b.cpp', 'test/b.hpp']), # Test install dir source ('c', ['examples/a.py', 'examples/b.py', 'examples/c.py', 'tests/d.py'], ['examples/b.py', 'tests'], ['examples/b.py', 'tests/d.py']), ]) def test_cache_extra_sources(install_mockery, spec, sources, extras, expect): """Test the package's cache extra test sources helper function.""" pkg = spack.repo.get(spec) pkg.spec.concretize() source_path = pkg.stage.source_path srcs = [fs.join_path(source_path, s) for s in sources] setup_install_test(srcs, pkg.install_test_root) emsg_dir = 'Expected {0} to be a directory' emsg_file = 'Expected {0} to be a file' for s in srcs: assert os.path.exists(s), 'Expected {0} to exist'.format(s) if os.path.splitext(s)[1]: assert os.path.isfile(s), emsg_file.format(s) else: assert os.path.isdir(s), emsg_dir.format(s) pkg.cache_extra_test_sources(extras) src_dests = [fs.join_path(pkg.install_test_root, s) for s in sources] exp_dests = [fs.join_path(pkg.install_test_root, e) for e in expect] poss_dests = set(src_dests) | set(exp_dests) msg = 'Expected {0} to{1} exist' for pd in poss_dests: if pd in exp_dests: assert os.path.exists(pd), msg.format(pd, '') if os.path.splitext(pd)[1]: assert os.path.isfile(pd), emsg_file.format(pd) else: assert os.path.isdir(pd), emsg_dir.format(pd) else: assert not os.path.exists(pd), msg.format(pd, ' not') # Perform a little cleanup shutil.rmtree(os.path.dirname(source_path))