hexlify_codec.py 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. #! python
  2. #
  3. # This is a codec to create and decode hexdumps with spaces between characters. used by miniterm.
  4. #
  5. # This file is part of pySerial. https://github.com/pyserial/pyserial
  6. # (C) 2015-2016 Chris Liechti <cliechti@gmx.net>
  7. #
  8. # SPDX-License-Identifier: BSD-3-Clause
  9. """\
  10. Python 'hex' Codec - 2-digit hex with spaces content transfer encoding.
  11. Encode and decode may be a bit missleading at first sight...
  12. The textual representation is a hex dump: e.g. "40 41"
  13. The "encoded" data of this is the binary form, e.g. b"@A"
  14. Therefore decoding is binary to text and thus converting binary data to hex dump.
  15. """
  16. import codecs
  17. import serial
  18. try:
  19. unicode
  20. except (NameError, AttributeError):
  21. unicode = str # for Python 3, pylint: disable=redefined-builtin,invalid-name
  22. HEXDIGITS = '0123456789ABCDEF'
  23. # Codec APIs
  24. def hex_encode(data, errors='strict'):
  25. """'40 41 42' -> b'@ab'"""
  26. return (serial.to_bytes([int(h, 16) for h in data.split()]), len(data))
  27. def hex_decode(data, errors='strict'):
  28. """b'@ab' -> '40 41 42'"""
  29. return (unicode(''.join('{:02X} '.format(ord(b)) for b in serial.iterbytes(data))), len(data))
  30. class Codec(codecs.Codec):
  31. def encode(self, data, errors='strict'):
  32. """'40 41 42' -> b'@ab'"""
  33. return serial.to_bytes([int(h, 16) for h in data.split()])
  34. def decode(self, data, errors='strict'):
  35. """b'@ab' -> '40 41 42'"""
  36. return unicode(''.join('{:02X} '.format(ord(b)) for b in serial.iterbytes(data)))
  37. class IncrementalEncoder(codecs.IncrementalEncoder):
  38. """Incremental hex encoder"""
  39. def __init__(self, errors='strict'):
  40. self.errors = errors
  41. self.state = 0
  42. def reset(self):
  43. self.state = 0
  44. def getstate(self):
  45. return self.state
  46. def setstate(self, state):
  47. self.state = state
  48. def encode(self, data, final=False):
  49. """\
  50. Incremental encode, keep track of digits and emit a byte when a pair
  51. of hex digits is found. The space is optional unless the error
  52. handling is defined to be 'strict'.
  53. """
  54. state = self.state
  55. encoded = []
  56. for c in data.upper():
  57. if c in HEXDIGITS:
  58. z = HEXDIGITS.index(c)
  59. if state:
  60. encoded.append(z + (state & 0xf0))
  61. state = 0
  62. else:
  63. state = 0x100 + (z << 4)
  64. elif c == ' ': # allow spaces to separate values
  65. if state and self.errors == 'strict':
  66. raise UnicodeError('odd number of hex digits')
  67. state = 0
  68. else:
  69. if self.errors == 'strict':
  70. raise UnicodeError('non-hex digit found: {!r}'.format(c))
  71. self.state = state
  72. return serial.to_bytes(encoded)
  73. class IncrementalDecoder(codecs.IncrementalDecoder):
  74. """Incremental decoder"""
  75. def decode(self, data, final=False):
  76. return unicode(''.join('{:02X} '.format(ord(b)) for b in serial.iterbytes(data)))
  77. class StreamWriter(Codec, codecs.StreamWriter):
  78. """Combination of hexlify codec and StreamWriter"""
  79. class StreamReader(Codec, codecs.StreamReader):
  80. """Combination of hexlify codec and StreamReader"""
  81. def getregentry():
  82. """encodings module API"""
  83. return codecs.CodecInfo(
  84. name='hexlify',
  85. encode=hex_encode,
  86. decode=hex_decode,
  87. incrementalencoder=IncrementalEncoder,
  88. incrementaldecoder=IncrementalDecoder,
  89. streamwriter=StreamWriter,
  90. streamreader=StreamReader,
  91. #~ _is_text_encoding=True,
  92. )