Вход Регистрация
Файл: vendor/vlucas/phpdotenv/src/Parser/EntryParser.php
Строк: 394
<?php

declare(strict_types=1);

namespace 
DotenvParser;

use 
DotenvUtilRegex;
use 
DotenvUtilStr;
use 
GrahamCampbellResultTypeError;
use 
GrahamCampbellResultTypeResult;
use 
GrahamCampbellResultTypeSuccess;

final class 
EntryParser
{
    private const 
INITIAL_STATE 0;
    private const 
UNQUOTED_STATE 1;
    private const 
SINGLE_QUOTED_STATE 2;
    private const 
DOUBLE_QUOTED_STATE 3;
    private const 
ESCAPE_SEQUENCE_STATE 4;
    private const 
WHITESPACE_STATE 5;
    private const 
COMMENT_STATE 6;
    private const 
REJECT_STATES = [self::SINGLE_QUOTED_STATEself::DOUBLE_QUOTED_STATEself::ESCAPE_SEQUENCE_STATE];

    
/**
     * This class is a singleton.
     *
     * @codeCoverageIgnore
     *
     * @return void
     */
    
private function __construct()
    {
        
//
    
}

    
/**
     * Parse a raw entry into a proper entry.
     *
     * That is, turn a raw environment variable entry into a name and possibly
     * a value. We wrap the answer in a result type.
     *
     * @param string $entry
     *
     * @return GrahamCampbellResultTypeResult<DotenvParserEntry,string>
     */
    
public static function parse(string $entry)
    {
        return 
self::splitStringIntoParts($entry)->flatMap(static function (array $parts) {
            [
$name$value] = $parts;

            return 
self::parseName($name)->flatMap(static function (string $name) use ($value) {
                
/** @var Result<Value|null,string> */
                
$parsedValue $value === null Success::create(null) : self::parseValue($value);

                return 
$parsedValue->map(static function (?Value $value) use ($name) {
                    return new 
Entry($name$value);
                });
            });
        });
    }

    
/**
     * Split the compound string into parts.
     *
     * @param string $line
     *
     * @return GrahamCampbellResultTypeResult<array{string,string|null},string>
     */
    
private static function splitStringIntoParts(string $line)
    {
        
/** @var array{string,string|null} */
        
$result Str::pos($line'=')->map(static function () use ($line) {
            return 
array_map('trim'explode('='$line2));
        })->
getOrElse([$linenull]);

        if (
$result[0] === '') {
            
/** @var GrahamCampbellResultTypeResult<array{string,string|null},string> */
            
return Error::create(self::getErrorMessage('an unexpected equals'$line));
        }

        
/** @var GrahamCampbellResultTypeResult<array{string,string|null},string> */
        
return Success::create($result);
    }

    
/**
     * Parse the given variable name.
     *
     * That is, strip the optional quotes and leading "export" from the
     * variable name. We wrap the answer in a result type.
     *
     * @param string $name
     *
     * @return GrahamCampbellResultTypeResult<string,string>
     */
    
private static function parseName(string $name)
    {
        if (
Str::len($name) > && Str::substr($name06) === 'export' && ctype_space(Str::substr($name61))) {
            
$name ltrim(Str::substr($name6));
        }

        if (
self::isQuotedName($name)) {
            
$name Str::substr($name1, -1);
        }

        if (!
self::isValidName($name)) {
            
/** @var GrahamCampbellResultTypeResult<string,string> */
            
return Error::create(self::getErrorMessage('an invalid name'$name));
        }

        
/** @var GrahamCampbellResultTypeResult<string,string> */
        
return Success::create($name);
    }

    
/**
     * Is the given variable name quoted?
     *
     * @param string $name
     *
     * @return bool
     */
    
private static function isQuotedName(string $name)
    {
        if (
Str::len($name) < 3) {
            return 
false;
        }

        
$first Str::substr($name01);
        
$last Str::substr($name, -11);

        return (
$first === '"' && $last === '"') || ($first === ''' && $last === ''');
    }

    
/**
     * Is the given variable name valid?
     *
     * @param string $name
     *
     * @return bool
     */
    
private static function isValidName(string $name)
    {
        return 
Regex::matches('~(*UTF8)A[p{Ll}p{Lu}p{M}p{N}_.]+z~'$name)->success()->getOrElse(false);
    }

    
/**
     * Parse the given variable value.
     *
     * This has the effect of stripping quotes and comments, dealing with
     * special characters, and locating nested variables, but not resolving
     * them. Formally, we run a finite state automaton with an output tape: a
     * transducer. We wrap the answer in a result type.
     *
     * @param string $value
     *
     * @return GrahamCampbellResultTypeResult<DotenvParserValue,string>
     */
    
private static function parseValue(string $value)
    {
        if (
trim($value) === '') {
            
/** @var GrahamCampbellResultTypeResult<DotenvParserValue,string> */
            
return Success::create(Value::blank());
        }

        return 
array_reduce(iterator_to_array(Lexer::lex($value)), static function (Result $datastring $token) {
            return 
$data->flatMap(static function (array $data) use ($token) {
                return 
self::processToken($data[1], $token)->map(static function (array $val) use ($data) {
                    return [
$data[0]->append($val[0], $val[1]), $val[2]];
                });
            });
        }, 
Success::create([Value::blank(), self::INITIAL_STATE]))->flatMap(static function (array $result) {
            
/** @psalm-suppress DocblockTypeContradiction */
            
if (in_array($result[1], self::REJECT_STATEStrue)) {
                
/** @var GrahamCampbellResultTypeResult<DotenvParserValue,string> */
                
return Error::create('a missing closing quote');
            }

            
/** @var GrahamCampbellResultTypeResult<DotenvParserValue,string> */
            
return Success::create($result[0]);
        })->
mapError(static function (string $err) use ($value) {
            return 
self::getErrorMessage($err$value);
        });
    }

    
/**
     * Process the given token.
     *
     * @param int    $state
     * @param string $token
     *
     * @return GrahamCampbellResultTypeResult<array{string,bool,int},string>
     */
    
private static function processToken(int $statestring $token)
    {
        switch (
$state) {
            case 
self::INITIAL_STATE:
                if (
$token === ''') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::SINGLE_QUOTED_STATE]);
                } elseif ($token === '"') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::DOUBLE_QUOTED_STATE]);
                } elseif (
$token === '#') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::COMMENT_STATE]);
                } elseif (
$token === '$') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, true, self::UNQUOTED_STATE]);
                } else {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, false, self::UNQUOTED_STATE]);
                }
            case self::UNQUOTED_STATE:
                if (
$token === '#') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::COMMENT_STATE]);
                } elseif (ctype_space(
$token)) {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::WHITESPACE_STATE]);
                } elseif (
$token === '$') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, true, self::UNQUOTED_STATE]);
                } else {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, false, self::UNQUOTED_STATE]);
                }
            case self::SINGLE_QUOTED_STATE:
                if (
$token === ''') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::WHITESPACE_STATE]);
                } else {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, false, self::SINGLE_QUOTED_STATE]);
                }
            case self::DOUBLE_QUOTED_STATE:
                if (
$token === '"') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::WHITESPACE_STATE]);
                } elseif ($token === '
\') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::ESCAPE_SEQUENCE_STATE]);
                } elseif ($token === '
$') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([$token, true, self::DOUBLE_QUOTED_STATE]);
                } else {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
                }
            case self::ESCAPE_SEQUENCE_STATE:
                if ($token === '"' || 
$token === '\') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, false, self::DOUBLE_QUOTED_STATE]);
                } elseif (
$token === '$') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create([
$token, false, self::DOUBLE_QUOTED_STATE]);
                } else {
                    
$first = Str::substr($token, 0, 1);
                    if (in_array(
$first, ['f', 'n', 'r', 't', 'v'], true)) {
                        /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                        return Success::create([stripcslashes('\'.
$first).Str::substr($token, 1), false, self::DOUBLE_QUOTED_STATE]);
                    } else {
                        /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                        return Error::create('an unexpected escape sequence');
                    }
                }
            case self::WHITESPACE_STATE:
                if (
$token === '#') {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::COMMENT_STATE]);
                } elseif (!ctype_space(
$token)) {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Error::create('unexpected whitespace');
                } else {
                    /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                    return Success::create(['', false, self::WHITESPACE_STATE]);
                }
            case self::COMMENT_STATE:
                /** @var GrahamCampbellResultTypeResult<array{string,bool,int},string> */
                return Success::create(['', false, self::COMMENT_STATE]);
            default:
                throw new Error('Parser entered invalid state.');
        }
    }

    /**
     * Generate a friendly error message.
     *
     * @param string 
$cause
     * @param string 
$subject
     *
     * @return string
     */
    private static function getErrorMessage(string 
$cause, string $subject)
    {
        return sprintf(
            'Encountered %s at [%s].',
            
$cause,
            strtok(
$subject, "n")
        );
    }
}
Онлайн: 1
Реклама