diff options
author | David Reid <dreid@dreid.org> | 2013-10-24 11:33:36 -0700 |
---|---|---|
committer | David Reid <dreid@dreid.org> | 2013-10-24 14:31:59 -0700 |
commit | 9188b01511849d3731e838decfcaece475d5d7ba (patch) | |
tree | db61e92d8321db441e72939dca15e67864133c0e | |
parent | 9f611ee88c02a4a4f407854e9589bb446ed1ec06 (diff) | |
download | cryptography-9188b01511849d3731e838decfcaece475d5d7ba.tar.gz cryptography-9188b01511849d3731e838decfcaece475d5d7ba.tar.bz2 cryptography-9188b01511849d3731e838decfcaece475d5d7ba.zip |
Initial refactoring of backend specific contexts.
-rw-r--r-- | cryptography/bindings/openssl/backend.py | 103 | ||||
-rw-r--r-- | cryptography/primitives/block/base.py | 38 |
2 files changed, 65 insertions, 76 deletions
diff --git a/cryptography/bindings/openssl/backend.py b/cryptography/bindings/openssl/backend.py index 6ddceb5a..99124e1c 100644 --- a/cryptography/bindings/openssl/backend.py +++ b/cryptography/bindings/openssl/backend.py @@ -111,10 +111,59 @@ class GetCipherByName(object): return backend.lib.EVP_get_cipherbyname(cipher_name.encode("ascii")) -class Ciphers(object): - OPENSSL_ENCRYPT = 1 - OPENSSL_DECRYPT = 0 +class _CipherContext(object): + _ENCRYPT = 1 + _DECRYPT = 0 + + def __init__(self, backend, cipher, mode, operation): + self._backend = backend + + ctx = self._backend.lib.EVP_CIPHER_CTX_new() + ctx = self._backend.ffi.gc(ctx, self._backend.lib.EVP_CIPHER_CTX_free) + + registry = self._backend.ciphers._cipher_registry + evp_cipher = registry[type(cipher), type(mode)]( + self._backend, cipher, mode + ) + assert evp_cipher != self._backend.ffi.NULL + if isinstance(mode, interfaces.ModeWithInitializationVector): + iv_nonce = mode.initialization_vector + elif isinstance(mode, interfaces.ModeWithNonce): + iv_nonce = mode.nonce + else: + iv_nonce = self._backend.ffi.NULL + res = self._backend.lib.EVP_CipherInit_ex(ctx, evp_cipher, + self._backend.ffi.NULL, + cipher.key, iv_nonce, + operation) + assert res != 0 + # We purposely disable padding here as it's handled higher up in the + # API. + self._backend.lib.EVP_CIPHER_CTX_set_padding(ctx, 0) + self._ctx = ctx + + def update(self, data): + block_size = self._backend.lib.EVP_CIPHER_CTX_block_size(self._ctx) + buf = self._backend.ffi.new("unsigned char[]", + len(data) + block_size - 1) + outlen = self._backend.ffi.new("int *") + res = self._backend.lib.EVP_CipherUpdate(self._ctx, buf, outlen, data, + len(data)) + assert res != 0 + return self._backend.ffi.buffer(buf)[:outlen[0]] + + def finalize(self): + block_size = self._backend.lib.EVP_CIPHER_CTX_block_size(self._ctx) + buf = self._backend.ffi.new("unsigned char[]", block_size) + outlen = self._backend.ffi.new("int *") + res = self._backend.lib.EVP_CipherFinal_ex(self._ctx, buf, outlen) + assert res != 0 + res = self._backend.lib.EVP_CIPHER_CTX_cleanup(self._ctx) + assert res == 1 + return self._backend.ffi.buffer(buf)[:outlen[0]] + +class Ciphers(object): def __init__(self, backend): super(Ciphers, self).__init__() self._backend = backend @@ -154,52 +203,12 @@ class Ciphers(object): ) def create_encrypt_ctx(self, cipher, mode): - return self._create_ctx(cipher, mode, self.OPENSSL_ENCRYPT) + return _CipherContext(self._backend, cipher, mode, + _CipherContext._ENCRYPT) def create_decrypt_ctx(self, cipher, mode): - return self._create_ctx(cipher, mode, self.OPENSSL_DECRYPT) - - def _create_ctx(self, cipher, mode, enc): - ctx = self._backend.lib.EVP_CIPHER_CTX_new() - ctx = self._backend.ffi.gc(ctx, self._backend.lib.EVP_CIPHER_CTX_free) - evp_cipher = self._cipher_registry[type(cipher), type(mode)]( - self._backend, cipher, mode - ) - assert evp_cipher != self._backend.ffi.NULL - if isinstance(mode, interfaces.ModeWithInitializationVector): - iv_nonce = mode.initialization_vector - elif isinstance(mode, interfaces.ModeWithNonce): - iv_nonce = mode.nonce - else: - iv_nonce = self._backend.ffi.NULL - res = self._backend.lib.EVP_CipherInit_ex(ctx, evp_cipher, - self._backend.ffi.NULL, - cipher.key, iv_nonce, enc) - assert res != 0 - # We purposely disable padding here as it's handled higher up in the - # API. - self._backend.lib.EVP_CIPHER_CTX_set_padding(ctx, 0) - return ctx - - def update_ctx(self, ctx, data): - block_size = self._backend.lib.EVP_CIPHER_CTX_block_size(ctx) - buf = self._backend.ffi.new("unsigned char[]", - len(data) + block_size - 1) - outlen = self._backend.ffi.new("int *") - res = self._backend.lib.EVP_CipherUpdate(ctx, buf, outlen, data, - len(data)) - assert res != 0 - return self._backend.ffi.buffer(buf)[:outlen[0]] - - def finalize_ctx(self, ctx): - block_size = self._backend.lib.EVP_CIPHER_CTX_block_size(ctx) - buf = self._backend.ffi.new("unsigned char[]", block_size) - outlen = self._backend.ffi.new("int *") - res = self._backend.lib.EVP_CipherFinal_ex(ctx, buf, outlen) - assert res != 0 - res = self._backend.lib.EVP_CIPHER_CTX_cleanup(ctx) - assert res == 1 - return self._backend.ffi.buffer(buf)[:outlen[0]] + return _CipherContext(self._backend, cipher, mode, + _CipherContext._DECRYPT) class Hashes(object): diff --git a/cryptography/primitives/block/base.py b/cryptography/primitives/block/base.py index 2d4623a8..e7559143 100644 --- a/cryptography/primitives/block/base.py +++ b/cryptography/primitives/block/base.py @@ -28,47 +28,27 @@ class BlockCipher(object): self._backend = backend def encryptor(self): - return _CipherEncryptionContext(self.cipher, self.mode, self._backend) + return _OneTimeCipherContext( + self._backend.ciphers.create_encrypt_ctx(self.cipher, self.mode)) def decryptor(self): - return _CipherDecryptionContext(self.cipher, self.mode, self._backend) + return _OneTimeCipherContext( + self._backend.ciphers.create_decrypt_ctx(self.cipher, self.mode)) @interfaces.register(interfaces.CipherContext) -class _CipherEncryptionContext(object): - def __init__(self, cipher, mode, backend): - super(_CipherEncryptionContext, self).__init__() - self._backend = backend - self._ctx = self._backend.ciphers.create_encrypt_ctx(cipher, mode) - - def update(self, data): - if self._ctx is None: - raise ValueError("Context was already finalized") - return self._backend.ciphers.update_ctx(self._ctx, data) - - def finalize(self): - if self._ctx is None: - raise ValueError("Context was already finalized") - data = self._backend.ciphers.finalize_ctx(self._ctx) - self._ctx = None - return data - - -@interfaces.register(interfaces.CipherContext) -class _CipherDecryptionContext(object): - def __init__(self, cipher, mode, backend): - super(_CipherDecryptionContext, self).__init__() - self._backend = backend - self._ctx = self._backend.ciphers.create_decrypt_ctx(cipher, mode) +class _OneTimeCipherContext(object): + def __init__(self, ctx): + self._ctx = ctx def update(self, data): if self._ctx is None: raise ValueError("Context was already finalized") - return self._backend.ciphers.update_ctx(self._ctx, data) + return self._ctx.update(data) def finalize(self): if self._ctx is None: raise ValueError("Context was already finalized") - data = self._backend.ciphers.finalize_ctx(self._ctx) + data = self._ctx.finalize() self._ctx = None return data |