Jean-Paul Calderone | 4f0467a | 2014-01-11 11:58:41 -0500 | [diff] [blame] | 1 | from six import PY3, binary_type, text_type |
| 2 | |
Jean-Paul Calderone | e36b31a | 2014-01-08 16:55:06 -0500 | [diff] [blame] | 3 | from cryptography.hazmat.bindings.openssl.binding import Binding |
| 4 | binding = Binding() |
| 5 | ffi = binding.ffi |
| 6 | lib = binding.lib |
Jean-Paul Calderone | c86bb7d | 2013-12-29 10:25:59 -0500 | [diff] [blame] | 7 | |
Stephen Holsapple | 0d9815f | 2014-08-27 19:36:53 -0700 | [diff] [blame^] | 8 | |
| 9 | |
| 10 | def text(charp): |
| 11 | return native(ffi.string(charp)) |
| 12 | |
| 13 | |
| 14 | |
| 15 | def exception_from_error_queue(exception_type): |
| 16 | """ |
| 17 | Convert an OpenSSL library failure into a Python exception. |
| 18 | |
| 19 | When a call to the native OpenSSL library fails, this is usually signalled |
| 20 | by the return value, and an error code is stored in an error queue |
| 21 | associated with the current thread. The err library provides functions to |
| 22 | obtain these error codes and textual error messages. |
| 23 | """ |
Jean-Paul Calderone | de07546 | 2014-01-18 10:34:12 -0500 | [diff] [blame] | 24 | |
Jean-Paul Calderone | c86bb7d | 2013-12-29 10:25:59 -0500 | [diff] [blame] | 25 | errors = [] |
Stephen Holsapple | 0d9815f | 2014-08-27 19:36:53 -0700 | [diff] [blame^] | 26 | |
Jean-Paul Calderone | c86bb7d | 2013-12-29 10:25:59 -0500 | [diff] [blame] | 27 | while True: |
| 28 | error = lib.ERR_get_error() |
| 29 | if error == 0: |
| 30 | break |
| 31 | errors.append(( |
Jean-Paul Calderone | de07546 | 2014-01-18 10:34:12 -0500 | [diff] [blame] | 32 | text(lib.ERR_lib_error_string(error)), |
| 33 | text(lib.ERR_func_error_string(error)), |
| 34 | text(lib.ERR_reason_error_string(error)))) |
Jean-Paul Calderone | c86bb7d | 2013-12-29 10:25:59 -0500 | [diff] [blame] | 35 | |
Stephen Holsapple | 0d9815f | 2014-08-27 19:36:53 -0700 | [diff] [blame^] | 36 | raise exception_type(errors) |
Jean-Paul Calderone | 4f0467a | 2014-01-11 11:58:41 -0500 | [diff] [blame] | 37 | |
| 38 | |
| 39 | |
| 40 | def native(s): |
| 41 | """ |
| 42 | Convert :py:class:`bytes` or :py:class:`unicode` to the native |
Jean-Paul Calderone | aca50f4 | 2014-01-11 14:43:37 -0500 | [diff] [blame] | 43 | :py:class:`str` type, using UTF-8 encoding if conversion is necessary. |
Jean-Paul Calderone | 4f0467a | 2014-01-11 11:58:41 -0500 | [diff] [blame] | 44 | |
| 45 | :raise UnicodeError: The input string is not UTF-8 decodeable. |
| 46 | |
| 47 | :raise TypeError: The input is neither :py:class:`bytes` nor |
| 48 | :py:class:`unicode`. |
| 49 | """ |
| 50 | if not isinstance(s, (binary_type, text_type)): |
| 51 | raise TypeError("%r is neither bytes nor unicode" % s) |
| 52 | if PY3: |
| 53 | if isinstance(s, binary_type): |
| 54 | return s.decode("utf-8") |
| 55 | else: |
| 56 | if isinstance(s, text_type): |
| 57 | return s.encode("utf-8") |
| 58 | return s |
| 59 | |
| 60 | |
| 61 | |
| 62 | if PY3: |
| 63 | def byte_string(s): |
| 64 | return s.encode("charmap") |
| 65 | else: |
| 66 | def byte_string(s): |
| 67 | return s |