You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

129 lines
5.3 KiB

  1. From 77bb21f6e06aabc81d672dbdd6f8834c40544351 Mon Sep 17 00:00:00 2001
  2. From: jpic <jpic@users.noreply.github.com>
  3. Date: Wed, 17 Jul 2019 23:54:25 +0200
  4. Subject: [PATCH] bpo-34155: Dont parse domains containing @ (GH-13079)
  5. Before:
  6. >>> email.message_from_string('From: a@malicious.org@important.com', policy=email.policy.default)['from'].addresses
  7. (Address(display_name='', username='a', domain='malicious.org'),)
  8. >>> parseaddr('a@malicious.org@important.com')
  9. ('', 'a@malicious.org')
  10. After:
  11. >>> email.message_from_string('From: a@malicious.org@important.com', policy=email.policy.default)['from'].addresses
  12. (Address(display_name='', username='', domain=''),)
  13. >>> parseaddr('a@malicious.org@important.com')
  14. ('', 'a@')
  15. https://bugs.python.org/issue34155
  16. (cherry picked from commit 8cb65d1381b027f0b09ee36bfed7f35bb4dec9a9)
  17. Co-authored-by: jpic <jpic@users.noreply.github.com>
  18. ---
  19. Lib/email/_header_value_parser.py | 2 ++
  20. Lib/email/_parseaddr.py | 11 ++++++++++-
  21. Lib/test/test_email/test__header_value_parser.py | 10 ++++++++++
  22. Lib/test/test_email/test_email.py | 14 ++++++++++++++
  23. .../2019-05-04-13-33-37.bpo-34155.MJll68.rst | 1 +
  24. 5 files changed, 37 insertions(+), 1 deletion(-)
  25. create mode 100644 Misc/NEWS.d/next/Security/2019-05-04-13-33-37.bpo-34155.MJll68.rst
  26. diff --git a/Lib/email/_header_value_parser.py b/Lib/email/_header_value_parser.py
  27. index 801ae728dd136..c09f4f121ffb6 100644
  28. --- a/Lib/email/_header_value_parser.py
  29. +++ b/Lib/email/_header_value_parser.py
  30. @@ -1585,6 +1585,8 @@ def get_domain(value):
  31. token, value = get_dot_atom(value)
  32. except errors.HeaderParseError:
  33. token, value = get_atom(value)
  34. + if value and value[0] == '@':
  35. + raise errors.HeaderParseError('Invalid Domain')
  36. if leader is not None:
  37. token[:0] = [leader]
  38. domain.append(token)
  39. diff --git a/Lib/email/_parseaddr.py b/Lib/email/_parseaddr.py
  40. index cdfa3729adc79..41ff6f8c000d5 100644
  41. --- a/Lib/email/_parseaddr.py
  42. +++ b/Lib/email/_parseaddr.py
  43. @@ -379,7 +379,12 @@ def getaddrspec(self):
  44. aslist.append('@')
  45. self.pos += 1
  46. self.gotonext()
  47. - return EMPTYSTRING.join(aslist) + self.getdomain()
  48. + domain = self.getdomain()
  49. + if not domain:
  50. + # Invalid domain, return an empty address instead of returning a
  51. + # local part to denote failed parsing.
  52. + return EMPTYSTRING
  53. + return EMPTYSTRING.join(aslist) + domain
  54. def getdomain(self):
  55. """Get the complete domain name from an address."""
  56. @@ -394,6 +399,10 @@ def getdomain(self):
  57. elif self.field[self.pos] == '.':
  58. self.pos += 1
  59. sdlist.append('.')
  60. + elif self.field[self.pos] == '@':
  61. + # bpo-34155: Don't parse domains with two `@` like
  62. + # `a@malicious.org@important.com`.
  63. + return EMPTYSTRING
  64. elif self.field[self.pos] in self.atomends:
  65. break
  66. else:
  67. diff --git a/Lib/test/test_email/test__header_value_parser.py b/Lib/test/test_email/test__header_value_parser.py
  68. index 9e862feab10c9..0f19f8bcc2e0f 100644
  69. --- a/Lib/test/test_email/test__header_value_parser.py
  70. +++ b/Lib/test/test_email/test__header_value_parser.py
  71. @@ -1448,6 +1448,16 @@ def test_get_addr_spec_dot_atom(self):
  72. self.assertEqual(addr_spec.domain, 'example.com')
  73. self.assertEqual(addr_spec.addr_spec, 'star.a.star@example.com')
  74. + def test_get_addr_spec_multiple_domains(self):
  75. + with self.assertRaises(errors.HeaderParseError):
  76. + parser.get_addr_spec('star@a.star@example.com')
  77. +
  78. + with self.assertRaises(errors.HeaderParseError):
  79. + parser.get_addr_spec('star@a@example.com')
  80. +
  81. + with self.assertRaises(errors.HeaderParseError):
  82. + parser.get_addr_spec('star@172.17.0.1@example.com')
  83. +
  84. # get_obs_route
  85. def test_get_obs_route_simple(self):
  86. diff --git a/Lib/test/test_email/test_email.py b/Lib/test/test_email/test_email.py
  87. index c29cc56203b1f..aa775881c5521 100644
  88. --- a/Lib/test/test_email/test_email.py
  89. +++ b/Lib/test/test_email/test_email.py
  90. @@ -3041,6 +3041,20 @@ def test_parseaddr_empty(self):
  91. self.assertEqual(utils.parseaddr('<>'), ('', ''))
  92. self.assertEqual(utils.formataddr(utils.parseaddr('<>')), '')
  93. + def test_parseaddr_multiple_domains(self):
  94. + self.assertEqual(
  95. + utils.parseaddr('a@b@c'),
  96. + ('', '')
  97. + )
  98. + self.assertEqual(
  99. + utils.parseaddr('a@b.c@c'),
  100. + ('', '')
  101. + )
  102. + self.assertEqual(
  103. + utils.parseaddr('a@172.17.0.1@c'),
  104. + ('', '')
  105. + )
  106. +
  107. def test_noquote_dump(self):
  108. self.assertEqual(
  109. utils.formataddr(('A Silly Person', 'person@dom.ain')),
  110. diff --git a/Misc/NEWS.d/next/Security/2019-05-04-13-33-37.bpo-34155.MJll68.rst b/Misc/NEWS.d/next/Security/2019-05-04-13-33-37.bpo-34155.MJll68.rst
  111. new file mode 100644
  112. index 0000000000000..50292e29ed1d2
  113. --- /dev/null
  114. +++ b/Misc/NEWS.d/next/Security/2019-05-04-13-33-37.bpo-34155.MJll68.rst
  115. @@ -0,0 +1 @@
  116. +Fix parsing of invalid email addresses with more than one ``@`` (e.g. a@b@c.com.) to not return the part before 2nd ``@`` as valid email address. Patch by maxking & jpic.