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,202 @@
<?php
declare(strict_types=1);
namespace Laravel\Boost;
use Illuminate\Foundation\Http\Middleware\VerifyCsrfToken;
use Illuminate\Http\Request;
use Illuminate\Log\Logger;
use Illuminate\Routing\Router;
use Illuminate\Support\Facades\Log;
use Illuminate\Support\Facades\Route;
use Illuminate\Support\ServiceProvider;
use Illuminate\View\Compilers\BladeCompiler;
use Laravel\Boost\Install\GuidelineAssist;
use Laravel\Boost\Install\GuidelineConfig;
use Laravel\Boost\Mcp\Boost;
use Laravel\Boost\Middleware\InjectBoost;
use Laravel\Boost\Services\BrowserLogger;
use Laravel\Mcp\Facades\Mcp;
use Laravel\Roster\Roster;
class BoostServiceProvider extends ServiceProvider
{
public function register(): void
{
$this->mergeConfigFrom(
__DIR__.'/../config/boost.php',
'boost'
);
if (! $this->shouldRun()) {
return;
}
$this->app->singleton(BoostManager::class, fn (): BoostManager => new BoostManager);
$this->app->singleton(Roster::class, fn (): Roster => Roster::scan(base_path()));
$this->app->singleton(GuidelineConfig::class, fn (): GuidelineConfig => new GuidelineConfig);
$this->app->singleton(GuidelineAssist::class, fn ($app): GuidelineAssist => new GuidelineAssist(
$app->make(Roster::class),
$app->make(GuidelineConfig::class)
));
}
public function boot(Router $router): void
{
if (! $this->shouldRun()) {
return;
}
Mcp::local('laravel-boost', Boost::class);
$this->registerPublishing();
$this->registerCommands();
$this->registerRoutes();
if (config('boost.browser_logs_watcher', true)) {
$this->registerBrowserLogger();
$this->callAfterResolving('blade.compiler', $this->registerBladeDirectives(...));
$this->hookIntoResponses($router);
}
}
protected function registerPublishing(): void
{
if ($this->app->runningInConsole()) {
$this->publishes([
__DIR__.'/../config/boost.php' => config_path('boost.php'),
], 'boost-config');
}
}
protected function registerCommands(): void
{
if ($this->app->runningInConsole()) {
$this->commands([
Console\StartCommand::class,
Console\InstallCommand::class,
Console\UpdateCommand::class,
Console\ExecuteToolCommand::class,
Console\AddSkillCommand::class,
Console\ListSkillCommand::class,
]);
}
}
protected function registerRoutes(): void
{
Route::post('/_boost/browser-logs', function (Request $request) {
$logs = $request->input('logs', []);
// Handle sendBeacon's text/plain content type.
if (empty($logs) && ! $request->isJson()) {
$decoded = json_decode($request->getContent(), true);
$logs = $decoded['logs'] ?? [];
}
/** @var Logger $logger */
$logger = Log::channel('browser');
/**
* @var array{
* type: 'error'|'warn'|'info'|'log'|'table'|'window_error'|'uncaught_error'|'unhandled_rejection',
* timestamp: string,
* data: array,
* url:string,
* userAgent:string
* } $log */
foreach ($logs as $log) {
$logger->write(
level: match ($log['type']) {
'warn' => 'warning',
'log', 'table' => 'debug',
'window_error', 'uncaught_error', 'unhandled_rejection' => 'error',
default => $log['type']
},
message: self::buildLogMessageFromData($log['data']),
context: [
'url' => $log['url'],
'user_agent' => $log['userAgent'] ?: null,
'timestamp' => $log['timestamp'] ?: now()->toIso8601String(),
]
);
}
return response()->json(['status' => 'logged']);
})
->name('boost.browser-logs')
->withoutMiddleware(VerifyCsrfToken::class);
}
/**
* Build a string message for the log based on various input types. Single-dimensional, and multi:
* "data": {"message":"Unhandled Promise Rejection","reason":{"name":"TypeError","message":"NetworkError when attempting to fetch resource.","stack":""}}]
*/
private static function buildLogMessageFromData(array $data): string
{
$messages = [];
foreach ($data as $value) {
$messages[] = match (true) {
is_array($value) => self::buildLogMessageFromData($value),
is_string($value), is_numeric($value) => (string) $value,
is_bool($value) => $value ? 'true' : 'false',
is_null($value) => 'null',
is_object($value) => json_encode($value),
default => $value,
};
}
return implode(' ', $messages);
}
protected function registerBrowserLogger(): void
{
if (config('logging.channels.browser') !== null) {
return;
}
config([
'logging.channels.browser' => [
'driver' => 'single',
'path' => storage_path('logs'.DIRECTORY_SEPARATOR.'browser.log'),
'level' => env('LOG_LEVEL', 'debug'),
'days' => 14,
],
]);
}
protected function registerBladeDirectives(BladeCompiler $bladeCompiler): void
{
$bladeCompiler->directive('boostJs', fn (): string => '<?php echo '.BrowserLogger::class.'::getScript(); ?>');
}
protected function hookIntoResponses(Router $router): void
{
$this->app->booted(function () use ($router): void {
$router->pushMiddlewareToGroup('web', InjectBoost::class);
});
}
protected function shouldRun(): bool
{
if (! config('boost.enabled', true)) {
return false;
}
if (app()->runningUnitTests()) {
return false;
}
// Only enable Boost on local environments or when debug is true
if (! app()->environment('local') && config('app.debug', false) !== true) {
return false;
}
return true;
}
}