From 8c72eae2378fde4e473feadd4c7139d9ec9cb8e3 Mon Sep 17 00:00:00 2001 From: Barry Warsaw Date: Mon, 1 Nov 2004 03:52:43 +0000 Subject: [PATCH] SF patch #1056967, changes the semantics of Template.safe_substitute() to not raise a ValueError for dangling delimiters (the delimiter itself is returned). --- Doc/lib/libstring.tex | 12 ++++++++++-- Lib/string.py | 2 +- Lib/test/test_pep292.py | 19 +++++++++---------- Misc/NEWS | 4 ++++ 4 files changed, 24 insertions(+), 13 deletions(-) diff --git a/Doc/lib/libstring.tex b/Doc/lib/libstring.tex index c9d27b5d575..1828b2e1cd7 100644 --- a/Doc/lib/libstring.tex +++ b/Doc/lib/libstring.tex @@ -131,8 +131,16 @@ precedence. Like \method{substitute()}, except that if placeholders are missing from \var{mapping} and \var{kws}, instead of raising a \exception{KeyError} exception, the original placeholder will appear in the resulting string -intact. Note that other exceptions may still be raised, including -\exception{ValueError} as described above. +intact. Also, unlike with \method{substitute()}, any other appearances of the +\samp{\$} will simply return \samp{\$} instead of raising +\exception{ValueError}. + +While other exceptions may still occur, this method is called ``safe'' because +substitutions always tries to return a usable string instead of raising an +exception. In another sense, \method{safe_substitute()} may be anything other +than safe, since it will silently ignore malformed templates containing +dangling delimiters, unmatched braces, or placeholders that are not valid +Python identifiers. \end{methoddesc} \class{Template} instances also provide one public data attribute: diff --git a/Lib/string.py b/Lib/string.py index e10087eb2a8..7c0e001a79a 100644 --- a/Lib/string.py +++ b/Lib/string.py @@ -199,7 +199,7 @@ class Template: if mo.group('escaped') is not None: return self.delimiter if mo.group('invalid') is not None: - self._invalid(mo) + return self.delimiter raise ValueError('Unrecognized named group in pattern', self.pattern) return self.pattern.sub(convert, self.template) diff --git a/Lib/test/test_pep292.py b/Lib/test/test_pep292.py index 19952e42399..2a4353adbd9 100644 --- a/Lib/test/test_pep292.py +++ b/Lib/test/test_pep292.py @@ -163,20 +163,19 @@ class TestTemplate(unittest.TestCase): raises(TypeError, s.safe_substitute, d, {}) def test_delimiter_override(self): + eq = self.assertEqual + raises = self.assertRaises class AmpersandTemplate(Template): delimiter = '&' s = AmpersandTemplate('this &gift is for &{who} &&') - self.assertEqual(s.substitute(gift='bud', who='you'), - 'this bud is for you &') - self.assertRaises(KeyError, s.substitute) - self.assertEqual(s.safe_substitute(gift='bud', who='you'), - 'this bud is for you &') - self.assertEqual(s.safe_substitute(), - 'this &gift is for &{who} &') + eq(s.substitute(gift='bud', who='you'), 'this bud is for you &') + raises(KeyError, s.substitute) + eq(s.safe_substitute(gift='bud', who='you'), 'this bud is for you &') + eq(s.safe_substitute(), 'this &gift is for &{who} &') s = AmpersandTemplate('this &gift is for &{who} &') - self.assertRaises(ValueError, s.substitute, - dict(gift='bud', who='you')) - self.assertRaises(ValueError, s.safe_substitute) + raises(ValueError, s.substitute, dict(gift='bud', who='you')) + eq(s.safe_substitute(), 'this &gift is for &{who} &') + def test_main(): from test import test_support diff --git a/Misc/NEWS b/Misc/NEWS index 001152a32ee..a75dff1b44d 100644 --- a/Misc/NEWS +++ b/Misc/NEWS @@ -56,6 +56,10 @@ Extension Modules Library ------- +- Patch #1056967 changes the semantics of Template.safe_substitute() so that + no ValueError is raised on an 'invalid' match group. Now the delimiter is + returned. + - Bug #1052503 pdb.runcall() was not passing along keyword arguments. - Bug #902037: XML.sax.saxutils.prepare_input_source() now combines relative