Source code for windows.crypto.encrypt_decrypt

from os import urandom

from windows import winproxy
from windows.crypto import DEFAULT_ENCODING
from windows.generated_def import *

__all__ = ["encrypt", "decrypt"]

def encode_init_vector(data):
    blob = CRYPT_DATA_BLOB.from_string(data)
    size = DWORD()
    buf = None
    winproxy.CryptEncodeObjectEx(DEFAULT_ENCODING, X509_OCTET_STRING, ctypes.byref(blob), 0, None, buf, size)
    buf = (BYTE * size.value)()
    winproxy.CryptEncodeObjectEx(DEFAULT_ENCODING, X509_OCTET_STRING, ctypes.byref(blob), 0, None, buf, size)
    return buf[:]


class GenerateInitVector(object):
    def __repr__(self):
        return "GenerateInitVector()"

    def generate_init_vector(self, algo):
        if algo in [szOID_OIWSEC_desCBC, szOID_RSA_DES_EDE3_CBC]:
            return urandom(8)
        if algo in [szOID_NIST_AES128_CBC, szOID_NIST_AES192_CBC, szOID_NIST_AES256_CBC]:
            return urandom(16)
        return None
geninitvector = GenerateInitVector()


[docs]def encrypt(cert_or_certlist, msg, algo=szOID_NIST_AES256_CBC, initvector=geninitvector): """Encrypt ``msg`` one or many :class:`Certificate` using ``algo`` with the initial vector ``initvector``. If ``geninitvector`` is left as it is, it will generate a random one. Algorithms supported by ``GenerateInitVector`` are: * ``szOID_OIWSEC_desCBC`` * ``szOID_RSA_DES_EDE3_CBC`` * ``szOID_NIST_AES128_CBC`` * ``szOID_NIST_AES192_CBC`` * ``szOID_NIST_AES256_CBC`` :param cert_or_certlist: One or many :class:`Certificate` used to encrypt the msg :type cert_or_certlist: :class:`Certificate` | [:class:`Certificate`] :return: :class:`bytearray`: The encrypted message """ alg_ident = CRYPT_ALGORITHM_IDENTIFIER() alg_ident.pszObjId = algo.encode("ascii") # We want to have automatique translation of Certificate -> PCERT_CONTEXT # In order to simple create the 'PCERT_CONTEXT[] certs' # For that we need a tuple of X * 1-item-tuple # as a (cert,) will be automaticly translatable to a PCERT_CONTEXT if isinstance(cert_or_certlist, CERT_CONTEXT): certlist = ((cert_or_certlist,),) else: certlist = tuple((c,) for c in cert_or_certlist) # Set (compute if needed) the IV if initvector is None: alg_ident.Parameters.cbData = 0 elif initvector is geninitvector: initvector = initvector.generate_init_vector(algo) if initvector is None: raise ValueError("I don't know how to generate an <initvector> for <{0}> please provide one (or None)".format(algo)) initvector_encoded = encode_init_vector(initvector) alg_ident.Parameters = CRYPT_DATA_BLOB.from_string(initvector_encoded) else: initvector_encoded = encode_init_vector(initvector) alg_ident.Parameters = CRYPT_DATA_BLOB.from_string(initvector_encoded) # Setup encryption parameters param = CRYPT_ENCRYPT_MESSAGE_PARA() param.cbSize = ctypes.sizeof(param) param.dwMsgEncodingType = DEFAULT_ENCODING param.hCryptProv = None param.ContentEncryptionAlgorithm = alg_ident param.pvEncryptionAuxInfo = None param.dwFlags = 0 param.dwInnerContentType = 0 certs = (PCERT_CONTEXT * len(certlist))(*certlist) #Ask the output buffer size size = DWORD() winproxy.CryptEncryptMessage(param, len(certs), certs, msg, len(msg), None, size) #Encrypt the msg buf = (BYTE * size.value)() winproxy.CryptEncryptMessage(param, len(certs), certs, msg, len(msg), buf, size) return bytearray(buf[:size.value])
[docs]def decrypt(cert_store, encrypted): """Try to decrypt the ``encrypted`` msg with any certificate in ``cert_store``. If there is no certificate able to decrypt the message ``WinproxyError(winerror=0x8009200c)`` is raised. :param cert_store: :type cert_store: :class:`CertificateStore` :return: :class:`str`: The decrypted message """ # Setup decryption parameters dparam = CRYPT_DECRYPT_MESSAGE_PARA() dparam.cbSize = ctypes.sizeof(dparam) dparam.dwMsgAndCertEncodingType = DEFAULT_ENCODING dparam.cCertStore = 1 dparam.rghCertStore = (cert_store,) dparam.dwFlags = 0 #Ask the output buffer size buf = (BYTE * len(encrypted)).from_buffer_copy(encrypted) dcryptsize = DWORD() winproxy.CryptDecryptMessage(dparam, buf, ctypes.sizeof(buf), None, dcryptsize, None) #Decrypt the msg dcryptbuff = (BYTE * (dcryptsize.value + 0x1000))() winproxy.CryptDecryptMessage(dparam, buf, ctypes.sizeof(buf), dcryptbuff, dcryptsize, None) return bytes(bytearray(dcryptbuff[:dcryptsize.value]))