Issue20132
This issue tracker has been migrated to GitHub,
and is currently read-only.
For more information,
see the GitHub FAQs in the Python's Developer Guide.
Created on 2014-01-05 13:48 by martin.panter, last changed 2022-04-11 14:57 by admin.
Files | ||||
---|---|---|---|---|
File name | Uploaded | Description | Edit | |
zlib-reader.patch | martin.panter, 2015-01-15 22:46 | zlib StreamReader + API clarification | review | |
inc-codecs.diff | martin.panter, 2015-01-25 04:47 | Fix bytes-bytes except StreamReader |
Messages (12) | |||
---|---|---|---|
msg207374 - (view) | Author: Martin Panter (martin.panter) * | Date: 2014-01-05 13:48 | |
Many of the incremental codecs do not handle fragmented data very well. In the past I think I was interested in using the Base-64 and Quoted-printable codecs, and playing with other codecs today reveals many more issues. A lot of the issues reflect missing functionality, so maybe the simplest solution would be to document the codecs that don’t work. Incremental decoding issues: >>> str().join(codecs.iterdecode(iter((b"\\", b"u2013")), "unicode-escape")) UnicodeDecodeError: 'unicodeescape' codec can't decode byte 0x5c in position 0: \ at end of string # Same deal for raw-unicode-escape. >>> bytes().join(codecs.iterdecode(iter((b"3", b"3")), "hex-codec")) binascii.Error: Odd-length string >>> bytes().join(codecs.iterdecode(iter((b"A", b"A==")), "base64-codec")) binascii.Error: Incorrect padding >>> bytes().join(codecs.iterdecode(iter((b"=", b"3D")), "quopri-codec")) b'3D' # Should return b"=" >>> codecs.getincrementaldecoder("uu-codec")().decode(b"begin ") ValueError: Truncated input data Incremental encoding issues: >>> e = codecs.getincrementalencoder("base64-codec")(); codecs.decode(e.encode(b"1") + e.encode(b"2", final=True), "base64-codec") b'1' # Should be b"12" >>> e = codecs.getincrementalencoder("quopri-codec")(); e.encode(b"1" * 50) + e.encode(b"2" * 50, final=True) b'1111111111111111111111111111111111111111111111111122222222222222222222222222222222222222222222222222' # I suspect the line should have been split in two >>> e = codecs.getincrementalencoder("uu-codec")(); codecs.decode(e.encode(b"1") + e.encode(b"2", final=True), "uu-codec") b'1' # Should be b"12" I also noticed iterdecode() does not work with “uu-codec”: >>> bytes().join(codecs.iterdecode(iter((b"begin 666 <data>\n \nend\n",)), "uu-codec")) Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/usr/lib/python3.3/codecs.py", line 1032, in iterdecode output = decoder.decode(b"", True) File "/usr/lib/python3.3/encodings/uu_codec.py", line 80, in decode return uu_decode(input, self.errors)[0] File "/usr/lib/python3.3/encodings/uu_codec.py", line 45, in uu_decode raise ValueError('Missing "begin" line in input data') ValueError: Missing "begin" line in input data And iterencode() does not work with any of the byte encoders, because it does not know what kind of empty string to pass to IncrementalEncoder.encode(final=True): >>> bytes().join(codecs.iterencode(iter(()), "base64-codec")) Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/usr/lib/python3.3/codecs.py", line 1014, in iterencode output = encoder.encode("", True) File "/usr/lib/python3.3/encodings/base64_codec.py", line 31, in encode return base64.encodebytes(input) File "/usr/lib/python3.3/base64.py", line 343, in encodebytes raise TypeError("expected bytes, not %s" % s.__class__.__name__) TypeError: expected bytes, not str Finally, incremental UTF-7 encoding is suboptimal, and the decoder seems to buffer unlimited data, both defeating the purpose of using an incremental codec: >>> bytes().join(codecs.iterencode("\xA9" * 2, "utf-7")) b'+AKk-+AKk-' # b"+AKkAqQ-" would be better >>> d = codecs.getincrementaldecoder("utf-7")() >>> d.decode(b"+") b'' >>> any(d.decode(b"AAAAAAAA" * 100000) for _ in range(10)) False # No data returned: everything must be buffered >>> d.decode(b"-") == "\x00" * 3000000 True # Returned all buffered data in one go |
|||
msg207850 - (view) | Author: Walter Dörwald (doerwalter) * | Date: 2014-01-10 11:26 | |
The best solution IMHO would be to implement real incremental codecs for all of those. Maybe iterencode() with an empty iterator should never call encode()? (But IMHO it would be better to document that iterencode()/iterdecode() should only be used with "real" codecs.) Note that the comment before PyUnicode_DecodeUTF7Stateful() in unicodeobject.c reads: /* The decoder. The only state we preserve is our read position, * i.e. how many characters we have consumed. So if we end in the * middle of a shift sequence we have to back off the read position * and the output to the beginning of the sequence, otherwise we lose * all the shift state (seen bits, number of bits seen, high * surrogate). */ Changing that would have to introduce a state object that the codec updates and from which it can be restarted. Also the encoder does not buffer anything. To implement the suggested behaviour, the encoder might have to buffer unlimited data. |
|||
msg207900 - (view) | Author: Martin Panter (martin.panter) * | Date: 2014-01-11 07:30 | |
I think calling iterencode() with an empty iterator is a side issue. Even with a non-empty iterator, it tries to encode an empty _text_ string to finalise the encoder: >>> bytes().join(codecs.iterencode(iter((b"data",)), "base64-codec")) Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/usr/lib/python3.3/codecs.py", line 1014, in iterencode output = encoder.encode("", True) File "/usr/lib/python3.3/encodings/base64_codec.py", line 31, in encode return base64.encodebytes(input) File "/usr/lib/python3.3/base64.py", line 343, in encodebytes raise TypeError("expected bytes, not %s" % s.__class__.__name__) TypeError: expected bytes, not str Similarly, iterdecode(encoding="rot-13") doesn’t work. I agree it would be good to document that iterencode() is limited to text encoders and iterdecode() is limited to byte decoders. |
|||
msg226679 - (view) | Author: Martin Panter (martin.panter) * | Date: 2014-09-10 05:16 | |
Stream reader interfaces suffer the same problem. Test cases: codecs.getreader("unicode-escape")(BytesIO(br"\u2013")).read(1) codecs.getreader("hex-codec")(BytesIO(b"33")).read(1) codecs.getreader("base64-codec")(BytesIO(b"AA==")).read(1) TestCase().assertEqual(b"=", codecs.getreader("quopri-codec")(BytesIO(b"=3D")).read(1)) Even though the “zlib” incremental decoder is okay, its stream reader still suffers this problem. I was going to check for zip bomb behaviour when you call read() with a limited number of input “characters”, but got this instead: >>> codecs.getreader("zlib-codec")(bomb).read(1, 1) zlib.error: Error -5 while decompressing data: incomplete or truncated stream Similar problems with stream writers, for instance: >>> w = codecs.getwriter("base64-codec")(BytesIO()) >>> w.write(b"1"); w.write(b"2"); w.reset(); w.getvalue() b'MQ==\nMg==\n' # Should be b"MTI=\n" I also noticed StreamWriter.writelines() assumes it is encoding text, and does not work at least with “base64-codec”: >>>> codecs.getwriter("base64-codec")(BytesIO()).writelines((b"1", b"2")) TypeError: sequence item 0: expected str instance, bytes found |
|||
msg232850 - (view) | Author: Martin Panter (martin.panter) * | Date: 2014-12-18 02:25 | |
The “unicode-escape” and “utf-7” cases affect the more widely-used TextIOWrapper interface: >>> TextIOWrapper(BytesIO(br"\u2013" * 2000), "unicode-escape").read(1) Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/usr/lib/python3.4/encodings/unicode_escape.py", line 26, in decode return codecs.unicode_escape_decode(input, self.errors)[0] UnicodeDecodeError: 'unicodeescape' codec can't decode bytes in position 8190-8191: truncated \uXXXX escape >>> w = TextIOWrapper(BytesIO(), "utf-7") >>> w.writelines("\xA9\xA9") # Write one character at a time >>> w.detach().getvalue() b'+AKk-+AKk-' >>> r = TextIOWrapper(BytesIO(b"+" + b"AAAAAAAA" * 100000 + b"-"), "utf-7") >>> r.read(1) # Short delay as all 800 kB are decoded to read one character '\x00' >>> r.buffer.tell() 800002 For UTF-7 decoding to work optimally I think the amount of data buffering necessary would be limited to only a few bytes. |
|||
msg234099 - (view) | Author: Martin Panter (martin.panter) * | Date: 2015-01-15 22:46 | |
I opened Issue 23231 about fixing iterencode() and iterdecode() in the general case. I added a patch to Issue 13881 to fix StreamWriter for zlib and bz2, and to fix StreamWriter.writelines() in general. I am adding a patch here to clarify the StreamReader API and fix the StreamReader for the zlib-codec. Plan of other things to do: bz2 StreamReader: Should be implemented similar to the zlib patch above, after Issue 15955 is resolved and we have a max_length parameter to use. Or could be based on Bz2File now. hex decoder: Shouldn’t be too hard to hack a stateful IncrementalDecoder that saves the leftover digit if given an odd number of digits. Create a generic codecs._IncrementalStreamReader class that uses an IncrementalDecoder and buffers unread decoded data, similar to my _IncrementalStreamWriter for Issue 13881. base64 encoder: IncrementalEncoder could encode in base64.MAXBINSIZE chunks base64 decoder: IncrementalDecoder could strip non-alphabet characters using regular expressions, decode in multiples of four characters quopri encoder: would require new implementation or major refactoring of quopri module quopri decoder: check for incomplete trailing escape codes (=, =<single hex digit>, =\r) and newlines (\r) uu encoder: write header and trailer via uu module; encode using b2a_uu() uu decoder: factor out header parsing from uu module; buffer and decode line by line based on encoded length unicode-escape, raw-unicode-escape: Stateful decoding would probably require a new -Stateful() function at the C level, though it might be easy to build from the existing implementation. I suggest documenting that stateful decoding is not supported for the time being. utf-7: As Walter said, proper stateful codec is not supported by the C API, despite PyUnicode_DecodeUTF7Stateful(); doing so would probably require significant changes. I suggest documenting a warning for stateful mode (including with TextIOWrapper) about suboptimal encoding and unlimited data buffering for the time being. punycode, unicode_internal: According to test_codecs.py, these also don’t work in stateful mode. Not sure on the details though. |
|||
msg234100 - (view) | Author: Marc-Andre Lemburg (lemburg) * | Date: 2015-01-15 23:02 | |
On 15.01.2015 23:46, Martin Panter wrote: > > I opened Issue 23231 about fixing iterencode() and iterdecode() in the general case. I added a patch to Issue 13881 to fix StreamWriter for zlib and bz2, and to fix StreamWriter.writelines() in general. > > I am adding a patch here to clarify the StreamReader API and fix the StreamReader for the zlib-codec. Martin, I must be missing some context: what's your master plan with all these codec patches and open tickets ? They are very hard to follow and you're making design changes which need more review and discussion than can easily be done on a few tickets. Regarding the patch: The doc patch seems to just change ordering of sentences and paragraphs. Without additional explanation, it's difficult to determine whether you are changing semantics or not. The "strict" error change is not correct. Most codecs raise a UnicodeError (which is a ValueError subclass) and code expects codecs that work on Unicode to return a UnicodeError, not a ValueError. Only codecs that do not work on Unicode are allowed to raise ValueErrors. |
|||
msg234102 - (view) | Author: Marc-Andre Lemburg (lemburg) * | Date: 2015-01-15 23:05 | |
This addition is wrong as well: The *stream* argument must be a file-like object open for reading - text or binary data, as appropriate for the specific codec. + text or binary data, as appropriate for the specific codec. This stream is + assumed to have ended as soon as a read from it returns no data. Where did you get this idea from ? |
|||
msg234104 - (view) | Author: Martin Panter (martin.panter) * | Date: 2015-01-16 00:28 | |
My “master plan” is basically to make most of the bytes-to-bytes codecs work as documented in the incremental (stateful) modes. I’m less interested in fixing the text codecs, and the quopri and uu codecs might be too hard, so I was going to propose some documentation warnings for those. If you have a suggestion on how to go about this better, let me know. With my doc change to StreamReader, I wanted to document the different modes that I saw in the base codecs.StreamReader.read() implementation: * read() or read(-1) reads everything * read(size) returns an arbitrary amount of data * read(size, chars) returns exactly *chars* length of data (unless EOF or similar) Previously the case of read(-1, chars) was ambiguous. Also I did not find the description “an approximate maximum number of decoded bytes” very helpful, considering more than this maximum was read if necessary to get enough *chars*. Regarding the end-of-stream behaviour, I made an assumption but I now realize it was wrong. Experimenting with the UTF-8 codec shows that its StreamReader.read() keeps returning "" when the underlying stream returns no data. But if it was in the middle of a multi-byte sequence, no “end of data” error is raised, and the multi-byte sequence can be completed if the underlying stream later returns more data. I think the lack of end-of-data checking should be documented. The different cases of ValueError and UnicodeError that you describe make sense. I think the various references to ValueError and UnicodeError should be updated (or replaced with pointers) to match. |
|||
msg234122 - (view) | Author: Martin Panter (martin.panter) * | Date: 2015-01-16 10:17 | |
There is a flaw with inheriting the readline() method in my patch, and I have decided to give up fixing the StreamReader classes. I did update the documentation in my copy of the patch based on Marc-Andre Lemburg’s feedback if anyone is interested in it, otherwise I am going to focus on some of the incremental codecs now. |
|||
msg234651 - (view) | Author: Martin Panter (martin.panter) * | Date: 2015-01-25 04:47 | |
Here is a new patch which fixes the bytes-to-bytes incremental codecs. It depends on my patches for these other issues being applied first: * Issue 23231: Bytes-to-bytes support for iteren/decode() * Issue 13881: Generic StreamWriter from IncrementalEncoder * Issue 16473: Clarify and test quopri-codec implementation In addition, without the fix for Issue 20121 (Quoted printable soft line breaking), the patch will apply, but there will be test failures. Summary of the changes in this patch: * Fix simple bz2-codec bug uncovered by new tests * Implement stateful hex-codec IncrementalDecoder * Add helpers for getstate() and setstate() to bijectively convert between arbitrary-length byte strings and integers * Implement stateful base64-codec IncrementalEncoder; use it for the StreamWriter * base64-codec IncrementalDecoder * quopri-codec IncrementalEncoder, StreamWriter, IncrementalDecoder * uu-codec IncrementalEncoder, StreamWriter, IncrementalDecoder * Document that bytes-to-bytes StreamReader is not supported * Document stateful raw-/unicode-escape decoding not supported * Warn that stateful UTF-7 encoding may not be optimal, and that stateful UTF-7 decoding may buffer unlimited input This patch is actually generated from a series of patches folded together. If anyone is interested, I could try pushing the original series to a Bitbucket repository or something. |
|||
msg273108 - (view) | Author: Martin Panter (martin.panter) * | Date: 2016-08-19 12:08 | |
Split off Issue 27799 for the base-64 incremental decoder |
History | |||
---|---|---|---|
Date | User | Action | Args |
2022-04-11 14:57:56 | admin | set | github: 64331 |
2016-08-19 12:08:31 | martin.panter | set | dependencies:
+ Fix base64-codec and bz2-codec incremental decoders messages: + msg273108 |
2015-07-23 01:54:39 | martin.panter | set | dependencies: + Stream encoder for zlib_codec doesn't use the incremental encoder, quopri module differences in quoted-printable text with whitespace, quopri_codec newline handling, Fix codecs.iterencode/decode() by allowing data parameter to be omitted |
2015-03-21 10:28:37 | serhiy.storchaka | set | nosy:
+ serhiy.storchaka stage: patch review versions: + Python 3.5, - Python 3.3 |
2015-01-25 04:47:50 | martin.panter | set | files:
+ inc-codecs.diff messages: + msg234651 |
2015-01-16 10:17:15 | martin.panter | set | messages: + msg234122 |
2015-01-16 00:28:31 | martin.panter | set | messages: + msg234104 |
2015-01-15 23:05:07 | lemburg | set | messages: + msg234102 |
2015-01-15 23:02:43 | lemburg | set | messages: + msg234100 |
2015-01-15 22:46:09 | martin.panter | set | files:
+ zlib-reader.patch keywords: + patch messages: + msg234099 |
2014-12-18 02:25:22 | martin.panter | set | messages: + msg232850 |
2014-09-10 05:16:57 | martin.panter | set | messages:
+ msg226679 versions: + Python 3.4 |
2014-01-11 07:30:21 | martin.panter | set | messages: + msg207900 |
2014-01-10 11:26:50 | doerwalter | set | messages: + msg207850 |
2014-01-05 15:16:19 | serhiy.storchaka | set | nosy:
+ lemburg, loewis, doerwalter, ncoghlan, vstinner |
2014-01-05 13:48:58 | martin.panter | create |