| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 | <?php/* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Symfony\Component\Finder;/** * Gitignore matches against text. * * @author Ahmed Abdou <mail@ahmd.io> */class Gitignore{    /**     * Returns a regexp which is the equivalent of the gitignore pattern.     *     * @return string The regexp     */    public static function toRegex(string $gitignoreFileContent): string    {        $gitignoreFileContent = preg_replace('/^[^\\\r\n]*#.*/m', '', $gitignoreFileContent);        $gitignoreLines = preg_split('/\r\n|\r|\n/', $gitignoreFileContent);        $positives = [];        $negatives = [];        foreach ($gitignoreLines as $i => $line) {            $line = trim($line);            if ('' === $line) {                continue;            }            if (1 === preg_match('/^!/', $line)) {                $positives[$i] = null;                $negatives[$i] = self::getRegexFromGitignore(preg_replace('/^!(.*)/', '${1}', $line), true);                continue;            }            $negatives[$i] = null;            $positives[$i] = self::getRegexFromGitignore($line);        }        $index = 0;        $patterns = [];        foreach ($positives as $pattern) {            if (null === $pattern) {                continue;            }            $negativesAfter = array_filter(\array_slice($negatives, ++$index));            if ([] !== $negativesAfter) {                $pattern .= sprintf('(?<!%s)', implode('|', $negativesAfter));            }            $patterns[] = $pattern;        }        return sprintf('/^((%s))$/', implode(')|(', $patterns));    }    private static function getRegexFromGitignore(string $gitignorePattern, bool $negative = false): string    {        $regex = '';        $isRelativePath = false;        // If there is a separator at the beginning or middle (or both) of the pattern, then the pattern is relative to the directory level of the particular .gitignore file itself        $slashPosition = strpos($gitignorePattern, '/');        if (false !== $slashPosition && \strlen($gitignorePattern) - 1 !== $slashPosition) {            if (0 === $slashPosition) {                $gitignorePattern = substr($gitignorePattern, 1);            }            $isRelativePath = true;            $regex .= '^';        }        if ('/' === $gitignorePattern[\strlen($gitignorePattern) - 1]) {            $gitignorePattern = substr($gitignorePattern, 0, -1);        }        $iMax = \strlen($gitignorePattern);        for ($i = 0; $i < $iMax; ++$i) {            $tripleChars = substr($gitignorePattern, $i, 3);            if ('**/' === $tripleChars || '/**' === $tripleChars) {                $regex .= '.*';                $i += 2;                continue;            }            $doubleChars = substr($gitignorePattern, $i, 2);            if ('**' === $doubleChars) {                $regex .= '.*';                ++$i;                continue;            }            if ('*/' === $doubleChars) {                $regex .= '[^\/]*\/?[^\/]*';                ++$i;                continue;            }            $c = $gitignorePattern[$i];            switch ($c) {                case '*':                    $regex .= $isRelativePath ? '[^\/]*' : '[^\/]*\/?[^\/]*';                    break;                case '/':                case '.':                case ':':                case '(':                case ')':                case '{':                case '}':                    $regex .= '\\'.$c;                    break;                default:                    $regex .= $c;            }        }        if ($negative) {            // a lookbehind assertion has to be a fixed width (it can not have nested '|' statements)            return sprintf('%s$|%s\/$', $regex, $regex);        }        return '(?>'.$regex.'($|\/.*))';    }}
 |