Fix checksumming in Python3; add more fetch tests (#3941)
* Checksum code wasn't opening binary files as binary. - Fixes Python 3 issue where files are opened as unicode text by default, and decoding fails for binary blobs. * Simplify fetch test parametrization. * - add tests for URL fetching and checksumming. - fix coverage on interface functions in FetchStrategy superclass - add some extra crypto tests.
This commit is contained in:
parent
c0356182b6
commit
63c3410370
@ -103,27 +103,42 @@ def set_stage(self, stage):
|
|||||||
|
|
||||||
# Subclasses need to implement these methods
|
# Subclasses need to implement these methods
|
||||||
def fetch(self):
|
def fetch(self):
|
||||||
pass # Return True on success, False on fail.
|
"""Fetch source code archive or repo.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: True on success, False on failure.
|
||||||
|
"""
|
||||||
|
|
||||||
def check(self):
|
def check(self):
|
||||||
pass # Do checksum.
|
"""Checksum the archive fetched by this FetchStrategy."""
|
||||||
|
|
||||||
def expand(self):
|
def expand(self):
|
||||||
pass # Expand archive.
|
"""Expand the downloaded archive."""
|
||||||
|
|
||||||
def reset(self):
|
def reset(self):
|
||||||
pass # Revert to freshly downloaded state.
|
"""Revert to freshly downloaded state.
|
||||||
|
|
||||||
|
For archive files, this may just re-expand the archive.
|
||||||
|
"""
|
||||||
|
|
||||||
def archive(self, destination):
|
def archive(self, destination):
|
||||||
pass # Used to create tarball for mirror.
|
"""Create an archive of the downloaded data for a mirror.
|
||||||
|
|
||||||
|
For downloaded files, this should preserve the checksum of the
|
||||||
|
original file. For repositories, it should just create an
|
||||||
|
expandable tarball out of the downloaded repository.
|
||||||
|
"""
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def cachable(self):
|
def cachable(self):
|
||||||
"""Return whether the fetcher is capable of caching the
|
"""Whether fetcher is capable of caching the resource it retrieves.
|
||||||
resource it retrieves. This generally is determined by
|
|
||||||
whether the resource is identifiably associated with a
|
This generally is determined by whether the resource is
|
||||||
specific package version."""
|
identifiably associated with a specific package version.
|
||||||
pass
|
|
||||||
|
Returns:
|
||||||
|
bool: True if can cache, False otherwise.
|
||||||
|
"""
|
||||||
|
|
||||||
def __str__(self): # Should be human readable URL.
|
def __str__(self): # Should be human readable URL.
|
||||||
return "FetchStrategy.__str___"
|
return "FetchStrategy.__str___"
|
||||||
@ -162,7 +177,8 @@ def __init__(self, url=None, digest=None, **kwargs):
|
|||||||
if not self.url:
|
if not self.url:
|
||||||
self.url = url
|
self.url = url
|
||||||
|
|
||||||
self.digest = kwargs.get('md5', None)
|
self.digest = next((kwargs[h] for h in crypto.hashes if h in kwargs),
|
||||||
|
None)
|
||||||
if not self.digest:
|
if not self.digest:
|
||||||
self.digest = digest
|
self.digest = digest
|
||||||
|
|
||||||
|
@ -300,6 +300,7 @@ def mock_archive():
|
|||||||
repo_name = 'mock-archive-repo'
|
repo_name = 'mock-archive-repo'
|
||||||
tmpdir.ensure(repo_name, dir=True)
|
tmpdir.ensure(repo_name, dir=True)
|
||||||
repodir = tmpdir.join(repo_name)
|
repodir = tmpdir.join(repo_name)
|
||||||
|
|
||||||
# Create the configure script
|
# Create the configure script
|
||||||
configure_path = str(tmpdir.join(repo_name, 'configure'))
|
configure_path = str(tmpdir.join(repo_name, 'configure'))
|
||||||
with open(configure_path, 'w') as f:
|
with open(configure_path, 'w') as f:
|
||||||
@ -315,15 +316,21 @@ def mock_archive():
|
|||||||
"EOF\n"
|
"EOF\n"
|
||||||
)
|
)
|
||||||
os.chmod(configure_path, 0o755)
|
os.chmod(configure_path, 0o755)
|
||||||
|
|
||||||
# Archive it
|
# Archive it
|
||||||
current = tmpdir.chdir()
|
current = tmpdir.chdir()
|
||||||
archive_name = '{0}.tar.gz'.format(repo_name)
|
archive_name = '{0}.tar.gz'.format(repo_name)
|
||||||
tar('-czf', archive_name, repo_name)
|
tar('-czf', archive_name, repo_name)
|
||||||
current.chdir()
|
current.chdir()
|
||||||
Archive = collections.namedtuple('Archive', ['url', 'path'])
|
Archive = collections.namedtuple('Archive',
|
||||||
url = 'file://' + str(tmpdir.join(archive_name))
|
['url', 'path', 'archive_file'])
|
||||||
|
archive_file = str(tmpdir.join(archive_name))
|
||||||
|
|
||||||
# Return the url
|
# Return the url
|
||||||
yield Archive(url=url, path=str(repodir))
|
yield Archive(
|
||||||
|
url=('file://' + archive_file),
|
||||||
|
archive_file=archive_file,
|
||||||
|
path=str(repodir))
|
||||||
stage.destroy()
|
stage.destroy()
|
||||||
|
|
||||||
|
|
||||||
|
@ -31,18 +31,8 @@
|
|||||||
from spack.version import ver
|
from spack.version import ver
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=['master', 'branch', 'tag', 'commit'])
|
@pytest.mark.parametrize("type_of_test", ['master', 'branch', 'tag', 'commit'])
|
||||||
def type_of_test(request):
|
@pytest.mark.parametrize("secure", [True, False])
|
||||||
"""Returns one of the test type available for the mock_git_repository"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=[True, False])
|
|
||||||
def secure(request):
|
|
||||||
"""Attempt both secure and insecure fetching"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
def test_fetch(
|
def test_fetch(
|
||||||
type_of_test,
|
type_of_test,
|
||||||
secure,
|
secure,
|
||||||
@ -62,11 +52,13 @@ def test_fetch(
|
|||||||
# Retrieve the right test parameters
|
# Retrieve the right test parameters
|
||||||
t = mock_git_repository.checks[type_of_test]
|
t = mock_git_repository.checks[type_of_test]
|
||||||
h = mock_git_repository.hash
|
h = mock_git_repository.hash
|
||||||
|
|
||||||
# Construct the package under test
|
# Construct the package under test
|
||||||
spec = Spec('git-test')
|
spec = Spec('git-test')
|
||||||
spec.concretize()
|
spec.concretize()
|
||||||
pkg = spack.repo.get(spec, new=True)
|
pkg = spack.repo.get(spec, new=True)
|
||||||
pkg.versions[ver('git')] = t.args
|
pkg.versions[ver('git')] = t.args
|
||||||
|
|
||||||
# Enter the stage directory and check some properties
|
# Enter the stage directory and check some properties
|
||||||
with pkg.stage:
|
with pkg.stage:
|
||||||
try:
|
try:
|
||||||
|
@ -31,18 +31,8 @@
|
|||||||
from spack.version import ver
|
from spack.version import ver
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=['default', 'rev0'])
|
@pytest.mark.parametrize("type_of_test", ['default', 'rev0'])
|
||||||
def type_of_test(request):
|
@pytest.mark.parametrize("secure", [True, False])
|
||||||
"""Returns one of the test type available for the mock_hg_repository"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=[True, False])
|
|
||||||
def secure(request):
|
|
||||||
"""Attempt both secure and insecure fetching"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
def test_fetch(
|
def test_fetch(
|
||||||
type_of_test,
|
type_of_test,
|
||||||
secure,
|
secure,
|
||||||
@ -62,11 +52,13 @@ def test_fetch(
|
|||||||
# Retrieve the right test parameters
|
# Retrieve the right test parameters
|
||||||
t = mock_hg_repository.checks[type_of_test]
|
t = mock_hg_repository.checks[type_of_test]
|
||||||
h = mock_hg_repository.hash
|
h = mock_hg_repository.hash
|
||||||
|
|
||||||
# Construct the package under test
|
# Construct the package under test
|
||||||
spec = Spec('hg-test')
|
spec = Spec('hg-test')
|
||||||
spec.concretize()
|
spec.concretize()
|
||||||
pkg = spack.repo.get(spec, new=True)
|
pkg = spack.repo.get(spec, new=True)
|
||||||
pkg.versions[ver('hg')] = t.args
|
pkg.versions[ver('hg')] = t.args
|
||||||
|
|
||||||
# Enter the stage directory and check some properties
|
# Enter the stage directory and check some properties
|
||||||
with pkg.stage:
|
with pkg.stage:
|
||||||
try:
|
try:
|
||||||
|
@ -31,18 +31,8 @@
|
|||||||
from spack.version import ver
|
from spack.version import ver
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=['default', 'rev0'])
|
@pytest.mark.parametrize("type_of_test", ['default', 'rev0'])
|
||||||
def type_of_test(request):
|
@pytest.mark.parametrize("secure", [True, False])
|
||||||
"""Returns one of the test type available for the mock_svn_repository"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=[True, False])
|
|
||||||
def secure(request):
|
|
||||||
"""Attempt both secure and insecure fetching"""
|
|
||||||
return request.param
|
|
||||||
|
|
||||||
|
|
||||||
def test_fetch(
|
def test_fetch(
|
||||||
type_of_test,
|
type_of_test,
|
||||||
secure,
|
secure,
|
||||||
@ -62,11 +52,13 @@ def test_fetch(
|
|||||||
# Retrieve the right test parameters
|
# Retrieve the right test parameters
|
||||||
t = mock_svn_repository.checks[type_of_test]
|
t = mock_svn_repository.checks[type_of_test]
|
||||||
h = mock_svn_repository.hash
|
h = mock_svn_repository.hash
|
||||||
|
|
||||||
# Construct the package under test
|
# Construct the package under test
|
||||||
spec = Spec('svn-test')
|
spec = Spec('svn-test')
|
||||||
spec.concretize()
|
spec.concretize()
|
||||||
pkg = spack.repo.get(spec, new=True)
|
pkg = spack.repo.get(spec, new=True)
|
||||||
pkg.versions[ver('svn')] = t.args
|
pkg.versions[ver('svn')] = t.args
|
||||||
|
|
||||||
# Enter the stage directory and check some properties
|
# Enter the stage directory and check some properties
|
||||||
with pkg.stage:
|
with pkg.stage:
|
||||||
try:
|
try:
|
||||||
|
93
lib/spack/spack/test/url_fetch.py
Normal file
93
lib/spack/spack/test/url_fetch.py
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
##############################################################################
|
||||||
|
# Copyright (c) 2013-2016, Lawrence Livermore National Security, LLC.
|
||||||
|
# Produced at the Lawrence Livermore National Laboratory.
|
||||||
|
#
|
||||||
|
# This file is part of Spack.
|
||||||
|
# Created by Todd Gamblin, tgamblin@llnl.gov, All rights reserved.
|
||||||
|
# LLNL-CODE-647188
|
||||||
|
#
|
||||||
|
# For details, see https://github.com/llnl/spack
|
||||||
|
# Please also see the LICENSE file for our notice and the LGPL.
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Lesser General Public License (as
|
||||||
|
# published by the Free Software Foundation) version 2.1, February 1999.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful, but
|
||||||
|
# WITHOUT ANY WARRANTY; without even the IMPLIED WARRANTY OF
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the terms and
|
||||||
|
# conditions of the GNU Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public
|
||||||
|
# License along with this program; if not, write to the Free Software
|
||||||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
##############################################################################
|
||||||
|
import os
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from llnl.util.filesystem import *
|
||||||
|
|
||||||
|
import spack
|
||||||
|
from spack.spec import Spec
|
||||||
|
from spack.version import ver
|
||||||
|
import spack.util.crypto as crypto
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(params=list(crypto.hashes.keys()))
|
||||||
|
def checksum_type(request):
|
||||||
|
return request.param
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('secure', [True, False])
|
||||||
|
def test_fetch(
|
||||||
|
mock_archive,
|
||||||
|
secure,
|
||||||
|
checksum_type,
|
||||||
|
config,
|
||||||
|
refresh_builtin_mock
|
||||||
|
):
|
||||||
|
"""Fetch an archive and make sure we can checksum it."""
|
||||||
|
mock_archive.url
|
||||||
|
mock_archive.path
|
||||||
|
|
||||||
|
algo = crypto.hashes[checksum_type]()
|
||||||
|
with open(mock_archive.archive_file, 'rb') as f:
|
||||||
|
algo.update(f.read())
|
||||||
|
checksum = algo.hexdigest()
|
||||||
|
|
||||||
|
# Get a spec and tweak the test package with new chcecksum params
|
||||||
|
spec = Spec('url-test')
|
||||||
|
spec.concretize()
|
||||||
|
|
||||||
|
pkg = spack.repo.get('url-test', new=True)
|
||||||
|
pkg.url = mock_archive.url
|
||||||
|
pkg.versions[ver('test')] = {checksum_type: checksum, 'url': pkg.url}
|
||||||
|
pkg.spec = spec
|
||||||
|
|
||||||
|
# Enter the stage directory and check some properties
|
||||||
|
with pkg.stage:
|
||||||
|
try:
|
||||||
|
spack.insecure = secure
|
||||||
|
pkg.do_stage()
|
||||||
|
finally:
|
||||||
|
spack.insecure = False
|
||||||
|
|
||||||
|
assert os.path.exists('configure')
|
||||||
|
assert is_exe('configure')
|
||||||
|
|
||||||
|
with open('configure') as f:
|
||||||
|
contents = f.read()
|
||||||
|
assert contents.startswith('#!/bin/sh')
|
||||||
|
assert 'echo Building...' in contents
|
||||||
|
|
||||||
|
|
||||||
|
def test_hash_detection(checksum_type):
|
||||||
|
algo = crypto.hashes[checksum_type]()
|
||||||
|
h = 'f' * (algo.digest_size * 2) # hex -> bytes
|
||||||
|
checker = crypto.Checker(h)
|
||||||
|
assert checker.hash_name == checksum_type
|
||||||
|
|
||||||
|
|
||||||
|
def test_unknown_hash(checksum_type):
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
crypto.Checker('a')
|
@ -26,16 +26,16 @@
|
|||||||
import hashlib
|
import hashlib
|
||||||
|
|
||||||
"""Set of acceptable hashes that Spack will use."""
|
"""Set of acceptable hashes that Spack will use."""
|
||||||
_acceptable_hashes = [
|
hashes = dict((h, getattr(hashlib, h)) for h in [
|
||||||
hashlib.md5,
|
'md5',
|
||||||
hashlib.sha1,
|
'sha1',
|
||||||
hashlib.sha224,
|
'sha224',
|
||||||
hashlib.sha256,
|
'sha256',
|
||||||
hashlib.sha384,
|
'sha384',
|
||||||
hashlib.sha512]
|
'sha512'])
|
||||||
|
|
||||||
"""Index for looking up hasher for a digest."""
|
"""Index for looking up hasher for a digest."""
|
||||||
_size_to_hash = dict((h().digest_size, h) for h in _acceptable_hashes)
|
_size_to_hash = dict((h().digest_size, h) for h in hashes.values())
|
||||||
|
|
||||||
|
|
||||||
def checksum(hashlib_algo, filename, **kwargs):
|
def checksum(hashlib_algo, filename, **kwargs):
|
||||||
@ -44,7 +44,7 @@ def checksum(hashlib_algo, filename, **kwargs):
|
|||||||
"""
|
"""
|
||||||
block_size = kwargs.get('block_size', 2**20)
|
block_size = kwargs.get('block_size', 2**20)
|
||||||
hasher = hashlib_algo()
|
hasher = hashlib_algo()
|
||||||
with open(filename) as file:
|
with open(filename, 'rb') as file:
|
||||||
while True:
|
while True:
|
||||||
data = file.read(block_size)
|
data = file.read(block_size)
|
||||||
if not data:
|
if not data:
|
||||||
|
35
var/spack/repos/builtin.mock/packages/url-test/package.py
Normal file
35
var/spack/repos/builtin.mock/packages/url-test/package.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
##############################################################################
|
||||||
|
# Copyright (c) 2013-2017, Lawrence Livermore National Security, LLC.
|
||||||
|
# Produced at the Lawrence Livermore National Laboratory.
|
||||||
|
#
|
||||||
|
# This file is part of Spack.
|
||||||
|
# Created by Todd Gamblin, tgamblin@llnl.gov, All rights reserved.
|
||||||
|
# LLNL-CODE-647188
|
||||||
|
#
|
||||||
|
# For details, see https://github.com/llnl/spack
|
||||||
|
# Please also see the LICENSE file for our notice and the LGPL.
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Lesser General Public License (as
|
||||||
|
# published by the Free Software Foundation) version 2.1, February 1999.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful, but
|
||||||
|
# WITHOUT ANY WARRANTY; without even the IMPLIED WARRANTY OF
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the terms and
|
||||||
|
# conditions of the GNU Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public
|
||||||
|
# License along with this program; if not, write to the Free Software
|
||||||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
##############################################################################
|
||||||
|
from spack import *
|
||||||
|
|
||||||
|
|
||||||
|
class UrlTest(Package):
|
||||||
|
"""Mock package that fetches from a URL."""
|
||||||
|
homepage = "http://www.url-fetch-example.com"
|
||||||
|
|
||||||
|
version('test', url='to-be-filled-in-by-test')
|
||||||
|
|
||||||
|
def install(self, spec, prefix):
|
||||||
|
pass
|
Loading…
Reference in New Issue
Block a user