mirror of
https://github.com/python/cpython.git
synced 2025-07-27 13:14:41 +00:00

of tuple) that provides incremental decoders and encoders (a way to use stateful codecs without the stream API). Functions codecs.getincrementaldecoder() and codecs.getincrementalencoder() have been added.
128 lines
3.6 KiB
Python
128 lines
3.6 KiB
Python
""" Python 'uu_codec' Codec - UU content transfer encoding
|
|
|
|
Unlike most of the other codecs which target Unicode, this codec
|
|
will return Python string objects for both encode and decode.
|
|
|
|
Written by Marc-Andre Lemburg (mal@lemburg.com). Some details were
|
|
adapted from uu.py which was written by Lance Ellinghouse and
|
|
modified by Jack Jansen and Fredrik Lundh.
|
|
|
|
"""
|
|
import codecs, binascii
|
|
|
|
### Codec APIs
|
|
|
|
def uu_encode(input,errors='strict',filename='<data>',mode=0666):
|
|
|
|
""" Encodes the object input and returns a tuple (output
|
|
object, length consumed).
|
|
|
|
errors defines the error handling to apply. It defaults to
|
|
'strict' handling which is the only currently supported
|
|
error handling for this codec.
|
|
|
|
"""
|
|
assert errors == 'strict'
|
|
from cStringIO import StringIO
|
|
from binascii import b2a_uu
|
|
infile = StringIO(input)
|
|
outfile = StringIO()
|
|
read = infile.read
|
|
write = outfile.write
|
|
|
|
# Encode
|
|
write('begin %o %s\n' % (mode & 0777, filename))
|
|
chunk = read(45)
|
|
while chunk:
|
|
write(b2a_uu(chunk))
|
|
chunk = read(45)
|
|
write(' \nend\n')
|
|
|
|
return (outfile.getvalue(), len(input))
|
|
|
|
def uu_decode(input,errors='strict'):
|
|
|
|
""" Decodes the object input and returns a tuple (output
|
|
object, length consumed).
|
|
|
|
input must be an object which provides the bf_getreadbuf
|
|
buffer slot. Python strings, buffer objects and memory
|
|
mapped files are examples of objects providing this slot.
|
|
|
|
errors defines the error handling to apply. It defaults to
|
|
'strict' handling which is the only currently supported
|
|
error handling for this codec.
|
|
|
|
Note: filename and file mode information in the input data is
|
|
ignored.
|
|
|
|
"""
|
|
assert errors == 'strict'
|
|
from cStringIO import StringIO
|
|
from binascii import a2b_uu
|
|
infile = StringIO(input)
|
|
outfile = StringIO()
|
|
readline = infile.readline
|
|
write = outfile.write
|
|
|
|
# Find start of encoded data
|
|
while 1:
|
|
s = readline()
|
|
if not s:
|
|
raise ValueError, 'Missing "begin" line in input data'
|
|
if s[:5] == 'begin':
|
|
break
|
|
|
|
# Decode
|
|
while 1:
|
|
s = readline()
|
|
if not s or \
|
|
s == 'end\n':
|
|
break
|
|
try:
|
|
data = a2b_uu(s)
|
|
except binascii.Error, v:
|
|
# Workaround for broken uuencoders by /Fredrik Lundh
|
|
nbytes = (((ord(s[0])-32) & 63) * 4 + 5) / 3
|
|
data = a2b_uu(s[:nbytes])
|
|
#sys.stderr.write("Warning: %s\n" % str(v))
|
|
write(data)
|
|
if not s:
|
|
raise ValueError, 'Truncated input data'
|
|
|
|
return (outfile.getvalue(), len(input))
|
|
|
|
class Codec(codecs.Codec):
|
|
|
|
def encode(self,input,errors='strict'):
|
|
return uu_encode(input,errors)
|
|
|
|
def decode(self,input,errors='strict'):
|
|
return uu_decode(input,errors)
|
|
|
|
class IncrementalEncoder(codecs.IncrementalEncoder):
|
|
def encode(self, input, final=False):
|
|
return uu_encode(input, errors)[0]
|
|
|
|
class IncrementalDecoder(codecs.IncrementalDecoder):
|
|
def decode(self, input, final=False):
|
|
return uu_decode(input, errors)[0]
|
|
|
|
class StreamWriter(Codec,codecs.StreamWriter):
|
|
pass
|
|
|
|
class StreamReader(Codec,codecs.StreamReader):
|
|
pass
|
|
|
|
### encodings module API
|
|
|
|
def getregentry():
|
|
return codecs.CodecInfo(
|
|
name='uu',
|
|
encode=uu_encode,
|
|
decode=uu_decode,
|
|
incrementalencoder=IncrementalEncoder,
|
|
incrementaldecoder=IncrementalDecoder,
|
|
streamreader=StreamReader,
|
|
streamwriter=StreamWriter,
|
|
)
|