Source code for tlslite.utils.compat

# Author: Trevor Perrin
# See the LICENSE file for legal information regarding use of this file.

"""Miscellaneous functions to mask Python version differences."""

import sys
import re
import os
import platform
import math
import binascii
import traceback
import time
import ecdsa

if sys.version_info >= (3,0):

    def compat26Str(x): return x

    # Python 3.3 requires bytes instead of bytearrays for HMAC
    # So, python 2.6 requires strings, python 3 requires 'bytes',
    # and python 2.7 and 3.5 can handle bytearrays...
    # pylint: disable=invalid-name
    # we need to keep compatHMAC and `x` for API compatibility
    if sys.version_info < (3, 4):
        def compatHMAC(x):
            """Convert bytes-like input to format acceptable for HMAC."""
            return bytes(x)
    else:
        def compatHMAC(x):
            """Convert bytes-like input to format acceptable for HMAC."""
            return x
    # pylint: enable=invalid-name

    def compatAscii2Bytes(val):
        """Convert ASCII string to bytes."""
        if isinstance(val, str):
            return bytes(val, 'ascii')
        return val

    def compat_b2a(val):
        """Convert an ASCII bytes string to string."""
        return str(val, 'ascii')

[docs] def raw_input(s): return input(s)
# So, the python3 binascii module deals with bytearrays, and python2 # deals with strings... I would rather deal with the "a" part as # strings, and the "b" part as bytearrays, regardless of python version, # so... def a2b_hex(s): try: b = bytearray(binascii.a2b_hex(bytearray(s, "ascii"))) except Exception as e: raise SyntaxError("base16 error: %s" % e) return b def a2b_base64(s): try: if isinstance(s, str): s = bytearray(s, "ascii") b = bytearray(binascii.a2b_base64(s)) except Exception as e: raise SyntaxError("base64 error: %s" % e) return b def b2a_hex(b): return binascii.b2a_hex(b).decode("ascii") def b2a_base64(b): return binascii.b2a_base64(b).decode("ascii")
[docs] def readStdinBinary(): return sys.stdin.buffer.read()
def compatLong(num): return int(num) int_types = tuple([int]) def formatExceptionTrace(e): """Return exception information formatted as string""" return str(e) def time_stamp(): """Returns system time as a float""" if sys.version_info >= (3, 3): return time.perf_counter() return time.clock() def remove_whitespace(text): """Removes all whitespace from passed in string""" return re.sub(r"\s+", "", text, flags=re.UNICODE) # pylint: disable=invalid-name # pylint is stupid here and deson't notice it's a function, not # constant bytes_to_int = int.from_bytes # pylint: enable=invalid-name def bit_length(val): """Return number of bits necessary to represent an integer.""" return val.bit_length() def int_to_bytes(val, length=None, byteorder="big"): """Return number converted to bytes""" if length is None: length = byte_length(val) # for gmpy we need to convert back to native int if type(val) != int: val = int(val) return bytearray(val.to_bytes(length=length, byteorder=byteorder)) else: # Python 2.6 requires strings instead of bytearrays in a couple places, # so we define this function so it does the conversion if needed. # same thing with very old 2.7 versions # or on Jython if sys.version_info < (2, 7) or sys.version_info < (2, 7, 4) \ or platform.system() == 'Java': def compat26Str(x): return str(x) def remove_whitespace(text): """Removes all whitespace from passed in string""" return re.sub(r"\s+", "", text) def bit_length(val): """Return number of bits necessary to represent an integer.""" if val == 0: return 0 return len(bin(val))-2 else:
[docs] def compat26Str(x): return x
[docs] def remove_whitespace(text): """Removes all whitespace from passed in string""" return re.sub(r"\s+", "", text, flags=re.UNICODE)
[docs] def bit_length(val): """Return number of bits necessary to represent an integer.""" return val.bit_length()
[docs] def compatAscii2Bytes(val): """Convert ASCII string to bytes.""" return val
[docs] def compat_b2a(val): """Convert an ASCII bytes string to string.""" return str(val)
# So, python 2.6 requires strings, python 3 requires 'bytes', # and python 2.7 can handle bytearrays...
[docs] def compatHMAC(x): return compat26Str(x)
[docs] def a2b_hex(s): try: b = bytearray(binascii.a2b_hex(s)) except Exception as e: raise SyntaxError("base16 error: %s" % e) return b
[docs] def a2b_base64(s): try: b = bytearray(binascii.a2b_base64(s)) except Exception as e: raise SyntaxError("base64 error: %s" % e) return b
[docs] def b2a_hex(b): return binascii.b2a_hex(compat26Str(b))
[docs] def b2a_base64(b): return binascii.b2a_base64(compat26Str(b))
[docs] def compatLong(num): return long(num)
int_types = (int, long) # pylint on Python3 goes nuts for the sys dereferences... #pylint: disable=no-member
[docs] def formatExceptionTrace(e): """Return exception information formatted as string""" newStr = "".join(traceback.format_exception(sys.exc_type, sys.exc_value, sys.exc_traceback)) return newStr
#pylint: enable=no-member
[docs] def time_stamp(): """Returns system time as a float""" return time.clock()
def bytes_to_int(val, byteorder): """Convert bytes to an int.""" if not val: return 0 if byteorder == "big": return int(b2a_hex(val), 16) if byteorder == "little": return int(b2a_hex(val[::-1]), 16) raise ValueError("Only 'big' and 'little' endian supported")
[docs] def int_to_bytes(val, length=None, byteorder="big"): """Return number converted to bytes""" if length is None: length = byte_length(val) if byteorder == "big": return bytearray((val >> i) & 0xff for i in reversed(range(0, length*8, 8))) if byteorder == "little": return bytearray((val >> i) & 0xff for i in range(0, length*8, 8)) raise ValueError("Only 'big' or 'little' endian supported")
[docs] def byte_length(val): """Return number of bytes necessary to represent an integer.""" length = bit_length(val) return (length + 7) // 8
try: # Fedora and Red Hat Enterprise Linux versions have small curves removed getattr(ecdsa, 'NIST192p') except AttributeError: ecdsaAllCurves = False else: ecdsaAllCurves = True