@@ -1669,11 +1669,58 @@ def AddMethod(obj, function, name=None):
1669
1669
1670
1670
1671
1671
# 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 = []
1673
1674
_HASH_FUNCTION = None
1674
1675
_HASH_FORMAT = None
1675
1676
1676
1677
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
+
1677
1724
def get_hash_format ():
1678
1725
"""Retrieves the hash format or ``None`` if not overridden.
1679
1726
@@ -1702,22 +1749,54 @@ def set_hash_format(hash_format):
1702
1749
if hash_format_lower not in ALLOWED_HASH_FORMATS :
1703
1750
from SCons .Errors import UserError # pylint: disable=import-outside-toplevel
1704
1751
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 '
1706
1768
'the following hash formats are supported: %s' %
1707
1769
(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
+ )
1709
1781
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.
1710
1785
_HASH_FUNCTION = getattr (hashlib , hash_format_lower , None )
1711
1786
if _HASH_FUNCTION is None :
1712
1787
from SCons .Errors import UserError # pylint: disable=import-outside-toplevel
1713
1788
1714
1789
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.'
1716
1793
% hash_format_lower
1717
1794
)
1718
1795
else :
1719
1796
# 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.
1721
1800
for choice in ALLOWED_HASH_FORMATS :
1722
1801
_HASH_FUNCTION = getattr (hashlib , choice , None )
1723
1802
if _HASH_FUNCTION is not None :
@@ -1728,7 +1807,9 @@ def set_hash_format(hash_format):
1728
1807
1729
1808
raise UserError (
1730
1809
'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
+ )
1732
1813
1733
1814
# Ensure that this is initialized in case either:
1734
1815
# 1. This code is running in a unit test.
0 commit comments