mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 21:51:50 +00:00 
			
		
		
		
	bpo-34155: Dont parse domains containing @ (GH-13079)
Before:
    
        >>> email.message_from_string('From: a@malicious.org@important.com', policy=email.policy.default)['from'].addresses
        (Address(display_name='', username='a', domain='malicious.org'),)
    
        >>> parseaddr('a@malicious.org@important.com')
        ('', 'a@malicious.org')
    
    After:
    
        >>> email.message_from_string('From: a@malicious.org@important.com', policy=email.policy.default)['from'].addresses
        (Address(display_name='', username='', domain=''),)
    
        >>> parseaddr('a@malicious.org@important.com')
        ('', 'a@')
https://bugs.python.org/issue34155
			
			
This commit is contained in:
		
							parent
							
								
									719a062bcb
								
							
						
					
					
						commit
						8cb65d1381
					
				
					 5 changed files with 37 additions and 1 deletions
				
			
		|  | @ -1587,6 +1587,8 @@ def get_domain(value): | ||||||
|         token, value = get_dot_atom(value) |         token, value = get_dot_atom(value) | ||||||
|     except errors.HeaderParseError: |     except errors.HeaderParseError: | ||||||
|         token, value = get_atom(value) |         token, value = get_atom(value) | ||||||
|  |     if value and value[0] == '@': | ||||||
|  |         raise errors.HeaderParseError('Invalid Domain') | ||||||
|     if leader is not None: |     if leader is not None: | ||||||
|         token[:0] = [leader] |         token[:0] = [leader] | ||||||
|     domain.append(token) |     domain.append(token) | ||||||
|  |  | ||||||
|  | @ -379,7 +379,12 @@ def getaddrspec(self): | ||||||
|         aslist.append('@') |         aslist.append('@') | ||||||
|         self.pos += 1 |         self.pos += 1 | ||||||
|         self.gotonext() |         self.gotonext() | ||||||
|         return EMPTYSTRING.join(aslist) + self.getdomain() |         domain = self.getdomain() | ||||||
|  |         if not domain: | ||||||
|  |             # Invalid domain, return an empty address instead of returning a | ||||||
|  |             # local part to denote failed parsing. | ||||||
|  |             return EMPTYSTRING | ||||||
|  |         return EMPTYSTRING.join(aslist) + domain | ||||||
| 
 | 
 | ||||||
|     def getdomain(self): |     def getdomain(self): | ||||||
|         """Get the complete domain name from an address.""" |         """Get the complete domain name from an address.""" | ||||||
|  | @ -394,6 +399,10 @@ def getdomain(self): | ||||||
|             elif self.field[self.pos] == '.': |             elif self.field[self.pos] == '.': | ||||||
|                 self.pos += 1 |                 self.pos += 1 | ||||||
|                 sdlist.append('.') |                 sdlist.append('.') | ||||||
|  |             elif self.field[self.pos] == '@': | ||||||
|  |                 # bpo-34155: Don't parse domains with two `@` like | ||||||
|  |                 # `a@malicious.org@important.com`. | ||||||
|  |                 return EMPTYSTRING | ||||||
|             elif self.field[self.pos] in self.atomends: |             elif self.field[self.pos] in self.atomends: | ||||||
|                 break |                 break | ||||||
|             else: |             else: | ||||||
|  |  | ||||||
|  | @ -1448,6 +1448,16 @@ def test_get_addr_spec_dot_atom(self): | ||||||
|         self.assertEqual(addr_spec.domain, 'example.com') |         self.assertEqual(addr_spec.domain, 'example.com') | ||||||
|         self.assertEqual(addr_spec.addr_spec, 'star.a.star@example.com') |         self.assertEqual(addr_spec.addr_spec, 'star.a.star@example.com') | ||||||
| 
 | 
 | ||||||
|  |     def test_get_addr_spec_multiple_domains(self): | ||||||
|  |         with self.assertRaises(errors.HeaderParseError): | ||||||
|  |             parser.get_addr_spec('star@a.star@example.com') | ||||||
|  | 
 | ||||||
|  |         with self.assertRaises(errors.HeaderParseError): | ||||||
|  |             parser.get_addr_spec('star@a@example.com') | ||||||
|  | 
 | ||||||
|  |         with self.assertRaises(errors.HeaderParseError): | ||||||
|  |             parser.get_addr_spec('star@172.17.0.1@example.com') | ||||||
|  | 
 | ||||||
|     # get_obs_route |     # get_obs_route | ||||||
| 
 | 
 | ||||||
|     def test_get_obs_route_simple(self): |     def test_get_obs_route_simple(self): | ||||||
|  |  | ||||||
|  | @ -3041,6 +3041,20 @@ def test_parseaddr_empty(self): | ||||||
|         self.assertEqual(utils.parseaddr('<>'), ('', '')) |         self.assertEqual(utils.parseaddr('<>'), ('', '')) | ||||||
|         self.assertEqual(utils.formataddr(utils.parseaddr('<>')), '') |         self.assertEqual(utils.formataddr(utils.parseaddr('<>')), '') | ||||||
| 
 | 
 | ||||||
|  |     def test_parseaddr_multiple_domains(self): | ||||||
|  |         self.assertEqual( | ||||||
|  |             utils.parseaddr('a@b@c'), | ||||||
|  |             ('', '') | ||||||
|  |         ) | ||||||
|  |         self.assertEqual( | ||||||
|  |             utils.parseaddr('a@b.c@c'), | ||||||
|  |             ('', '') | ||||||
|  |         ) | ||||||
|  |         self.assertEqual( | ||||||
|  |             utils.parseaddr('a@172.17.0.1@c'), | ||||||
|  |             ('', '') | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|     def test_noquote_dump(self): |     def test_noquote_dump(self): | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             utils.formataddr(('A Silly Person', 'person@dom.ain')), |             utils.formataddr(('A Silly Person', 'person@dom.ain')), | ||||||
|  |  | ||||||
|  | @ -0,0 +1 @@ | ||||||
|  | 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. | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 jpic
						jpic