refactor: susun semula struktur folder — Laravel source ke src/

This commit is contained in:
Saufi
2026-05-19 15:58:35 +08:00
parent f052251b94
commit bf53c71b45
10806 changed files with 1385379 additions and 121 deletions

View 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);
}
}

View 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);
}
}