test_parser.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. """
  2. Testing some internals of the template processing.
  3. These are *not* examples to be copied in user code.
  4. """
  5. from django.template import Library, TemplateSyntaxError
  6. from django.template.base import (
  7. FilterExpression,
  8. Lexer,
  9. Parser,
  10. Token,
  11. TokenType,
  12. Variable,
  13. VariableDoesNotExist,
  14. )
  15. from django.template.defaultfilters import register as filter_library
  16. from django.test import SimpleTestCase
  17. class ParserTests(SimpleTestCase):
  18. def test_token_smart_split(self):
  19. """
  20. #7027 -- _() syntax should work with spaces
  21. """
  22. token = Token(
  23. TokenType.BLOCK, 'sometag _("Page not found") value|yesno:_("yes,no")'
  24. )
  25. split = token.split_contents()
  26. self.assertEqual(
  27. split, ["sometag", '_("Page not found")', 'value|yesno:_("yes,no")']
  28. )
  29. def test_repr(self):
  30. token = Token(TokenType.BLOCK, "some text")
  31. self.assertEqual(repr(token), '<Block token: "some text...">')
  32. parser = Parser([token], builtins=[filter_library])
  33. self.assertEqual(
  34. repr(parser),
  35. '<Parser tokens=[<Block token: "some text...">]>',
  36. )
  37. filter_expression = FilterExpression("news|upper", parser)
  38. self.assertEqual(repr(filter_expression), "<FilterExpression 'news|upper'>")
  39. lexer = Lexer("{% for i in 1 %}{{ a }}\n{% endfor %}")
  40. self.assertEqual(
  41. repr(lexer),
  42. '<Lexer template_string="{% for i in 1 %}{{ a...", verbatim=False>',
  43. )
  44. def test_filter_parsing(self):
  45. c = {"article": {"section": "News"}}
  46. p = Parser("", builtins=[filter_library])
  47. def fe_test(s, val):
  48. self.assertEqual(FilterExpression(s, p).resolve(c), val)
  49. fe_test("article.section", "News")
  50. fe_test("article.section|upper", "NEWS")
  51. fe_test('"News"', "News")
  52. fe_test("'News'", "News")
  53. fe_test(r'"Some \"Good\" News"', 'Some "Good" News')
  54. fe_test(r'"Some \"Good\" News"', 'Some "Good" News')
  55. fe_test(r"'Some \'Bad\' News'", "Some 'Bad' News")
  56. fe = FilterExpression(r'"Some \"Good\" News"', p)
  57. self.assertEqual(fe.filters, [])
  58. self.assertEqual(fe.var, 'Some "Good" News')
  59. # Filtered variables should reject access of attributes beginning with
  60. # underscores.
  61. msg = (
  62. "Variables and attributes may not begin with underscores: 'article._hidden'"
  63. )
  64. with self.assertRaisesMessage(TemplateSyntaxError, msg):
  65. FilterExpression("article._hidden|upper", p)
  66. def test_cannot_parse_characters(self):
  67. p = Parser("", builtins=[filter_library])
  68. for filter_expression, characters in [
  69. ('<>|default:"Default"|upper', '|<>||default:"Default"|upper'),
  70. ("test|<>|upper", "test||<>||upper"),
  71. ]:
  72. with self.subTest(filter_expression=filter_expression):
  73. with self.assertRaisesMessage(
  74. TemplateSyntaxError,
  75. f"Could not parse some characters: {characters}",
  76. ):
  77. FilterExpression(filter_expression, p)
  78. def test_cannot_find_variable(self):
  79. p = Parser("", builtins=[filter_library])
  80. with self.assertRaisesMessage(
  81. TemplateSyntaxError,
  82. 'Could not find variable at start of |default:"Default"',
  83. ):
  84. FilterExpression('|default:"Default"', p)
  85. def test_variable_parsing(self):
  86. c = {"article": {"section": "News"}}
  87. self.assertEqual(Variable("article.section").resolve(c), "News")
  88. self.assertEqual(Variable('"News"').resolve(c), "News")
  89. self.assertEqual(Variable("'News'").resolve(c), "News")
  90. # Translated strings are handled correctly.
  91. self.assertEqual(Variable("_(article.section)").resolve(c), "News")
  92. self.assertEqual(Variable('_("Good News")').resolve(c), "Good News")
  93. self.assertEqual(Variable("_('Better News')").resolve(c), "Better News")
  94. # Escaped quotes work correctly as well.
  95. self.assertEqual(
  96. Variable(r'"Some \"Good\" News"').resolve(c), 'Some "Good" News'
  97. )
  98. self.assertEqual(
  99. Variable(r"'Some \'Better\' News'").resolve(c), "Some 'Better' News"
  100. )
  101. # Variables should reject access of attributes and variables beginning
  102. # with underscores.
  103. for name in ["article._hidden", "_article"]:
  104. msg = f"Variables and attributes may not begin with underscores: '{name}'"
  105. with self.assertRaisesMessage(TemplateSyntaxError, msg):
  106. Variable(name)
  107. # Variables should raise on non string type
  108. with self.assertRaisesMessage(
  109. TypeError, "Variable must be a string or number, got <class 'dict'>"
  110. ):
  111. Variable({})
  112. def test_filter_args_count(self):
  113. parser = Parser("")
  114. register = Library()
  115. @register.filter
  116. def no_arguments(value):
  117. pass
  118. @register.filter
  119. def one_argument(value, arg):
  120. pass
  121. @register.filter
  122. def one_opt_argument(value, arg=False):
  123. pass
  124. @register.filter
  125. def two_arguments(value, arg, arg2):
  126. pass
  127. @register.filter
  128. def two_one_opt_arg(value, arg, arg2=False):
  129. pass
  130. parser.add_library(register)
  131. for expr in (
  132. '1|no_arguments:"1"',
  133. "1|two_arguments",
  134. '1|two_arguments:"1"',
  135. "1|two_one_opt_arg",
  136. ):
  137. with self.assertRaises(TemplateSyntaxError):
  138. FilterExpression(expr, parser)
  139. for expr in (
  140. # Correct number of arguments
  141. "1|no_arguments",
  142. '1|one_argument:"1"',
  143. # One optional
  144. "1|one_opt_argument",
  145. '1|one_opt_argument:"1"',
  146. # Not supplying all
  147. '1|two_one_opt_arg:"1"',
  148. ):
  149. FilterExpression(expr, parser)
  150. def test_filter_numeric_argument_parsing(self):
  151. p = Parser("", builtins=[filter_library])
  152. cases = {
  153. "5": 5,
  154. "-5": -5,
  155. "5.2": 5.2,
  156. ".4": 0.4,
  157. "5.2e3": 5200.0, # 5.2 × 10³ = 5200.0.
  158. "5.2E3": 5200.0, # Case-insensitive.
  159. "5.2e-3": 0.0052, # Negative exponent.
  160. "-1.5E4": -15000.0,
  161. "+3.0e2": 300.0,
  162. ".5e2": 50.0, # 0.5 × 10² = 50.0
  163. }
  164. for num, expected in cases.items():
  165. with self.subTest(num=num):
  166. self.assertEqual(FilterExpression(num, p).resolve({}), expected)
  167. self.assertEqual(
  168. FilterExpression(f"0|default:{num}", p).resolve({}), expected
  169. )
  170. invalid_numbers = [
  171. "abc123",
  172. "123abc",
  173. "foo",
  174. "error",
  175. "1e",
  176. "e400",
  177. "1e.2",
  178. "1e2.",
  179. "1e2.0",
  180. "1e2a",
  181. "1e2e3",
  182. "1e-",
  183. "1e-a",
  184. ]
  185. for num in invalid_numbers:
  186. with self.subTest(num=num):
  187. self.assertIsNone(
  188. FilterExpression(num, p).resolve({}, ignore_failures=True)
  189. )
  190. with self.assertRaises(VariableDoesNotExist):
  191. FilterExpression(f"0|default:{num}", p).resolve({})