Commit 6bccbe7d authored by Greg Price's avatar Greg Price Committed by Victor Stinner

bpo-36502: Correct documentation of str.isspace() (GH-15019)

The documented definition was much broader than the real one:
there are tons of characters with general category "Other",
and we don't (and shouldn't) treat most of them as whitespace.

Rewrite the definition to agree with the comment on
_PyUnicode_IsWhitespace, and with the logic in makeunicodedata.py,
which is what generates that function and so ultimately governs.

Add suitable breadcrumbs so that a reader who wants to pin down
exactly what this definition means (what's a "bidirectional class"
of "B"?) can do so.  The `unicodedata` module documentation is an
appropriate central place for our references to Unicode's own copious
documentation, so point there.

Also add to the isspace() test a thorough check that the
implementation agrees with the intended definition.
parent 077af8c2
...@@ -1763,9 +1763,13 @@ expression support in the :mod:`re` module). ...@@ -1763,9 +1763,13 @@ expression support in the :mod:`re` module).
.. method:: str.isspace() .. method:: str.isspace()
Return true if there are only whitespace characters in the string and there is Return true if there are only whitespace characters in the string and there is
at least one character, false otherwise. Whitespace characters are those at least one character, false otherwise.
characters defined in the Unicode character database as "Other" or "Separator"
and those with bidirectional property being one of "WS", "B", or "S". A character is *whitespace* if in the Unicode character database
(see :mod:`unicodedata`), either its general category is ``Zs``
("Separator, space"), or its bidirectional class is one of ``WS``,
``B``, or ``S``.
.. method:: str.istitle() .. method:: str.istitle()
......
...@@ -12,6 +12,7 @@ import operator ...@@ -12,6 +12,7 @@ import operator
import struct import struct
import sys import sys
import textwrap import textwrap
import unicodedata
import unittest import unittest
import warnings import warnings
from test import support, string_tests from test import support, string_tests
...@@ -617,11 +618,21 @@ class UnicodeTest(string_tests.CommonTest, ...@@ -617,11 +618,21 @@ class UnicodeTest(string_tests.CommonTest,
self.checkequalnofix(True, '\u2000', 'isspace') self.checkequalnofix(True, '\u2000', 'isspace')
self.checkequalnofix(True, '\u200a', 'isspace') self.checkequalnofix(True, '\u200a', 'isspace')
self.checkequalnofix(False, '\u2014', 'isspace') self.checkequalnofix(False, '\u2014', 'isspace')
# apparently there are no non-BMP spaces chars in Unicode 6 # There are no non-BMP whitespace chars as of Unicode 12.
for ch in ['\U00010401', '\U00010427', '\U00010429', '\U0001044E', for ch in ['\U00010401', '\U00010427', '\U00010429', '\U0001044E',
'\U0001F40D', '\U0001F46F']: '\U0001F40D', '\U0001F46F']:
self.assertFalse(ch.isspace(), '{!a} is not space.'.format(ch)) self.assertFalse(ch.isspace(), '{!a} is not space.'.format(ch))
@support.requires_resource('cpu')
def test_isspace_invariant(self):
for codepoint in range(sys.maxunicode + 1):
char = chr(codepoint)
bidirectional = unicodedata.bidirectional(char)
category = unicodedata.category(char)
self.assertEqual(char.isspace(),
(bidirectional in ('WS', 'B', 'S')
or category == 'Zs'))
def test_isalnum(self): def test_isalnum(self):
super().test_isalnum() super().test_isalnum()
for ch in ['\U00010401', '\U00010427', '\U00010429', '\U0001044E', for ch in ['\U00010401', '\U00010427', '\U00010429', '\U0001044E',
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment