mirror of
				https://github.com/yiisoft/yii2.git
				synced 2025-11-04 06:37:55 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			202 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			202 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
/**
 | 
						|
 * @link http://www.yiiframework.com/
 | 
						|
 * @copyright Copyright (c) 2008 Yii Software LLC
 | 
						|
 * @license http://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()
 | 
						|
    {
 | 
						|
        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->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->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->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->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));
 | 
						|
    }
 | 
						|
}
 |