mirror of
				https://github.com/django/django.git
				synced 2025-10-31 09:41:08 +00:00 
			
		
		
		
	Regression in 5183f7c287.
Thank you to Jon Banafato and Baptiste Mispelon for the report.
		
	
		
			
				
	
	
		
			243 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			243 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | ||
| Testing some internals of the template processing.
 | ||
| These are *not* examples to be copied in user code.
 | ||
| """
 | ||
| 
 | ||
| from django.template import Library, TemplateSyntaxError
 | ||
| from django.template.base import (
 | ||
|     FilterExpression,
 | ||
|     Lexer,
 | ||
|     Parser,
 | ||
|     Token,
 | ||
|     TokenType,
 | ||
|     Variable,
 | ||
|     VariableDoesNotExist,
 | ||
| )
 | ||
| from django.template.defaultfilters import register as filter_library
 | ||
| from django.test import SimpleTestCase
 | ||
| 
 | ||
| 
 | ||
| class ParserTests(SimpleTestCase):
 | ||
|     def test_token_smart_split(self):
 | ||
|         """
 | ||
|         #7027 -- _() syntax should work with spaces
 | ||
|         """
 | ||
|         token = Token(
 | ||
|             TokenType.BLOCK, 'sometag _("Page not found") value|yesno:_("yes,no")'
 | ||
|         )
 | ||
|         split = token.split_contents()
 | ||
|         self.assertEqual(
 | ||
|             split, ["sometag", '_("Page not found")', 'value|yesno:_("yes,no")']
 | ||
|         )
 | ||
| 
 | ||
|     def test_repr(self):
 | ||
|         token = Token(TokenType.BLOCK, "some text")
 | ||
|         self.assertEqual(repr(token), '<Block token: "some text...">')
 | ||
|         parser = Parser([token], builtins=[filter_library])
 | ||
|         self.assertEqual(
 | ||
|             repr(parser),
 | ||
|             '<Parser tokens=[<Block token: "some text...">]>',
 | ||
|         )
 | ||
|         filter_expression = FilterExpression("news|upper", parser)
 | ||
|         self.assertEqual(repr(filter_expression), "<FilterExpression 'news|upper'>")
 | ||
|         lexer = Lexer("{% for i in 1 %}{{ a }}\n{% endfor %}")
 | ||
|         self.assertEqual(
 | ||
|             repr(lexer),
 | ||
|             '<Lexer template_string="{% for i in 1 %}{{ a...", verbatim=False>',
 | ||
|         )
 | ||
| 
 | ||
|     def test_filter_parsing(self):
 | ||
|         c = {"article": {"section": "News"}}
 | ||
|         p = Parser("", builtins=[filter_library])
 | ||
| 
 | ||
|         def fe_test(s, val):
 | ||
|             self.assertEqual(FilterExpression(s, p).resolve(c), val)
 | ||
| 
 | ||
|         fe_test("article.section", "News")
 | ||
|         fe_test("article.section|upper", "NEWS")
 | ||
|         fe_test('"News"', "News")
 | ||
|         fe_test("'News'", "News")
 | ||
|         fe_test(r'"Some \"Good\" News"', 'Some "Good" News')
 | ||
|         fe_test(r'"Some \"Good\" News"', 'Some "Good" News')
 | ||
|         fe_test(r"'Some \'Bad\' News'", "Some 'Bad' News")
 | ||
| 
 | ||
|         fe = FilterExpression(r'"Some \"Good\" News"', p)
 | ||
|         self.assertEqual(fe.filters, [])
 | ||
|         self.assertEqual(fe.var, 'Some "Good" News')
 | ||
| 
 | ||
|         # Filtered variables should reject access of attributes beginning with
 | ||
|         # underscores.
 | ||
|         msg = (
 | ||
|             "Variables and attributes may not begin with underscores: 'article._hidden'"
 | ||
|         )
 | ||
|         with self.assertRaisesMessage(TemplateSyntaxError, msg):
 | ||
|             FilterExpression("article._hidden|upper", p)
 | ||
| 
 | ||
|     def test_cannot_parse_characters(self):
 | ||
|         p = Parser("", builtins=[filter_library])
 | ||
|         for filter_expression, characters in [
 | ||
|             ('<>|default:"Default"|upper', '|<>||default:"Default"|upper'),
 | ||
|             ("test|<>|upper", "test||<>||upper"),
 | ||
|         ]:
 | ||
|             with self.subTest(filter_expression=filter_expression):
 | ||
|                 with self.assertRaisesMessage(
 | ||
|                     TemplateSyntaxError,
 | ||
|                     f"Could not parse some characters: {characters}",
 | ||
|                 ):
 | ||
|                     FilterExpression(filter_expression, p)
 | ||
| 
 | ||
|     def test_cannot_find_variable(self):
 | ||
|         p = Parser("", builtins=[filter_library])
 | ||
|         with self.assertRaisesMessage(
 | ||
|             TemplateSyntaxError,
 | ||
|             'Could not find variable at start of |default:"Default"',
 | ||
|         ):
 | ||
|             FilterExpression('|default:"Default"', p)
 | ||
| 
 | ||
|     def test_variable_parsing(self):
 | ||
|         c = {"article": {"section": "News"}}
 | ||
|         self.assertEqual(Variable("article.section").resolve(c), "News")
 | ||
|         self.assertEqual(Variable('"News"').resolve(c), "News")
 | ||
|         self.assertEqual(Variable("'News'").resolve(c), "News")
 | ||
| 
 | ||
|         # Translated strings are handled correctly.
 | ||
|         self.assertEqual(Variable("_(article.section)").resolve(c), "News")
 | ||
|         self.assertEqual(Variable('_("Good News")').resolve(c), "Good News")
 | ||
|         self.assertEqual(Variable("_('Better News')").resolve(c), "Better News")
 | ||
| 
 | ||
|         # Escaped quotes work correctly as well.
 | ||
|         self.assertEqual(
 | ||
|             Variable(r'"Some \"Good\" News"').resolve(c), 'Some "Good" News'
 | ||
|         )
 | ||
|         self.assertEqual(
 | ||
|             Variable(r"'Some \'Better\' News'").resolve(c), "Some 'Better' News"
 | ||
|         )
 | ||
| 
 | ||
|         # Variables should reject access of attributes and variables beginning
 | ||
|         # with underscores.
 | ||
|         for name in ["article._hidden", "_article"]:
 | ||
|             msg = f"Variables and attributes may not begin with underscores: '{name}'"
 | ||
|             with self.assertRaisesMessage(TemplateSyntaxError, msg):
 | ||
|                 Variable(name)
 | ||
| 
 | ||
|         # Variables should raise on non string type
 | ||
|         with self.assertRaisesMessage(
 | ||
|             TypeError, "Variable must be a string or number, got <class 'dict'>"
 | ||
|         ):
 | ||
|             Variable({})
 | ||
| 
 | ||
|         # Variables should raise when invalid characters in name.
 | ||
|         for c in ["+", "-"]:
 | ||
|             with self.subTest(invalid_character=c):
 | ||
|                 variable_name = f"variable{c}name"
 | ||
|                 with self.assertRaisesMessage(
 | ||
|                     TemplateSyntaxError,
 | ||
|                     f"Invalid character ('{c}') in variable name: '{variable_name}'",
 | ||
|                 ):
 | ||
|                     Variable(variable_name)
 | ||
| 
 | ||
|     def test_filter_args_count(self):
 | ||
|         parser = Parser("")
 | ||
|         register = Library()
 | ||
| 
 | ||
|         @register.filter
 | ||
|         def no_arguments(value):
 | ||
|             pass
 | ||
| 
 | ||
|         @register.filter
 | ||
|         def one_argument(value, arg):
 | ||
|             pass
 | ||
| 
 | ||
|         @register.filter
 | ||
|         def one_opt_argument(value, arg=False):
 | ||
|             pass
 | ||
| 
 | ||
|         @register.filter
 | ||
|         def two_arguments(value, arg, arg2):
 | ||
|             pass
 | ||
| 
 | ||
|         @register.filter
 | ||
|         def two_one_opt_arg(value, arg, arg2=False):
 | ||
|             pass
 | ||
| 
 | ||
|         parser.add_library(register)
 | ||
|         for expr in (
 | ||
|             '1|no_arguments:"1"',
 | ||
|             "1|two_arguments",
 | ||
|             '1|two_arguments:"1"',
 | ||
|             "1|two_one_opt_arg",
 | ||
|         ):
 | ||
|             with self.assertRaises(TemplateSyntaxError):
 | ||
|                 FilterExpression(expr, parser)
 | ||
|         for expr in (
 | ||
|             # Correct number of arguments
 | ||
|             "1|no_arguments",
 | ||
|             '1|one_argument:"1"',
 | ||
|             # One optional
 | ||
|             "1|one_opt_argument",
 | ||
|             '1|one_opt_argument:"1"',
 | ||
|             # Not supplying all
 | ||
|             '1|two_one_opt_arg:"1"',
 | ||
|         ):
 | ||
|             FilterExpression(expr, parser)
 | ||
| 
 | ||
|     def test_filter_numeric_argument_parsing(self):
 | ||
|         p = Parser("", builtins=[filter_library])
 | ||
| 
 | ||
|         # Values that resolve to a numeric literal.
 | ||
|         cases = {
 | ||
|             "5": 5,
 | ||
|             "-5": -5,
 | ||
|             "5.2": 5.2,
 | ||
|             ".4": 0.4,
 | ||
|             "5.2e3": 5200.0,  # 5.2 × 10³ = 5200.0.
 | ||
|             "5.2E3": 5200.0,  # Case-insensitive.
 | ||
|             "5.2e-3": 0.0052,  # Negative exponent.
 | ||
|             "-1.5E4": -15000.0,
 | ||
|             "+3.0e2": 300.0,
 | ||
|             ".5e2": 50.0,  # 0.5 × 10² = 50.0
 | ||
|         }
 | ||
|         for num, expected in cases.items():
 | ||
|             with self.subTest(num=num):
 | ||
|                 self.assertEqual(FilterExpression(num, p).resolve({}), expected)
 | ||
|                 self.assertEqual(
 | ||
|                     FilterExpression(f"0|default:{num}", p).resolve({}), expected
 | ||
|                 )
 | ||
| 
 | ||
|         # Values that are interpreted as names of variables that do not exist.
 | ||
|         invalid_numbers = [
 | ||
|             "abc123",
 | ||
|             "123abc",
 | ||
|             "foo",
 | ||
|             "error",
 | ||
|             "1e",
 | ||
|             "e400",
 | ||
|             "1e.2",
 | ||
|             "1e2.",
 | ||
|             "1e2.0",
 | ||
|             "1e2a",
 | ||
|             "1e2e3",
 | ||
|         ]
 | ||
| 
 | ||
|         for num in invalid_numbers:
 | ||
|             with self.subTest(num=num):
 | ||
|                 self.assertIsNone(
 | ||
|                     FilterExpression(num, p).resolve({}, ignore_failures=True)
 | ||
|                 )
 | ||
|                 with self.assertRaises(VariableDoesNotExist):
 | ||
|                     FilterExpression(f"0|default:{num}", p).resolve({})
 | ||
| 
 | ||
|         # Values that are interpreted as an invalid variable name.
 | ||
|         invalid_numbers_and_var_names = [
 | ||
|             "1e-",
 | ||
|             "1e-a",
 | ||
|             "1+1",
 | ||
|             "1-1",
 | ||
|         ]
 | ||
|         for num in invalid_numbers_and_var_names:
 | ||
|             with self.subTest(num=num):
 | ||
|                 with self.assertRaises(TemplateSyntaxError):
 | ||
|                     FilterExpression(num, p).resolve({})
 | ||
|                 with self.assertRaises(TemplateSyntaxError):
 | ||
|                     FilterExpression(f"0|default:{num}", p).resolve({})
 |