Commit 87be28f4 authored by Serhiy Storchaka's avatar Serhiy Storchaka Committed by GitHub

bpo-31672: Restore the former behavior when override flags in Template. (#5099)

Overriding flags to 0 will make the default pattern matching only
lower case letters.
parent 78e24d44
...@@ -755,19 +755,14 @@ attributes: ...@@ -755,19 +755,14 @@ attributes:
* *idpattern* -- This is the regular expression describing the pattern for * *idpattern* -- This is the regular expression describing the pattern for
non-braced placeholders. The default value is the regular expression non-braced placeholders. The default value is the regular expression
``(?a:[_a-zA-Z][_a-zA-Z0-9]*)``. If this is given and *braceidpattern* is ``(?a:[_a-z][_a-z0-9]*)``. If this is given and *braceidpattern* is
``None`` this pattern will also apply to braced placeholders. ``None`` this pattern will also apply to braced placeholders.
.. note:: .. note::
Since default *flags* is ``re.IGNORECASE``, pattern ``[a-z]`` can match Since default *flags* is ``re.IGNORECASE``, pattern ``[a-z]`` can match
with some non-ASCII characters. That's why we use the local ``a`` flag with some non-ASCII characters. That's why we use the local ``a`` flag
here. Further, with the default *flags* value, including ``A-Z`` in the here.
ranges is redundant, but required for backward compatibility.
While *flags* is kept to ``re.IGNORECASE`` for backward compatibility,
you can override it to ``0`` or ``re.IGNORECASE | re.ASCII`` when
subclassing.
.. versionchanged:: 3.7 .. versionchanged:: 3.7
*braceidpattern* can be used to define separate patterns used inside and *braceidpattern* can be used to define separate patterns used inside and
......
...@@ -82,11 +82,8 @@ class Template(metaclass=_TemplateMetaclass): ...@@ -82,11 +82,8 @@ class Template(metaclass=_TemplateMetaclass):
# r'[a-z]' matches to non-ASCII letters when used with IGNORECASE, but # r'[a-z]' matches to non-ASCII letters when used with IGNORECASE, but
# without the ASCII flag. We can't add re.ASCII to flags because of # without the ASCII flag. We can't add re.ASCII to flags because of
# backward compatibility. So we use the ?a local flag and [a-z] pattern. # backward compatibility. So we use the ?a local flag and [a-z] pattern.
# We also can't remove the A-Z ranges, because although they are
# technically redundant with the IGNORECASE flag, the value is part of the
# publicly documented API.
# See https://bugs.python.org/issue31672 # See https://bugs.python.org/issue31672
idpattern = r'(?a:[_a-zA-Z][_a-zA-Z0-9]*)' idpattern = r'(?a:[_a-z][_a-z0-9]*)'
braceidpattern = None braceidpattern = None
flags = _re.IGNORECASE flags = _re.IGNORECASE
......
...@@ -219,6 +219,16 @@ class TestTemplate(unittest.TestCase): ...@@ -219,6 +219,16 @@ class TestTemplate(unittest.TestCase):
self.assertRaises(KeyError, s.substitute, self.assertRaises(KeyError, s.substitute,
dict(who='tim', what='ham')) dict(who='tim', what='ham'))
def test_regular_templates_with_upper_case(self):
s = Template('$WHO likes ${WHAT} for ${MEAL}')
d = dict(WHO='tim', WHAT='ham', MEAL='dinner')
self.assertEqual(s.substitute(d), 'tim likes ham for dinner')
def test_regular_templates_with_non_letters(self):
s = Template('$_wh0_ likes ${_w_h_a_t_} for ${mea1}')
d = dict(_wh0_='tim', _w_h_a_t_='ham', mea1='dinner')
self.assertEqual(s.substitute(d), 'tim likes ham for dinner')
def test_escapes(self): def test_escapes(self):
eq = self.assertEqual eq = self.assertEqual
s = Template('$who likes to eat a bag of $$what worth $$100') s = Template('$who likes to eat a bag of $$what worth $$100')
...@@ -288,6 +298,14 @@ class TestTemplate(unittest.TestCase): ...@@ -288,6 +298,14 @@ class TestTemplate(unittest.TestCase):
s = PathPattern('$bag.foo.who likes to eat a bag of $bag.what') s = PathPattern('$bag.foo.who likes to eat a bag of $bag.what')
self.assertEqual(s.substitute(m), 'tim likes to eat a bag of ham') self.assertEqual(s.substitute(m), 'tim likes to eat a bag of ham')
def test_flags_override(self):
class MyPattern(Template):
flags = 0
s = MyPattern('$wHO likes ${WHAT} for ${meal}')
d = dict(wHO='tim', WHAT='ham', meal='dinner', w='fred')
self.assertRaises(ValueError, s.substitute, d)
self.assertEqual(s.safe_substitute(d), 'fredHO likes ${WHAT} for dinner')
def test_idpattern_override_inside_outside(self): def test_idpattern_override_inside_outside(self):
# bpo-1198569: Allow the regexp inside and outside braces to be # bpo-1198569: Allow the regexp inside and outside braces to be
# different when deriving from Template. # different when deriving from Template.
......
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