mirror of
				https://github.com/yiisoft/yii2.git
				synced 2025-11-01 03:26:36 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			207 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * @link https://www.yiiframework.com/
 | |
|  * @copyright Copyright (c) 2008 Yii Software LLC
 | |
|  * @license https://www.yiiframework.com/license/
 | |
|  */
 | |
| 
 | |
| namespace yiiunit\framework\validators;
 | |
| 
 | |
| use yii\validators\EmailValidator;
 | |
| use yiiunit\data\validators\models\FakedValidationModel;
 | |
| use yiiunit\TestCase;
 | |
| 
 | |
| /**
 | |
|  * @group validators
 | |
|  */
 | |
| class EmailValidatorTest extends TestCase
 | |
| {
 | |
|     protected function setUp(): void
 | |
|     {
 | |
|         parent::setUp();
 | |
| 
 | |
|         // destroy application, Validator must work without Yii::$app
 | |
|         $this->destroyApplication();
 | |
|     }
 | |
| 
 | |
|     public function testValidateValue()
 | |
|     {
 | |
|         $validator = new EmailValidator();
 | |
| 
 | |
|         $this->assertTrue($validator->validate('sam@rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('5011@gmail.com'));
 | |
|         $this->assertTrue($validator->validate('Abc.123@example.com'));
 | |
|         $this->assertTrue($validator->validate('user+mailbox/department=shipping@example.com'));
 | |
|         $this->assertTrue($validator->validate('!#$%&\'*+-/=?^_`.{|}~@example.com'));
 | |
|         $this->assertTrue($validator->validate('firstName.x.lastName.-nd@example.com'));
 | |
|         $this->assertFalse($validator->validate('rmcreative.ru'));
 | |
|         $this->assertFalse($validator->validate('Carsten Brandt <mail@cebe.cc>'));
 | |
|         $this->assertFalse($validator->validate('"Carsten Brandt" <mail@cebe.cc>'));
 | |
|         $this->assertFalse($validator->validate('<mail@cebe.cc>'));
 | |
|         $this->assertFalse($validator->validate('info@örtliches.de'));
 | |
|         $this->assertFalse($validator->validate('sam@рмкреатиф.ru'));
 | |
|         $this->assertFalse($validator->validate('ex..ample@example.com'));
 | |
|         $this->assertFalse($validator->validate(['developer@yiiframework.com']));
 | |
| 
 | |
|         $validator->allowName = true;
 | |
| 
 | |
|         $this->assertTrue($validator->validate('sam@rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('5011@gmail.com'));
 | |
|         $this->assertFalse($validator->validate('rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('Carsten Brandt <mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('"Carsten Brandt" <mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('<mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('"FirstName LastName" <firstName.x.lastName.-nd@example.com>'));
 | |
|         $this->assertFalse($validator->validate('info@örtliches.de'));
 | |
|         $this->assertFalse($validator->validate('üñîçøðé@üñîçøðé.com'));
 | |
|         $this->assertFalse($validator->validate('sam@рмкреатиф.ru'));
 | |
|         $this->assertFalse($validator->validate('Informtation info@oertliches.de'));
 | |
|         $this->assertTrue($validator->validate('test@example.com'));
 | |
|         $this->assertTrue($validator->validate('John Smith <john.smith@example.com>'));
 | |
|         $this->assertTrue($validator->validate('"This name is longer than 64 characters. Blah blah blah blah blah" <shortmail@example.com>'));
 | |
|         $this->assertFalse($validator->validate('John Smith <example.com>'));
 | |
|         $this->assertFalse($validator->validate('Short Name <localPartMoreThan64Characters-blah-blah-blah-blah-blah-blah-blah-blah@example.com>'));
 | |
|         $this->assertFalse($validator->validate('Short Name <domainNameIsMoreThan254Characters@example-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah-blah.com>'));
 | |
|         $this->assertFalse($validator->validate(['developer@yiiframework.com']));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     public function testValidateValueIdn()
 | |
|     {
 | |
|         if (!function_exists('idn_to_ascii')) {
 | |
|             $this->markTestSkipped('Intl extension required');
 | |
| 
 | |
|             return;
 | |
|         }
 | |
|         $validator = new EmailValidator();
 | |
|         $validator->enableIDN = true;
 | |
| 
 | |
|         $this->assertTrue($validator->validate('5011@example.com'));
 | |
|         $this->assertTrue($validator->validate('example@äüößìà.de'));
 | |
|         $this->assertTrue($validator->validate('example@xn--zcack7ayc9a.de'));
 | |
|         $this->assertTrue($validator->validate('info@örtliches.de'));
 | |
|         $this->assertTrue($validator->validate('sam@рмкреатиф.ru'));
 | |
|         $this->assertTrue($validator->validate('sam@rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('5011@gmail.com'));
 | |
|         $this->assertTrue($validator->validate('üñîçøðé@üñîçøðé.com'));
 | |
|         $this->assertTrue($validator->validate('firstName.x.lastName.-nd@example.com'));
 | |
|         $this->assertFalse($validator->validate('rmcreative.ru'));
 | |
|         $this->assertFalse($validator->validate('Carsten Brandt <mail@cebe.cc>'));
 | |
|         $this->assertFalse($validator->validate('"Carsten Brandt" <mail@cebe.cc>'));
 | |
|         $this->assertFalse($validator->validate('<mail@cebe.cc>'));
 | |
| 
 | |
|         $validator->allowName = true;
 | |
| 
 | |
|         $this->assertTrue($validator->validate('info@örtliches.de'));
 | |
|         $this->assertTrue($validator->validate('Informtation <info@örtliches.de>'));
 | |
|         $this->assertFalse($validator->validate('Informtation info@örtliches.de'));
 | |
|         $this->assertTrue($validator->validate('sam@рмкреатиф.ru'));
 | |
|         $this->assertTrue($validator->validate('sam@rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('5011@gmail.com'));
 | |
|         $this->assertFalse($validator->validate('rmcreative.ru'));
 | |
|         $this->assertTrue($validator->validate('Carsten Brandt <mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('"Carsten Brandt" <mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('üñîçøðé 日本国 <üñîçøðé@üñîçøðé.com>'));
 | |
|         $this->assertTrue($validator->validate('<mail@cebe.cc>'));
 | |
|         $this->assertTrue($validator->validate('test@example.com'));
 | |
|         $this->assertTrue($validator->validate('John Smith <john.smith@example.com>'));
 | |
|         $this->assertTrue($validator->validate('"Такое имя достаточно длинное, но оно все равно может пройти валидацию" <shortmail@example.com>'));
 | |
|         $this->assertTrue($validator->validate('"FirstName LastName" <firstName.x.lastName.-nd@example.com>'));
 | |
|         $this->assertFalse($validator->validate('John Smith <example.com>'));
 | |
|         $this->assertFalse($validator->validate('Короткое имя <после-преобразования-в-idn-тут-будет-больше-чем-64-символа@пример.com>'));
 | |
|         $this->assertFalse($validator->validate('Короткое имя <тест@это-доменное-имя.после-преобразования-в-idn.будет-содержать-больше-254-символов.бла-бла-бла-бла-бла-бла-бла-бла.бла-бла-бла-бла-бла-бла.бла-бла-бла-бла-бла-бла.бла-бла-бла-бла-бла-бла.com>'));
 | |
|     }
 | |
| 
 | |
|     public function testValidateValueMx()
 | |
|     {
 | |
|         $validator = new EmailValidator();
 | |
| 
 | |
|         $validator->checkDNS = true;
 | |
|         $this->assertTrue($validator->validate('5011@gmail.com'));
 | |
| 
 | |
|         $validator->checkDNS = false;
 | |
|         $this->assertTrue($validator->validate('test@nonexistingsubdomain.example.com'));
 | |
|         $validator->checkDNS = true;
 | |
|         $this->assertFalse($validator->validate('test@nonexistingsubdomain.example.com'));
 | |
| 
 | |
|         $validator->checkDNS = true;
 | |
|         $validator->allowName = true;
 | |
|         $emails = [
 | |
|             'ipetrov@gmail.com',
 | |
|             'Ivan Petrov <ipetrov@gmail.com>',
 | |
|         ];
 | |
|         foreach ($emails as $email) {
 | |
|             $this->assertTrue($validator->validate($email), "Email: '$email' failed to validate(checkDNS=true, allowName=true)");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function testValidateAttribute()
 | |
|     {
 | |
|         $validator = new EmailValidator();
 | |
|         $model = new FakedValidationModel();
 | |
|         $model->attr_email = '5011@gmail.com';
 | |
|         $validator->validateAttribute($model, 'attr_email');
 | |
|         $this->assertFalse($model->hasErrors('attr_email'));
 | |
|     }
 | |
| 
 | |
|     public function malformedAddressesProvider()
 | |
|     {
 | |
|         return [
 | |
|             // this is the demo email used in the proof of concept of the exploit
 | |
|             ['"attacker\" -oQ/tmp/ -X/var/www/cache/phpcode.php "@email.com'],
 | |
|             // trying more adresses
 | |
|             ['"Attacker -Param2 -Param3"@test.com'],
 | |
|             ['\'Attacker -Param2 -Param3\'@test.com'],
 | |
|             ['"Attacker \" -Param2 -Param3"@test.com'],
 | |
|             ["'Attacker \\' -Param2 -Param3'@test.com"],
 | |
|             ['"attacker\" -oQ/tmp/ -X/var/www/cache/phpcode.php "@email.com'],
 | |
|             // and even more variants
 | |
|             ['"attacker\"\ -oQ/tmp/\ -X/var/www/cache/phpcode.php"@email.com'],
 | |
|             ["\"attacker\\\"\0-oQ/tmp/\0-X/var/www/cache/phpcode.php\"@email.com"],
 | |
|             ['"attacker@cebe.cc\"-Xbeep"@email.com'],
 | |
| 
 | |
|             ["'attacker\\' -oQ/tmp/ -X/var/www/cache/phpcode.php'@email.com"],
 | |
|             ["'attacker\\\\' -oQ/tmp/ -X/var/www/cache/phpcode.php'@email.com"],
 | |
|             ["'attacker\\\\'\\ -oQ/tmp/ -X/var/www/cache/phpcode.php'@email.com"],
 | |
|             ["'attacker\\';touch /tmp/hackme'@email.com"],
 | |
|             ["'attacker\\\\';touch /tmp/hackme'@email.com"],
 | |
|             ["'attacker\\';touch/tmp/hackme'@email.com"],
 | |
|             ["'attacker\\\\';touch/tmp/hackme'@email.com"],
 | |
|             ['"attacker\" -oQ/tmp/ -X/var/www/cache/phpcode.php "@email.com'],
 | |
|         ];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Test malicious email addresses that can be used to exploit SwiftMailer vulnerability CVE-2016-10074 while IDN is disabled.
 | |
|      * @see https://legalhackers.com/advisories/SwiftMailer-Exploit-Remote-Code-Exec-CVE-2016-10074-Vuln.html
 | |
|      *
 | |
|      * @dataProvider malformedAddressesProvider
 | |
|      * @param string $value
 | |
|      */
 | |
|     public function testMalformedAddressesIdnDisabled($value)
 | |
|     {
 | |
|         $validator = new EmailValidator();
 | |
|         $validator->enableIDN = false;
 | |
|         $this->assertFalse($validator->validate($value));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Test malicious email addresses that can be used to exploit SwiftMailer vulnerability CVE-2016-10074 while IDN is enabled.
 | |
|      * @see https://legalhackers.com/advisories/SwiftMailer-Exploit-Remote-Code-Exec-CVE-2016-10074-Vuln.html
 | |
|      *
 | |
|      * @dataProvider malformedAddressesProvider
 | |
|      * @param string $value
 | |
|      */
 | |
|     public function testMalformedAddressesIdnEnabled($value)
 | |
|     {
 | |
|         if (!function_exists('idn_to_ascii')) {
 | |
|             $this->markTestSkipped('Intl extension required');
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $val = new EmailValidator();
 | |
|         $val->enableIDN = true;
 | |
|         $this->assertFalse($val->validate($value));
 | |
|     }
 | |
| }
 | 
