6392 changed files with 700972 additions and 2 deletions
			
			
		| @ -0,0 +1,31 @@ | |||
| APP_NAME=IFAI | |||
| APP_ENV=local | |||
| APP_KEY=base64:I206O8ibx+GQyRE7BeOxDobn04Mfmyyc5Ptzns/C0mY= | |||
| APP_DEBUG=true | |||
| APP_LOG_LEVEL=debug | |||
| APP_URL=http://localhost | |||
| 
 | |||
| DB_CONNECTION=sqlite | |||
| DB_DATABASE=app.sqlite | |||
| 
 | |||
| BROADCAST_DRIVER=log | |||
| CACHE_DRIVER=file | |||
| SESSION_DRIVER=file | |||
| SESSION_LIFETIME=120 | |||
| QUEUE_DRIVER=sync | |||
| 
 | |||
| REDIS_HOST=127.0.0.1 | |||
| REDIS_PASSWORD=null | |||
| REDIS_PORT=6379 | |||
| 
 | |||
| MAIL_DRIVER=smtp | |||
| MAIL_HOST=smtp.mailtrap.io | |||
| MAIL_PORT=2525 | |||
| MAIL_USERNAME=null | |||
| MAIL_PASSWORD=null | |||
| MAIL_ENCRYPTION=null | |||
| 
 | |||
| PUSHER_APP_ID= | |||
| PUSHER_APP_KEY= | |||
| PUSHER_APP_SECRET= | |||
| PUSHER_APP_CLUSTER=mt1 | |||
| @ -0,0 +1,7 @@ | |||
| <?php | |||
| 
 | |||
| // autoload.php @generated by Composer | |||
| 
 | |||
| require_once __DIR__ . '/composer/autoload_real.php'; | |||
| 
 | |||
| return ComposerAutoloaderInit4b6fb9210a1ea37c2db27b8ff53a1ecf::getLoader(); | |||
| @ -0,0 +1 @@ | |||
| ../nikic/php-parser/bin/php-parse | |||
| @ -0,0 +1 @@ | |||
| ../phpunit/phpunit/phpunit | |||
| @ -0,0 +1 @@ | |||
| ../psy/psysh/bin/psysh | |||
| @ -0,0 +1,445 @@ | |||
| <?php | |||
| 
 | |||
| /* | |||
|  * This file is part of Composer. | |||
|  * | |||
|  * (c) Nils Adermann <naderman@naderman.de> | |||
|  *     Jordi Boggiano <j.boggiano@seld.be> | |||
|  * | |||
|  * For the full copyright and license information, please view the LICENSE | |||
|  * file that was distributed with this source code. | |||
|  */ | |||
| 
 | |||
| namespace Composer\Autoload; | |||
| 
 | |||
| /** | |||
|  * ClassLoader implements a PSR-0, PSR-4 and classmap class loader. | |||
|  * | |||
|  *     $loader = new \Composer\Autoload\ClassLoader(); | |||
|  * | |||
|  *     // register classes with namespaces | |||
|  *     $loader->add('Symfony\Component', __DIR__.'/component'); | |||
|  *     $loader->add('Symfony',           __DIR__.'/framework'); | |||
|  * | |||
|  *     // activate the autoloader | |||
|  *     $loader->register(); | |||
|  * | |||
|  *     // to enable searching the include path (eg. for PEAR packages) | |||
|  *     $loader->setUseIncludePath(true); | |||
|  * | |||
|  * In this example, if you try to use a class in the Symfony\Component | |||
|  * namespace or one of its children (Symfony\Component\Console for instance), | |||
|  * the autoloader will first look for the class under the component/ | |||
|  * directory, and it will then fallback to the framework/ directory if not | |||
|  * found before giving up. | |||
|  * | |||
|  * This class is loosely based on the Symfony UniversalClassLoader. | |||
|  * | |||
|  * @author Fabien Potencier <fabien@symfony.com> | |||
|  * @author Jordi Boggiano <j.boggiano@seld.be> | |||
|  * @see    http://www.php-fig.org/psr/psr-0/ | |||
|  * @see    http://www.php-fig.org/psr/psr-4/ | |||
|  */ | |||
| class ClassLoader | |||
| { | |||
|     // PSR-4 | |||
|     private $prefixLengthsPsr4 = array(); | |||
|     private $prefixDirsPsr4 = array(); | |||
|     private $fallbackDirsPsr4 = array(); | |||
| 
 | |||
|     // PSR-0 | |||
|     private $prefixesPsr0 = array(); | |||
|     private $fallbackDirsPsr0 = array(); | |||
| 
 | |||
|     private $useIncludePath = false; | |||
|     private $classMap = array(); | |||
|     private $classMapAuthoritative = false; | |||
|     private $missingClasses = array(); | |||
|     private $apcuPrefix; | |||
| 
 | |||
|     public function getPrefixes() | |||
|     { | |||
|         if (!empty($this->prefixesPsr0)) { | |||
|             return call_user_func_array('array_merge', $this->prefixesPsr0); | |||
|         } | |||
| 
 | |||
|         return array(); | |||
|     } | |||
| 
 | |||
|     public function getPrefixesPsr4() | |||
|     { | |||
|         return $this->prefixDirsPsr4; | |||
|     } | |||
| 
 | |||
|     public function getFallbackDirs() | |||
|     { | |||
|         return $this->fallbackDirsPsr0; | |||
|     } | |||
| 
 | |||
|     public function getFallbackDirsPsr4() | |||
|     { | |||
|         return $this->fallbackDirsPsr4; | |||
|     } | |||
| 
 | |||
|     public function getClassMap() | |||
|     { | |||
|         return $this->classMap; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param array $classMap Class to filename map | |||
|      */ | |||
|     public function addClassMap(array $classMap) | |||
|     { | |||
|         if ($this->classMap) { | |||
|             $this->classMap = array_merge($this->classMap, $classMap); | |||
|         } else { | |||
|             $this->classMap = $classMap; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Registers a set of PSR-0 directories for a given prefix, either | |||
|      * appending or prepending to the ones previously set for this prefix. | |||
|      * | |||
|      * @param string       $prefix  The prefix | |||
|      * @param array|string $paths   The PSR-0 root directories | |||
|      * @param bool         $prepend Whether to prepend the directories | |||
|      */ | |||
|     public function add($prefix, $paths, $prepend = false) | |||
|     { | |||
|         if (!$prefix) { | |||
|             if ($prepend) { | |||
|                 $this->fallbackDirsPsr0 = array_merge( | |||
|                     (array) $paths, | |||
|                     $this->fallbackDirsPsr0 | |||
|                 ); | |||
|             } else { | |||
|                 $this->fallbackDirsPsr0 = array_merge( | |||
|                     $this->fallbackDirsPsr0, | |||
|                     (array) $paths | |||
|                 ); | |||
|             } | |||
| 
 | |||
|             return; | |||
|         } | |||
| 
 | |||
|         $first = $prefix[0]; | |||
|         if (!isset($this->prefixesPsr0[$first][$prefix])) { | |||
|             $this->prefixesPsr0[$first][$prefix] = (array) $paths; | |||
| 
 | |||
|             return; | |||
|         } | |||
|         if ($prepend) { | |||
|             $this->prefixesPsr0[$first][$prefix] = array_merge( | |||
|                 (array) $paths, | |||
|                 $this->prefixesPsr0[$first][$prefix] | |||
|             ); | |||
|         } else { | |||
|             $this->prefixesPsr0[$first][$prefix] = array_merge( | |||
|                 $this->prefixesPsr0[$first][$prefix], | |||
|                 (array) $paths | |||
|             ); | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Registers a set of PSR-4 directories for a given namespace, either | |||
|      * appending or prepending to the ones previously set for this namespace. | |||
|      * | |||
|      * @param string       $prefix  The prefix/namespace, with trailing '\\' | |||
|      * @param array|string $paths   The PSR-4 base directories | |||
|      * @param bool         $prepend Whether to prepend the directories | |||
|      * | |||
|      * @throws \InvalidArgumentException | |||
|      */ | |||
|     public function addPsr4($prefix, $paths, $prepend = false) | |||
|     { | |||
|         if (!$prefix) { | |||
|             // Register directories for the root namespace. | |||
|             if ($prepend) { | |||
|                 $this->fallbackDirsPsr4 = array_merge( | |||
|                     (array) $paths, | |||
|                     $this->fallbackDirsPsr4 | |||
|                 ); | |||
|             } else { | |||
|                 $this->fallbackDirsPsr4 = array_merge( | |||
|                     $this->fallbackDirsPsr4, | |||
|                     (array) $paths | |||
|                 ); | |||
|             } | |||
|         } elseif (!isset($this->prefixDirsPsr4[$prefix])) { | |||
|             // Register directories for a new namespace. | |||
|             $length = strlen($prefix); | |||
|             if ('\\' !== $prefix[$length - 1]) { | |||
|                 throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); | |||
|             } | |||
|             $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length; | |||
|             $this->prefixDirsPsr4[$prefix] = (array) $paths; | |||
|         } elseif ($prepend) { | |||
|             // Prepend directories for an already registered namespace. | |||
|             $this->prefixDirsPsr4[$prefix] = array_merge( | |||
|                 (array) $paths, | |||
|                 $this->prefixDirsPsr4[$prefix] | |||
|             ); | |||
|         } else { | |||
|             // Append directories for an already registered namespace. | |||
|             $this->prefixDirsPsr4[$prefix] = array_merge( | |||
|                 $this->prefixDirsPsr4[$prefix], | |||
|                 (array) $paths | |||
|             ); | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Registers a set of PSR-0 directories for a given prefix, | |||
|      * replacing any others previously set for this prefix. | |||
|      * | |||
|      * @param string       $prefix The prefix | |||
|      * @param array|string $paths  The PSR-0 base directories | |||
|      */ | |||
|     public function set($prefix, $paths) | |||
|     { | |||
|         if (!$prefix) { | |||
|             $this->fallbackDirsPsr0 = (array) $paths; | |||
|         } else { | |||
|             $this->prefixesPsr0[$prefix[0]][$prefix] = (array) $paths; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Registers a set of PSR-4 directories for a given namespace, | |||
|      * replacing any others previously set for this namespace. | |||
|      * | |||
|      * @param string       $prefix The prefix/namespace, with trailing '\\' | |||
|      * @param array|string $paths  The PSR-4 base directories | |||
|      * | |||
|      * @throws \InvalidArgumentException | |||
|      */ | |||
|     public function setPsr4($prefix, $paths) | |||
|     { | |||
|         if (!$prefix) { | |||
|             $this->fallbackDirsPsr4 = (array) $paths; | |||
|         } else { | |||
|             $length = strlen($prefix); | |||
|             if ('\\' !== $prefix[$length - 1]) { | |||
|                 throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); | |||
|             } | |||
|             $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length; | |||
|             $this->prefixDirsPsr4[$prefix] = (array) $paths; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Turns on searching the include path for class files. | |||
|      * | |||
|      * @param bool $useIncludePath | |||
|      */ | |||
|     public function setUseIncludePath($useIncludePath) | |||
|     { | |||
|         $this->useIncludePath = $useIncludePath; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Can be used to check if the autoloader uses the include path to check | |||
|      * for classes. | |||
|      * | |||
|      * @return bool | |||
|      */ | |||
|     public function getUseIncludePath() | |||
|     { | |||
|         return $this->useIncludePath; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Turns off searching the prefix and fallback directories for classes | |||
|      * that have not been registered with the class map. | |||
|      * | |||
|      * @param bool $classMapAuthoritative | |||
|      */ | |||
|     public function setClassMapAuthoritative($classMapAuthoritative) | |||
|     { | |||
|         $this->classMapAuthoritative = $classMapAuthoritative; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Should class lookup fail if not found in the current class map? | |||
|      * | |||
|      * @return bool | |||
|      */ | |||
|     public function isClassMapAuthoritative() | |||
|     { | |||
|         return $this->classMapAuthoritative; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * APCu prefix to use to cache found/not-found classes, if the extension is enabled. | |||
|      * | |||
|      * @param string|null $apcuPrefix | |||
|      */ | |||
|     public function setApcuPrefix($apcuPrefix) | |||
|     { | |||
|         $this->apcuPrefix = function_exists('apcu_fetch') && ini_get('apc.enabled') ? $apcuPrefix : null; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * The APCu prefix in use, or null if APCu caching is not enabled. | |||
|      * | |||
|      * @return string|null | |||
|      */ | |||
|     public function getApcuPrefix() | |||
|     { | |||
|         return $this->apcuPrefix; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Registers this instance as an autoloader. | |||
|      * | |||
|      * @param bool $prepend Whether to prepend the autoloader or not | |||
|      */ | |||
|     public function register($prepend = false) | |||
|     { | |||
|         spl_autoload_register(array($this, 'loadClass'), true, $prepend); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Unregisters this instance as an autoloader. | |||
|      */ | |||
|     public function unregister() | |||
|     { | |||
|         spl_autoload_unregister(array($this, 'loadClass')); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Loads the given class or interface. | |||
|      * | |||
|      * @param  string    $class The name of the class | |||
|      * @return bool|null True if loaded, null otherwise | |||
|      */ | |||
|     public function loadClass($class) | |||
|     { | |||
|         if ($file = $this->findFile($class)) { | |||
|             includeFile($file); | |||
| 
 | |||
|             return true; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Finds the path to the file where the class is defined. | |||
|      * | |||
|      * @param string $class The name of the class | |||
|      * | |||
|      * @return string|false The path if found, false otherwise | |||
|      */ | |||
|     public function findFile($class) | |||
|     { | |||
|         // class map lookup | |||
|         if (isset($this->classMap[$class])) { | |||
|             return $this->classMap[$class]; | |||
|         } | |||
|         if ($this->classMapAuthoritative || isset($this->missingClasses[$class])) { | |||
|             return false; | |||
|         } | |||
|         if (null !== $this->apcuPrefix) { | |||
|             $file = apcu_fetch($this->apcuPrefix.$class, $hit); | |||
|             if ($hit) { | |||
|                 return $file; | |||
|             } | |||
|         } | |||
| 
 | |||
|         $file = $this->findFileWithExtension($class, '.php'); | |||
| 
 | |||
|         // Search for Hack files if we are running on HHVM | |||
|         if (false === $file && defined('HHVM_VERSION')) { | |||
|             $file = $this->findFileWithExtension($class, '.hh'); | |||
|         } | |||
| 
 | |||
|         if (null !== $this->apcuPrefix) { | |||
|             apcu_add($this->apcuPrefix.$class, $file); | |||
|         } | |||
| 
 | |||
|         if (false === $file) { | |||
|             // Remember that this class does not exist. | |||
|             $this->missingClasses[$class] = true; | |||
|         } | |||
| 
 | |||
|         return $file; | |||
|     } | |||
| 
 | |||
|     private function findFileWithExtension($class, $ext) | |||
|     { | |||
|         // PSR-4 lookup | |||
|         $logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . $ext; | |||
| 
 | |||
|         $first = $class[0]; | |||
|         if (isset($this->prefixLengthsPsr4[$first])) { | |||
|             $subPath = $class; | |||
|             while (false !== $lastPos = strrpos($subPath, '\\')) { | |||
|                 $subPath = substr($subPath, 0, $lastPos); | |||
|                 $search = $subPath.'\\'; | |||
|                 if (isset($this->prefixDirsPsr4[$search])) { | |||
|                     foreach ($this->prefixDirsPsr4[$search] as $dir) { | |||
|                         $length = $this->prefixLengthsPsr4[$first][$search]; | |||
|                         if (file_exists($file = $dir . DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $length))) { | |||
|                             return $file; | |||
|                         } | |||
|                     } | |||
|                 } | |||
|             } | |||
|         } | |||
| 
 | |||
|         // PSR-4 fallback dirs | |||
|         foreach ($this->fallbackDirsPsr4 as $dir) { | |||
|             if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) { | |||
|                 return $file; | |||
|             } | |||
|         } | |||
| 
 | |||
|         // PSR-0 lookup | |||
|         if (false !== $pos = strrpos($class, '\\')) { | |||
|             // namespaced class name | |||
|             $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1) | |||
|                 . strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR); | |||
|         } else { | |||
|             // PEAR-like class name | |||
|             $logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . $ext; | |||
|         } | |||
| 
 | |||
|         if (isset($this->prefixesPsr0[$first])) { | |||
|             foreach ($this->prefixesPsr0[$first] as $prefix => $dirs) { | |||
|                 if (0 === strpos($class, $prefix)) { | |||
|                     foreach ($dirs as $dir) { | |||
|                         if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) { | |||
|                             return $file; | |||
|                         } | |||
|                     } | |||
|                 } | |||
|             } | |||
|         } | |||
| 
 | |||
|         // PSR-0 fallback dirs | |||
|         foreach ($this->fallbackDirsPsr0 as $dir) { | |||
|             if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) { | |||
|                 return $file; | |||
|             } | |||
|         } | |||
| 
 | |||
|         // PSR-0 include paths. | |||
|         if ($this->useIncludePath && $file = stream_resolve_include_path($logicalPathPsr0)) { | |||
|             return $file; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| } | |||
| 
 | |||
| /** | |||
|  * Scope isolated include. | |||
|  * | |||
|  * Prevents access to $this/self from included files. | |||
|  */ | |||
| function includeFile($file) | |||
| { | |||
|     include $file; | |||
| } | |||
| @ -0,0 +1,21 @@ | |||
| 
 | |||
| Copyright (c) Nils Adermann, Jordi Boggiano | |||
| 
 | |||
| Permission is hereby granted, free of charge, to any person obtaining a copy | |||
| of this software and associated documentation files (the "Software"), to deal | |||
| in the Software without restriction, including without limitation the rights | |||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |||
| copies of the Software, and to permit persons to whom the Software is furnished | |||
| to do so, subject to the following conditions: | |||
| 
 | |||
| The above copyright notice and this permission notice shall be included in all | |||
| copies or substantial portions of the Software. | |||
| 
 | |||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |||
| THE SOFTWARE. | |||
| 
 | |||
								
									
										File diff suppressed because it is too large
									
								
							
						
					| @ -0,0 +1,19 @@ | |||
| <?php | |||
| 
 | |||
| // autoload_files.php @generated by Composer | |||
| 
 | |||
| $vendorDir = dirname(dirname(__FILE__)); | |||
| $baseDir = dirname($vendorDir); | |||
| 
 | |||
| return array( | |||
|     '0e6d7bf4a5811bfa5cf40c5ccd6fae6a' => $vendorDir . '/symfony/polyfill-mbstring/bootstrap.php', | |||
|     '5255c38a0faeba867671b61dfda6d864' => $vendorDir . '/paragonie/random_compat/lib/random.php', | |||
|     '023d27dca8066ef29e6739335ea73bad' => $vendorDir . '/symfony/polyfill-php70/bootstrap.php', | |||
|     '667aeda72477189d0494fecd327c3641' => $vendorDir . '/symfony/var-dumper/Resources/functions/dump.php', | |||
|     '2c102faa651ef8ea5874edb585946bce' => $vendorDir . '/swiftmailer/swiftmailer/lib/swift_required.php', | |||
|     'e7223560d890eab89cda23685e711e2c' => $vendorDir . '/psy/psysh/src/Psy/functions.php', | |||
|     '6124b4c8570aa390c21fafd04a26c69f' => $vendorDir . '/myclabs/deep-copy/src/DeepCopy/deep_copy.php', | |||
|     'f0906e6318348a765ffb6eb24e0d0938' => $vendorDir . '/laravel/framework/src/Illuminate/Foundation/helpers.php', | |||
|     '58571171fd5812e6e447dce228f52f4d' => $vendorDir . '/laravel/framework/src/Illuminate/Support/helpers.php', | |||
|     'f18cc91337d49233e5754e93f3ed9ec3' => $vendorDir . '/laravelcollective/html/src/helpers.php', | |||
| ); | |||
| @ -0,0 +1,15 @@ | |||
| <?php | |||
| 
 | |||
| // autoload_namespaces.php @generated by Composer | |||
| 
 | |||
| $vendorDir = dirname(dirname(__FILE__)); | |||
| $baseDir = dirname($vendorDir); | |||
| 
 | |||
| return array( | |||
|     'Prophecy\\' => array($vendorDir . '/phpspec/prophecy/src'), | |||
|     'Parsedown' => array($vendorDir . '/erusev/parsedown'), | |||
|     'Mockery' => array($vendorDir . '/mockery/mockery/library'), | |||
|     'JakubOnderka\\PhpConsoleHighlighter' => array($vendorDir . '/jakub-onderka/php-console-highlighter/src'), | |||
|     'JakubOnderka\\PhpConsoleColor' => array($vendorDir . '/jakub-onderka/php-console-color/src'), | |||
|     'Doctrine\\Common\\Lexer\\' => array($vendorDir . '/doctrine/lexer/lib'), | |||
| ); | |||
| @ -0,0 +1,50 @@ | |||
| <?php | |||
| 
 | |||
| // autoload_psr4.php @generated by Composer | |||
| 
 | |||
| $vendorDir = dirname(dirname(__FILE__)); | |||
| $baseDir = dirname($vendorDir); | |||
| 
 | |||
| return array( | |||
|     'phpDocumentor\\Reflection\\' => array($vendorDir . '/phpdocumentor/reflection-common/src', $vendorDir . '/phpdocumentor/type-resolver/src', $vendorDir . '/phpdocumentor/reflection-docblock/src'), | |||
|     'XdgBaseDir\\' => array($vendorDir . '/dnoegel/php-xdg-base-dir/src'), | |||
|     'Whoops\\' => array($vendorDir . '/filp/whoops/src/Whoops'), | |||
|     'Webmozart\\Assert\\' => array($vendorDir . '/webmozart/assert/src'), | |||
|     'TijsVerkoyen\\CssToInlineStyles\\' => array($vendorDir . '/tijsverkoyen/css-to-inline-styles/src'), | |||
|     'Tests\\' => array($baseDir . '/tests'), | |||
|     'Symfony\\Thanks\\' => array($vendorDir . '/symfony/thanks/src'), | |||
|     'Symfony\\Polyfill\\Php70\\' => array($vendorDir . '/symfony/polyfill-php70'), | |||
|     'Symfony\\Polyfill\\Mbstring\\' => array($vendorDir . '/symfony/polyfill-mbstring'), | |||
|     'Symfony\\Component\\VarDumper\\' => array($vendorDir . '/symfony/var-dumper'), | |||
|     'Symfony\\Component\\Translation\\' => array($vendorDir . '/symfony/translation'), | |||
|     'Symfony\\Component\\Routing\\' => array($vendorDir . '/symfony/routing'), | |||
|     'Symfony\\Component\\Process\\' => array($vendorDir . '/symfony/process'), | |||
|     'Symfony\\Component\\HttpKernel\\' => array($vendorDir . '/symfony/http-kernel'), | |||
|     'Symfony\\Component\\HttpFoundation\\' => array($vendorDir . '/symfony/http-foundation'), | |||
|     'Symfony\\Component\\Finder\\' => array($vendorDir . '/symfony/finder'), | |||
|     'Symfony\\Component\\EventDispatcher\\' => array($vendorDir . '/symfony/event-dispatcher'), | |||
|     'Symfony\\Component\\Debug\\' => array($vendorDir . '/symfony/debug'), | |||
|     'Symfony\\Component\\CssSelector\\' => array($vendorDir . '/symfony/css-selector'), | |||
|     'Symfony\\Component\\Console\\' => array($vendorDir . '/symfony/console'), | |||
|     'Ramsey\\Uuid\\' => array($vendorDir . '/ramsey/uuid/src'), | |||
|     'Psy\\' => array($vendorDir . '/psy/psysh/src/Psy'), | |||
|     'Psr\\SimpleCache\\' => array($vendorDir . '/psr/simple-cache/src'), | |||
|     'Psr\\Log\\' => array($vendorDir . '/psr/log/Psr/Log'), | |||
|     'Psr\\Container\\' => array($vendorDir . '/psr/container/src'), | |||
|     'PhpParser\\' => array($vendorDir . '/nikic/php-parser/lib/PhpParser'), | |||
|     'Monolog\\' => array($vendorDir . '/monolog/monolog/src/Monolog'), | |||
|     'League\\Flysystem\\' => array($vendorDir . '/league/flysystem/src'), | |||
|     'Laravel\\Tinker\\' => array($vendorDir . '/laravel/tinker/src'), | |||
|     'Illuminate\\' => array($vendorDir . '/laravel/framework/src/Illuminate'), | |||
|     'Fideloper\\Proxy\\' => array($vendorDir . '/fideloper/proxy/src'), | |||
|     'Faker\\' => array($vendorDir . '/fzaninotto/faker/src/Faker'), | |||
|     'Egulias\\EmailValidator\\' => array($vendorDir . '/egulias/email-validator/EmailValidator'), | |||
|     'Dotenv\\' => array($vendorDir . '/vlucas/phpdotenv/src'), | |||
|     'Doctrine\\Instantiator\\' => array($vendorDir . '/doctrine/instantiator/src/Doctrine/Instantiator'), | |||
|     'Doctrine\\Common\\Inflector\\' => array($vendorDir . '/doctrine/inflector/lib/Doctrine/Common/Inflector'), | |||
|     'DeepCopy\\' => array($vendorDir . '/myclabs/deep-copy/src/DeepCopy'), | |||
|     'Cron\\' => array($vendorDir . '/mtdowling/cron-expression/src/Cron'), | |||
|     'Collective\\Html\\' => array($vendorDir . '/laravelcollective/html/src'), | |||
|     'Carbon\\' => array($vendorDir . '/nesbot/carbon/src/Carbon'), | |||
|     'App\\' => array($baseDir . '/app'), | |||
| ); | |||
| @ -0,0 +1,70 @@ | |||
| <?php | |||
| 
 | |||
| // autoload_real.php @generated by Composer | |||
| 
 | |||
| class ComposerAutoloaderInit4b6fb9210a1ea37c2db27b8ff53a1ecf | |||
| { | |||
|     private static $loader; | |||
| 
 | |||
|     public static function loadClassLoader($class) | |||
|     { | |||
|         if ('Composer\Autoload\ClassLoader' === $class) { | |||
|             require __DIR__ . '/ClassLoader.php'; | |||
|         } | |||
|     } | |||
| 
 | |||
|     public static function getLoader() | |||
|     { | |||
|         if (null !== self::$loader) { | |||
|             return self::$loader; | |||
|         } | |||
| 
 | |||
|         spl_autoload_register(array('ComposerAutoloaderInit4b6fb9210a1ea37c2db27b8ff53a1ecf', 'loadClassLoader'), true, true); | |||
|         self::$loader = $loader = new \Composer\Autoload\ClassLoader(); | |||
|         spl_autoload_unregister(array('ComposerAutoloaderInit4b6fb9210a1ea37c2db27b8ff53a1ecf', 'loadClassLoader')); | |||
| 
 | |||
|         $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded()); | |||
|         if ($useStaticLoader) { | |||
|             require_once __DIR__ . '/autoload_static.php'; | |||
| 
 | |||
|             call_user_func(\Composer\Autoload\ComposerStaticInit4b6fb9210a1ea37c2db27b8ff53a1ecf::getInitializer($loader)); | |||
|         } else { | |||
|             $map = require __DIR__ . '/autoload_namespaces.php'; | |||
|             foreach ($map as $namespace => $path) { | |||
|                 $loader->set($namespace, $path); | |||
|             } | |||
| 
 | |||
|             $map = require __DIR__ . '/autoload_psr4.php'; | |||
|             foreach ($map as $namespace => $path) { | |||
|                 $loader->setPsr4($namespace, $path); | |||
|             } | |||
| 
 | |||
|             $classMap = require __DIR__ . '/autoload_classmap.php'; | |||
|             if ($classMap) { | |||
|                 $loader->addClassMap($classMap); | |||
|             } | |||
|         } | |||
| 
 | |||
|         $loader->register(true); | |||
| 
 | |||
|         if ($useStaticLoader) { | |||
|             $includeFiles = Composer\Autoload\ComposerStaticInit4b6fb9210a1ea37c2db27b8ff53a1ecf::$files; | |||
|         } else { | |||
|             $includeFiles = require __DIR__ . '/autoload_files.php'; | |||
|         } | |||
|         foreach ($includeFiles as $fileIdentifier => $file) { | |||
|             composerRequire4b6fb9210a1ea37c2db27b8ff53a1ecf($fileIdentifier, $file); | |||
|         } | |||
| 
 | |||
|         return $loader; | |||
|     } | |||
| } | |||
| 
 | |||
| function composerRequire4b6fb9210a1ea37c2db27b8ff53a1ecf($fileIdentifier, $file) | |||
| { | |||
|     if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) { | |||
|         require $file; | |||
| 
 | |||
|         $GLOBALS['__composer_autoload_files'][$fileIdentifier] = true; | |||
|     } | |||
| } | |||
								
									
										File diff suppressed because it is too large
									
								
							
						
					
								
									
										File diff suppressed because it is too large
									
								
							
						
					| @ -0,0 +1 @@ | |||
| /vendor/ | |||
| @ -0,0 +1,19 @@ | |||
| Copyright (c) 2014 Daniel Nögel | |||
| 
 | |||
| Permission is hereby granted, free of charge, to any person obtaining a copy | |||
| of this software and associated documentation files (the "Software"), to deal | |||
| in the Software without restriction, including without limitation the rights | |||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |||
| copies of the Software, and to permit persons to whom the Software is | |||
| furnished to do so, subject to the following conditions: | |||
| 
 | |||
| The above copyright notice and this permission notice shall be included in | |||
| all copies or substantial portions of the Software. | |||
| 
 | |||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |||
| THE SOFTWARE. | |||
| @ -0,0 +1,38 @@ | |||
| # XDG Base Directory | |||
| 
 | |||
| [](LICENSE.md) | |||
| 
 | |||
| Implementation of XDG Base Directory  specification for php | |||
| 
 | |||
| ## Install | |||
| 
 | |||
| Via Composer | |||
| 
 | |||
| ``` bash | |||
| $ composer require dnoegel/php-xdg-base-dir | |||
| ``` | |||
| 
 | |||
| ## Usage | |||
| 
 | |||
| ``` php | |||
| $xdg = \XdgBaseDir\Xdg(); | |||
| 
 | |||
| echo $xdg->getHomeDir(); | |||
| echo $xdg->getHomeConfigDir() | |||
| echo $xdg->getHomeDataDir() | |||
| echo $xdg->getHomeCacheDir() | |||
| echo $xdg->getRuntimeDir() | |||
| 
 | |||
| $xdg->getDataDirs() // returns array | |||
| $xdg->getConfigDirs() // returns array | |||
| ``` | |||
| 
 | |||
| ## Testing | |||
| 
 | |||
| ``` bash | |||
| $ phpunit | |||
| ``` | |||
| 
 | |||
| ## License | |||
| 
 | |||
| The MIT License (MIT). Please see [License File](https://github.com/dnoegel/php-xdg-base-dir/blob/master/LICENSE) for more information. | |||
| @ -0,0 +1,17 @@ | |||
| { | |||
|     "name": "dnoegel/php-xdg-base-dir", | |||
|     "description": "implementation of xdg base directory specification for php", | |||
|     "type": "project", | |||
|     "license": "MIT", | |||
|     "require": { | |||
|         "php": ">=5.3.2" | |||
|     }, | |||
|     "require-dev": { | |||
|         "phpunit/phpunit": "@stable" | |||
|     }, | |||
|     "autoload": { | |||
|         "psr-4": { | |||
|             "XdgBaseDir\\": "src/" | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,24 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | |||
| 
 | |||
| <phpunit colors="true" | |||
|          convertErrorsToExceptions="true" | |||
|          convertNoticesToExceptions="true" | |||
|          convertWarningsToExceptions="true" | |||
|          processIsolation="false" | |||
|          stopOnFailure="false" | |||
|          syntaxCheck="false" | |||
|          bootstrap="vendor/autoload.php" | |||
|         > | |||
| 
 | |||
|     <testsuites> | |||
|         <testsuite name="php-xdg-base-dir unit tests"> | |||
|             <directory>./tests/</directory> | |||
|         </testsuite> | |||
|     </testsuites> | |||
| 
 | |||
|     <filter> | |||
|         <whitelist> | |||
|             <directory>./src/</directory> | |||
|         </whitelist> | |||
|     </filter> | |||
| </phpunit> | |||
| @ -0,0 +1,121 @@ | |||
| <?php | |||
| 
 | |||
| namespace XdgBaseDir; | |||
| 
 | |||
| /** | |||
|  * Simple implementation of the XDG standard http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html | |||
|  * | |||
|  * Based on the python implementation https://github.com/takluyver/pyxdg/blob/master/xdg/BaseDirectory.py | |||
|  * | |||
|  * Class Xdg | |||
|  * @package ShopwareCli\Application | |||
|  */ | |||
| class Xdg | |||
| { | |||
|     const S_IFDIR = 040000; // directory | |||
|     const S_IRWXO = 00007;  // rwx other | |||
|     const S_IRWXG = 00056;  // rwx group | |||
|     const RUNTIME_DIR_FALLBACK = 'php-xdg-runtime-dir-fallback-'; | |||
| 
 | |||
|     /** | |||
|      * @return string | |||
|      */ | |||
|     public function getHomeDir() | |||
|     { | |||
|         return getenv('HOME') ?: (getenv('HOMEDRIVE') . DIRECTORY_SEPARATOR . getenv('HOMEPATH')); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return string | |||
|      */ | |||
|     public function getHomeConfigDir() | |||
|     { | |||
|         $path = getenv('XDG_CONFIG_HOME') ?: $this->getHomeDir() . DIRECTORY_SEPARATOR . '.config'; | |||
| 
 | |||
|         return $path; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return string | |||
|      */ | |||
|     public function getHomeDataDir() | |||
|     { | |||
|         $path = getenv('XDG_DATA_HOME') ?: $this->getHomeDir() . DIRECTORY_SEPARATOR . '.local' . DIRECTORY_SEPARATOR . 'share'; | |||
| 
 | |||
|         return $path; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return array | |||
|      */ | |||
|     public function getConfigDirs() | |||
|     { | |||
|         $configDirs = getenv('XDG_CONFIG_DIRS') ? explode(':', getenv('XDG_CONFIG_DIRS')) : array('/etc/xdg'); | |||
| 
 | |||
|         $paths = array_merge(array($this->getHomeConfigDir()), $configDirs); | |||
| 
 | |||
|         return $paths; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return array | |||
|      */ | |||
|     public function getDataDirs() | |||
|     { | |||
|         $dataDirs = getenv('XDG_DATA_DIRS') ? explode(':', getenv('XDG_DATA_DIRS')) : array('/usr/local/share', '/usr/share'); | |||
| 
 | |||
|         $paths = array_merge(array($this->getHomeDataDir()), $dataDirs); | |||
| 
 | |||
|         return $paths; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return string | |||
|      */ | |||
|     public function getHomeCacheDir() | |||
|     { | |||
|         $path = getenv('XDG_CACHE_HOME') ?: $this->getHomeDir() . DIRECTORY_SEPARATOR . '.cache'; | |||
| 
 | |||
|         return $path; | |||
| 
 | |||
|     } | |||
| 
 | |||
|     public function getRuntimeDir($strict=true) | |||
|     { | |||
|         if ($runtimeDir = getenv('XDG_RUNTIME_DIR')) { | |||
|             return $runtimeDir; | |||
|         } | |||
| 
 | |||
|         if ($strict) { | |||
|             throw new \RuntimeException('XDG_RUNTIME_DIR was not set'); | |||
|         } | |||
| 
 | |||
|         $fallback = sys_get_temp_dir() . DIRECTORY_SEPARATOR . self::RUNTIME_DIR_FALLBACK . getenv('USER'); | |||
| 
 | |||
|         $create = false; | |||
| 
 | |||
|         if (!is_dir($fallback)) { | |||
|             mkdir($fallback, 0700, true); | |||
|         } | |||
| 
 | |||
|         $st = lstat($fallback); | |||
| 
 | |||
|         # The fallback must be a directory | |||
|         if (!$st['mode'] & self::S_IFDIR) { | |||
|             rmdir($fallback); | |||
|             $create = true; | |||
|         } elseif ($st['uid'] != getmyuid() || | |||
|             $st['mode'] & (self::S_IRWXG | self::S_IRWXO) | |||
|         ) { | |||
|             rmdir($fallback); | |||
|             $create = true; | |||
|         } | |||
| 
 | |||
|         if ($create) { | |||
|             mkdir($fallback, 0700, true); | |||
|         } | |||
| 
 | |||
|         return $fallback; | |||
|     } | |||
| 
 | |||
| } | |||
| @ -0,0 +1,116 @@ | |||
| <?php | |||
| 
 | |||
| class XdgTest extends PHPUnit_Framework_TestCase | |||
| { | |||
|     /** | |||
|      * @return \XdgBaseDir\Xdg | |||
|      */ | |||
|     public function getXdg() | |||
|     { | |||
|         return new \XdgBaseDir\Xdg(); | |||
|     } | |||
| 
 | |||
|     public function testGetHomeDir() | |||
|     { | |||
|          putenv('HOME=/fake-dir'); | |||
|          $this->assertEquals('/fake-dir', $this->getXdg()->getHomeDir()); | |||
|     } | |||
| 
 | |||
|     public function testGetFallbackHomeDir() | |||
|     { | |||
|         putenv('HOME='); | |||
|         putenv('HOMEDRIVE=C:'); | |||
|         putenv('HOMEPATH=fake-dir'); | |||
|         $this->assertEquals('C:/fake-dir', $this->getXdg()->getHomeDir()); | |||
|     } | |||
| 
 | |||
|     public function testXdgPutCache() | |||
|     { | |||
|         putenv('XDG_DATA_HOME=tmp/'); | |||
|         putenv('XDG_CONFIG_HOME=tmp/'); | |||
|         putenv('XDG_CACHE_HOME=tmp/'); | |||
|         $this->assertEquals('tmp/', $this->getXdg()->getHomeCacheDir()); | |||
|     } | |||
| 
 | |||
|     public function testXdgPutData() | |||
|     { | |||
|         putenv('XDG_DATA_HOME=tmp/'); | |||
|         $this->assertEquals('tmp/', $this->getXdg()->getHomeDataDir()); | |||
|     } | |||
| 
 | |||
|     public function testXdgPutConfig() | |||
|     { | |||
|         putenv('XDG_CONFIG_HOME=tmp/'); | |||
|         $this->assertEquals('tmp/', $this->getXdg()->getHomeConfigDir()); | |||
|     } | |||
| 
 | |||
|     public function testXdgDataDirsShouldIncludeHomeDataDir() | |||
|     { | |||
|         putenv('XDG_DATA_HOME=tmp/'); | |||
|         putenv('XDG_CONFIG_HOME=tmp/'); | |||
| 
 | |||
|         $this->assertArrayHasKey('tmp/', array_flip($this->getXdg()->getDataDirs())); | |||
|     } | |||
| 
 | |||
|     public function testXdgConfigDirsShouldIncludeHomeConfigDir() | |||
|     { | |||
|         putenv('XDG_CONFIG_HOME=tmp/'); | |||
| 
 | |||
|         $this->assertArrayHasKey('tmp/', array_flip($this->getXdg()->getConfigDirs())); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * If XDG_RUNTIME_DIR is set, it should be returned | |||
|      */ | |||
|     public function testGetRuntimeDir() | |||
|     { | |||
|         putenv('XDG_RUNTIME_DIR=/tmp/'); | |||
|         $runtimeDir = $this->getXdg()->getRuntimeDir(); | |||
| 
 | |||
|         $this->assertEquals(is_dir($runtimeDir), true); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * In strict mode, an exception should be shown if XDG_RUNTIME_DIR does not exist | |||
|      * | |||
|      * @expectedException \RuntimeException | |||
|      */ | |||
|     public function testGetRuntimeDirShouldThrowException() | |||
|     { | |||
|         putenv('XDG_RUNTIME_DIR='); | |||
|         $this->getXdg()->getRuntimeDir(true); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * In fallback mode a directory should be created | |||
|      */ | |||
|     public function testGetRuntimeDirShouldCreateDirectory() | |||
|     { | |||
|         putenv('XDG_RUNTIME_DIR='); | |||
|         $dir = $this->getXdg()->getRuntimeDir(false); | |||
|         $permission = decoct(fileperms($dir) & 0777); | |||
|         $this->assertEquals(700, $permission); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Ensure, that the fallback directories are created with correct permission | |||
|      */ | |||
|     public function testGetRuntimeShouldDeleteDirsWithWrongPermission() | |||
|     { | |||
|         $runtimeDir = sys_get_temp_dir() . DIRECTORY_SEPARATOR . XdgBaseDir\Xdg::RUNTIME_DIR_FALLBACK . getenv('USER'); | |||
| 
 | |||
|         rmdir($runtimeDir); | |||
|         mkdir($runtimeDir, 0764, true); | |||
| 
 | |||
|         // Permission should be wrong now | |||
|         $permission = decoct(fileperms($runtimeDir) & 0777); | |||
|         $this->assertEquals(764, $permission); | |||
| 
 | |||
|         putenv('XDG_RUNTIME_DIR='); | |||
|         $dir = $this->getXdg()->getRuntimeDir(false); | |||
| 
 | |||
|         // Permission should be fixed | |||
|         $permission = decoct(fileperms($dir) & 0777); | |||
|         $this->assertEquals(700, $permission); | |||
|     } | |||
| } | |||
| @ -0,0 +1,19 @@ | |||
| Copyright (c) 2006-2015 Doctrine Project | |||
| 
 | |||
| Permission is hereby granted, free of charge, to any person obtaining a copy of | |||
| this software and associated documentation files (the "Software"), to deal in | |||
| the Software without restriction, including without limitation the rights to | |||
| use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | |||
| of the Software, and to permit persons to whom the Software is furnished to do | |||
| so, subject to the following conditions: | |||
| 
 | |||
| The above copyright notice and this permission notice shall be included in all | |||
| copies or substantial portions of the Software. | |||
| 
 | |||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |||
| SOFTWARE. | |||
| @ -0,0 +1,6 @@ | |||
| # Doctrine Inflector | |||
| 
 | |||
| Doctrine Inflector is a small library that can perform string manipulations | |||
| with regard to upper-/lowercase and singular/plural forms of words. | |||
| 
 | |||
| [](https://travis-ci.org/doctrine/inflector) | |||
| @ -0,0 +1,32 @@ | |||
| { | |||
|     "name": "doctrine/inflector", | |||
|     "type": "library", | |||
|     "description": "Common String Manipulations with regard to casing and singular/plural rules.", | |||
|     "keywords": ["string", "inflection", "singularize", "pluralize"], | |||
|     "homepage": "http://www.doctrine-project.org", | |||
|     "license": "MIT", | |||
|     "authors": [ | |||
|         {"name": "Guilherme Blanco", "email": "guilhermeblanco@gmail.com"}, | |||
|         {"name": "Roman Borschel", "email": "roman@code-factory.org"}, | |||
|         {"name": "Benjamin Eberlei", "email": "kontakt@beberlei.de"}, | |||
|         {"name": "Jonathan Wage", "email": "jonwage@gmail.com"}, | |||
|         {"name": "Johannes Schmitt", "email": "schmittjoh@gmail.com"} | |||
|     ], | |||
|     "require": { | |||
|         "php": "^7.1" | |||
|     }, | |||
|     "require-dev": { | |||
|         "phpunit/phpunit": "^6.2" | |||
|     }, | |||
|     "autoload": { | |||
|         "psr-4": { "Doctrine\\Common\\Inflector\\": "lib/Doctrine/Common/Inflector" } | |||
|     }, | |||
|     "autoload-dev": { | |||
|         "psr-4": { "Doctrine\\Tests\\Common\\Inflector\\": "tests/Doctrine/Tests/Common/Inflector" } | |||
|     }, | |||
|     "extra": { | |||
|         "branch-alias": { | |||
|             "dev-master": "1.3.x-dev" | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,490 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Common\Inflector; | |||
| 
 | |||
| /** | |||
|  * Doctrine inflector has static methods for inflecting text. | |||
|  * | |||
|  * The methods in these classes are from several different sources collected | |||
|  * across several different php projects and several different authors. The | |||
|  * original author names and emails are not known. | |||
|  * | |||
|  * Pluralize & Singularize implementation are borrowed from CakePHP with some modifications. | |||
|  * | |||
|  * @link   www.doctrine-project.org | |||
|  * @since  1.0 | |||
|  * @author Konsta Vesterinen <kvesteri@cc.hut.fi> | |||
|  * @author Jonathan H. Wage <jonwage@gmail.com> | |||
|  */ | |||
| class Inflector | |||
| { | |||
|     /** | |||
|      * Plural inflector rules. | |||
|      * | |||
|      * @var string[][] | |||
|      */ | |||
|     private static $plural = array( | |||
|         'rules' => array( | |||
|             '/(s)tatus$/i' => '\1\2tatuses', | |||
|             '/(quiz)$/i' => '\1zes', | |||
|             '/^(ox)$/i' => '\1\2en', | |||
|             '/([m|l])ouse$/i' => '\1ice', | |||
|             '/(matr|vert|ind)(ix|ex)$/i' => '\1ices', | |||
|             '/(x|ch|ss|sh)$/i' => '\1es', | |||
|             '/([^aeiouy]|qu)y$/i' => '\1ies', | |||
|             '/(hive|gulf)$/i' => '\1s', | |||
|             '/(?:([^f])fe|([lr])f)$/i' => '\1\2ves', | |||
|             '/sis$/i' => 'ses', | |||
|             '/([ti])um$/i' => '\1a', | |||
|             '/(c)riterion$/i' => '\1riteria', | |||
|             '/(p)erson$/i' => '\1eople', | |||
|             '/(m)an$/i' => '\1en', | |||
|             '/(c)hild$/i' => '\1hildren', | |||
|             '/(f)oot$/i' => '\1eet', | |||
|             '/(buffal|her|potat|tomat|volcan)o$/i' => '\1\2oes', | |||
|             '/(alumn|bacill|cact|foc|fung|nucle|radi|stimul|syllab|termin|vir)us$/i' => '\1i', | |||
|             '/us$/i' => 'uses', | |||
|             '/(alias)$/i' => '\1es', | |||
|             '/(analys|ax|cris|test|thes)is$/i' => '\1es', | |||
|             '/s$/' => 's', | |||
|             '/^$/' => '', | |||
|             '/$/' => 's', | |||
|         ), | |||
|         'uninflected' => array( | |||
|             '.*[nrlm]ese', | |||
|             '.*deer', | |||
|             '.*fish', | |||
|             '.*measles', | |||
|             '.*ois', | |||
|             '.*pox', | |||
|             '.*sheep', | |||
|             'people', | |||
|             'cookie', | |||
|             'police', | |||
|         ), | |||
|         'irregular' => array( | |||
|             'atlas' => 'atlases', | |||
|             'axe' => 'axes', | |||
|             'beef' => 'beefs', | |||
|             'brother' => 'brothers', | |||
|             'cafe' => 'cafes', | |||
|             'chateau' => 'chateaux', | |||
|             'niveau' => 'niveaux', | |||
|             'child' => 'children', | |||
|             'cookie' => 'cookies', | |||
|             'corpus' => 'corpuses', | |||
|             'cow' => 'cows', | |||
|             'criterion' => 'criteria', | |||
|             'curriculum' => 'curricula', | |||
|             'demo' => 'demos', | |||
|             'domino' => 'dominoes', | |||
|             'echo' => 'echoes', | |||
|             'foot' => 'feet', | |||
|             'fungus' => 'fungi', | |||
|             'ganglion' => 'ganglions', | |||
|             'genie' => 'genies', | |||
|             'genus' => 'genera', | |||
|             'goose' => 'geese', | |||
|             'graffito' => 'graffiti', | |||
|             'hippopotamus' => 'hippopotami', | |||
|             'hoof' => 'hoofs', | |||
|             'human' => 'humans', | |||
|             'iris' => 'irises', | |||
|             'larva' => 'larvae', | |||
|             'leaf' => 'leaves', | |||
|             'loaf' => 'loaves', | |||
|             'man' => 'men', | |||
|             'medium' => 'media', | |||
|             'memorandum' => 'memoranda', | |||
|             'money' => 'monies', | |||
|             'mongoose' => 'mongooses', | |||
|             'motto' => 'mottoes', | |||
|             'move' => 'moves', | |||
|             'mythos' => 'mythoi', | |||
|             'niche' => 'niches', | |||
|             'nucleus' => 'nuclei', | |||
|             'numen' => 'numina', | |||
|             'occiput' => 'occiputs', | |||
|             'octopus' => 'octopuses', | |||
|             'opus' => 'opuses', | |||
|             'ox' => 'oxen', | |||
|             'passerby' => 'passersby', | |||
|             'penis' => 'penises', | |||
|             'person' => 'people', | |||
|             'plateau' => 'plateaux', | |||
|             'runner-up' => 'runners-up', | |||
|             'sex' => 'sexes', | |||
|             'soliloquy' => 'soliloquies', | |||
|             'son-in-law' => 'sons-in-law', | |||
|             'syllabus' => 'syllabi', | |||
|             'testis' => 'testes', | |||
|             'thief' => 'thieves', | |||
|             'tooth' => 'teeth', | |||
|             'tornado' => 'tornadoes', | |||
|             'trilby' => 'trilbys', | |||
|             'turf' => 'turfs', | |||
|             'valve' => 'valves', | |||
|             'volcano' => 'volcanoes', | |||
|         ) | |||
|     ); | |||
| 
 | |||
|     /** | |||
|      * Singular inflector rules. | |||
|      * | |||
|      * @var string[][] | |||
|      */ | |||
|     private static $singular = array( | |||
|         'rules' => array( | |||
|             '/(s)tatuses$/i' => '\1\2tatus', | |||
|             '/^(.*)(menu)s$/i' => '\1\2', | |||
|             '/(quiz)zes$/i' => '\\1', | |||
|             '/(matr)ices$/i' => '\1ix', | |||
|             '/(vert|ind)ices$/i' => '\1ex', | |||
|             '/^(ox)en/i' => '\1', | |||
|             '/(alias)(es)*$/i' => '\1', | |||
|             '/(buffal|her|potat|tomat|volcan)oes$/i' => '\1o', | |||
|             '/(alumn|bacill|cact|foc|fung|nucle|radi|stimul|syllab|termin|viri?)i$/i' => '\1us', | |||
|             '/([ftw]ax)es/i' => '\1', | |||
|             '/(analys|ax|cris|test|thes)es$/i' => '\1is', | |||
|             '/(shoe|slave)s$/i' => '\1', | |||
|             '/(o)es$/i' => '\1', | |||
|             '/ouses$/' => 'ouse', | |||
|             '/([^a])uses$/' => '\1us', | |||
|             '/([m|l])ice$/i' => '\1ouse', | |||
|             '/(x|ch|ss|sh)es$/i' => '\1', | |||
|             '/(m)ovies$/i' => '\1\2ovie', | |||
|             '/(s)eries$/i' => '\1\2eries', | |||
|             '/([^aeiouy]|qu)ies$/i' => '\1y', | |||
|             '/([lr])ves$/i' => '\1f', | |||
|             '/(tive)s$/i' => '\1', | |||
|             '/(hive)s$/i' => '\1', | |||
|             '/(drive)s$/i' => '\1', | |||
|             '/(dive)s$/i' => '\1', | |||
|             '/(olive)s$/i' => '\1', | |||
|             '/([^fo])ves$/i' => '\1fe', | |||
|             '/(^analy)ses$/i' => '\1sis', | |||
|             '/(analy|diagno|^ba|(p)arenthe|(p)rogno|(s)ynop|(t)he)ses$/i' => '\1\2sis', | |||
|             '/(c)riteria$/i' => '\1riterion', | |||
|             '/([ti])a$/i' => '\1um', | |||
|             '/(p)eople$/i' => '\1\2erson', | |||
|             '/(m)en$/i' => '\1an', | |||
|             '/(c)hildren$/i' => '\1\2hild', | |||
|             '/(f)eet$/i' => '\1oot', | |||
|             '/(n)ews$/i' => '\1\2ews', | |||
|             '/eaus$/' => 'eau', | |||
|             '/^(.*us)$/' => '\\1', | |||
|             '/s$/i' => '', | |||
|         ), | |||
|         'uninflected' => array( | |||
|             '.*[nrlm]ese', | |||
|             '.*deer', | |||
|             '.*fish', | |||
|             '.*measles', | |||
|             '.*ois', | |||
|             '.*pox', | |||
|             '.*sheep', | |||
|             '.*ss', | |||
|             'data', | |||
|             'police', | |||
|             'pants', | |||
|             'clothes', | |||
|         ), | |||
|         'irregular' => array( | |||
|             'abuses'     => 'abuse', | |||
|             'avalanches' => 'avalanche', | |||
|             'caches'     => 'cache', | |||
|             'criteria'   => 'criterion', | |||
|             'curves'     => 'curve', | |||
|             'emphases'   => 'emphasis', | |||
|             'foes'       => 'foe', | |||
|             'geese'      => 'goose', | |||
|             'graves'     => 'grave', | |||
|             'hoaxes'     => 'hoax', | |||
|             'media'      => 'medium', | |||
|             'neuroses'   => 'neurosis', | |||
|             'waves'      => 'wave', | |||
|             'oases'      => 'oasis', | |||
|             'valves'     => 'valve', | |||
|         ) | |||
|     ); | |||
| 
 | |||
|     /** | |||
|      * Words that should not be inflected. | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     private static $uninflected = array( | |||
|         '.*?media', 'Amoyese', 'audio', 'bison', 'Borghese', 'bream', 'breeches', | |||
|         'britches', 'buffalo', 'cantus', 'carp', 'chassis', 'clippers', 'cod', 'coitus', 'compensation', 'Congoese', | |||
|         'contretemps', 'coreopsis', 'corps', 'data', 'debris', 'deer', 'diabetes', 'djinn', 'education', 'eland', | |||
|         'elk', 'emoji', 'equipment', 'evidence', 'Faroese', 'feedback', 'fish', 'flounder', 'Foochowese', | |||
|         'Furniture', 'furniture', 'gallows', 'Genevese', 'Genoese', 'Gilbertese', 'gold',  | |||
|         'headquarters', 'herpes', 'hijinks', 'Hottentotese', 'information', 'innings', 'jackanapes', 'jedi', | |||
|         'Kiplingese', 'knowledge', 'Kongoese', 'love', 'Lucchese', 'Luggage', 'mackerel', 'Maltese', 'metadata', | |||
|         'mews', 'moose', 'mumps', 'Nankingese', 'news', 'nexus', 'Niasese', 'nutrition', 'offspring', | |||
|         'Pekingese', 'Piedmontese', 'pincers', 'Pistoiese', 'plankton', 'pliers', 'pokemon', 'police', 'Portuguese', | |||
|         'proceedings', 'rabies', 'rain', 'rhinoceros', 'rice', 'salmon', 'Sarawakese', 'scissors', 'sea[- ]bass', | |||
|         'series', 'Shavese', 'shears', 'sheep', 'siemens', 'species', 'staff', 'swine', 'traffic', | |||
|         'trousers', 'trout', 'tuna', 'us', 'Vermontese', 'Wenchowese', 'wheat', 'whiting', 'wildebeest', 'Yengeese' | |||
|     ); | |||
| 
 | |||
|     /** | |||
|      * Method cache array. | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     private static $cache = array(); | |||
| 
 | |||
|     /** | |||
|      * The initial state of Inflector so reset() works. | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     private static $initialState = array(); | |||
| 
 | |||
|     /** | |||
|      * Converts a word into the format for a Doctrine table name. Converts 'ModelName' to 'model_name'. | |||
|      */ | |||
|     public static function tableize(string $word) : string | |||
|     { | |||
|         return strtolower(preg_replace('~(?<=\\w)([A-Z])~', '_$1', $word)); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Converts a word into the format for a Doctrine class name. Converts 'table_name' to 'TableName'. | |||
|      */ | |||
|     public static function classify(string $word) : string | |||
|     { | |||
|         return str_replace([' ', '_', '-'], '', ucwords($word, ' _-')); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Camelizes a word. This uses the classify() method and turns the first character to lowercase. | |||
|      */ | |||
|     public static function camelize(string $word) : string | |||
|     { | |||
|         return lcfirst(self::classify($word)); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Uppercases words with configurable delimeters between words. | |||
|      * | |||
|      * Takes a string and capitalizes all of the words, like PHP's built-in | |||
|      * ucwords function. This extends that behavior, however, by allowing the | |||
|      * word delimeters to be configured, rather than only separating on | |||
|      * whitespace. | |||
|      * | |||
|      * Here is an example: | |||
|      * <code> | |||
|      * <?php | |||
|      * $string = 'top-o-the-morning to all_of_you!'; | |||
|      * echo \Doctrine\Common\Inflector\Inflector::ucwords($string); | |||
|      * // Top-O-The-Morning To All_of_you! | |||
|      * | |||
|      * echo \Doctrine\Common\Inflector\Inflector::ucwords($string, '-_ '); | |||
|      * // Top-O-The-Morning To All_Of_You! | |||
|      * ?> | |||
|      * </code> | |||
|      * | |||
|      * @param string $string The string to operate on. | |||
|      * @param string $delimiters A list of word separators. | |||
|      * | |||
|      * @return string The string with all delimeter-separated words capitalized. | |||
|      */ | |||
|     public static function ucwords(string $string, string $delimiters = " \n\t\r\0\x0B-") : string | |||
|     { | |||
|         return ucwords($string, $delimiters); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Clears Inflectors inflected value caches, and resets the inflection | |||
|      * rules to the initial values. | |||
|      */ | |||
|     public static function reset() : void | |||
|     { | |||
|         if (empty(self::$initialState)) { | |||
|             self::$initialState = get_class_vars('Inflector'); | |||
| 
 | |||
|             return; | |||
|         } | |||
| 
 | |||
|         foreach (self::$initialState as $key => $val) { | |||
|             if ($key !== 'initialState') { | |||
|                 self::${$key} = $val; | |||
|             } | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Adds custom inflection $rules, of either 'plural' or 'singular' $type. | |||
|      * | |||
|      * ### Usage: | |||
|      * | |||
|      * {{{ | |||
|      * Inflector::rules('plural', array('/^(inflect)or$/i' => '\1ables')); | |||
|      * Inflector::rules('plural', array( | |||
|      *     'rules' => array('/^(inflect)ors$/i' => '\1ables'), | |||
|      *     'uninflected' => array('dontinflectme'), | |||
|      *     'irregular' => array('red' => 'redlings') | |||
|      * )); | |||
|      * }}} | |||
|      * | |||
|      * @param string  $type         The type of inflection, either 'plural' or 'singular' | |||
|      * @param array|iterable $rules An array of rules to be added. | |||
|      * @param boolean $reset        If true, will unset default inflections for all | |||
|      *                              new rules that are being defined in $rules. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public static function rules(string $type, iterable $rules, bool $reset = false) : void | |||
|     { | |||
|         foreach ($rules as $rule => $pattern) { | |||
|             if ( ! is_array($pattern)) { | |||
|                 continue; | |||
|             } | |||
| 
 | |||
|             if ($reset) { | |||
|                 self::${$type}[$rule] = $pattern; | |||
|             } else { | |||
|                 self::${$type}[$rule] = ($rule === 'uninflected') | |||
|                     ? array_merge($pattern, self::${$type}[$rule]) | |||
|                     : $pattern + self::${$type}[$rule]; | |||
|             } | |||
| 
 | |||
|             unset($rules[$rule], self::${$type}['cache' . ucfirst($rule)]); | |||
| 
 | |||
|             if (isset(self::${$type}['merged'][$rule])) { | |||
|                 unset(self::${$type}['merged'][$rule]); | |||
|             } | |||
| 
 | |||
|             if ($type === 'plural') { | |||
|                 self::$cache['pluralize'] = self::$cache['tableize'] = array(); | |||
|             } elseif ($type === 'singular') { | |||
|                 self::$cache['singularize'] = array(); | |||
|             } | |||
|         } | |||
| 
 | |||
|         self::${$type}['rules'] = $rules + self::${$type}['rules']; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Returns a word in plural form. | |||
|      * | |||
|      * @param string $word The word in singular form. | |||
|      * | |||
|      * @return string The word in plural form. | |||
|      */ | |||
|     public static function pluralize(string $word) : string | |||
|     { | |||
|         if (isset(self::$cache['pluralize'][$word])) { | |||
|             return self::$cache['pluralize'][$word]; | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$plural['merged']['irregular'])) { | |||
|             self::$plural['merged']['irregular'] = self::$plural['irregular']; | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$plural['merged']['uninflected'])) { | |||
|             self::$plural['merged']['uninflected'] = array_merge(self::$plural['uninflected'], self::$uninflected); | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$plural['cacheUninflected']) || !isset(self::$plural['cacheIrregular'])) { | |||
|             self::$plural['cacheUninflected'] = '(?:' . implode('|', self::$plural['merged']['uninflected']) . ')'; | |||
|             self::$plural['cacheIrregular']   = '(?:' . implode('|', array_keys(self::$plural['merged']['irregular'])) . ')'; | |||
|         } | |||
| 
 | |||
|         if (preg_match('/(.*)\\b(' . self::$plural['cacheIrregular'] . ')$/i', $word, $regs)) { | |||
|             self::$cache['pluralize'][$word] = $regs[1] . $word[0] . substr(self::$plural['merged']['irregular'][strtolower($regs[2])], 1); | |||
| 
 | |||
|             return self::$cache['pluralize'][$word]; | |||
|         } | |||
| 
 | |||
|         if (preg_match('/^(' . self::$plural['cacheUninflected'] . ')$/i', $word, $regs)) { | |||
|             self::$cache['pluralize'][$word] = $word; | |||
| 
 | |||
|             return $word; | |||
|         } | |||
| 
 | |||
|         foreach (self::$plural['rules'] as $rule => $replacement) { | |||
|             if (preg_match($rule, $word)) { | |||
|                 self::$cache['pluralize'][$word] = preg_replace($rule, $replacement, $word); | |||
| 
 | |||
|                 return self::$cache['pluralize'][$word]; | |||
|             } | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Returns a word in singular form. | |||
|      * | |||
|      * @param string $word The word in plural form. | |||
|      * | |||
|      * @return string The word in singular form. | |||
|      */ | |||
|     public static function singularize(string $word) : string | |||
|     { | |||
|         if (isset(self::$cache['singularize'][$word])) { | |||
|             return self::$cache['singularize'][$word]; | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$singular['merged']['uninflected'])) { | |||
|             self::$singular['merged']['uninflected'] = array_merge( | |||
|                 self::$singular['uninflected'], | |||
|                 self::$uninflected | |||
|             ); | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$singular['merged']['irregular'])) { | |||
|             self::$singular['merged']['irregular'] = array_merge( | |||
|                 self::$singular['irregular'], | |||
|                 array_flip(self::$plural['irregular']) | |||
|             ); | |||
|         } | |||
| 
 | |||
|         if (!isset(self::$singular['cacheUninflected']) || !isset(self::$singular['cacheIrregular'])) { | |||
|             self::$singular['cacheUninflected'] = '(?:' . implode('|', self::$singular['merged']['uninflected']) . ')'; | |||
|             self::$singular['cacheIrregular'] = '(?:' . implode('|', array_keys(self::$singular['merged']['irregular'])) . ')'; | |||
|         } | |||
| 
 | |||
|         if (preg_match('/(.*)\\b(' . self::$singular['cacheIrregular'] . ')$/i', $word, $regs)) { | |||
|             self::$cache['singularize'][$word] = $regs[1] . $word[0] . substr(self::$singular['merged']['irregular'][strtolower($regs[2])], 1); | |||
| 
 | |||
|             return self::$cache['singularize'][$word]; | |||
|         } | |||
| 
 | |||
|         if (preg_match('/^(' . self::$singular['cacheUninflected'] . ')$/i', $word, $regs)) { | |||
|             self::$cache['singularize'][$word] = $word; | |||
| 
 | |||
|             return $word; | |||
|         } | |||
| 
 | |||
|         foreach (self::$singular['rules'] as $rule => $replacement) { | |||
|             if (preg_match($rule, $word)) { | |||
|                 self::$cache['singularize'][$word] = preg_replace($rule, $replacement, $word); | |||
| 
 | |||
|                 return self::$cache['singularize'][$word]; | |||
|             } | |||
|         } | |||
| 
 | |||
|         self::$cache['singularize'][$word] = $word; | |||
| 
 | |||
|         return $word; | |||
|     } | |||
| } | |||
| @ -0,0 +1,35 @@ | |||
| # Contributing | |||
| 
 | |||
|  * Coding standard for the project is [PSR-2](https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-2-coding-style-guide.md) | |||
|  * The project will follow strict [object calisthenics](http://www.slideshare.net/guilhermeblanco/object-calisthenics-applied-to-php) | |||
|  * Any contribution must provide tests for additional introduced conditions | |||
|  * Any un-confirmed issue needs a failing test case before being accepted | |||
|  * Pull requests must be sent from a new hotfix/feature branch, not from `master`. | |||
| 
 | |||
| ## Installation | |||
| 
 | |||
| To install the project and run the tests, you need to clone it first: | |||
| 
 | |||
| ```sh | |||
| $ git clone git://github.com/doctrine/instantiator.git | |||
| ``` | |||
| 
 | |||
| You will then need to run a composer installation: | |||
| 
 | |||
| ```sh | |||
| $ cd Instantiator | |||
| $ curl -s https://getcomposer.org/installer | php | |||
| $ php composer.phar update | |||
| ``` | |||
| 
 | |||
| ## Testing | |||
| 
 | |||
| The PHPUnit version to be used is the one installed as a dev- dependency via composer: | |||
| 
 | |||
| ```sh | |||
| $ ./vendor/bin/phpunit | |||
| ``` | |||
| 
 | |||
| Accepted coverage for new contributions is 80%. Any contribution not satisfying this requirement  | |||
| won't be merged. | |||
| 
 | |||
| @ -0,0 +1,19 @@ | |||
| Copyright (c) 2014 Doctrine Project | |||
| 
 | |||
| Permission is hereby granted, free of charge, to any person obtaining a copy of | |||
| this software and associated documentation files (the "Software"), to deal in | |||
| the Software without restriction, including without limitation the rights to | |||
| use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | |||
| of the Software, and to permit persons to whom the Software is furnished to do | |||
| so, subject to the following conditions: | |||
| 
 | |||
| The above copyright notice and this permission notice shall be included in all | |||
| copies or substantial portions of the Software. | |||
| 
 | |||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |||
| SOFTWARE. | |||
| @ -0,0 +1,40 @@ | |||
| # Instantiator | |||
| 
 | |||
| This library provides a way of avoiding usage of constructors when instantiating PHP classes. | |||
| 
 | |||
| [](https://travis-ci.org/doctrine/instantiator) | |||
| [](https://scrutinizer-ci.com/g/doctrine/instantiator/?branch=master) | |||
| [](https://scrutinizer-ci.com/g/doctrine/instantiator/?branch=master) | |||
| [](https://www.versioneye.com/package/php--doctrine--instantiator) | |||
| [](http://hhvm.h4cc.de/package/doctrine/instantiator) | |||
| 
 | |||
| [](https://packagist.org/packages/doctrine/instantiator) | |||
| [](https://packagist.org/packages/doctrine/instantiator) | |||
| 
 | |||
| ## Installation | |||
| 
 | |||
| The suggested installation method is via [composer](https://getcomposer.org/): | |||
| 
 | |||
| ```sh | |||
| php composer.phar require "doctrine/instantiator:~1.0.3" | |||
| ``` | |||
| 
 | |||
| ## Usage | |||
| 
 | |||
| The instantiator is able to create new instances of any class without using the constructor or any API of the class | |||
| itself: | |||
| 
 | |||
| ```php | |||
| $instantiator = new \Doctrine\Instantiator\Instantiator(); | |||
| 
 | |||
| $instance = $instantiator->instantiate(\My\ClassName\Here::class); | |||
| ``` | |||
| 
 | |||
| ## Contributing | |||
| 
 | |||
| Please read the [CONTRIBUTING.md](CONTRIBUTING.md) contents if you wish to help out! | |||
| 
 | |||
| ## Credits | |||
| 
 | |||
| This library was migrated from [ocramius/instantiator](https://github.com/Ocramius/Instantiator), which | |||
| has been donated to the doctrine organization, and which is now deprecated in favour of this package. | |||
| @ -0,0 +1,45 @@ | |||
| { | |||
|     "name":              "doctrine/instantiator", | |||
|     "description":       "A small, lightweight utility to instantiate objects in PHP without invoking their constructors", | |||
|     "type":              "library", | |||
|     "license":           "MIT", | |||
|     "homepage":          "https://github.com/doctrine/instantiator", | |||
|     "keywords":          [ | |||
|         "instantiate", | |||
|         "constructor" | |||
|     ], | |||
|     "authors": [ | |||
|         { | |||
|             "name":     "Marco Pivetta", | |||
|             "email":    "ocramius@gmail.com", | |||
|             "homepage": "http://ocramius.github.com/" | |||
|         } | |||
|     ], | |||
|     "require": { | |||
|         "php": "^7.1" | |||
|     }, | |||
|     "require-dev": { | |||
|         "ext-phar":                  "*", | |||
|         "ext-pdo":                   "*", | |||
|         "phpunit/phpunit":           "^6.2.3", | |||
|         "squizlabs/php_codesniffer": "^3.0.2", | |||
|         "athletic/athletic":         "~0.1.8" | |||
|     }, | |||
|     "autoload": { | |||
|         "psr-4": { | |||
|             "Doctrine\\Instantiator\\": "src/Doctrine/Instantiator/" | |||
|         } | |||
|     }, | |||
|     "autoload-dev": { | |||
|         "psr-0": { | |||
|             "DoctrineTest\\InstantiatorPerformance\\": "tests", | |||
|             "DoctrineTest\\InstantiatorTest\\": "tests", | |||
|             "DoctrineTest\\InstantiatorTestAsset\\": "tests" | |||
|         } | |||
|     }, | |||
|     "extra": { | |||
|         "branch-alias": { | |||
|             "dev-master": "1.2.x-dev" | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,29 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Instantiator\Exception; | |||
| 
 | |||
| /** | |||
|  * Base exception marker interface for the instantiator component | |||
|  * | |||
|  * @author Marco Pivetta <ocramius@gmail.com> | |||
|  */ | |||
| interface ExceptionInterface | |||
| { | |||
| } | |||
| @ -0,0 +1,52 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Instantiator\Exception; | |||
| 
 | |||
| use InvalidArgumentException as BaseInvalidArgumentException; | |||
| use ReflectionClass; | |||
| 
 | |||
| /** | |||
|  * Exception for invalid arguments provided to the instantiator | |||
|  * | |||
|  * @author Marco Pivetta <ocramius@gmail.com> | |||
|  */ | |||
| class InvalidArgumentException extends BaseInvalidArgumentException implements ExceptionInterface | |||
| { | |||
|     public static function fromNonExistingClass(string $className) : self | |||
|     { | |||
|         if (interface_exists($className)) { | |||
|             return new self(sprintf('The provided type "%s" is an interface, and can not be instantiated', $className)); | |||
|         } | |||
| 
 | |||
|         if (PHP_VERSION_ID >= 50400 && trait_exists($className)) { | |||
|             return new self(sprintf('The provided type "%s" is a trait, and can not be instantiated', $className)); | |||
|         } | |||
| 
 | |||
|         return new self(sprintf('The provided class "%s" does not exist', $className)); | |||
|     } | |||
| 
 | |||
|     public static function fromAbstractClass(ReflectionClass $reflectionClass) : self | |||
|     { | |||
|         return new self(sprintf( | |||
|             'The provided class "%s" is abstract, and can not be instantiated', | |||
|             $reflectionClass->getName() | |||
|         )); | |||
|     } | |||
| } | |||
| @ -0,0 +1,66 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Instantiator\Exception; | |||
| 
 | |||
| use Exception; | |||
| use ReflectionClass; | |||
| use UnexpectedValueException as BaseUnexpectedValueException; | |||
| 
 | |||
| /** | |||
|  * Exception for given parameters causing invalid/unexpected state on instantiation | |||
|  * | |||
|  * @author Marco Pivetta <ocramius@gmail.com> | |||
|  */ | |||
| class UnexpectedValueException extends BaseUnexpectedValueException implements ExceptionInterface | |||
| { | |||
|     public static function fromSerializationTriggeredException( | |||
|         ReflectionClass $reflectionClass, | |||
|         Exception $exception | |||
|     ) : self { | |||
|         return new self( | |||
|             sprintf( | |||
|                 'An exception was raised while trying to instantiate an instance of "%s" via un-serialization', | |||
|                 $reflectionClass->getName() | |||
|             ), | |||
|             0, | |||
|             $exception | |||
|         ); | |||
|     } | |||
| 
 | |||
|     public static function fromUncleanUnSerialization( | |||
|         ReflectionClass $reflectionClass, | |||
|         string $errorString, | |||
|         int $errorCode, | |||
|         string $errorFile, | |||
|         int $errorLine | |||
|     ) : self { | |||
|         return new self( | |||
|             sprintf( | |||
|                 'Could not produce an instance of "%s" via un-serialization, since an error was triggered ' | |||
|                 . 'in file "%s" at line "%d"', | |||
|                 $reflectionClass->getName(), | |||
|                 $errorFile, | |||
|                 $errorLine | |||
|             ), | |||
|             0, | |||
|             new Exception($errorString, $errorCode) | |||
|         ); | |||
|     } | |||
| } | |||
| @ -0,0 +1,216 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Instantiator; | |||
| 
 | |||
| use Doctrine\Instantiator\Exception\InvalidArgumentException; | |||
| use Doctrine\Instantiator\Exception\UnexpectedValueException; | |||
| use Exception; | |||
| use ReflectionClass; | |||
| 
 | |||
| /** | |||
|  * {@inheritDoc} | |||
|  * | |||
|  * @author Marco Pivetta <ocramius@gmail.com> | |||
|  */ | |||
| final class Instantiator implements InstantiatorInterface | |||
| { | |||
|     /** | |||
|      * Markers used internally by PHP to define whether {@see \unserialize} should invoke | |||
|      * the method {@see \Serializable::unserialize()} when dealing with classes implementing | |||
|      * the {@see \Serializable} interface. | |||
|      */ | |||
|     const SERIALIZATION_FORMAT_USE_UNSERIALIZER   = 'C'; | |||
|     const SERIALIZATION_FORMAT_AVOID_UNSERIALIZER = 'O'; | |||
| 
 | |||
|     /** | |||
|      * @var \callable[] used to instantiate specific classes, indexed by class name | |||
|      */ | |||
|     private static $cachedInstantiators = []; | |||
| 
 | |||
|     /** | |||
|      * @var object[] of objects that can directly be cloned, indexed by class name | |||
|      */ | |||
|     private static $cachedCloneables = []; | |||
| 
 | |||
|     /** | |||
|      * {@inheritDoc} | |||
|      */ | |||
|     public function instantiate($className) | |||
|     { | |||
|         if (isset(self::$cachedCloneables[$className])) { | |||
|             return clone self::$cachedCloneables[$className]; | |||
|         } | |||
| 
 | |||
|         if (isset(self::$cachedInstantiators[$className])) { | |||
|             $factory = self::$cachedInstantiators[$className]; | |||
| 
 | |||
|             return $factory(); | |||
|         } | |||
| 
 | |||
|         return $this->buildAndCacheFromFactory($className); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Builds the requested object and caches it in static properties for performance | |||
|      * | |||
|      * @return object | |||
|      */ | |||
|     private function buildAndCacheFromFactory(string $className) | |||
|     { | |||
|         $factory  = self::$cachedInstantiators[$className] = $this->buildFactory($className); | |||
|         $instance = $factory(); | |||
| 
 | |||
|         if ($this->isSafeToClone(new ReflectionClass($instance))) { | |||
|             self::$cachedCloneables[$className] = clone $instance; | |||
|         } | |||
| 
 | |||
|         return $instance; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Builds a callable capable of instantiating the given $className without | |||
|      * invoking its constructor. | |||
|      * | |||
|      * @throws InvalidArgumentException | |||
|      * @throws UnexpectedValueException | |||
|      * @throws \ReflectionException | |||
|      */ | |||
|     private function buildFactory(string $className) : callable | |||
|     { | |||
|         $reflectionClass = $this->getReflectionClass($className); | |||
| 
 | |||
|         if ($this->isInstantiableViaReflection($reflectionClass)) { | |||
|             return [$reflectionClass, 'newInstanceWithoutConstructor']; | |||
|         } | |||
| 
 | |||
|         $serializedString = sprintf( | |||
|             '%s:%d:"%s":0:{}', | |||
|             self::SERIALIZATION_FORMAT_AVOID_UNSERIALIZER, | |||
|             strlen($className), | |||
|             $className | |||
|         ); | |||
| 
 | |||
|         $this->checkIfUnSerializationIsSupported($reflectionClass, $serializedString); | |||
| 
 | |||
|         return function () use ($serializedString) { | |||
|             return unserialize($serializedString); | |||
|         }; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param string $className | |||
|      * | |||
|      * @return ReflectionClass | |||
|      * | |||
|      * @throws InvalidArgumentException | |||
|      * @throws \ReflectionException | |||
|      */ | |||
|     private function getReflectionClass($className) : ReflectionClass | |||
|     { | |||
|         if (! class_exists($className)) { | |||
|             throw InvalidArgumentException::fromNonExistingClass($className); | |||
|         } | |||
| 
 | |||
|         $reflection = new ReflectionClass($className); | |||
| 
 | |||
|         if ($reflection->isAbstract()) { | |||
|             throw InvalidArgumentException::fromAbstractClass($reflection); | |||
|         } | |||
| 
 | |||
|         return $reflection; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param ReflectionClass $reflectionClass | |||
|      * @param string          $serializedString | |||
|      * | |||
|      * @throws UnexpectedValueException | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     private function checkIfUnSerializationIsSupported(ReflectionClass $reflectionClass, $serializedString) : void | |||
|     { | |||
|         set_error_handler(function ($code, $message, $file, $line) use ($reflectionClass, & $error) : void { | |||
|             $error = UnexpectedValueException::fromUncleanUnSerialization( | |||
|                 $reflectionClass, | |||
|                 $message, | |||
|                 $code, | |||
|                 $file, | |||
|                 $line | |||
|             ); | |||
|         }); | |||
| 
 | |||
|         $this->attemptInstantiationViaUnSerialization($reflectionClass, $serializedString); | |||
| 
 | |||
|         restore_error_handler(); | |||
| 
 | |||
|         if ($error) { | |||
|             throw $error; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param ReflectionClass $reflectionClass | |||
|      * @param string          $serializedString | |||
|      * | |||
|      * @throws UnexpectedValueException | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     private function attemptInstantiationViaUnSerialization(ReflectionClass $reflectionClass, $serializedString) : void | |||
|     { | |||
|         try { | |||
|             unserialize($serializedString); | |||
|         } catch (Exception $exception) { | |||
|             restore_error_handler(); | |||
| 
 | |||
|             throw UnexpectedValueException::fromSerializationTriggeredException($reflectionClass, $exception); | |||
|         } | |||
|     } | |||
| 
 | |||
|     private function isInstantiableViaReflection(ReflectionClass $reflectionClass) : bool | |||
|     { | |||
|         return ! ($this->hasInternalAncestors($reflectionClass) && $reflectionClass->isFinal()); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Verifies whether the given class is to be considered internal | |||
|      */ | |||
|     private function hasInternalAncestors(ReflectionClass $reflectionClass) : bool | |||
|     { | |||
|         do { | |||
|             if ($reflectionClass->isInternal()) { | |||
|                 return true; | |||
|             } | |||
|         } while ($reflectionClass = $reflectionClass->getParentClass()); | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Checks if a class is cloneable | |||
|      * | |||
|      * Classes implementing `__clone` cannot be safely cloned, as that may cause side-effects. | |||
|      */ | |||
|     private function isSafeToClone(ReflectionClass $reflection) : bool | |||
|     { | |||
|         return $reflection->isCloneable() && ! $reflection->hasMethod('__clone'); | |||
|     } | |||
| } | |||
| @ -0,0 +1,37 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Instantiator; | |||
| 
 | |||
| /** | |||
|  * Instantiator provides utility methods to build objects without invoking their constructors | |||
|  * | |||
|  * @author Marco Pivetta <ocramius@gmail.com> | |||
|  */ | |||
| interface InstantiatorInterface | |||
| { | |||
|     /** | |||
|      * @param string $className | |||
|      * | |||
|      * @return object | |||
|      * | |||
|      * @throws \Doctrine\Instantiator\Exception\ExceptionInterface | |||
|      */ | |||
|     public function instantiate($className); | |||
| } | |||
| @ -0,0 +1,19 @@ | |||
| Copyright (c) 2006-2013 Doctrine Project | |||
| 
 | |||
| Permission is hereby granted, free of charge, to any person obtaining a copy of | |||
| this software and associated documentation files (the "Software"), to deal in | |||
| the Software without restriction, including without limitation the rights to | |||
| use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | |||
| of the Software, and to permit persons to whom the Software is furnished to do | |||
| so, subject to the following conditions: | |||
| 
 | |||
| The above copyright notice and this permission notice shall be included in all | |||
| copies or substantial portions of the Software. | |||
| 
 | |||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |||
| SOFTWARE. | |||
| @ -0,0 +1,5 @@ | |||
| # Doctrine Lexer | |||
| 
 | |||
| Base library for a lexer that can be used in Top-Down, Recursive Descent Parsers. | |||
| 
 | |||
| This lexer is used in Doctrine Annotations and in Doctrine ORM (DQL). | |||
| @ -0,0 +1,24 @@ | |||
| { | |||
|     "name": "doctrine/lexer", | |||
|     "type": "library", | |||
|     "description": "Base library for a lexer that can be used in Top-Down, Recursive Descent Parsers.", | |||
|     "keywords": ["lexer", "parser"], | |||
|     "homepage": "http://www.doctrine-project.org", | |||
|     "license": "MIT", | |||
|     "authors": [ | |||
|         {"name": "Guilherme Blanco", "email": "guilhermeblanco@gmail.com"}, | |||
|         {"name": "Roman Borschel", "email": "roman@code-factory.org"}, | |||
|         {"name": "Johannes Schmitt", "email": "schmittjoh@gmail.com"} | |||
|     ], | |||
|     "require": { | |||
|         "php": ">=5.3.2" | |||
|     }, | |||
|     "autoload": { | |||
|         "psr-0": { "Doctrine\\Common\\Lexer\\": "lib/" } | |||
|     }, | |||
|     "extra": { | |||
|         "branch-alias": { | |||
|             "dev-master": "1.0.x-dev" | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,327 @@ | |||
| <?php | |||
| /* | |||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
|  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
|  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
|  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
|  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
|  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
|  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
|  * | |||
|  * This software consists of voluntary contributions made by many individuals | |||
|  * and is licensed under the MIT license. For more information, see | |||
|  * <http://www.doctrine-project.org>. | |||
|  */ | |||
| 
 | |||
| namespace Doctrine\Common\Lexer; | |||
| 
 | |||
| /** | |||
|  * Base class for writing simple lexers, i.e. for creating small DSLs. | |||
|  * | |||
|  * @since  2.0 | |||
|  * @author Guilherme Blanco <guilhermeblanco@hotmail.com> | |||
|  * @author Jonathan Wage <jonwage@gmail.com> | |||
|  * @author Roman Borschel <roman@code-factory.org> | |||
|  */ | |||
| abstract class AbstractLexer | |||
| { | |||
|     /** | |||
|      * Lexer original input string. | |||
|      * | |||
|      * @var string | |||
|      */ | |||
|     private $input; | |||
| 
 | |||
|     /** | |||
|      * Array of scanned tokens. | |||
|      * | |||
|      * Each token is an associative array containing three items: | |||
|      *  - 'value'    : the string value of the token in the input string | |||
|      *  - 'type'     : the type of the token (identifier, numeric, string, input | |||
|      *                 parameter, none) | |||
|      *  - 'position' : the position of the token in the input string | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     private $tokens = array(); | |||
| 
 | |||
|     /** | |||
|      * Current lexer position in input string. | |||
|      * | |||
|      * @var integer | |||
|      */ | |||
|     private $position = 0; | |||
| 
 | |||
|     /** | |||
|      * Current peek of current lexer position. | |||
|      * | |||
|      * @var integer | |||
|      */ | |||
|     private $peek = 0; | |||
| 
 | |||
|     /** | |||
|      * The next token in the input. | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     public $lookahead; | |||
| 
 | |||
|     /** | |||
|      * The last matched/seen token. | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     public $token; | |||
| 
 | |||
|     /** | |||
|      * Sets the input data to be tokenized. | |||
|      * | |||
|      * The Lexer is immediately reset and the new input tokenized. | |||
|      * Any unprocessed tokens from any previous input are lost. | |||
|      * | |||
|      * @param string $input The input to be tokenized. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public function setInput($input) | |||
|     { | |||
|         $this->input  = $input; | |||
|         $this->tokens = array(); | |||
| 
 | |||
|         $this->reset(); | |||
|         $this->scan($input); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Resets the lexer. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public function reset() | |||
|     { | |||
|         $this->lookahead = null; | |||
|         $this->token = null; | |||
|         $this->peek = 0; | |||
|         $this->position = 0; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Resets the peek pointer to 0. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public function resetPeek() | |||
|     { | |||
|         $this->peek = 0; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Resets the lexer position on the input to the given position. | |||
|      * | |||
|      * @param integer $position Position to place the lexical scanner. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public function resetPosition($position = 0) | |||
|     { | |||
|         $this->position = $position; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Retrieve the original lexer's input until a given position.  | |||
|      * | |||
|      * @param integer $position | |||
|      * | |||
|      * @return string | |||
|      */ | |||
|     public function getInputUntilPosition($position) | |||
|     { | |||
|         return substr($this->input, 0, $position); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Checks whether a given token matches the current lookahead. | |||
|      * | |||
|      * @param integer|string $token | |||
|      * | |||
|      * @return boolean | |||
|      */ | |||
|     public function isNextToken($token) | |||
|     { | |||
|         return null !== $this->lookahead && $this->lookahead['type'] === $token; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Checks whether any of the given tokens matches the current lookahead. | |||
|      * | |||
|      * @param array $tokens | |||
|      * | |||
|      * @return boolean | |||
|      */ | |||
|     public function isNextTokenAny(array $tokens) | |||
|     { | |||
|         return null !== $this->lookahead && in_array($this->lookahead['type'], $tokens, true); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Moves to the next token in the input string. | |||
|      * | |||
|      * @return boolean | |||
|      */ | |||
|     public function moveNext() | |||
|     { | |||
|         $this->peek = 0; | |||
|         $this->token = $this->lookahead; | |||
|         $this->lookahead = (isset($this->tokens[$this->position])) | |||
|             ? $this->tokens[$this->position++] : null; | |||
| 
 | |||
|         return $this->lookahead !== null; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Tells the lexer to skip input tokens until it sees a token with the given value. | |||
|      * | |||
|      * @param string $type The token type to skip until. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     public function skipUntil($type) | |||
|     { | |||
|         while ($this->lookahead !== null && $this->lookahead['type'] !== $type) { | |||
|             $this->moveNext(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Checks if given value is identical to the given token. | |||
|      * | |||
|      * @param mixed   $value | |||
|      * @param integer $token | |||
|      * | |||
|      * @return boolean | |||
|      */ | |||
|     public function isA($value, $token) | |||
|     { | |||
|         return $this->getType($value) === $token; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Moves the lookahead token forward. | |||
|      * | |||
|      * @return array|null The next token or NULL if there are no more tokens ahead. | |||
|      */ | |||
|     public function peek() | |||
|     { | |||
|         if (isset($this->tokens[$this->position + $this->peek])) { | |||
|             return $this->tokens[$this->position + $this->peek++]; | |||
|         } else { | |||
|             return null; | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Peeks at the next token, returns it and immediately resets the peek. | |||
|      * | |||
|      * @return array|null The next token or NULL if there are no more tokens ahead. | |||
|      */ | |||
|     public function glimpse() | |||
|     { | |||
|         $peek = $this->peek(); | |||
|         $this->peek = 0; | |||
|         return $peek; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Scans the input string for tokens. | |||
|      * | |||
|      * @param string $input A query string. | |||
|      * | |||
|      * @return void | |||
|      */ | |||
|     protected function scan($input) | |||
|     { | |||
|         static $regex; | |||
| 
 | |||
|         if ( ! isset($regex)) { | |||
|             $regex = sprintf( | |||
|                 '/(%s)|%s/%s', | |||
|                 implode(')|(', $this->getCatchablePatterns()), | |||
|                 implode('|', $this->getNonCatchablePatterns()), | |||
|                 $this->getModifiers() | |||
|             ); | |||
|         } | |||
| 
 | |||
|         $flags = PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_OFFSET_CAPTURE; | |||
|         $matches = preg_split($regex, $input, -1, $flags); | |||
| 
 | |||
|         foreach ($matches as $match) { | |||
|             // Must remain before 'value' assignment since it can change content | |||
|             $type = $this->getType($match[0]); | |||
| 
 | |||
|             $this->tokens[] = array( | |||
|                 'value' => $match[0], | |||
|                 'type'  => $type, | |||
|                 'position' => $match[1], | |||
|             ); | |||
|         } | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Gets the literal for a given token. | |||
|      * | |||
|      * @param integer $token | |||
|      * | |||
|      * @return string | |||
|      */ | |||
|     public function getLiteral($token) | |||
|     { | |||
|         $className = get_class($this); | |||
|         $reflClass = new \ReflectionClass($className); | |||
|         $constants = $reflClass->getConstants(); | |||
| 
 | |||
|         foreach ($constants as $name => $value) { | |||
|             if ($value === $token) { | |||
|                 return $className . '::' . $name; | |||
|             } | |||
|         } | |||
| 
 | |||
|         return $token; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Regex modifiers | |||
|      * | |||
|      * @return string | |||
|      */ | |||
|     protected function getModifiers() | |||
|     { | |||
|         return 'i'; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Lexical catchable patterns. | |||
|      * | |||
|      * @return array | |||
|      */ | |||
|     abstract protected function getCatchablePatterns(); | |||
| 
 | |||
|     /** | |||
|      * Lexical non-catchable patterns. | |||
|      * | |||
|      * @return array | |||
|      */ | |||
|     abstract protected function getNonCatchablePatterns(); | |||
| 
 | |||
|     /** | |||
|      * Retrieve token type. Also processes the token value if necessary. | |||
|      * | |||
|      * @param string $value | |||
|      * | |||
|      * @return integer | |||
|      */ | |||
|     abstract protected function getType(&$value); | |||
| } | |||
| @ -0,0 +1,221 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator; | |||
| 
 | |||
| use Doctrine\Common\Lexer\AbstractLexer; | |||
| 
 | |||
| class EmailLexer extends AbstractLexer | |||
| { | |||
|     //ASCII values | |||
|     const C_DEL              = 127; | |||
|     const C_NUL              = 0; | |||
|     const S_AT               = 64; | |||
|     const S_BACKSLASH        = 92; | |||
|     const S_DOT              = 46; | |||
|     const S_DQUOTE           = 34; | |||
|     const S_OPENPARENTHESIS  = 49; | |||
|     const S_CLOSEPARENTHESIS = 261; | |||
|     const S_OPENBRACKET      = 262; | |||
|     const S_CLOSEBRACKET     = 263; | |||
|     const S_HYPHEN           = 264; | |||
|     const S_COLON            = 265; | |||
|     const S_DOUBLECOLON      = 266; | |||
|     const S_SP               = 267; | |||
|     const S_HTAB             = 268; | |||
|     const S_CR               = 269; | |||
|     const S_LF               = 270; | |||
|     const S_IPV6TAG          = 271; | |||
|     const S_LOWERTHAN        = 272; | |||
|     const S_GREATERTHAN      = 273; | |||
|     const S_COMMA            = 274; | |||
|     const S_SEMICOLON        = 275; | |||
|     const S_OPENQBRACKET     = 276; | |||
|     const S_CLOSEQBRACKET    = 277; | |||
|     const S_SLASH            = 278; | |||
|     const S_EMPTY            = null; | |||
|     const GENERIC            = 300; | |||
|     const CRLF               = 301; | |||
|     const INVALID            = 302; | |||
|     const ASCII_INVALID_FROM = 127; | |||
|     const ASCII_INVALID_TO   = 199; | |||
| 
 | |||
|     /** | |||
|      * US-ASCII visible characters not valid for atext (@link http://tools.ietf.org/html/rfc5322#section-3.2.3) | |||
|      * | |||
|      * @var array | |||
|      */ | |||
|     protected $charValue = array( | |||
|         '('    => self::S_OPENPARENTHESIS, | |||
|         ')'    => self::S_CLOSEPARENTHESIS, | |||
|         '<'    => self::S_LOWERTHAN, | |||
|         '>'    => self::S_GREATERTHAN, | |||
|         '['    => self::S_OPENBRACKET, | |||
|         ']'    => self::S_CLOSEBRACKET, | |||
|         ':'    => self::S_COLON, | |||
|         ';'    => self::S_SEMICOLON, | |||
|         '@'    => self::S_AT, | |||
|         '\\'   => self::S_BACKSLASH, | |||
|         '/'    => self::S_SLASH, | |||
|         ','    => self::S_COMMA, | |||
|         '.'    => self::S_DOT, | |||
|         '"'    => self::S_DQUOTE, | |||
|         '-'    => self::S_HYPHEN, | |||
|         '::'   => self::S_DOUBLECOLON, | |||
|         ' '    => self::S_SP, | |||
|         "\t"   => self::S_HTAB, | |||
|         "\r"   => self::S_CR, | |||
|         "\n"   => self::S_LF, | |||
|         "\r\n" => self::CRLF, | |||
|         'IPv6' => self::S_IPV6TAG, | |||
|         '{'    => self::S_OPENQBRACKET, | |||
|         '}'    => self::S_CLOSEQBRACKET, | |||
|         ''     => self::S_EMPTY, | |||
|         '\0'   => self::C_NUL, | |||
|     ); | |||
| 
 | |||
|     protected $hasInvalidTokens = false; | |||
| 
 | |||
|     protected $previous; | |||
| 
 | |||
|     public function reset() | |||
|     { | |||
|         $this->hasInvalidTokens = false; | |||
|         parent::reset(); | |||
|     } | |||
| 
 | |||
|     public function hasInvalidTokens() | |||
|     { | |||
|         return $this->hasInvalidTokens; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param $type | |||
|      * @throws \UnexpectedValueException | |||
|      * @return boolean | |||
|      */ | |||
|     public function find($type) | |||
|     { | |||
|         $search = clone $this; | |||
|         $search->skipUntil($type); | |||
| 
 | |||
|         if (!$search->lookahead) { | |||
|             throw new \UnexpectedValueException($type . ' not found'); | |||
|         } | |||
|         return true; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * getPrevious | |||
|      * | |||
|      * @return array token | |||
|      */ | |||
|     public function getPrevious() | |||
|     { | |||
|         return $this->previous; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * moveNext | |||
|      * | |||
|      * @return boolean | |||
|      */ | |||
|     public function moveNext() | |||
|     { | |||
|         $this->previous = $this->token; | |||
| 
 | |||
|         return parent::moveNext(); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Lexical catchable patterns. | |||
|      * | |||
|      * @return string[] | |||
|      */ | |||
|     protected function getCatchablePatterns() | |||
|     { | |||
|         return array( | |||
|             '[a-zA-Z_]+[46]?', //ASCII and domain literal | |||
|             '[^\x00-\x7F]',  //UTF-8 | |||
|             '[0-9]+', | |||
|             '\r\n', | |||
|             '::', | |||
|             '\s+?', | |||
|             '.', | |||
|             ); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Lexical non-catchable patterns. | |||
|      * | |||
|      * @return string[] | |||
|      */ | |||
|     protected function getNonCatchablePatterns() | |||
|     { | |||
|         return array('[\xA0-\xff]+'); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * Retrieve token type. Also processes the token value if necessary. | |||
|      * | |||
|      * @param string $value | |||
|      * @throws \InvalidArgumentException | |||
|      * @return integer | |||
|      */ | |||
|     protected function getType(&$value) | |||
|     { | |||
|         if ($this->isNullType($value)) { | |||
|             return self::C_NUL; | |||
|         } | |||
| 
 | |||
|         if ($this->isValid($value)) { | |||
|             return $this->charValue[$value]; | |||
|         } | |||
| 
 | |||
|         if ($this->isUTF8Invalid($value)) { | |||
|             $this->hasInvalidTokens = true; | |||
|             return self::INVALID; | |||
|         } | |||
| 
 | |||
|         return  self::GENERIC; | |||
|     } | |||
| 
 | |||
|     protected function isValid($value) | |||
|     { | |||
|         if (isset($this->charValue[$value])) { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param $value | |||
|      * @return bool | |||
|      */ | |||
|     protected function isNullType($value) | |||
|     { | |||
|         if ($value === "\0") { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param $value | |||
|      * @return bool | |||
|      */ | |||
|     protected function isUTF8Invalid($value) | |||
|     { | |||
|         if (preg_match('/\p{Cc}+/u', $value)) { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     protected function getModifiers() | |||
|     { | |||
|         return 'iu'; | |||
|     } | |||
| } | |||
| @ -0,0 +1,104 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\ExpectingATEXT; | |||
| use Egulias\EmailValidator\Exception\NoLocalPart; | |||
| use Egulias\EmailValidator\Parser\DomainPart; | |||
| use Egulias\EmailValidator\Parser\LocalPart; | |||
| use Egulias\EmailValidator\Warning\EmailTooLong; | |||
| 
 | |||
| /** | |||
|  * EmailParser | |||
|  * | |||
|  * @author Eduardo Gulias Davis <me@egulias.com> | |||
|  */ | |||
| class EmailParser | |||
| { | |||
|     const EMAIL_MAX_LENGTH = 254; | |||
| 
 | |||
|     protected $warnings; | |||
|     protected $domainPart = ''; | |||
|     protected $localPart = ''; | |||
|     protected $lexer; | |||
|     protected $localPartParser; | |||
|     protected $domainPartParser; | |||
| 
 | |||
|     public function __construct(EmailLexer $lexer) | |||
|     { | |||
|         $this->lexer = $lexer; | |||
|         $this->localPartParser = new LocalPart($this->lexer); | |||
|         $this->domainPartParser = new DomainPart($this->lexer); | |||
|         $this->warnings = new \SplObjectStorage(); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param $str | |||
|      * @return array | |||
|      */ | |||
|     public function parse($str) | |||
|     { | |||
|         $this->lexer->setInput($str); | |||
| 
 | |||
|         if (!$this->hasAtToken()) { | |||
|             throw new NoLocalPart(); | |||
|         } | |||
| 
 | |||
| 
 | |||
|         $this->localPartParser->parse($str); | |||
|         $this->domainPartParser->parse($str); | |||
| 
 | |||
|         $this->setParts($str); | |||
| 
 | |||
|         if ($this->lexer->hasInvalidTokens()) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         return array('local' => $this->localPart, 'domain' => $this->domainPart); | |||
|     } | |||
| 
 | |||
|     public function getWarnings() | |||
|     { | |||
|         $localPartWarnings = $this->localPartParser->getWarnings(); | |||
|         $domainPartWarnings = $this->domainPartParser->getWarnings(); | |||
|         $this->warnings = array_merge($localPartWarnings, $domainPartWarnings); | |||
| 
 | |||
|         $this->addLongEmailWarning($this->localPart, $this->domainPart); | |||
| 
 | |||
|         return $this->warnings; | |||
|     } | |||
| 
 | |||
|     public function getParsedDomainPart() | |||
|     { | |||
|         return $this->domainPart; | |||
|     } | |||
| 
 | |||
|     protected function setParts($email) | |||
|     { | |||
|         $parts = explode('@', $email); | |||
|         $this->domainPart = $this->domainPartParser->getDomainPart(); | |||
|         $this->localPart = $parts[0]; | |||
|     } | |||
| 
 | |||
|     protected function hasAtToken() | |||
|     { | |||
|         $this->lexer->moveNext(); | |||
|         $this->lexer->moveNext(); | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_AT) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         return true; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param string $localPart | |||
|      * @param string $parsedDomainPart | |||
|      */ | |||
|     protected function addLongEmailWarning($localPart, $parsedDomainPart) | |||
|     { | |||
|         if (strlen($localPart . '@' . $parsedDomainPart) > self::EMAIL_MAX_LENGTH) { | |||
|             $this->warnings[EmailTooLong::CODE] = new EmailTooLong(); | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,67 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| use Egulias\EmailValidator\Validation\EmailValidation; | |||
| 
 | |||
| class EmailValidator | |||
| { | |||
|     /** | |||
|      * @var EmailLexer | |||
|      */ | |||
|     private $lexer; | |||
| 
 | |||
|     /** | |||
|      * @var array | |||
|      */ | |||
|     protected $warnings; | |||
| 
 | |||
|     /** | |||
|      * @var InvalidEmail | |||
|      */ | |||
|     protected $error; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->lexer = new EmailLexer(); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @param                 $email | |||
|      * @param EmailValidation $emailValidation | |||
|      * @return bool | |||
|      */ | |||
|     public function isValid($email, EmailValidation $emailValidation) | |||
|     { | |||
|         $isValid = $emailValidation->isValid($email, $this->lexer); | |||
|         $this->warnings = $emailValidation->getWarnings(); | |||
|         $this->error = $emailValidation->getError(); | |||
| 
 | |||
|         return $isValid; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return boolean | |||
|      */ | |||
|     public function hasWarnings() | |||
|     { | |||
|         return !empty($this->warnings); | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return array | |||
|      */ | |||
|     public function getWarnings() | |||
|     { | |||
|         return $this->warnings; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * @return InvalidEmail | |||
|      */ | |||
|     public function getError() | |||
|     { | |||
|         return $this->error; | |||
|     } | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class AtextAfterCFWS extends InvalidEmail | |||
| { | |||
|     const CODE = 133; | |||
|     const REASON = "ATEXT found after CFWS"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class CRLFAtTheEnd extends InvalidEmail | |||
| { | |||
|     const CODE = 149; | |||
|     const REASON = "CRLF at the end"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class CRLFX2 extends InvalidEmail | |||
| { | |||
|     const CODE = 148; | |||
|     const REASON = "Folding whitespace CR LF found twice"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class CRNoLF extends InvalidEmail | |||
| { | |||
|     const CODE = 150; | |||
|     const REASON = "Missing LF after CR"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class CharNotAllowed extends InvalidEmail | |||
| { | |||
|     const CODE = 201; | |||
|     const REASON = "Non allowed character in domain"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class CommaInDomain extends InvalidEmail | |||
| { | |||
|     const CODE = 200; | |||
|     const REASON = "Comma ',' is not allowed in domain part"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ConsecutiveAt extends InvalidEmail | |||
| { | |||
|     const CODE = 128; | |||
|     const REASON = "Consecutive AT"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ConsecutiveDot extends InvalidEmail | |||
| { | |||
|     const CODE = 132; | |||
|     const REASON = "Consecutive DOT"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class DomainHyphened extends InvalidEmail | |||
| { | |||
|     const CODE = 144; | |||
|     const REASON = "Hyphen found in domain"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class DotAtEnd extends InvalidEmail | |||
| { | |||
|     const CODE = 142; | |||
|     const REASON = "Dot at the end"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class DotAtStart extends InvalidEmail | |||
| { | |||
|     const CODE = 141; | |||
|     const REASON = "Found DOT at start"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectingAT extends InvalidEmail | |||
| { | |||
|     const CODE = 202; | |||
|     const REASON = "Expecting AT '@' "; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectingATEXT extends InvalidEmail | |||
| { | |||
|     const CODE = 137; | |||
|     const REASON = "Expecting ATEXT"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectingCTEXT extends InvalidEmail | |||
| { | |||
|     const CODE = 139; | |||
|     const REASON = "Expecting CTEXT"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectingDTEXT extends InvalidEmail | |||
| { | |||
|     const CODE = 129; | |||
|     const REASON = "Expected DTEXT"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectingDomainLiteralClose extends InvalidEmail | |||
| { | |||
|     const CODE = 137; | |||
|     const REASON = "Closing bracket ']' for domain literal not found"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class ExpectedQPair extends InvalidEmail | |||
| { | |||
|     const CODE = 136; | |||
|     const REASON = "Expecting QPAIR"; | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| abstract class InvalidEmail extends \InvalidArgumentException | |||
| { | |||
|     const REASON = "Invalid email"; | |||
|     const CODE = 0; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         parent::__construct(static::REASON, static::CODE); | |||
|     } | |||
| } | |||
| @ -0,0 +1,11 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| 
 | |||
| class NoDNSRecord extends InvalidEmail | |||
| { | |||
|     const CODE = 5; | |||
|     const REASON = 'No MX or A DSN record was found for this email'; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class NoDomainPart extends InvalidEmail | |||
| { | |||
|     const CODE = 131; | |||
|     const REASON = "No Domain part"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class NoLocalPart extends InvalidEmail | |||
| { | |||
|     const CODE = 130; | |||
|     const REASON = "No local part"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class UnclosedComment extends InvalidEmail | |||
| { | |||
|     const CODE = 146; | |||
|     const REASON = "No colosing comment token found"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class UnclosedQuotedString extends InvalidEmail | |||
| { | |||
|     const CODE = 145; | |||
|     const REASON = "Unclosed quoted string"; | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Exception; | |||
| 
 | |||
| class UnopenedComment extends InvalidEmail | |||
| { | |||
|     const CODE = 152; | |||
|     const REASON = "No opening comment token found"; | |||
| } | |||
| @ -0,0 +1,368 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Parser; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\CharNotAllowed; | |||
| use Egulias\EmailValidator\Exception\CommaInDomain; | |||
| use Egulias\EmailValidator\Exception\ConsecutiveAt; | |||
| use Egulias\EmailValidator\Exception\CRLFAtTheEnd; | |||
| use Egulias\EmailValidator\Exception\CRNoLF; | |||
| use Egulias\EmailValidator\Exception\DomainHyphened; | |||
| use Egulias\EmailValidator\Exception\DotAtEnd; | |||
| use Egulias\EmailValidator\Exception\DotAtStart; | |||
| use Egulias\EmailValidator\Exception\ExpectingATEXT; | |||
| use Egulias\EmailValidator\Exception\ExpectingDomainLiteralClose; | |||
| use Egulias\EmailValidator\Exception\ExpectingDTEXT; | |||
| use Egulias\EmailValidator\Exception\NoDomainPart; | |||
| use Egulias\EmailValidator\Exception\UnopenedComment; | |||
| use Egulias\EmailValidator\Warning\AddressLiteral; | |||
| use Egulias\EmailValidator\Warning\CFWSWithFWS; | |||
| use Egulias\EmailValidator\Warning\DeprecatedComment; | |||
| use Egulias\EmailValidator\Warning\DomainLiteral; | |||
| use Egulias\EmailValidator\Warning\DomainTooLong; | |||
| use Egulias\EmailValidator\Warning\IPV6BadChar; | |||
| use Egulias\EmailValidator\Warning\IPV6ColonEnd; | |||
| use Egulias\EmailValidator\Warning\IPV6ColonStart; | |||
| use Egulias\EmailValidator\Warning\IPV6Deprecated; | |||
| use Egulias\EmailValidator\Warning\IPV6DoubleColon; | |||
| use Egulias\EmailValidator\Warning\IPV6GroupCount; | |||
| use Egulias\EmailValidator\Warning\IPV6MaxGroups; | |||
| use Egulias\EmailValidator\Warning\LabelTooLong; | |||
| use Egulias\EmailValidator\Warning\ObsoleteDTEXT; | |||
| use Egulias\EmailValidator\Warning\TLD; | |||
| 
 | |||
| class DomainPart extends Parser | |||
| { | |||
|     const DOMAIN_MAX_LENGTH = 254; | |||
|     protected $domainPart = ''; | |||
| 
 | |||
|     public function parse($domainPart) | |||
|     { | |||
|         $this->lexer->moveNext(); | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_DOT) { | |||
|             throw new DotAtStart(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_EMPTY) { | |||
|             throw new NoDomainPart(); | |||
|         } | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_HYPHEN) { | |||
|             throw new DomainHyphened(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_OPENPARENTHESIS) { | |||
|             $this->warnings[DeprecatedComment::CODE] = new DeprecatedComment(); | |||
|             $this->parseDomainComments(); | |||
|         } | |||
| 
 | |||
|         $domain = $this->doParseDomainPart(); | |||
| 
 | |||
|         $prev = $this->lexer->getPrevious(); | |||
|         $length = strlen($domain); | |||
| 
 | |||
|         if ($prev['type'] === EmailLexer::S_DOT) { | |||
|             throw new DotAtEnd(); | |||
|         } | |||
|         if ($prev['type'] === EmailLexer::S_HYPHEN) { | |||
|             throw new DomainHyphened(); | |||
|         } | |||
|         if ($length > self::DOMAIN_MAX_LENGTH) { | |||
|             $this->warnings[DomainTooLong::CODE] = new DomainTooLong(); | |||
|         } | |||
|         if ($prev['type'] === EmailLexer::S_CR) { | |||
|             throw new CRLFAtTheEnd(); | |||
|         } | |||
|         $this->domainPart = $domain; | |||
|     } | |||
| 
 | |||
|     public function getDomainPart() | |||
|     { | |||
|         return $this->domainPart; | |||
|     } | |||
| 
 | |||
|     public function checkIPV6Tag($addressLiteral, $maxGroups = 8) | |||
|     { | |||
|         $prev = $this->lexer->getPrevious(); | |||
|         if ($prev['type'] === EmailLexer::S_COLON) { | |||
|             $this->warnings[IPV6ColonEnd::CODE] = new IPV6ColonEnd(); | |||
|         } | |||
| 
 | |||
|         $IPv6       = substr($addressLiteral, 5); | |||
|         //Daniel Marschall's new IPv6 testing strategy | |||
|         $matchesIP  = explode(':', $IPv6); | |||
|         $groupCount = count($matchesIP); | |||
|         $colons     = strpos($IPv6, '::'); | |||
| 
 | |||
|         if (count(preg_grep('/^[0-9A-Fa-f]{0,4}$/', $matchesIP, PREG_GREP_INVERT)) !== 0) { | |||
|             $this->warnings[IPV6BadChar::CODE] = new IPV6BadChar(); | |||
|         } | |||
| 
 | |||
|         if ($colons === false) { | |||
|             // We need exactly the right number of groups | |||
|             if ($groupCount !== $maxGroups) { | |||
|                 $this->warnings[IPV6GroupCount::CODE] = new IPV6GroupCount(); | |||
|             } | |||
|             return; | |||
|         } | |||
| 
 | |||
|         if ($colons !== strrpos($IPv6, '::')) { | |||
|             $this->warnings[IPV6DoubleColon::CODE] = new IPV6DoubleColon(); | |||
|             return; | |||
|         } | |||
| 
 | |||
|         if ($colons === 0 || $colons === (strlen($IPv6) - 2)) { | |||
|             // RFC 4291 allows :: at the start or end of an address | |||
|             //with 7 other groups in addition | |||
|             ++$maxGroups; | |||
|         } | |||
| 
 | |||
|         if ($groupCount > $maxGroups) { | |||
|             $this->warnings[IPV6MaxGroups::CODE] = new IPV6MaxGroups(); | |||
|         } elseif ($groupCount === $maxGroups) { | |||
|             $this->warnings[IPV6Deprecated::CODE] = new IPV6Deprecated(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function doParseDomainPart() | |||
|     { | |||
|         $domain = ''; | |||
|         $openedParenthesis = 0; | |||
|         do { | |||
|             $prev = $this->lexer->getPrevious(); | |||
| 
 | |||
|             $this->checkNotAllowedChars($this->lexer->token); | |||
| 
 | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_OPENPARENTHESIS) { | |||
|                 $this->parseComments(); | |||
|                 $openedParenthesis += $this->getOpenedParenthesis(); | |||
|                 $this->lexer->moveNext(); | |||
|                 $tmpPrev = $this->lexer->getPrevious(); | |||
|                 if ($tmpPrev['type'] === EmailLexer::S_CLOSEPARENTHESIS) { | |||
|                     $openedParenthesis--; | |||
|                 } | |||
|             } | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_CLOSEPARENTHESIS) { | |||
|                 if ($openedParenthesis === 0) { | |||
|                     throw new UnopenedComment(); | |||
|                 } else { | |||
|                     $openedParenthesis--; | |||
|                 } | |||
|             } | |||
| 
 | |||
|             $this->checkConsecutiveDots(); | |||
|             $this->checkDomainPartExceptions($prev); | |||
| 
 | |||
|             if ($this->hasBrackets()) { | |||
|                 $this->parseDomainLiteral(); | |||
|             } | |||
| 
 | |||
|             $this->checkLabelLength($prev); | |||
| 
 | |||
|             if ($this->isFWS()) { | |||
|                 $this->parseFWS(); | |||
|             } | |||
| 
 | |||
|             $domain .= $this->lexer->token['value']; | |||
|             $this->lexer->moveNext(); | |||
|         } while ($this->lexer->token); | |||
| 
 | |||
|         return $domain; | |||
|     } | |||
| 
 | |||
|     private function checkNotAllowedChars($token) | |||
|     { | |||
|         $notAllowed = [EmailLexer::S_BACKSLASH => true, EmailLexer::S_SLASH=> true]; | |||
|         if (isset($notAllowed[$token['type']])) { | |||
|             throw new CharNotAllowed(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function parseDomainLiteral() | |||
|     { | |||
|         if ($this->lexer->isNextToken(EmailLexer::S_COLON)) { | |||
|             $this->warnings[IPV6ColonStart::CODE] = new IPV6ColonStart(); | |||
|         } | |||
|         if ($this->lexer->isNextToken(EmailLexer::S_IPV6TAG)) { | |||
|             $lexer = clone $this->lexer; | |||
|             $lexer->moveNext(); | |||
|             if ($lexer->isNextToken(EmailLexer::S_DOUBLECOLON)) { | |||
|                 $this->warnings[IPV6ColonStart::CODE] = new IPV6ColonStart(); | |||
|             } | |||
|         } | |||
| 
 | |||
|         return $this->doParseDomainLiteral(); | |||
|     } | |||
| 
 | |||
|     protected function doParseDomainLiteral() | |||
|     { | |||
|         $IPv6TAG = false; | |||
|         $addressLiteral = ''; | |||
|         do { | |||
|             if ($this->lexer->token['type'] === EmailLexer::C_NUL) { | |||
|                 throw new ExpectingDTEXT(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->token['type'] === EmailLexer::INVALID || | |||
|                 $this->lexer->token['type'] === EmailLexer::C_DEL   || | |||
|                 $this->lexer->token['type'] === EmailLexer::S_LF | |||
|             ) { | |||
|                 $this->warnings[ObsoleteDTEXT::CODE] = new ObsoleteDTEXT(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->isNextTokenAny(array(EmailLexer::S_OPENQBRACKET, EmailLexer::S_OPENBRACKET))) { | |||
|                 throw new ExpectingDTEXT(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->isNextTokenAny( | |||
|                 array(EmailLexer::S_HTAB, EmailLexer::S_SP, $this->lexer->token['type'] === EmailLexer::CRLF) | |||
|             )) { | |||
|                 $this->warnings[CFWSWithFWS::CODE] = new CFWSWithFWS(); | |||
|                 $this->parseFWS(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->isNextToken(EmailLexer::S_CR)) { | |||
|                 throw new CRNoLF(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_BACKSLASH) { | |||
|                 $this->warnings[ObsoleteDTEXT::CODE] = new ObsoleteDTEXT(); | |||
|                 $addressLiteral .= $this->lexer->token['value']; | |||
|                 $this->lexer->moveNext(); | |||
|                 $this->validateQuotedPair(); | |||
|             } | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_IPV6TAG) { | |||
|                 $IPv6TAG = true; | |||
|             } | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_CLOSEQBRACKET) { | |||
|                 break; | |||
|             } | |||
| 
 | |||
|             $addressLiteral .= $this->lexer->token['value']; | |||
| 
 | |||
|         } while ($this->lexer->moveNext()); | |||
| 
 | |||
|         $addressLiteral = str_replace('[', '', $addressLiteral); | |||
|         $addressLiteral = $this->checkIPV4Tag($addressLiteral); | |||
| 
 | |||
|         if (false === $addressLiteral) { | |||
|             return $addressLiteral; | |||
|         } | |||
| 
 | |||
|         if (!$IPv6TAG) { | |||
|             $this->warnings[DomainLiteral::CODE] = new DomainLiteral(); | |||
|             return $addressLiteral; | |||
|         } | |||
| 
 | |||
|         $this->warnings[AddressLiteral::CODE] = new AddressLiteral(); | |||
| 
 | |||
|         $this->checkIPV6Tag($addressLiteral); | |||
| 
 | |||
|         return $addressLiteral; | |||
|     } | |||
| 
 | |||
|     protected function checkIPV4Tag($addressLiteral) | |||
|     { | |||
|         $matchesIP  = array(); | |||
| 
 | |||
|         // Extract IPv4 part from the end of the address-literal (if there is one) | |||
|         if (preg_match( | |||
|             '/\\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/', | |||
|             $addressLiteral, | |||
|             $matchesIP | |||
|         ) > 0 | |||
|         ) { | |||
|             $index = strrpos($addressLiteral, $matchesIP[0]); | |||
|             if ($index === 0) { | |||
|                 $this->warnings[AddressLiteral::CODE] = new AddressLiteral(); | |||
|                 return false; | |||
|             } | |||
|             // Convert IPv4 part to IPv6 format for further testing | |||
|             $addressLiteral = substr($addressLiteral, 0, $index) . '0:0'; | |||
|         } | |||
| 
 | |||
|         return $addressLiteral; | |||
|     } | |||
| 
 | |||
|     protected function checkDomainPartExceptions($prev) | |||
|     { | |||
|         $invalidDomainTokens = array( | |||
|             EmailLexer::S_DQUOTE => true, | |||
|             EmailLexer::S_SEMICOLON => true, | |||
|             EmailLexer::S_GREATERTHAN => true, | |||
|             EmailLexer::S_LOWERTHAN => true, | |||
|         ); | |||
| 
 | |||
|         if (isset($invalidDomainTokens[$this->lexer->token['type']])) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_COMMA) { | |||
|             throw new CommaInDomain(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_AT) { | |||
|             throw new ConsecutiveAt(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_OPENQBRACKET && $prev['type'] !== EmailLexer::S_AT) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_HYPHEN && $this->lexer->isNextToken(EmailLexer::S_DOT)) { | |||
|             throw new DomainHyphened(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_BACKSLASH | |||
|             && $this->lexer->isNextToken(EmailLexer::GENERIC)) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function hasBrackets() | |||
|     { | |||
|         if ($this->lexer->token['type'] !== EmailLexer::S_OPENBRACKET) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         try { | |||
|             $this->lexer->find(EmailLexer::S_CLOSEBRACKET); | |||
|         } catch (\RuntimeException $e) { | |||
|             throw new ExpectingDomainLiteralClose(); | |||
|         } | |||
| 
 | |||
|         return true; | |||
|     } | |||
| 
 | |||
|     protected function checkLabelLength($prev) | |||
|     { | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_DOT && | |||
|             $prev['type'] === EmailLexer::GENERIC && | |||
|             strlen($prev['value']) > 63 | |||
|         ) { | |||
|             $this->warnings[LabelTooLong::CODE] = new LabelTooLong(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function parseDomainComments() | |||
|     { | |||
|         $this->isUnclosedComment(); | |||
|         while (!$this->lexer->isNextToken(EmailLexer::S_CLOSEPARENTHESIS)) { | |||
|             $this->warnEscaping(); | |||
|             $this->lexer->moveNext(); | |||
|         } | |||
| 
 | |||
|         $this->lexer->moveNext(); | |||
|         if ($this->lexer->isNextToken(EmailLexer::S_DOT)) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function addTLDWarnings() | |||
|     { | |||
|         if ($this->warnings[DomainLiteral::CODE]) { | |||
|             $this->warnings[TLD::CODE] = new TLD(); | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,138 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Parser; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\DotAtEnd; | |||
| use Egulias\EmailValidator\Exception\DotAtStart; | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\EmailValidator; | |||
| use Egulias\EmailValidator\Exception\ExpectingAT; | |||
| use Egulias\EmailValidator\Exception\ExpectingATEXT; | |||
| use Egulias\EmailValidator\Exception\UnclosedQuotedString; | |||
| use Egulias\EmailValidator\Exception\UnopenedComment; | |||
| use Egulias\EmailValidator\Warning\CFWSWithFWS; | |||
| use Egulias\EmailValidator\Warning\LocalTooLong; | |||
| 
 | |||
| class LocalPart extends Parser | |||
| { | |||
|     public function parse($localPart) | |||
|     { | |||
|         $parseDQuote = true; | |||
|         $closingQuote = false; | |||
|         $openedParenthesis = 0; | |||
| 
 | |||
|         while ($this->lexer->token['type'] !== EmailLexer::S_AT && $this->lexer->token) { | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_DOT && !$this->lexer->getPrevious()) { | |||
|                 throw new DotAtStart(); | |||
|             } | |||
| 
 | |||
|             $closingQuote = $this->checkDQUOTE($closingQuote); | |||
|             if ($closingQuote && $parseDQuote) { | |||
|                 $parseDQuote = $this->parseDoubleQuote(); | |||
|             } | |||
| 
 | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_OPENPARENTHESIS) { | |||
|                 $this->parseComments(); | |||
|                 $openedParenthesis += $this->getOpenedParenthesis(); | |||
|             } | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_CLOSEPARENTHESIS) { | |||
|                 if ($openedParenthesis === 0) { | |||
|                     throw new UnopenedComment(); | |||
|                 } else { | |||
|                     $openedParenthesis--; | |||
|                 } | |||
|             } | |||
| 
 | |||
|             $this->checkConsecutiveDots(); | |||
| 
 | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_DOT && | |||
|                 $this->lexer->isNextToken(EmailLexer::S_AT) | |||
|             ) { | |||
|                 throw new DotAtEnd(); | |||
|             } | |||
| 
 | |||
|             $this->warnEscaping(); | |||
|             $this->isInvalidToken($this->lexer->token, $closingQuote); | |||
| 
 | |||
|             if ($this->isFWS()) { | |||
|                 $this->parseFWS(); | |||
|             } | |||
| 
 | |||
|             $this->lexer->moveNext(); | |||
|         } | |||
| 
 | |||
|         $prev = $this->lexer->getPrevious(); | |||
|         if (strlen($prev['value']) > LocalTooLong::LOCAL_PART_LENGTH) { | |||
|             $this->warnings[LocalTooLong::CODE] = new LocalTooLong(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function parseDoubleQuote() | |||
|     { | |||
|         $parseAgain = true; | |||
|         $special = array( | |||
|             EmailLexer::S_CR => true, | |||
|             EmailLexer::S_HTAB => true, | |||
|             EmailLexer::S_LF => true | |||
|         ); | |||
| 
 | |||
|         $invalid = array( | |||
|             EmailLexer::C_NUL => true, | |||
|             EmailLexer::S_HTAB => true, | |||
|             EmailLexer::S_CR => true, | |||
|             EmailLexer::S_LF => true | |||
|         ); | |||
|         $setSpecialsWarning = true; | |||
| 
 | |||
|         $this->lexer->moveNext(); | |||
| 
 | |||
|         while ($this->lexer->token['type'] !== EmailLexer::S_DQUOTE && $this->lexer->token) { | |||
|             $parseAgain = false; | |||
|             if (isset($special[$this->lexer->token['type']]) && $setSpecialsWarning) { | |||
|                 $this->warnings[CFWSWithFWS::CODE] = new CFWSWithFWS(); | |||
|                 $setSpecialsWarning = false; | |||
|             } | |||
|             if ($this->lexer->token['type'] === EmailLexer::S_BACKSLASH && $this->lexer->isNextToken(EmailLexer::S_DQUOTE)) { | |||
|                 $this->lexer->moveNext(); | |||
|             } | |||
| 
 | |||
|             $this->lexer->moveNext(); | |||
| 
 | |||
|             if (!$this->escaped() && isset($invalid[$this->lexer->token['type']])) { | |||
|                 throw new ExpectingATEXT(); | |||
|             } | |||
|         } | |||
| 
 | |||
|         $prev = $this->lexer->getPrevious(); | |||
| 
 | |||
|         if ($prev['type'] === EmailLexer::S_BACKSLASH) { | |||
|             if (!$this->checkDQUOTE(false)) { | |||
|                 throw new UnclosedQuotedString(); | |||
|             } | |||
|         } | |||
| 
 | |||
|         if (!$this->lexer->isNextToken(EmailLexer::S_AT) && $prev['type'] !== EmailLexer::S_BACKSLASH) { | |||
|             throw new ExpectingAT(); | |||
|         } | |||
| 
 | |||
|         return $parseAgain; | |||
|     } | |||
| 
 | |||
|     protected function isInvalidToken($token, $closingQuote) | |||
|     { | |||
|         $forbidden = array( | |||
|             EmailLexer::S_COMMA, | |||
|             EmailLexer::S_CLOSEBRACKET, | |||
|             EmailLexer::S_OPENBRACKET, | |||
|             EmailLexer::S_GREATERTHAN, | |||
|             EmailLexer::S_LOWERTHAN, | |||
|             EmailLexer::S_COLON, | |||
|             EmailLexer::S_SEMICOLON, | |||
|             EmailLexer::INVALID | |||
|         ); | |||
| 
 | |||
|         if (in_array($token['type'], $forbidden) && !$closingQuote) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,215 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Parser; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\AtextAfterCFWS; | |||
| use Egulias\EmailValidator\Exception\ConsecutiveDot; | |||
| use Egulias\EmailValidator\Exception\CRLFAtTheEnd; | |||
| use Egulias\EmailValidator\Exception\CRLFX2; | |||
| use Egulias\EmailValidator\Exception\CRNoLF; | |||
| use Egulias\EmailValidator\Exception\ExpectedQPair; | |||
| use Egulias\EmailValidator\Exception\ExpectingATEXT; | |||
| use Egulias\EmailValidator\Exception\ExpectingCTEXT; | |||
| use Egulias\EmailValidator\Exception\UnclosedComment; | |||
| use Egulias\EmailValidator\Exception\UnclosedQuotedString; | |||
| use Egulias\EmailValidator\Warning\CFWSNearAt; | |||
| use Egulias\EmailValidator\Warning\CFWSWithFWS; | |||
| use Egulias\EmailValidator\Warning\Comment; | |||
| use Egulias\EmailValidator\Warning\QuotedPart; | |||
| use Egulias\EmailValidator\Warning\QuotedString; | |||
| 
 | |||
| abstract class Parser | |||
| { | |||
|     protected $warnings = []; | |||
|     protected $lexer; | |||
|     protected $openedParenthesis = 0; | |||
| 
 | |||
|     public function __construct(EmailLexer $lexer) | |||
|     { | |||
|         $this->lexer = $lexer; | |||
|     } | |||
| 
 | |||
|     public function getWarnings() | |||
|     { | |||
|         return $this->warnings; | |||
|     } | |||
| 
 | |||
|     abstract public function parse($str); | |||
| 
 | |||
|     /** @return int */ | |||
|     public function getOpenedParenthesis() | |||
|     { | |||
|         return $this->openedParenthesis; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * validateQuotedPair | |||
|      */ | |||
|     protected function validateQuotedPair() | |||
|     { | |||
|         if (!($this->lexer->token['type'] === EmailLexer::INVALID | |||
|             || $this->lexer->token['type'] === EmailLexer::C_DEL)) { | |||
|             throw new ExpectedQPair(); | |||
|         } | |||
| 
 | |||
|         $this->warnings[QuotedPart::CODE] = | |||
|             new QuotedPart($this->lexer->getPrevious()['type'], $this->lexer->token['type']); | |||
|     } | |||
| 
 | |||
|     protected function parseComments() | |||
|     { | |||
|         $this->openedParenthesis = 1; | |||
|         $this->isUnclosedComment(); | |||
|         $this->warnings[Comment::CODE] = new Comment(); | |||
|         while (!$this->lexer->isNextToken(EmailLexer::S_CLOSEPARENTHESIS)) { | |||
|             if ($this->lexer->isNextToken(EmailLexer::S_OPENPARENTHESIS)) { | |||
|                 $this->openedParenthesis++; | |||
|             } | |||
|             $this->warnEscaping(); | |||
|             $this->lexer->moveNext(); | |||
|         } | |||
| 
 | |||
|         $this->lexer->moveNext(); | |||
|         if ($this->lexer->isNextTokenAny(array(EmailLexer::GENERIC, EmailLexer::S_EMPTY))) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->isNextToken(EmailLexer::S_AT)) { | |||
|             $this->warnings[CFWSNearAt::CODE] = new CFWSNearAt(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function isUnclosedComment() | |||
|     { | |||
|         try { | |||
|             $this->lexer->find(EmailLexer::S_CLOSEPARENTHESIS); | |||
|             return true; | |||
|         } catch (\RuntimeException $e) { | |||
|             throw new UnclosedComment(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function parseFWS() | |||
|     { | |||
|         $previous = $this->lexer->getPrevious(); | |||
| 
 | |||
|         $this->checkCRLFInFWS(); | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_CR) { | |||
|             throw new CRNoLF(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->isNextToken(EmailLexer::GENERIC) && $previous['type']  !== EmailLexer::S_AT) { | |||
|             throw new AtextAfterCFWS(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_LF || $this->lexer->token['type'] === EmailLexer::C_NUL) { | |||
|             throw new ExpectingCTEXT(); | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->isNextToken(EmailLexer::S_AT) || $previous['type']  === EmailLexer::S_AT) { | |||
|             $this->warnings[CFWSNearAt::CODE] = new CFWSNearAt(); | |||
|         } else { | |||
|             $this->warnings[CFWSWithFWS::CODE] = new CFWSWithFWS(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function checkConsecutiveDots() | |||
|     { | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_DOT && $this->lexer->isNextToken(EmailLexer::S_DOT)) { | |||
|             throw new ConsecutiveDot(); | |||
|         } | |||
|     } | |||
| 
 | |||
|     protected function isFWS() | |||
|     { | |||
|         if ($this->escaped()) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->token['type'] === EmailLexer::S_SP || | |||
|             $this->lexer->token['type'] === EmailLexer::S_HTAB || | |||
|             $this->lexer->token['type'] === EmailLexer::S_CR || | |||
|             $this->lexer->token['type'] === EmailLexer::S_LF || | |||
|             $this->lexer->token['type'] === EmailLexer::CRLF | |||
|         ) { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     protected function escaped() | |||
|     { | |||
|         $previous = $this->lexer->getPrevious(); | |||
| 
 | |||
|         if ($previous['type'] === EmailLexer::S_BACKSLASH | |||
|             && | |||
|             $this->lexer->token['type'] !== EmailLexer::GENERIC | |||
|         ) { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     protected function warnEscaping() | |||
|     { | |||
|         if ($this->lexer->token['type'] !== EmailLexer::S_BACKSLASH) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         if ($this->lexer->isNextToken(EmailLexer::GENERIC)) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         if (!$this->lexer->isNextTokenAny(array(EmailLexer::S_SP, EmailLexer::S_HTAB, EmailLexer::C_DEL))) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         $this->warnings[QuotedPart::CODE] = | |||
|             new QuotedPart($this->lexer->getPrevious()['type'], $this->lexer->token['type']); | |||
|         return true; | |||
| 
 | |||
|     } | |||
| 
 | |||
|     protected function checkDQUOTE($hasClosingQuote) | |||
|     { | |||
|         if ($this->lexer->token['type'] !== EmailLexer::S_DQUOTE) { | |||
|             return $hasClosingQuote; | |||
|         } | |||
|         if ($hasClosingQuote) { | |||
|             return $hasClosingQuote; | |||
|         } | |||
|         $previous = $this->lexer->getPrevious(); | |||
|         if ($this->lexer->isNextToken(EmailLexer::GENERIC) && $previous['type'] === EmailLexer::GENERIC) { | |||
|             throw new ExpectingATEXT(); | |||
|         } | |||
| 
 | |||
|         try { | |||
|             $this->lexer->find(EmailLexer::S_DQUOTE); | |||
|             $hasClosingQuote = true; | |||
|         } catch (\Exception $e) { | |||
|             throw new UnclosedQuotedString(); | |||
|         } | |||
|         $this->warnings[QuotedString::CODE] = new QuotedString($previous['value'], $this->lexer->token['value']); | |||
| 
 | |||
|         return $hasClosingQuote; | |||
|     } | |||
| 
 | |||
|     protected function checkCRLFInFWS() | |||
|     { | |||
|         if ($this->lexer->token['type'] !== EmailLexer::CRLF) { | |||
|             return; | |||
|         } | |||
| 
 | |||
|         if (!$this->lexer->isNextTokenAny(array(EmailLexer::S_SP, EmailLexer::S_HTAB))) { | |||
|             throw new CRLFX2(); | |||
|         } | |||
| 
 | |||
|         if (!$this->lexer->isNextTokenAny(array(EmailLexer::S_SP, EmailLexer::S_HTAB))) { | |||
|             throw new CRLFAtTheEnd(); | |||
|         } | |||
|     } | |||
| } | |||
| @ -0,0 +1,61 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| use Egulias\EmailValidator\Warning\NoDNSMXRecord; | |||
| use Egulias\EmailValidator\Exception\NoDNSRecord; | |||
| 
 | |||
| class DNSCheckValidation implements EmailValidation | |||
| { | |||
|     /** | |||
|      * @var array | |||
|      */ | |||
|     private $warnings = []; | |||
| 
 | |||
|     /** | |||
|      * @var InvalidEmail | |||
|      */ | |||
|     private $error; | |||
| 
 | |||
|     public function isValid($email, EmailLexer $emailLexer) | |||
|     { | |||
|         // use the input to check DNS if we cannot extract something similar to a domain | |||
|         $host = $email; | |||
| 
 | |||
|         // Arguable pattern to extract the domain. Not aiming to validate the domain nor the email | |||
|         if (false !== $lastAtPos = strrpos($email, '@')) { | |||
|             $host = substr($email, $lastAtPos + 1); | |||
|         } | |||
| 
 | |||
|         return $this->checkDNS($host); | |||
|     } | |||
| 
 | |||
|     public function getError() | |||
|     { | |||
|         return $this->error; | |||
|     } | |||
| 
 | |||
|     public function getWarnings() | |||
|     { | |||
|         return $this->warnings; | |||
|     } | |||
| 
 | |||
|     protected function checkDNS($host) | |||
|     { | |||
|         $host = rtrim($host, '.') . '.'; | |||
| 
 | |||
|         $Aresult = true; | |||
|         $MXresult = checkdnsrr($host, 'MX'); | |||
| 
 | |||
|         if (!$MXresult) { | |||
|             $this->warnings[NoDNSMXRecord::CODE] = new NoDNSMXRecord(); | |||
|             $Aresult = checkdnsrr($host, 'A') || checkdnsrr($host, 'AAAA'); | |||
|             if (!$Aresult) { | |||
|                 $this->error = new NoDNSRecord(); | |||
|             } | |||
|         } | |||
|         return $MXresult || $Aresult; | |||
|     } | |||
| } | |||
| @ -0,0 +1,34 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| use Egulias\EmailValidator\Warning\Warning; | |||
| 
 | |||
| interface EmailValidation | |||
| { | |||
|     /** | |||
|      * Returns true if the given email is valid. | |||
|      * | |||
|      * @param string     $email      The email you want to validate. | |||
|      * @param EmailLexer $emailLexer The email lexer. | |||
|      * | |||
|      * @return bool | |||
|      */ | |||
|     public function isValid($email, EmailLexer $emailLexer); | |||
| 
 | |||
|     /** | |||
|      * Returns the validation error. | |||
|      * | |||
|      * @return InvalidEmail|null | |||
|      */ | |||
|     public function getError(); | |||
| 
 | |||
|     /** | |||
|      * Returns the validation warnings. | |||
|      * | |||
|      * @return Warning[] | |||
|      */ | |||
|     public function getWarnings(); | |||
| } | |||
| @ -0,0 +1,11 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation\Error; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| 
 | |||
| class RFCWarnings extends InvalidEmail | |||
| { | |||
|     const CODE = 997; | |||
|     const REASON = 'Warnings were found.'; | |||
| } | |||
| @ -0,0 +1,11 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation\Error; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| 
 | |||
| class SpoofEmail extends InvalidEmail | |||
| { | |||
|     const CODE = 998; | |||
|     const REASON = "The email contains mixed UTF8 chars that makes it suspicious"; | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation\Exception; | |||
| 
 | |||
| use Exception; | |||
| 
 | |||
| class EmptyValidationList extends \InvalidArgumentException | |||
| { | |||
|     public function __construct($code = 0, Exception $previous = null) | |||
|     { | |||
|         parent::__construct("Empty validation list is not allowed", $code, $previous); | |||
|     } | |||
| } | |||
| @ -0,0 +1,26 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| 
 | |||
| class MultipleErrors extends InvalidEmail | |||
| { | |||
|     const CODE = 999; | |||
|     const REASON = "Accumulated errors for multiple validations"; | |||
|     /** | |||
|      * @var array | |||
|      */ | |||
|     private $errors = []; | |||
| 
 | |||
|     public function __construct(array $errors) | |||
|     { | |||
|         $this->errors = $errors; | |||
|         parent::__construct(); | |||
|     } | |||
| 
 | |||
|     public function getErrors() | |||
|     { | |||
|         return $this->errors; | |||
|     } | |||
| } | |||
| @ -0,0 +1,110 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Validation\Exception\EmptyValidationList; | |||
| 
 | |||
| class MultipleValidationWithAnd implements EmailValidation | |||
| { | |||
|     /** | |||
|      * If one of validations gets failure skips all succeeding validation. | |||
|      * This means MultipleErrors will only contain a single error which first found. | |||
|      */ | |||
|     const STOP_ON_ERROR = 0; | |||
| 
 | |||
|     /** | |||
|      * All of validations will be invoked even if one of them got failure. | |||
|      * So MultipleErrors will contain all causes. | |||
|      */ | |||
|     const ALLOW_ALL_ERRORS = 1; | |||
| 
 | |||
|     /** | |||
|      * @var EmailValidation[] | |||
|      */ | |||
|     private $validations = []; | |||
| 
 | |||
|     /** | |||
|      * @var array | |||
|      */ | |||
|     private $warnings = []; | |||
| 
 | |||
|     /** | |||
|      * @var MultipleErrors | |||
|      */ | |||
|     private $error; | |||
| 
 | |||
|     /** | |||
|      * @var bool | |||
|      */ | |||
|     private $mode; | |||
| 
 | |||
|     /** | |||
|      * @param EmailValidation[] $validations The validations. | |||
|      * @param int               $mode        The validation mode (one of the constants). | |||
|      */ | |||
|     public function __construct(array $validations, $mode = self::ALLOW_ALL_ERRORS) | |||
|     { | |||
|         if (count($validations) == 0) { | |||
|             throw new EmptyValidationList(); | |||
|         } | |||
| 
 | |||
|         $this->validations = $validations; | |||
|         $this->mode = $mode; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * {@inheritdoc} | |||
|      */ | |||
|     public function isValid($email, EmailLexer $emailLexer) | |||
|     { | |||
|         $result = true; | |||
|         $errors = []; | |||
|         foreach ($this->validations as $validation) { | |||
|             $emailLexer->reset(); | |||
|             $result = $result && $validation->isValid($email, $emailLexer); | |||
|             $this->warnings = array_merge($this->warnings, $validation->getWarnings()); | |||
|             $errors = $this->addNewError($validation->getError(), $errors); | |||
| 
 | |||
|             if ($this->shouldStop($result)) { | |||
|                 break; | |||
|             } | |||
|         } | |||
| 
 | |||
|         if (!empty($errors)) { | |||
|             $this->error = new MultipleErrors($errors); | |||
|         } | |||
| 
 | |||
|         return $result; | |||
|     } | |||
| 
 | |||
|     private function addNewError($possibleError, array $errors) | |||
|     { | |||
|         if (null !== $possibleError) { | |||
|             $errors[] = $possibleError; | |||
|         } | |||
| 
 | |||
|         return $errors; | |||
|     } | |||
| 
 | |||
|     private function shouldStop($result) | |||
|     { | |||
|         return !$result && $this->mode === self::STOP_ON_ERROR; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * {@inheritdoc} | |||
|      */ | |||
|     public function getError() | |||
|     { | |||
|         return $this->error; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * {@inheritdoc} | |||
|      */ | |||
|     public function getWarnings() | |||
|     { | |||
|         return $this->warnings; | |||
|     } | |||
| } | |||
| @ -0,0 +1,41 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| use Egulias\EmailValidator\Validation\Error\RFCWarnings; | |||
| 
 | |||
| class NoRFCWarningsValidation extends RFCValidation | |||
| { | |||
|     /** | |||
|      * @var InvalidEmail | |||
|      */ | |||
|     private $error; | |||
| 
 | |||
|     /** | |||
|      * {@inheritdoc} | |||
|      */ | |||
|     public function isValid($email, EmailLexer $emailLexer) | |||
|     { | |||
|         if (!parent::isValid($email, $emailLexer)) { | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         if (empty($this->getWarnings())) { | |||
|             return true; | |||
|         } | |||
| 
 | |||
|         $this->error = new RFCWarnings(); | |||
| 
 | |||
|         return false; | |||
|     } | |||
| 
 | |||
|     /** | |||
|      * {@inheritdoc} | |||
|      */ | |||
|     public function getError() | |||
|     { | |||
|         return $this->error ?: parent::getError(); | |||
|     } | |||
| } | |||
| @ -0,0 +1,49 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\EmailParser; | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| 
 | |||
| class RFCValidation implements EmailValidation | |||
| { | |||
|     /** | |||
|      * @var EmailParser | |||
|      */ | |||
|     private $parser; | |||
| 
 | |||
|     /** | |||
|      * @var array | |||
|      */ | |||
|     private $warnings = []; | |||
| 
 | |||
|     /** | |||
|      * @var InvalidEmail | |||
|      */ | |||
|     private $error; | |||
| 
 | |||
|     public function isValid($email, EmailLexer $emailLexer) | |||
|     { | |||
|         $this->parser = new EmailParser($emailLexer); | |||
|         try { | |||
|             $this->parser->parse((string)$email); | |||
|         } catch (InvalidEmail $invalid) { | |||
|             $this->error = $invalid; | |||
|             return false; | |||
|         } | |||
| 
 | |||
|         $this->warnings = $this->parser->getWarnings(); | |||
|         return true; | |||
|     } | |||
| 
 | |||
|     public function getError() | |||
|     { | |||
|         return $this->error; | |||
|     } | |||
| 
 | |||
|     public function getWarnings() | |||
|     { | |||
|         return $this->warnings; | |||
|     } | |||
| } | |||
| @ -0,0 +1,45 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Validation; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailLexer; | |||
| use Egulias\EmailValidator\Exception\InvalidEmail; | |||
| use Egulias\EmailValidator\Validation\Error\SpoofEmail; | |||
| use \Spoofchecker; | |||
| 
 | |||
| class SpoofCheckValidation implements EmailValidation | |||
| { | |||
|     /** | |||
|      * @var InvalidEmail | |||
|      */ | |||
|     private $error; | |||
|      | |||
|     public function __construct() | |||
|     { | |||
|         if (!class_exists(Spoofchecker::class)) { | |||
|             throw new \LogicException(sprintf('The %s class requires the Intl extension.', __CLASS__)); | |||
|         } | |||
|     } | |||
| 
 | |||
|     public function isValid($email, EmailLexer $emailLexer) | |||
|     { | |||
|         $checker = new Spoofchecker(); | |||
|         $checker->setChecks(Spoofchecker::SINGLE_SCRIPT); | |||
| 
 | |||
|         if ($checker->isSuspicious($email)) { | |||
|             $this->error = new SpoofEmail(); | |||
|         } | |||
| 
 | |||
|         return $this->error === null; | |||
|     } | |||
| 
 | |||
|     public function getError() | |||
|     { | |||
|         return $this->error; | |||
|     } | |||
| 
 | |||
|     public function getWarnings() | |||
|     { | |||
|         return []; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class AddressLiteral extends Warning | |||
| { | |||
|     const CODE = 12; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Address literal in domain part'; | |||
|         $this->rfcNumber = 5321; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class CFWSNearAt extends Warning | |||
| { | |||
|     const CODE = 49; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = "Deprecated folding white space near @"; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class CFWSWithFWS extends Warning | |||
| { | |||
|     const CODE = 18; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Folding whites space followed by folding white space'; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class Comment extends Warning | |||
| { | |||
|     const CODE = 17; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = "Comments found in this email"; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class DeprecatedComment extends Warning | |||
| { | |||
|     const CODE = 37; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Deprecated comments'; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class DomainLiteral extends Warning | |||
| { | |||
|     const CODE = 70; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Domain Literal'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class DomainTooLong extends Warning | |||
| { | |||
|     const CODE = 255; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Domain is too long, exceeds 255 chars'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,15 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| use Egulias\EmailValidator\EmailParser; | |||
| 
 | |||
| class EmailTooLong extends Warning | |||
| { | |||
|     const CODE = 66; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Email is too long, exceeds ' . EmailParser::EMAIL_MAX_LENGTH; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6BadChar extends Warning | |||
| { | |||
|     const CODE = 74; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Bad char in IPV6 domain literal'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6ColonEnd extends Warning | |||
| { | |||
|     const CODE = 77; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = ':: found at the end of the domain literal'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6ColonStart extends Warning | |||
| { | |||
|     const CODE = 76; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = ':: found at the start of the domain literal'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6Deprecated extends Warning | |||
| { | |||
|     const CODE = 13; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Deprecated form of IPV6'; | |||
|         $this->rfcNumber = 5321; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6DoubleColon extends Warning | |||
| { | |||
|     const CODE = 73; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Double colon found after IPV6 tag'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6GroupCount extends Warning | |||
| { | |||
|     const CODE = 72; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Group count is not IPV6 valid'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class IPV6MaxGroups extends Warning | |||
| { | |||
|     const CODE = 75; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Reached the maximum number of IPV6 groups allowed'; | |||
|         $this->rfcNumber = 5321; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class LabelTooLong extends Warning | |||
| { | |||
|     const CODE = 63; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Label too long'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,15 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class LocalTooLong extends Warning | |||
| { | |||
|     const CODE = 64; | |||
|     const LOCAL_PART_LENGTH = 64; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'Local part is too long, exceeds 64 chars (octets)'; | |||
|         $this->rfcNumber = 5322; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class NoDNSMXRecord extends Warning | |||
| { | |||
|     const CODE = 6; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->message = 'No MX DSN record was found for this email'; | |||
|         $this->rfcNumber = 5321; | |||
|     } | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class ObsoleteDTEXT extends Warning | |||
| { | |||
|     const CODE = 71; | |||
| 
 | |||
|     public function __construct() | |||
|     { | |||
|         $this->rfcNumber = 5322; | |||
|         $this->message = 'Obsolete DTEXT in domain literal'; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class QuotedPart extends Warning | |||
| { | |||
|     const CODE = 36; | |||
| 
 | |||
|     public function __construct($prevToken, $postToken) | |||
|     { | |||
|         $this->message = "Deprecated Quoted String found between $prevToken and $postToken"; | |||
|     } | |||
| } | |||
| @ -0,0 +1,13 @@ | |||
| <?php | |||
| 
 | |||
| namespace Egulias\EmailValidator\Warning; | |||
| 
 | |||
| class QuotedString extends Warning | |||
| { | |||
|     const CODE = 11; | |||
| 
 | |||
|     public function __construct($prevToken, $postToken) | |||
|     { | |||
|         $this->message = "Quoted String found between $prevToken and $postToken"; | |||
|     } | |||
| } | |||
Some files were not shown because too many files changed in this diff
					Loading…
					
					
				
		Reference in new issue