lthn.io/app/Core/Config/Console/ConfigImportCommand.php
Claude 41a90cbff8
feat: lthn.io API serving live chain data
Fixed: basePath self→static binding, namespace detection, event wiring,
SQLite cache, file cache driver. All Mod Boot classes converted to
$listens pattern for lifecycle event discovery.

Working endpoints:
- /v1/explorer/info — live chain height, difficulty, aliases
- /v1/explorer/stats — formatted chain statistics
- /v1/names/directory — alias directory grouped by type
- /v1/names/available/{name} — name availability check
- /v1/names/lookup/{name} — name details

Co-Authored-By: Charon <charon@lethean.io>
2026-04-03 17:17:42 +01:00

187 lines
6.2 KiB
PHP

<?php
/*
* Core PHP Framework
*
* Licensed under the European Union Public Licence (EUPL) v1.2.
* See LICENSE file for details.
*/
declare(strict_types=1);
namespace Core\Config\Console;
use Core\Config\ConfigExporter;
use Core\Config\ConfigVersioning;
use Core\Tenant\Models\Workspace;
use Illuminate\Console\Command;
use Symfony\Component\Console\Completion\CompletionInput;
use Symfony\Component\Console\Completion\CompletionSuggestions;
/**
* Import config from JSON or YAML file.
*
* Usage:
* php artisan config:import config.json
* php artisan config:import config.yaml --workspace=myworkspace
* php artisan config:import backup.json --dry-run
*/
class ConfigImportCommand extends Command
{
protected $signature = 'config:import
{file : Input file path (.json or .yaml/.yml)}
{--workspace= : Import config for specific workspace slug}
{--dry-run : Preview changes without applying}
{--no-backup : Skip creating a version backup before import}
{--force : Skip confirmation prompt}';
protected $description = 'Import config from JSON or YAML file';
public function handle(ConfigExporter $exporter, ConfigVersioning $versioning): int
{
$file = $this->argument('file');
$workspaceSlug = $this->option('workspace');
$dryRun = $this->option('dry-run');
$skipBackup = $this->option('no-backup');
$force = $this->option('force');
// Check file exists
if (! file_exists($file)) {
$this->components->error("File not found: {$file}");
return self::FAILURE;
}
// Resolve workspace
$workspace = null;
if ($workspaceSlug) {
if (! class_exists(Workspace::class)) {
$this->components->error('Tenant module not installed. Cannot import workspace config.');
return self::FAILURE;
}
$workspace = Workspace::where('slug', $workspaceSlug)->first();
if (! $workspace) {
$this->components->error("Workspace not found: {$workspaceSlug}");
return self::FAILURE;
}
}
// Read file content
$content = file_get_contents($file);
if ($content === false) {
$this->components->error("Failed to read file: {$file}");
return self::FAILURE;
}
// Determine format from extension
$extension = strtolower(pathinfo($file, PATHINFO_EXTENSION));
$format = match ($extension) {
'yaml', 'yml' => 'YAML',
default => 'JSON',
};
$scope = $workspace ? "workspace: {$workspace->slug}" : 'system';
if ($dryRun) {
$this->components->info("Dry-run import from {$file} ({$scope}):");
} else {
if (! $force) {
$this->components->warn("This will import config from {$file} to {$scope}.");
if (! $this->confirm('Are you sure you want to continue?')) {
$this->components->info('Import cancelled.');
return self::SUCCESS;
}
}
// Create backup before import
if (! $skipBackup && ! $dryRun) {
$this->components->task('Creating backup version', function () use ($versioning, $workspace, $file) {
$versioning->createVersion(
$workspace,
'Backup before import from '.basename($file)
);
});
}
}
// Perform import
$result = null;
$this->components->task("Importing {$format} config", function () use ($exporter, $content, $extension, $workspace, $dryRun, &$result) {
$result = match ($extension) {
'yaml', 'yml' => $exporter->importYaml($content, $workspace, $dryRun),
default => $exporter->importJson($content, $workspace, $dryRun),
};
});
// Show results
$this->newLine();
if ($dryRun) {
$this->components->info('Dry-run results (no changes applied):');
}
// Display created items
if ($result->createdCount() > 0) {
$this->components->twoColumnDetail('<fg=green>Created</>', $result->createdCount().' items');
foreach ($result->getCreated() as $item) {
$this->components->bulletList(["{$item['type']}: {$item['code']}"]);
}
}
// Display updated items
if ($result->updatedCount() > 0) {
$this->components->twoColumnDetail('<fg=yellow>Updated</>', $result->updatedCount().' items');
foreach ($result->getUpdated() as $item) {
$this->components->bulletList(["{$item['type']}: {$item['code']}"]);
}
}
// Display skipped items
if ($result->skippedCount() > 0) {
$this->components->twoColumnDetail('<fg=gray>Skipped</>', $result->skippedCount().' items');
foreach ($result->getSkipped() as $reason) {
$this->components->bulletList([$reason]);
}
}
// Display errors
if ($result->hasErrors()) {
$this->newLine();
$this->components->error('Errors:');
foreach ($result->getErrors() as $error) {
$this->components->bulletList(["<fg=red>{$error}</>"]);
}
return self::FAILURE;
}
$this->newLine();
if ($dryRun) {
$this->components->info("Dry-run complete: {$result->getSummary()}");
} else {
$this->components->info("Import complete: {$result->getSummary()}");
}
return self::SUCCESS;
}
/**
* Get autocompletion suggestions.
*/
public function complete(CompletionInput $input, CompletionSuggestions $suggestions): void
{
if ($input->mustSuggestOptionValuesFor('workspace')) {
if (class_exists(Workspace::class)) {
$suggestions->suggestValues(Workspace::pluck('slug')->toArray());
}
}
}
}