refactor: susun semula struktur folder — Laravel source ke src/
This commit is contained in:
188
vendor/psy/psysh/src/CodeAnalysis/BufferAnalysis.php
vendored
Normal file
188
vendor/psy/psysh/src/CodeAnalysis/BufferAnalysis.php
vendored
Normal file
@@ -0,0 +1,188 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of Psy Shell.
|
||||
*
|
||||
* (c) 2012-2026 Justin Hileman
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Psy\CodeAnalysis;
|
||||
|
||||
use PhpParser\Error;
|
||||
use Psy\Readline\Interactive\Helper\TokenHelper;
|
||||
|
||||
/**
|
||||
* Cached analysis data for a code buffer snapshot.
|
||||
*/
|
||||
class BufferAnalysis
|
||||
{
|
||||
private string $code;
|
||||
|
||||
/** @var array<int, array|string> */
|
||||
private array $tokens;
|
||||
|
||||
/** @var array<int, array{start: int, end: int}> */
|
||||
private array $tokenPositions;
|
||||
|
||||
/** @var array<int, mixed>|null */
|
||||
private ?array $ast;
|
||||
private ?Error $lastError;
|
||||
|
||||
/**
|
||||
* @param array<int, array|string> $tokens
|
||||
* @param array<int, array{start: int, end: int}> $tokenPositions
|
||||
* @param array<int, mixed>|null $ast
|
||||
*/
|
||||
public function __construct(string $code, array $tokens, array $tokenPositions, ?array $ast, ?Error $lastError)
|
||||
{
|
||||
$this->code = $code;
|
||||
$this->tokens = $tokens;
|
||||
$this->tokenPositions = $tokenPositions;
|
||||
$this->ast = $ast;
|
||||
$this->lastError = $lastError;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the token_get_all() tokens for the buffer.
|
||||
*
|
||||
* @return array<int, array|string>
|
||||
*/
|
||||
public function getTokens(): array
|
||||
{
|
||||
return $this->tokens;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get start/end code-point positions for each token.
|
||||
*
|
||||
* @return array<int, array{start: int, end: int}>
|
||||
*/
|
||||
public function getTokenPositions(): array
|
||||
{
|
||||
return $this->tokenPositions;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the parsed AST, or null if parsing failed.
|
||||
*
|
||||
* @return array<int, mixed>|null
|
||||
*/
|
||||
public function getAst(): ?array
|
||||
{
|
||||
return $this->ast;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the last parse error, if any.
|
||||
*/
|
||||
public function getLastError(): ?Error
|
||||
{
|
||||
return $this->lastError;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether the buffer has balanced (), [] and {} pairs.
|
||||
*/
|
||||
public function hasBalancedBrackets(): bool
|
||||
{
|
||||
$stack = [];
|
||||
$pairs = ['(' => ')', '[' => ']', '{' => '}'];
|
||||
|
||||
foreach ($this->tokens as $token) {
|
||||
if (\is_array($token)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (isset($pairs[$token])) {
|
||||
$stack[] = $token;
|
||||
} elseif (\in_array($token, $pairs, true)) {
|
||||
if (empty($stack)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$last = \array_pop($stack);
|
||||
if ($pairs[$last] !== $token) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return empty($stack);
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether the buffer ends with an operator that requires more input.
|
||||
*/
|
||||
public function hasTrailingOperator(): bool
|
||||
{
|
||||
return TokenHelper::hasTrailingOperator($this->tokens);
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether the token stream ends inside a string or comment context.
|
||||
*/
|
||||
public function endsInOpenStringOrComment(): bool
|
||||
{
|
||||
if ($this->tokens === []) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$last = $this->tokens[\count($this->tokens) - 1];
|
||||
|
||||
return $last === '"' || $last === '`' ||
|
||||
(\is_array($last) && \in_array($last[0], [\T_ENCAPSED_AND_WHITESPACE, \T_START_HEREDOC, \T_COMMENT], true));
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether the buffer ends with a control structure header that still needs a body.
|
||||
*/
|
||||
public function hasControlStructureWithoutBody(): bool
|
||||
{
|
||||
$trimmed = \rtrim($this->code);
|
||||
|
||||
if (\preg_match('/\b(if|while|for|foreach|elseif)\s*\(.*\)\s*$/', $trimmed)) {
|
||||
$lastParen = \strrpos($trimmed, ')');
|
||||
if ($lastParen !== false) {
|
||||
$afterParen = \trim(\substr($trimmed, $lastParen + 1));
|
||||
if ($afterParen === '') {
|
||||
if ($this->lastError !== null && !$this->isEOFError($this->lastError)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$isElseAfterBrace = \preg_match('/\}\s*else\s*$/', $trimmed);
|
||||
$isBareElse = \preg_match('/^\s*else\s*$/', $trimmed);
|
||||
|
||||
if ($isElseAfterBrace || $isBareElse) {
|
||||
if ($isBareElse && $this->lastError !== null && !$this->isEOFError($this->lastError)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether the last parse error is an unexpected-EOF error.
|
||||
*/
|
||||
public function hasEOFError(): bool
|
||||
{
|
||||
return $this->lastError !== null && $this->isEOFError($this->lastError);
|
||||
}
|
||||
|
||||
private function isEOFError(Error $error): bool
|
||||
{
|
||||
$msg = $error->getRawMessage();
|
||||
|
||||
return ($msg === 'Unexpected token EOF') || (\strpos($msg, 'Syntax error, unexpected EOF') !== false);
|
||||
}
|
||||
}
|
||||
117
vendor/psy/psysh/src/CodeAnalysis/BufferAnalyzer.php
vendored
Normal file
117
vendor/psy/psysh/src/CodeAnalysis/BufferAnalyzer.php
vendored
Normal file
@@ -0,0 +1,117 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of Psy Shell.
|
||||
*
|
||||
* (c) 2012-2026 Justin Hileman
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Psy\CodeAnalysis;
|
||||
|
||||
use PhpParser\Error;
|
||||
use PhpParser\Parser;
|
||||
use Psy\ParserFactory;
|
||||
|
||||
/**
|
||||
* Maintains a cached analysis snapshot for code buffers.
|
||||
*
|
||||
* Uses a small LRU cache (2 entries) to avoid thrashing when callers
|
||||
* alternate between full-buffer and partial (before-cursor) text.
|
||||
*/
|
||||
class BufferAnalyzer
|
||||
{
|
||||
private const CACHE_SIZE = 2;
|
||||
|
||||
private Parser $parser;
|
||||
|
||||
/**
|
||||
* LRU cache of recent analyses, most-recently-used first.
|
||||
*
|
||||
* @var array<int, array{code: string, analysis: BufferAnalysis}>
|
||||
*/
|
||||
private array $cache = [];
|
||||
|
||||
public function __construct(?Parser $parser = null)
|
||||
{
|
||||
$this->parser = $parser ?? (new ParserFactory())->createParser();
|
||||
}
|
||||
|
||||
/**
|
||||
* Analyze the given code, using cached data when possible.
|
||||
*/
|
||||
public function analyze(string $code): BufferAnalysis
|
||||
{
|
||||
foreach ($this->cache as $i => $entry) {
|
||||
if ($entry['code'] === $code) {
|
||||
if ($i > 0) {
|
||||
\array_splice($this->cache, $i, 1);
|
||||
\array_unshift($this->cache, $entry);
|
||||
}
|
||||
|
||||
return $entry['analysis'];
|
||||
}
|
||||
}
|
||||
|
||||
$analysis = $this->buildAnalysis($code);
|
||||
|
||||
\array_unshift($this->cache, ['code' => $code, 'analysis' => $analysis]);
|
||||
if (\count($this->cache) > self::CACHE_SIZE) {
|
||||
\array_pop($this->cache);
|
||||
}
|
||||
|
||||
return $analysis;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check whether appending a semicolon would make the code parseable.
|
||||
*/
|
||||
public function canBeFixedWithSemicolon(string $code): bool
|
||||
{
|
||||
try {
|
||||
$this->parser->parse('<?php '.$code.";\n");
|
||||
|
||||
return true;
|
||||
} catch (Error $e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private function buildAnalysis(string $code): BufferAnalysis
|
||||
{
|
||||
$tokens = @\token_get_all('<?php '.$code);
|
||||
|
||||
$tokenPositions = [];
|
||||
$position = 0;
|
||||
foreach ($tokens as $index => $token) {
|
||||
$text = \is_array($token) ? $token[1] : $token;
|
||||
$length = \mb_strlen($text);
|
||||
$tokenPositions[$index] = ['start' => $position, 'end' => $position + $length];
|
||||
$position += $length;
|
||||
}
|
||||
|
||||
$ast = null;
|
||||
$lastError = null;
|
||||
try {
|
||||
// Trailing newline improves heredoc EOF behavior to match runtime checks.
|
||||
$ast = $this->parser->parse('<?php '.$code."\n");
|
||||
} catch (Error $e) {
|
||||
$lastError = $e;
|
||||
}
|
||||
|
||||
return $this->createAnalysis($code, $tokens, $tokenPositions, $ast, $lastError);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $code
|
||||
* @param array<int, array|string> $tokens
|
||||
* @param array<int, array{start: int, end: int}> $tokenPositions
|
||||
* @param array<int, mixed>|null $ast
|
||||
*/
|
||||
protected function createAnalysis(string $code, array $tokens, array $tokenPositions, ?array $ast, ?Error $lastError): BufferAnalysis
|
||||
{
|
||||
return new BufferAnalysis($code, $tokens, $tokenPositions, $ast, $lastError);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user