## Summary - move the Laravel application into `apps/platform` and keep the repository root for orchestration, docs, and tooling - update the local command model, Sail/Docker wiring, runtime paths, and ignore rules around the new platform location - add relocation quickstart/contracts plus focused smoke coverage for bootstrap, command model, routes, and runtime behavior ## Validation - `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Feature/PlatformRelocation` - integrated browser smoke validated `/up`, `/`, `/admin`, `/admin/choose-workspace`, and tenant route semantics for `200`, `403`, and `404` ## Remaining Rollout Checks - validate Dokploy build context and working-directory assumptions against the new `apps/platform` layout - confirm web, queue, and scheduler processes all start from the expected working directory in staging/production - verify no legacy volume mounts or asset-publish paths still point at the old root-level `public/` or `storage/` locations Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #213
114 lines
2.9 KiB
PHP
114 lines
2.9 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Services\Baselines\SnapshotRendering;
|
|
|
|
use Illuminate\Support\Str;
|
|
|
|
enum FidelityState: string
|
|
{
|
|
case Full = 'full';
|
|
case Partial = 'partial';
|
|
case ReferenceOnly = 'reference_only';
|
|
case Unsupported = 'unsupported';
|
|
|
|
public static function fromEvidence(mixed $value, bool $usesFallback = false): self
|
|
{
|
|
$normalized = Str::of((string) ($value ?? ''))
|
|
->trim()
|
|
->lower()
|
|
->replace(['-', ' '], '_')
|
|
->toString();
|
|
|
|
return match ($normalized) {
|
|
'content', 'full' => self::Full,
|
|
'partial' => self::Partial,
|
|
'meta', 'metadata_only', 'reference_only' => self::ReferenceOnly,
|
|
'unsupported' => self::Unsupported,
|
|
default => $usesFallback ? self::Unsupported : self::Partial,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $summary
|
|
*/
|
|
public static function fromSummary(array $summary, bool $hasItems): self
|
|
{
|
|
if (! $hasItems) {
|
|
return self::Unsupported;
|
|
}
|
|
|
|
$counts = is_array($summary['fidelity_counts'] ?? null)
|
|
? $summary['fidelity_counts']
|
|
: [];
|
|
|
|
$content = is_numeric($counts['content'] ?? null) ? (int) $counts['content'] : 0;
|
|
$meta = is_numeric($counts['meta'] ?? null) ? (int) $counts['meta'] : 0;
|
|
|
|
if ($content > 0 && $meta === 0) {
|
|
return self::Full;
|
|
}
|
|
|
|
if ($content > 0 && $meta > 0) {
|
|
return self::Partial;
|
|
}
|
|
|
|
if ($meta > 0) {
|
|
return self::ReferenceOnly;
|
|
}
|
|
|
|
return self::Unsupported;
|
|
}
|
|
|
|
/**
|
|
* @param array<int, self> $states
|
|
*/
|
|
public static function aggregate(array $states): self
|
|
{
|
|
if ($states === []) {
|
|
return self::Unsupported;
|
|
}
|
|
|
|
$worst = self::Full;
|
|
|
|
foreach ($states as $state) {
|
|
if ($state->rank() < $worst->rank()) {
|
|
$worst = $state;
|
|
}
|
|
}
|
|
|
|
return $worst;
|
|
}
|
|
|
|
public function label(): string
|
|
{
|
|
return match ($this) {
|
|
self::Full => 'Full',
|
|
self::Partial => 'Partial',
|
|
self::ReferenceOnly => 'Reference only',
|
|
self::Unsupported => 'Unsupported',
|
|
};
|
|
}
|
|
|
|
public function coverageHint(): ?string
|
|
{
|
|
return match ($this) {
|
|
self::Full => null,
|
|
self::Partial => 'Mixed evidence detail is available for this group.',
|
|
self::ReferenceOnly => 'Metadata-only evidence is available for this group.',
|
|
self::Unsupported => 'Support is limited for this policy type. Fallback rendering is being used.',
|
|
};
|
|
}
|
|
|
|
private function rank(): int
|
|
{
|
|
return match ($this) {
|
|
self::Full => 4,
|
|
self::Partial => 3,
|
|
self::ReferenceOnly => 2,
|
|
self::Unsupported => 1,
|
|
};
|
|
}
|
|
}
|