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 ')); $this->assertFalse($validator->validate('"Carsten Brandt" ')); $this->assertFalse($validator->validate('')); $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 ')); $this->assertTrue($validator->validate('"Carsten Brandt" ')); $this->assertTrue($validator->validate('')); $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 ')); $this->assertTrue($validator->validate('"This name is longer than 64 characters. Blah blah blah blah blah" ')); $this->assertFalse($validator->validate('John Smith ')); $this->assertFalse($validator->validate('Short Name ')); $this->assertFalse($validator->validate('Short Name ')); $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 ')); $this->assertFalse($validator->validate('"Carsten Brandt" ')); $this->assertFalse($validator->validate('')); $validator->allowName = true; $this->assertTrue($validator->validate('info@örtliches.de')); $this->assertTrue($validator->validate('Informtation ')); $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 ')); $this->assertTrue($validator->validate('"Carsten Brandt" ')); $this->assertTrue($validator->validate('üñîçøðé 日本国 <üñîçøðé@üñîçøðé.com>')); $this->assertTrue($validator->validate('')); $this->assertTrue($validator->validate('test@example.com')); $this->assertTrue($validator->validate('John Smith ')); $this->assertTrue($validator->validate('"Такое имя достаточно длинное, но оно все равно может пройти валидацию" ')); $this->assertFalse($validator->validate('John Smith ')); $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 ', ]; 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)); } }