crypt.py 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162
  1. """Wrapper to the POSIX crypt library call and associated functionality."""
  2. import _crypt
  3. import string as _string
  4. from random import SystemRandom as _SystemRandom
  5. from collections import namedtuple as _namedtuple
  6. _saltchars = _string.ascii_letters + _string.digits + './'
  7. _sr = _SystemRandom()
  8. class _Method(_namedtuple('_Method', 'name ident salt_chars total_size')):
  9. """Class representing a salt method per the Modular Crypt Format or the
  10. legacy 2-character crypt method."""
  11. def __repr__(self):
  12. return '<crypt.METHOD_{}>'.format(self.name)
  13. def mksalt(method=None):
  14. """Generate a salt for the specified method.
  15. If not specified, the strongest available method will be used.
  16. """
  17. if method is None:
  18. method = methods[0]
  19. s = '${}$'.format(method.ident) if method.ident else ''
  20. s += ''.join(_sr.choice(_saltchars) for char in range(method.salt_chars))
  21. return s
  22. def crypt(word, salt=None):
  23. """Return a string representing the one-way hash of a password, with a salt
  24. prepended.
  25. If ``salt`` is not specified or is ``None``, the strongest
  26. available method will be selected and a salt generated. Otherwise,
  27. ``salt`` may be one of the ``crypt.METHOD_*`` values, or a string as
  28. returned by ``crypt.mksalt()``.
  29. """
  30. if salt is None or isinstance(salt, _Method):
  31. salt = mksalt(salt)
  32. return _crypt.crypt(word, salt)
  33. # available salting/crypto methods
  34. METHOD_CRYPT = _Method('CRYPT', None, 2, 13)
  35. METHOD_MD5 = _Method('MD5', '1', 8, 34)
  36. METHOD_SHA256 = _Method('SHA256', '5', 16, 63)
  37. METHOD_SHA512 = _Method('SHA512', '6', 16, 106)
  38. methods = []
  39. for _method in (METHOD_SHA512, METHOD_SHA256, METHOD_MD5):
  40. _result = crypt('', _method)
  41. if _result and len(_result) == _method.total_size:
  42. methods.append(_method)
  43. methods.append(METHOD_CRYPT)
  44. del _result, _method