-
-
Notifications
You must be signed in to change notification settings - Fork 30.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
octal escapes applied inconsistently throughout the interpreter and lib #81548
Comments
At present, the bytecode compiler can generate 512 different unicode characters, one for each integral from the range [0-511), 512 being the total number of syntactically valid permutations of 3 octal digits preceded by a backslash. However, this does not match the regex compiler, which raises an error regardless of the input type when it encounters an an octal escape character with a decimal value greater than 255. On the other hand... the bytes literal:
is somehow valid, and can lead to extremely difficult bugs to track down, such as this nonsense: >>> re.compile(b'\407').search(b'\a')
<re.Match object; span=(0, 1), match=b'\x07'> I propose that the regex parser be augmented, enabling for unicode patterns the interpretation of three character octal escapes from the range(256, 512), while the bytecode parser be adjusted to match the behavior of the regex parser, raising an error for bytes literals > b"\400", rather than truncating the 9th bit. |
https://docs.python.org/3/reference/lexical_analysis.html#string-and-bytes-literals I agree that sometimes truncating an invalid integer instead of always raising ValueError is strange. >>> ord(b'\407')
7
>>> bytes((0o407,))
...
ValueError: bytes must be in range(0, 256) I don't know is there was an intentional back-compatibility reason for this. Without an example of re raising, I don't understand the re complaint. |
>>> b'\407'
b'\x07'
>>> ord(b'\407')
7 This is the object structure passed to builtin_ord(): (lldb) p *((PyBytesObject *)(c)) If two bytes were stored (0x107), I would expect ob_sval[0] to be 7 ('\a') and ob_sval[1] to be 1 on a little endian system, but ob_sval[1] is 0: (lldb) p (long)(unsigned char) (((PyBytesObject *)(c))->ob_sval[0]) This means the truncation to a single byte is happening when the byte string object is created. |
Here is the problematic code in _PyBytes_DecodeEscape in Objects/bytesobject.c: c = s[-1] - '0';
if (s < end && '0' <= *s && *s <= '7') {
c = (c<<3) + *s++ - '0';
if (s < end && '0' <= *s && *s <= '7')
c = (c<<3) + *s++ - '0';
}
*p++ = c; c is an int, and p is a char pointer to the new bytes object's string buffer. For b'\407', c gets correctly calculated as 263 (0x107), but the upper bits are lost when it gets recast as a char and stored in the location pointed to by p. Hence, b'\407' becomes b'\x07' when the object is created. IMO, this should raise "ValueError: bytes must be in range(0, 256)" instead of silently throwing away the upper bits. I will work on a PR. I also took a look at how escaped hex values are handled by the same function. It may seem at first glance that >>> b'\x107'
b'\x107' is returning the hex value 0x107, but in reality it is returning '\x10' as the first character and '7' as the second character. While visually misleading, it is syntactically and semantically correct. |
The PR is ready for review. It raises ValueError if the escaped octal value in a byte string is greater than 255. |
The PR looks pretty good, but the question is whether we want to break backwards compatibility in the name of correctness. In this case, the silent buggy behavior of keeping the (mod 256) of the value seems worth fixing to me. |
I can't find who wrote this block treating octal escapes beginning with 4-7 the same as those beginning with 0-3. case '0': case '1': case '2': case '3':
case '4': case '5': case '6': case '7':
c = s[-1] - '0';
if (s < end && '0' <= *s && *s <= '7') {
c = (c<<3) + *s++ - '0';
if (s < end && '0' <= *s && *s <= '7')
c = (c<<3) + *s++ - '0';
}
*p++ = c;
break; Antoine Pitrou merged from somewhere in 2010 and Christiqn Heimes renamed something in 2008 and before that, ???. Guido wrote the initial bytesobject.c in 2006. Guido, do you agree that the current behavior, treating the same int differently when input into a bytes in different ways, is a bug, and if so, should we backport the fix? >>> b'\407'
b'\x07'
>>> bytes((0o407,))
Traceback (most recent call last):
File "<pyshell#7>", line 1, in <module>
bytes((0o407,))
ValueError: bytes must be in range(0, 256) |
For sure the Python tokenizer/parser should reject octal escapes that produce values >= 256. Certainly in bytes strings. Probably also in text strings (nobody using Unicode thinks in octal). This is ancient behavior though (it's the same in 2.7) so it may require a deprecation for the text string case. (For byte strings, it should become an error in 3.9 -- dropping the top bit is nonsensical.) The regex parser should not be changed. |
Alright, so let's push through the existing PR for rejecting such octal escapes in byte strings. We'll also need another PR for raising a deprecation warning for such octal escapes in strings. |
#91668 is an alternate PR which deprecates invalid octal sequences in bytes and string literals, in the same way as invalid escape sequences are deprecated. |
Given that the current code matches the doc, which says "As in Standard C, up to three octal digits are accepted." without qualification, it seems reasonable to treat this as a design bug (to be deprecated) rather than an implementation bug (to be immediately changed). I think it definitely preferable to deprecate now, before the upcoming beta, and close #14654, than to do nothing before 3.12. |
…ngs (GH-92643) Automerge-Triggered-By: GH:pablogsal
…e strings (pythonGH-92643) Automerge-Triggered-By: GH:pablogsal (cherry picked from commit 0d8500c) Co-authored-by: Pablo Galindo Salgado <[email protected]>
…ngs (GH-92643) Automerge-Triggered-By: GH:pablogsal (cherry picked from commit 0d8500c) Co-authored-by: Pablo Galindo Salgado <[email protected]>
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
The text was updated successfully, but these errors were encountered: