65 lines
2.1 KiB
Python
65 lines
2.1 KiB
Python
|
#!/usr/bin/env python3
|
||
|
|
||
|
""" Python implementation of ASCII85/ASCIIHex decoder (Adobe version).
|
||
|
|
||
|
This code is in the public domain.
|
||
|
|
||
|
"""
|
||
|
|
||
|
import re
|
||
|
import struct
|
||
|
|
||
|
def ascii85decode(data):
|
||
|
"""
|
||
|
In ASCII85 encoding, every four bytes are encoded with five ASCII
|
||
|
letters, using 85 different types of characters (as 256**4 < 85**5).
|
||
|
When the length of the original bytes is not a multiple of 4, a special
|
||
|
rule is used for round up.
|
||
|
|
||
|
The Adobe's ASCII85 implementation is slightly different from
|
||
|
its original in handling the last characters.
|
||
|
|
||
|
The sample string is taken from:
|
||
|
http://en.wikipedia.org/w/index.php?title=Ascii85
|
||
|
"""
|
||
|
if isinstance(data, str):
|
||
|
data = data.encode('ascii')
|
||
|
n = b = 0
|
||
|
out = bytearray()
|
||
|
for c in data:
|
||
|
if ord('!') <= c and c <= ord('u'):
|
||
|
n += 1
|
||
|
b = b*85+(c-33)
|
||
|
if n == 5:
|
||
|
out += struct.pack(b'>L',b)
|
||
|
n = b = 0
|
||
|
elif c == ord('z'):
|
||
|
assert n == 0
|
||
|
out += b'\0\0\0\0'
|
||
|
elif c == ord('~'):
|
||
|
if n:
|
||
|
for _ in range(5-n):
|
||
|
b = b*85+84
|
||
|
out += struct.pack(b'>L',b)[:n-1]
|
||
|
break
|
||
|
return bytes(out)
|
||
|
|
||
|
hex_re = re.compile(r'([a-f\d]{2})', re.IGNORECASE)
|
||
|
trail_re = re.compile(r'^(?:[a-f\d]{2}|\s)*([a-f\d])[\s>]*$', re.IGNORECASE)
|
||
|
def asciihexdecode(data):
|
||
|
"""
|
||
|
ASCIIHexDecode filter: PDFReference v1.4 section 3.3.1
|
||
|
For each pair of ASCII hexadecimal digits (0-9 and A-F or a-f), the
|
||
|
ASCIIHexDecode filter produces one byte of binary data. All white-space
|
||
|
characters are ignored. A right angle bracket character (>) indicates
|
||
|
EOD. Any other characters will cause an error. If the filter encounters
|
||
|
the EOD marker after reading an odd number of hexadecimal digits, it
|
||
|
will behave as if a 0 followed the last digit.
|
||
|
"""
|
||
|
decode = (lambda hx: chr(int(hx, 16)))
|
||
|
out = list(map(decode, hex_re.findall(data)))
|
||
|
m = trail_re.search(data)
|
||
|
if m:
|
||
|
out.append(decode("%c0" % m.group(1)))
|
||
|
return ''.join(out)
|