Skip to content

Commit d8f3bc2

Browse files
Code now works during FIPS mode.
Test suite currently failing with FIPS enabled due to hard-coded MD5 tests.
1 parent 86e06a8 commit d8f3bc2

File tree

4 files changed

+96
-9
lines changed

4 files changed

+96
-9
lines changed

.gitignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ venv.bak/
3434

3535

3636
# SCons files
37-
.sconsign.*
37+
.sconsign*
3838

3939
# Tool output
4040
.coverage

CHANGES.txt

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,12 @@ NOTE: The 4.2.0 Release of SCons will deprecate Python 3.5 Support. Python 3.5 s
99

1010
RELEASE VERSION/DATE TO BE FILLED IN LATER
1111

12+
From Jacob Cassagnol:
13+
- Default hash algorithm check updated for Scons FIPS compliance. Now checks for hash viability
14+
first and then walks the tree to use the first viable hash as the default one. This typically
15+
selects SHA1 on FIPS-enabled systems as the new default instead of MD5, unless SHA1 has also
16+
been disabled by security policy, at which point SCons selects SHA256 as the default.
17+
1218
From Joseph Brill:
1319
- Fix MSVS tests (vs-N.N-exec.py) for MSVS 6.0, 7.0, and 7.1 (import missing module).
1420
- Add support for Visual Studio 2022 (release and prerelease).

SCons/Util.py

+87-6
Original file line numberDiff line numberDiff line change
@@ -1669,11 +1669,58 @@ def AddMethod(obj, function, name=None):
16691669

16701670

16711671
# Default hash function and format. SCons-internal.
1672-
ALLOWED_HASH_FORMATS = ['md5', 'sha1', 'sha256']
1672+
_DEFAULT_HASH_FORMATS = ['md5', 'sha1', 'sha256']
1673+
ALLOWED_HASH_FORMATS = []
16731674
_HASH_FUNCTION = None
16741675
_HASH_FORMAT = None
16751676

16761677

1678+
def set_allowed_viable_default_hashes():
1679+
"""Checks if SCons has ability to call the default algorithms normally supported.
1680+
1681+
This util class is sometimes called prior to setting the user-selected hash algorithm,
1682+
meaning that on FIPS-compliant systems the library would default-initialize MD5
1683+
and throw an exception in set_hash_format. A common case is using the SConf options,
1684+
which can run prior to main, and thus ignore the options.hash_format variable.
1685+
1686+
This function checks the _DEFAULT_HASH_FORMATS and sets the ALLOWED_HASH_FORMATS
1687+
to only the ones that can be called.
1688+
1689+
Throws if no allowed hash formats are detected.
1690+
"""
1691+
global ALLOWED_HASH_FORMATS
1692+
_last_error = None
1693+
# note: if you call this method repeatedly, example using timeout, this is needed.
1694+
# otherwise it keeps appending valid formats to the string
1695+
ALLOWED_HASH_FORMATS = []
1696+
1697+
for test_algorithm in _DEFAULT_HASH_FORMATS:
1698+
_test_hash = getattr(hashlib, test_algorithm, None)
1699+
# we know hashlib claims to support it... check to see if we can call it.
1700+
if _test_hash is not None:
1701+
# the hashing library will throw an exception on initialization in FIPS mode,
1702+
# meaning if we call the default algorithm returned with no parameters, it'll
1703+
# throw if it's a bad algorithm, otherwise it will append it to the known
1704+
# good formats.
1705+
try:
1706+
_test_hash()
1707+
ALLOWED_HASH_FORMATS.append(test_algorithm)
1708+
except ValueError as e:
1709+
_last_error = e
1710+
continue
1711+
1712+
if len(ALLOWED_HASH_FORMATS) == 0:
1713+
from SCons.Errors import SConsEnvironmentError # pylint: disable=import-outside-toplevel
1714+
# chain the exception thrown with the most recent error from hashlib.
1715+
raise SConsEnvironmentError(
1716+
'No usable hash algorithms found.'
1717+
'Most recent error from hashlib attached in trace.'
1718+
) from _last_error
1719+
return
1720+
1721+
set_allowed_viable_default_hashes()
1722+
1723+
16771724
def get_hash_format():
16781725
"""Retrieves the hash format or ``None`` if not overridden.
16791726
@@ -1702,22 +1749,54 @@ def set_hash_format(hash_format):
17021749
if hash_format_lower not in ALLOWED_HASH_FORMATS:
17031750
from SCons.Errors import UserError # pylint: disable=import-outside-toplevel
17041751

1705-
raise UserError('Hash format "%s" is not supported by SCons. Only '
1752+
# user can select something not supported by their OS but normally supported by
1753+
# SCons, example, selecting MD5 in an OS with FIPS-mode turned on. Therefore we first
1754+
# check if SCons supports it, and then if their local OS supports it.
1755+
if hash_format_lower in _DEFAULT_HASH_FORMATS:
1756+
raise UserError('While hash format "%s" is supported by SCons, the '
1757+
'local system indicates only the following hash '
1758+
'formats are supported by the hashlib library: %s' %
1759+
(hash_format_lower,
1760+
', '.join(ALLOWED_HASH_FORMATS))
1761+
)
1762+
# the hash format isn't supported by SCons in any case. Warn the user, and
1763+
# if we detect that SCons supports more algorithms than their local system
1764+
# supports, warn the user about that too.
1765+
else:
1766+
if ALLOWED_HASH_FORMATS == _DEFAULT_HASH_FORMATS:
1767+
raise UserError('Hash format "%s" is not supported by SCons. Only '
17061768
'the following hash formats are supported: %s' %
17071769
(hash_format_lower,
1708-
', '.join(ALLOWED_HASH_FORMATS)))
1770+
', '.join(ALLOWED_HASH_FORMATS))
1771+
)
1772+
else:
1773+
raise UserError('Hash format "%s" is not supported by SCons. '
1774+
'SCons supports more hash formats than your local system '
1775+
'is reporting; SCons supports: %s. Your local system only '
1776+
'supports: %s' %
1777+
(hash_format_lower,
1778+
', '.join(_DEFAULT_HASH_FORMATS),
1779+
', '.join(ALLOWED_HASH_FORMATS))
1780+
)
17091781

1782+
# this is not expected to fail. If this fails it means the set_allowed_viable_default_hashes
1783+
# function did not throw, or when it threw, the exception was caught and ignored, or
1784+
# the global ALLOWED_HASH_FORMATS was changed by an external user.
17101785
_HASH_FUNCTION = getattr(hashlib, hash_format_lower, None)
17111786
if _HASH_FUNCTION is None:
17121787
from SCons.Errors import UserError # pylint: disable=import-outside-toplevel
17131788

17141789
raise UserError(
1715-
'Hash format "%s" is not available in your Python interpreter.'
1790+
'Hash format "%s" is not available in your Python interpreter. '
1791+
'Expected to be supported algorithm by set_allowed_viable_default_hashes, '
1792+
'Assertion error in SCons.'
17161793
% hash_format_lower
17171794
)
17181795
else:
17191796
# Set the default hash format based on what is available, defaulting
1720-
# to md5 for backwards compatibility.
1797+
# to the first supported hash algorithm (usually md5) for backwards compatibility.
1798+
# in FIPS-compliant systems this usually defaults to SHA1, unless that too has been
1799+
# disabled.
17211800
for choice in ALLOWED_HASH_FORMATS:
17221801
_HASH_FUNCTION = getattr(hashlib, choice, None)
17231802
if _HASH_FUNCTION is not None:
@@ -1728,7 +1807,9 @@ def set_hash_format(hash_format):
17281807

17291808
raise UserError(
17301809
'Your Python interpreter does not have MD5, SHA1, or SHA256. '
1731-
'SCons requires at least one.')
1810+
'SCons requires at least one. Expected to support one or more '
1811+
'during set_allowed_viable_default_hashes.'
1812+
)
17321813

17331814
# Ensure that this is initialized in case either:
17341815
# 1. This code is running in a unit test.

doc/man/scons.xml

+2-2
Original file line numberDiff line numberDiff line change
@@ -1185,8 +1185,8 @@ to use the specified algorithm.</para>
11851185
For example, <option>--hash-format=sha256</option> will create a SConsign
11861186
database with name <filename>.sconsign_sha256.dblite</filename>.</para>
11871187

1188-
<para>If this option is not specified, a hash format of
1189-
md5 is used, and the SConsign database is
1188+
<para>If this option is not specified, a the first supported hash format found
1189+
is selected, and the SConsign database is
11901190
<filename>.sconsign.dblite</filename>.</para>
11911191

11921192
<para><emphasis>Available since &scons; 4.2.</emphasis></para>

0 commit comments

Comments
 (0)