Compare commits
15 Commits
147-tenant
...
dev
| Author | SHA1 | Date | |
|---|---|---|---|
| 845d21db6d | |||
| 8426741068 | |||
| e7c9b4b853 | |||
| 92f39d9749 | |||
| 3c3daae405 | |||
| a4f2629493 | |||
| b1e1e06861 | |||
| a74ab12f04 | |||
| 5ec62cd117 | |||
| ec71c2d4e7 | |||
| 1f3619bd16 | |||
| 5bcb4f6ab8 | |||
| ede4cc363d | |||
| 417df4f9aa | |||
| 73a879d061 |
25
.github/agents/copilot-instructions.md
vendored
25
.github/agents/copilot-instructions.md
vendored
@ -83,6 +83,25 @@ ## Active Technologies
|
|||||||
- PostgreSQL for tenants, onboarding sessions, audit logs, operation runs, and workspace membership data (145-tenant-action-taxonomy-lifecycle-safe-visibility)
|
- PostgreSQL for tenants, onboarding sessions, audit logs, operation runs, and workspace membership data (145-tenant-action-taxonomy-lifecycle-safe-visibility)
|
||||||
- PostgreSQL (existing tenant and operation records only; no schema changes planned) (146-central-tenant-status-presentation)
|
- PostgreSQL (existing tenant and operation records only; no schema changes planned) (146-central-tenant-status-presentation)
|
||||||
- PostgreSQL plus existing session-backed workspace and remembered-tenant context; no schema change planned (147-tenant-selector-remembered-context-enforcement)
|
- PostgreSQL plus existing session-backed workspace and remembered-tenant context; no schema change planned (147-tenant-selector-remembered-context-enforcement)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Pest 4, existing support-layer helpers such as `UiEnforcement`, `CapabilityResolver`, `WorkspaceContext`, `OperateHubShell`, `TenantOperabilityService`, and `TenantActionPolicySurface` (148-central-tenant-operability-policy)
|
||||||
|
- PostgreSQL plus existing session-backed workspace and remembered-tenant context; no schema change planned for the first implementation slice (148-central-tenant-operability-policy)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Pest 4, existing `OperationRunService`, `TrackOperationRun`, `ProviderOperationStartGate`, `TenantOperabilityService`, `CapabilityResolver`, and `WriteGateInterface` seams (149-queued-execution-reauthorization)
|
||||||
|
- PostgreSQL-backed application data plus queue-serialized `OperationRun` context; no schema migration planned for the first implementation slice (149-queued-execution-reauthorization)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament v5, Livewire v4, PostgreSQL, Pest 4 (150-tenant-owned-query-canon-and-wrong-tenant-guards)
|
||||||
|
- PostgreSQL with existing `findings` and `audit_logs` tables; no new storage engine or external log store (151-findings-workflow-backstop)
|
||||||
|
- PostgreSQL with existing workspace-, tenant-, onboarding-, and audit-related tables; no new persistent storage planned for the first slice (152-livewire-context-locking)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament v5, Livewire v4, Pest v4, existing `StoredReport`, `Finding`, `OperationRun`, and `AuditLog` infrastructure (153-evidence-domain-foundation)
|
||||||
|
- PostgreSQL with JSONB-backed snapshot metadata; existing private storage remains a downstream-consumer concern, not a primary evidence-foundation store (153-evidence-domain-foundation)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament v5, Livewire v4, Pest v4, existing Finding, AuditLog, EvidenceSnapshot, CapabilityResolver, WorkspaceCapabilityResolver, and UiEnforcement patterns (001-finding-risk-acceptance)
|
||||||
|
- PostgreSQL with new tenant-owned exception tables and JSONB-backed supporting metadata (001-finding-risk-acceptance)
|
||||||
|
- PHP 8.4, Laravel 12, Livewire 4, Filament 5 + Filament resources/pages/actions, Eloquent models, queued Laravel jobs, existing `EvidenceSnapshotService`, existing `ReviewPackService`, capability registry, `OperationRunService` (155-tenant-review-layer)
|
||||||
|
- PostgreSQL with JSONB-backed summary payloads and tenant/workspace ownership columns (155-tenant-review-layer)
|
||||||
|
- PostgreSQL-backed existing domain records; no new business-domain table is required for the first slice; shared taxonomy reference will live in repository documentation and code-level metadata (156-operator-outcome-taxonomy)
|
||||||
|
- PostgreSQL-backed existing records such as `operation_runs`, tenant governance records, onboarding workflow state, and provider connection state; no new business-domain table is required for the first slice (157-reason-code-translation)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament v5, Livewire v4, Pest v4, existing `BadgeCatalog` / `BadgeRenderer` / `OperatorOutcomeTaxonomy`, `ReasonPresenter`, `OperationRunService`, `TenantReviewReadinessGate`, existing baseline/evidence/review/review-pack resources and canonical pages (158-artifact-truth-semantics)
|
||||||
|
- PostgreSQL with existing JSONB-backed `summary`, `summary_jsonb`, and `context` payloads on baseline snapshots, evidence snapshots, tenant reviews, review packs, and operation runs; no new primary storage required for the first slice (158-artifact-truth-semantics)
|
||||||
|
- PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Pest 4, Laravel queue workers, existing `OperationRunService`, `TrackOperationRun`, `OperationUxPresenter`, `ReasonPresenter`, `BadgeCatalog` domain badges, and current Operations Monitoring pages (160-operation-lifecycle-guarantees)
|
||||||
|
- PostgreSQL for `operation_runs`, `jobs`, and `failed_jobs`; JSONB-backed `context`, `summary_counts`, and `failure_summary`; configuration in `config/queue.php` and `config/tenantpilot.php` (160-operation-lifecycle-guarantees)
|
||||||
|
|
||||||
- PHP 8.4.15 (feat/005-bulk-operations)
|
- PHP 8.4.15 (feat/005-bulk-operations)
|
||||||
|
|
||||||
@ -102,8 +121,8 @@ ## Code Style
|
|||||||
PHP 8.4.15: Follow standard conventions
|
PHP 8.4.15: Follow standard conventions
|
||||||
|
|
||||||
## Recent Changes
|
## Recent Changes
|
||||||
- 147-tenant-selector-remembered-context-enforcement: Added PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Tailwind CSS 4
|
- 160-operation-lifecycle-guarantees: Added PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Pest 4, Laravel queue workers, existing `OperationRunService`, `TrackOperationRun`, `OperationUxPresenter`, `ReasonPresenter`, `BadgeCatalog` domain badges, and current Operations Monitoring pages
|
||||||
- 146-central-tenant-status-presentation: Added PHP 8.4.15 + Laravel 12, Filament 5, Livewire 4, Tailwind CSS 4
|
- 159-baseline-snapshot-truth: Added PHP 8.4 + Laravel 12, Filament v5, Livewire v4
|
||||||
- 145-tenant-action-taxonomy-lifecycle-safe-visibility: Added PHP 8.4.15 with Laravel 12, Filament v5, Livewire v4.0+ + Filament Actions/Tables/Infolists, Laravel Gates/Policies, `UiEnforcement`, `WorkspaceUiEnforcement`, `ActionSurfaceDeclaration`, `BadgeCatalog`, `TenantOperabilityService`, `OnboardingLifecycleService`
|
- 158-artifact-truth-semantics: Added PHP 8.4.15 + Laravel 12, Filament v5, Livewire v4, Pest v4, existing `BadgeCatalog` / `BadgeRenderer` / `OperatorOutcomeTaxonomy`, `ReasonPresenter`, `OperationRunService`, `TenantReviewReadinessGate`, existing baseline/evidence/review/review-pack resources and canonical pages
|
||||||
<!-- MANUAL ADDITIONS START -->
|
<!-- MANUAL ADDITIONS START -->
|
||||||
<!-- MANUAL ADDITIONS END -->
|
<!-- MANUAL ADDITIONS END -->
|
||||||
|
|||||||
@ -3,12 +3,17 @@
|
|||||||
|
|
||||||
- Version change: 1.11.0 → 1.12.0
|
- Version change: 1.11.0 → 1.12.0
|
||||||
- Modified principles:
|
- Modified principles:
|
||||||
- Scope & Ownership Clarification (SCOPE-001)
|
|
||||||
- Added sections:
|
|
||||||
- None
|
- None
|
||||||
|
- Added sections:
|
||||||
|
- Operator Surface Principles (OPSURF-001)
|
||||||
- Removed sections: None
|
- Removed sections: None
|
||||||
- Templates requiring updates:
|
- Templates requiring updates:
|
||||||
- None
|
- ✅ .specify/templates/spec-template.md
|
||||||
|
- ✅ .specify/templates/plan-template.md
|
||||||
|
- ✅ .specify/templates/tasks-template.md
|
||||||
|
- ✅ docs/product/principles.md
|
||||||
|
- ✅ docs/product/standards/README.md
|
||||||
|
- ✅ docs/HANDOVER.md
|
||||||
- Follow-up TODOs:
|
- Follow-up TODOs:
|
||||||
- None.
|
- None.
|
||||||
-->
|
-->
|
||||||
@ -330,6 +335,65 @@ ### Operator-facing UI Naming Standards (UI-NAMING-001)
|
|||||||
- The visible run label for that action MUST be `Policy sync`.
|
- The visible run label for that action MUST be `Policy sync`.
|
||||||
- The audit prose for that action MUST be `{actor} queued policy sync`.
|
- The audit prose for that action MUST be `{actor} queued policy sync`.
|
||||||
|
|
||||||
|
### Operator Surface Principles (OPSURF-001)
|
||||||
|
|
||||||
|
Goal: operator-facing surfaces MUST optimize for the primary working audience rather than raw implementation visibility.
|
||||||
|
|
||||||
|
Operator-first default surfaces
|
||||||
|
- `/admin` is operator-first.
|
||||||
|
- Default-visible content MUST use operator-facing language, clear scope, and actionable status communication.
|
||||||
|
- Raw implementation details MUST NOT be default-visible on primary operator surfaces.
|
||||||
|
|
||||||
|
Progressive disclosure for diagnostics
|
||||||
|
- Diagnostic detail MAY be available where needed, but it MUST be secondary and explicitly revealed.
|
||||||
|
- JSON payloads, raw IDs, internal field names, provider error details, and low-level technical metadata belong in diagnostics surfaces, drawers, tabs, accordions, or modals rather than primary content.
|
||||||
|
- A surface MUST NOT require operators to parse raw JSON or provider-specific field names to understand the primary state or next action.
|
||||||
|
|
||||||
|
Distinct status dimensions
|
||||||
|
- Operator-facing surfaces MUST distinguish at least the following dimensions when they all exist in the domain:
|
||||||
|
- execution outcome
|
||||||
|
- data completeness
|
||||||
|
- governance result
|
||||||
|
- lifecycle or readiness state
|
||||||
|
- These dimensions MUST NOT be collapsed into a single ambiguous status model.
|
||||||
|
- If a surface summarizes multiple status dimensions, the default-visible presentation MUST label each dimension explicitly.
|
||||||
|
|
||||||
|
Explicit mutation scope
|
||||||
|
- Every action that changes state MUST communicate before execution whether it affects:
|
||||||
|
- TenantPilot only
|
||||||
|
- the Microsoft tenant
|
||||||
|
- simulation only
|
||||||
|
- Mutation scope MUST be understandable from the action label, helper text, confirmation copy, preview, or nearby status copy before the operator commits.
|
||||||
|
- A mutating action MUST NOT rely on hidden implementation knowledge to communicate its blast radius.
|
||||||
|
|
||||||
|
Safe execution for dangerous actions
|
||||||
|
- Dangerous actions MUST follow a consistent safe-execution pattern:
|
||||||
|
- configuration
|
||||||
|
- safety checks or simulation
|
||||||
|
- preview
|
||||||
|
- hard confirmation where required
|
||||||
|
- execute
|
||||||
|
- One-click destructive actions are not acceptable for high-blast-radius operations.
|
||||||
|
- When a full multi-step flow is not feasible, the spec MUST document the explicit exemption and the replacement safeguards.
|
||||||
|
|
||||||
|
Explicit workspace and tenant context
|
||||||
|
- Workspace and tenant scope MUST remain explicit in navigation, actions, and page semantics.
|
||||||
|
- Tenant-scoped surfaces MUST NOT silently expose workspace-wide actions.
|
||||||
|
- Canonical workspace views that reference tenant-owned records MUST make the workspace and tenant context legible before the operator acts.
|
||||||
|
|
||||||
|
Page contract requirement
|
||||||
|
- Every new or materially refactored operator-facing page MUST define:
|
||||||
|
- primary persona
|
||||||
|
- surface type
|
||||||
|
- primary operator question
|
||||||
|
- default-visible information
|
||||||
|
- diagnostics-only information
|
||||||
|
- status dimensions used
|
||||||
|
- mutation scope
|
||||||
|
- primary actions
|
||||||
|
- dangerous actions
|
||||||
|
- This page contract MUST be recorded in the governing spec and kept in sync when the page semantics materially change.
|
||||||
|
|
||||||
Spec Scope Fields (SCOPE-002)
|
Spec Scope Fields (SCOPE-002)
|
||||||
|
|
||||||
- Every feature spec MUST declare:
|
- Every feature spec MUST declare:
|
||||||
@ -387,4 +451,4 @@ ### Versioning Policy (SemVer)
|
|||||||
- **MINOR**: new principle/section or materially expanded guidance.
|
- **MINOR**: new principle/section or materially expanded guidance.
|
||||||
- **MAJOR**: removing/redefining principles in a backward-incompatible way.
|
- **MAJOR**: removing/redefining principles in a backward-incompatible way.
|
||||||
|
|
||||||
**Version**: 1.11.0 | **Ratified**: 2026-01-03 | **Last Amended**: 2026-03-10
|
**Version**: 1.12.0 | **Ratified**: 2026-01-03 | **Last Amended**: 2026-03-21
|
||||||
|
|||||||
@ -50,6 +50,12 @@ ## Constitution Check
|
|||||||
- Data minimization: Inventory stores metadata + whitelisted meta; logs contain no secrets/tokens
|
- Data minimization: Inventory stores metadata + whitelisted meta; logs contain no secrets/tokens
|
||||||
- Badge semantics (BADGE-001): status-like badges use `BadgeCatalog` / `BadgeRenderer`; no ad-hoc mappings; new values include tests
|
- Badge semantics (BADGE-001): status-like badges use `BadgeCatalog` / `BadgeRenderer`; no ad-hoc mappings; new values include tests
|
||||||
- UI naming (UI-NAMING-001): operator-facing labels use `Verb + Object`; scope (`Workspace`, `Tenant`) is never the primary action label; source/domain is secondary unless disambiguation is required; runs/toasts/audit prose use the same domain vocabulary; implementation-first terms do not appear in primary operator UI
|
- UI naming (UI-NAMING-001): operator-facing labels use `Verb + Object`; scope (`Workspace`, `Tenant`) is never the primary action label; source/domain is secondary unless disambiguation is required; runs/toasts/audit prose use the same domain vocabulary; implementation-first terms do not appear in primary operator UI
|
||||||
|
- Operator surfaces (OPSURF-001): `/admin` defaults are operator-first; default-visible content avoids raw implementation detail; diagnostics are explicitly revealed secondarily
|
||||||
|
- Operator surfaces (OPSURF-001): execution outcome, data completeness, governance result, and lifecycle/readiness are modeled as distinct status dimensions when all apply; they are not collapsed into one ambiguous status
|
||||||
|
- Operator surfaces (OPSURF-001): every mutating action communicates whether it changes TenantPilot only, the Microsoft tenant, or simulation only before execution
|
||||||
|
- Operator surfaces (OPSURF-001): dangerous actions follow configuration → safety checks/simulation → preview → hard confirmation where required → execute, unless a spec documents an explicit exemption and replacement safeguards
|
||||||
|
- Operator surfaces (OPSURF-001): workspace and tenant context remain explicit in navigation, actions, and page semantics; tenant surfaces do not silently expose workspace-wide actions
|
||||||
|
- Operator surfaces (OPSURF-001): each new or materially refactored operator-facing page defines a page contract covering persona, surface type, operator question, default-visible info, diagnostics-only info, status dimensions, mutation scope, primary actions, and dangerous actions
|
||||||
- Filament UI Action Surface Contract: for any new/modified Filament Resource/RelationManager/Page, define Header/Row/Bulk/Empty-State actions, ensure every List/Table has a record inspection affordance (prefer `recordUrl()` clickable rows; do not render a lone View row action), keep max 2 visible row actions with the rest in “More”, group bulk actions, require confirmations for destructive actions (typed confirmation for large/bulk where applicable), write audit logs for mutations, enforce RBAC via central helpers (non-member 404, member missing capability 403), and ensure CI blocks merges if the contract is violated or not explicitly exempted
|
- Filament UI Action Surface Contract: for any new/modified Filament Resource/RelationManager/Page, define Header/Row/Bulk/Empty-State actions, ensure every List/Table has a record inspection affordance (prefer `recordUrl()` clickable rows; do not render a lone View row action), keep max 2 visible row actions with the rest in “More”, group bulk actions, require confirmations for destructive actions (typed confirmation for large/bulk where applicable), write audit logs for mutations, enforce RBAC via central helpers (non-member 404, member missing capability 403), and ensure CI blocks merges if the contract is violated or not explicitly exempted
|
||||||
- Filament UI UX-001 (Layout & IA): Create/Edit uses Main/Aside (3-col grid, Main=columnSpan(2), Aside=columnSpan(1)); all fields inside Sections/Cards (no naked inputs); View uses Infolists (not disabled edit forms); status badges use BADGE-001; empty states have specific title + explanation + 1 CTA; max 1 primary + 1 secondary header action; tables provide search/sort/filters for core dimensions; shared layout builders preferred for consistency
|
- Filament UI UX-001 (Layout & IA): Create/Edit uses Main/Aside (3-col grid, Main=columnSpan(2), Aside=columnSpan(1)); all fields inside Sections/Cards (no naked inputs); View uses Infolists (not disabled edit forms); status badges use BADGE-001; empty states have specific title + explanation + 1 CTA; max 1 primary + 1 secondary header action; tables provide search/sort/filters for core dimensions; shared layout builders preferred for consistency
|
||||||
## Project Structure
|
## Project Structure
|
||||||
|
|||||||
@ -17,6 +17,14 @@ ## Spec Scope Fields *(mandatory)*
|
|||||||
- **Default filter behavior when tenant-context is active**: [e.g., prefilter to current tenant]
|
- **Default filter behavior when tenant-context is active**: [e.g., prefilter to current tenant]
|
||||||
- **Explicit entitlement checks preventing cross-tenant leakage**: [Describe checks]
|
- **Explicit entitlement checks preventing cross-tenant leakage**: [Describe checks]
|
||||||
|
|
||||||
|
## Operator Surface Contract *(mandatory when operator-facing surfaces are changed)*
|
||||||
|
|
||||||
|
If this feature adds a new operator-facing page or materially refactors one, fill out one row per affected page/surface.
|
||||||
|
|
||||||
|
| Surface | Primary Persona | Surface Type | Primary Operator Question | Default-visible Information | Diagnostics-only Information | Status Dimensions Used | Mutation Scope | Primary Actions | Dangerous Actions |
|
||||||
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
|
| e.g. Tenant policies page | Tenant operator | List/detail | What needs action right now? | Policy health, drift, assignment coverage | Raw payloads, provider IDs, low-level API details | lifecycle, data completeness, governance result | TenantPilot only / Microsoft tenant / simulation only | Sync policies, View policy | Restore policy |
|
||||||
|
|
||||||
## User Scenarios & Testing *(mandatory)*
|
## User Scenarios & Testing *(mandatory)*
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
@ -127,6 +135,15 @@ ## Requirements *(mandatory)*
|
|||||||
- how the same domain vocabulary is preserved across button labels, modal titles, run titles, notifications, and audit prose,
|
- how the same domain vocabulary is preserved across button labels, modal titles, run titles, notifications, and audit prose,
|
||||||
- and how implementation-first terms are kept out of primary operator-facing labels.
|
- and how implementation-first terms are kept out of primary operator-facing labels.
|
||||||
|
|
||||||
|
**Constitution alignment (OPSURF-001):** If this feature adds or materially refactors an operator-facing surface, the spec MUST describe:
|
||||||
|
- how the default-visible content stays operator-first on `/admin` and avoids raw implementation detail,
|
||||||
|
- which diagnostics are secondary and how they are explicitly revealed,
|
||||||
|
- which status dimensions are shown separately (execution outcome, data completeness, governance result, lifecycle/readiness) and why,
|
||||||
|
- how each mutating action communicates its mutation scope before execution (`TenantPilot only`, `Microsoft tenant`, or `simulation only`),
|
||||||
|
- how dangerous actions follow the safe-execution pattern (configuration, safety checks/simulation, preview, hard confirmation where required, execute),
|
||||||
|
- how workspace and tenant context remain explicit in navigation, action copy, and page semantics,
|
||||||
|
- and the page contract for each new or materially refactored operator-facing page.
|
||||||
|
|
||||||
**Constitution alignment (Filament Action Surfaces):** If this feature adds or modifies any Filament Resource / RelationManager / Page,
|
**Constitution alignment (Filament Action Surfaces):** If this feature adds or modifies any Filament Resource / RelationManager / Page,
|
||||||
the spec MUST include a “UI Action Matrix” (see below) and explicitly state whether the Action Surface Contract is satisfied.
|
the spec MUST include a “UI Action Matrix” (see below) and explicitly state whether the Action Surface Contract is satisfied.
|
||||||
If the contract is not satisfied, the spec MUST include an explicit exemption with rationale.
|
If the contract is not satisfied, the spec MUST include an explicit exemption with rationale.
|
||||||
|
|||||||
@ -38,6 +38,13 @@ # Tasks: [FEATURE NAME]
|
|||||||
- using source/domain terms only where same-screen disambiguation is required,
|
- using source/domain terms only where same-screen disambiguation is required,
|
||||||
- aligning button labels, modal titles, run titles, notifications, and audit prose to the same domain vocabulary,
|
- aligning button labels, modal titles, run titles, notifications, and audit prose to the same domain vocabulary,
|
||||||
- removing implementation-first wording from primary operator-facing copy.
|
- removing implementation-first wording from primary operator-facing copy.
|
||||||
|
**Operator Surfaces**: If this feature adds or materially refactors an operator-facing page or flow, tasks MUST include:
|
||||||
|
- filling the spec’s Operator Surface Contract for every affected page,
|
||||||
|
- making default-visible content operator-first and moving JSON payloads, raw IDs, internal field names, provider error details, and low-level metadata into explicitly revealed diagnostics surfaces,
|
||||||
|
- modeling execution outcome, data completeness, governance result, and lifecycle/readiness as distinct status dimensions when applicable,
|
||||||
|
- making mutation scope legible before execution for every state-changing action (`TenantPilot only`, `Microsoft tenant`, or `simulation only`),
|
||||||
|
- implementing the safe-execution flow for dangerous actions (configuration, safety checks/simulation, preview, hard confirmation where required, execute) or documenting an approved exemption,
|
||||||
|
- keeping workspace and tenant context explicit in navigation, actions, and page semantics so tenant pages do not silently expose workspace-wide actions.
|
||||||
**Filament UI Action Surfaces**: If this feature adds/modifies any Filament Resource / RelationManager / Page, tasks MUST include:
|
**Filament UI Action Surfaces**: If this feature adds/modifies any Filament Resource / RelationManager / Page, tasks MUST include:
|
||||||
- filling the spec’s “UI Action Matrix” for all changed surfaces,
|
- filling the spec’s “UI Action Matrix” for all changed surfaces,
|
||||||
- implementing required action surfaces (header/row/bulk/empty-state CTA for lists; header actions for view; consistent save/cancel on create/edit),
|
- implementing required action surfaces (header/row/bulk/empty-state CTA for lists; header actions for view; consistent save/cancel on create/edit),
|
||||||
|
|||||||
@ -6,6 +6,7 @@
|
|||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
|
use App\Services\Operations\OperationLifecycleReconciler;
|
||||||
use App\Support\OperationRunOutcome;
|
use App\Support\OperationRunOutcome;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
@ -18,8 +19,10 @@ class TenantpilotReconcileBackupScheduleOperationRuns extends Command
|
|||||||
|
|
||||||
protected $description = 'Reconcile stuck backup schedule OperationRuns without legacy run-table lookups.';
|
protected $description = 'Reconcile stuck backup schedule OperationRuns without legacy run-table lookups.';
|
||||||
|
|
||||||
public function handle(OperationRunService $operationRunService): int
|
public function handle(
|
||||||
{
|
OperationRunService $operationRunService,
|
||||||
|
OperationLifecycleReconciler $operationLifecycleReconciler,
|
||||||
|
): int {
|
||||||
$tenantIdentifiers = array_values(array_filter((array) $this->option('tenant')));
|
$tenantIdentifiers = array_values(array_filter((array) $this->option('tenant')));
|
||||||
$olderThanMinutes = max(0, (int) $this->option('older-than'));
|
$olderThanMinutes = max(0, (int) $this->option('older-than'));
|
||||||
$dryRun = (bool) $this->option('dry-run');
|
$dryRun = (bool) $this->option('dry-run');
|
||||||
@ -96,31 +99,9 @@ public function handle(OperationRunService $operationRunService): int
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($operationRun->status === 'queued' && $operationRunService->isStaleQueuedRun($operationRun, max(1, $olderThanMinutes))) {
|
$change = $operationLifecycleReconciler->reconcileRun($operationRun, $dryRun);
|
||||||
if (! $dryRun) {
|
|
||||||
$operationRunService->failStaleQueuedRun($operationRun, 'Backup schedule run was queued but never started.');
|
|
||||||
}
|
|
||||||
|
|
||||||
$reconciled++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($operationRun->status === 'running') {
|
|
||||||
if (! $dryRun) {
|
|
||||||
$operationRunService->updateRun(
|
|
||||||
$operationRun,
|
|
||||||
status: 'completed',
|
|
||||||
outcome: OperationRunOutcome::Failed->value,
|
|
||||||
failures: [
|
|
||||||
[
|
|
||||||
'code' => 'backup_schedule.stalled',
|
|
||||||
'message' => 'Backup schedule run exceeded reconciliation timeout and was marked failed.',
|
|
||||||
],
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if ($change !== null) {
|
||||||
$reconciled++;
|
$reconciled++;
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
|
|||||||
105
app/Console/Commands/TenantpilotReconcileOperationRuns.php
Normal file
105
app/Console/Commands/TenantpilotReconcileOperationRuns.php
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Services\Operations\OperationLifecycleReconciler;
|
||||||
|
use App\Support\Operations\OperationLifecyclePolicy;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
|
class TenantpilotReconcileOperationRuns extends Command
|
||||||
|
{
|
||||||
|
protected $signature = 'tenantpilot:operation-runs:reconcile
|
||||||
|
{--type=* : Limit reconciliation to one or more covered operation types}
|
||||||
|
{--tenant=* : Limit reconciliation to tenant_id or tenant external_id}
|
||||||
|
{--workspace=* : Limit reconciliation to workspace ids}
|
||||||
|
{--limit=100 : Maximum number of active runs to inspect}
|
||||||
|
{--dry-run : Report the changes without writing them}';
|
||||||
|
|
||||||
|
protected $description = 'Reconcile stale covered operation runs back to deterministic terminal truth.';
|
||||||
|
|
||||||
|
public function handle(
|
||||||
|
OperationLifecycleReconciler $reconciler,
|
||||||
|
OperationLifecyclePolicy $policy,
|
||||||
|
): int {
|
||||||
|
$types = array_values(array_filter(
|
||||||
|
(array) $this->option('type'),
|
||||||
|
static fn (mixed $type): bool => is_string($type) && trim($type) !== '',
|
||||||
|
));
|
||||||
|
$workspaceIds = array_values(array_filter(
|
||||||
|
array_map(
|
||||||
|
static fn (mixed $workspaceId): int => is_numeric($workspaceId) ? (int) $workspaceId : 0,
|
||||||
|
(array) $this->option('workspace'),
|
||||||
|
),
|
||||||
|
static fn (int $workspaceId): bool => $workspaceId > 0,
|
||||||
|
));
|
||||||
|
$tenantIds = $this->resolveTenantIds(array_values(array_filter((array) $this->option('tenant'))));
|
||||||
|
$dryRun = (bool) $this->option('dry-run');
|
||||||
|
|
||||||
|
if ($types === []) {
|
||||||
|
$types = $policy->coveredTypeNames();
|
||||||
|
}
|
||||||
|
|
||||||
|
$result = $reconciler->reconcile([
|
||||||
|
'types' => $types,
|
||||||
|
'tenant_ids' => $tenantIds,
|
||||||
|
'workspace_ids' => $workspaceIds,
|
||||||
|
'limit' => max(1, (int) $this->option('limit')),
|
||||||
|
'dry_run' => $dryRun,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$rows = collect($result['changes'] ?? [])
|
||||||
|
->map(static function (array $change): array {
|
||||||
|
return [
|
||||||
|
'Run' => (string) ($change['operation_run_id'] ?? '—'),
|
||||||
|
'Type' => (string) ($change['type'] ?? '—'),
|
||||||
|
'Reason' => (string) ($change['reason_code'] ?? '—'),
|
||||||
|
'Applied' => (($change['applied'] ?? false) === true) ? 'yes' : 'no',
|
||||||
|
];
|
||||||
|
})
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
|
||||||
|
if ($rows !== []) {
|
||||||
|
$this->table(['Run', 'Type', 'Reason', 'Applied'], $rows);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->info(sprintf(
|
||||||
|
'Inspected %d run(s); reconciled %d; skipped %d.',
|
||||||
|
(int) ($result['candidates'] ?? 0),
|
||||||
|
(int) ($result['reconciled'] ?? 0),
|
||||||
|
(int) ($result['skipped'] ?? 0),
|
||||||
|
));
|
||||||
|
|
||||||
|
if ($dryRun) {
|
||||||
|
$this->comment('Dry-run: no changes written.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $tenantIdentifiers
|
||||||
|
* @return array<int, int>
|
||||||
|
*/
|
||||||
|
private function resolveTenantIds(array $tenantIdentifiers): array
|
||||||
|
{
|
||||||
|
if ($tenantIdentifiers === []) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenantIds = [];
|
||||||
|
|
||||||
|
foreach ($tenantIdentifiers as $identifier) {
|
||||||
|
$tenant = Tenant::query()->forTenant($identifier)->first();
|
||||||
|
|
||||||
|
if ($tenant instanceof Tenant) {
|
||||||
|
$tenantIds[] = (int) $tenant->getKey();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values(array_unique($tenantIds));
|
||||||
|
}
|
||||||
|
}
|
||||||
27
app/Exceptions/ReviewPackEvidenceResolutionException.php
Normal file
27
app/Exceptions/ReviewPackEvidenceResolutionException.php
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Exceptions;
|
||||||
|
|
||||||
|
use App\Services\Evidence\EvidenceResolutionResult;
|
||||||
|
use RuntimeException;
|
||||||
|
|
||||||
|
class ReviewPackEvidenceResolutionException extends RuntimeException
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
public readonly EvidenceResolutionResult $result,
|
||||||
|
?string $message = null,
|
||||||
|
) {
|
||||||
|
parent::__construct($message ?? self::defaultMessage($result));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function defaultMessage(EvidenceResolutionResult $result): string
|
||||||
|
{
|
||||||
|
return match ($result->outcome) {
|
||||||
|
'missing_snapshot' => 'No eligible evidence snapshot is available for this review pack.',
|
||||||
|
'snapshot_ineligible' => 'The latest evidence snapshot is not eligible for review-pack generation.',
|
||||||
|
default => 'Evidence snapshot resolution failed.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
72
app/Filament/Concerns/InteractsWithTenantOwnedRecords.php
Normal file
72
app/Filament/Concerns/InteractsWithTenantOwnedRecords.php
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Concerns;
|
||||||
|
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Support\WorkspaceIsolation\TenantOwnedQueryScope;
|
||||||
|
use App\Support\WorkspaceIsolation\TenantOwnedRecordResolver;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
|
trait InteractsWithTenantOwnedRecords
|
||||||
|
{
|
||||||
|
protected static function tenantOwnedRelationshipName(): string
|
||||||
|
{
|
||||||
|
$relationshipName = property_exists(static::class, 'tenantOwnershipRelationshipName')
|
||||||
|
? static::$tenantOwnershipRelationshipName
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return is_string($relationshipName) && $relationshipName !== ''
|
||||||
|
? $relationshipName
|
||||||
|
: 'tenant';
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function resolveTenantContextForTenantOwnedRecords(): ?Tenant
|
||||||
|
{
|
||||||
|
if (method_exists(static::class, 'resolveTenantContextForCurrentPanel')) {
|
||||||
|
return static::resolveTenantContextForCurrentPanel();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (method_exists(static::class, 'panelTenantContext')) {
|
||||||
|
return static::panelTenantContext();
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getTenantOwnedEloquentQuery(): Builder
|
||||||
|
{
|
||||||
|
return static::scopeTenantOwnedQuery(parent::getEloquentQuery());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function scopeTenantOwnedQuery(Builder $query, ?Tenant $tenant = null): Builder
|
||||||
|
{
|
||||||
|
return app(TenantOwnedQueryScope::class)->apply(
|
||||||
|
$query,
|
||||||
|
$tenant ?? static::resolveTenantContextForTenantOwnedRecords(),
|
||||||
|
static::tenantOwnedRelationshipName(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function resolveTenantOwnedRecord(Model|int|string|null $record, ?Builder $query = null, ?Tenant $tenant = null): ?Model
|
||||||
|
{
|
||||||
|
$scopedQuery = static::scopeTenantOwnedQuery(
|
||||||
|
$query ?? parent::getEloquentQuery(),
|
||||||
|
$tenant,
|
||||||
|
);
|
||||||
|
|
||||||
|
return app(TenantOwnedRecordResolver::class)->resolve($scopedQuery, $record);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function resolveTenantOwnedRecordOrFail(Model|int|string|null $record, ?Builder $query = null, ?Tenant $tenant = null): Model
|
||||||
|
{
|
||||||
|
$scopedQuery = static::scopeTenantOwnedQuery(
|
||||||
|
$query ?? parent::getEloquentQuery(),
|
||||||
|
$tenant,
|
||||||
|
);
|
||||||
|
|
||||||
|
return app(TenantOwnedRecordResolver::class)->resolveOrFail($scopedQuery, $record);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -14,7 +14,7 @@ trait ResolvesPanelTenantContext
|
|||||||
protected static function resolveTenantContextForCurrentPanel(): ?Tenant
|
protected static function resolveTenantContextForCurrentPanel(): ?Tenant
|
||||||
{
|
{
|
||||||
if (Filament::getCurrentPanel()?->getId() === 'admin') {
|
if (Filament::getCurrentPanel()?->getId() === 'admin') {
|
||||||
$tenant = app(OperateHubShell::class)->activeEntitledTenant(request());
|
$tenant = app(OperateHubShell::class)->tenantOwnedPanelContext(request());
|
||||||
|
|
||||||
return $tenant instanceof Tenant ? $tenant : null;
|
return $tenant instanceof Tenant ? $tenant : null;
|
||||||
}
|
}
|
||||||
@ -24,6 +24,16 @@ protected static function resolveTenantContextForCurrentPanel(): ?Tenant
|
|||||||
return $tenant instanceof Tenant ? $tenant : null;
|
return $tenant instanceof Tenant ? $tenant : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function panelTenantContext(): ?Tenant
|
||||||
|
{
|
||||||
|
return static::resolveTenantContextForCurrentPanel();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function trustedPanelTenantContext(): ?Tenant
|
||||||
|
{
|
||||||
|
return static::panelTenantContext();
|
||||||
|
}
|
||||||
|
|
||||||
protected static function resolveTenantContextForCurrentPanelOrFail(): Tenant
|
protected static function resolveTenantContextForCurrentPanelOrFail(): Tenant
|
||||||
{
|
{
|
||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
@ -34,4 +44,9 @@ protected static function resolveTenantContextForCurrentPanelOrFail(): Tenant
|
|||||||
|
|
||||||
return $tenant;
|
return $tenant;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected static function resolveTrustedPanelTenantContextOrFail(): Tenant
|
||||||
|
{
|
||||||
|
return static::resolveTenantContextForCurrentPanelOrFail();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Support\OperateHub\OperateHubShell;
|
use App\Support\OperateHub\OperateHubShell;
|
||||||
|
use App\Support\WorkspaceIsolation\TenantOwnedModelFamilies;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
@ -21,6 +22,10 @@ public static function getGlobalSearchEloquentQuery(): Builder
|
|||||||
{
|
{
|
||||||
$query = static::getModel()::query();
|
$query = static::getModel()::query();
|
||||||
|
|
||||||
|
if (! TenantOwnedModelFamilies::supportsScopedGlobalSearch(static::getModel())) {
|
||||||
|
return $query->whereRaw('1 = 0');
|
||||||
|
}
|
||||||
|
|
||||||
if (! static::isScopedToTenant()) {
|
if (! static::isScopedToTenant()) {
|
||||||
$panel = Filament::getCurrentOrDefaultPanel();
|
$panel = Filament::getCurrentOrDefaultPanel();
|
||||||
|
|
||||||
|
|||||||
@ -307,9 +307,22 @@ private function compareNowAction(): Action
|
|||||||
$result = $service->startCompare($tenant, $user);
|
$result = $service->startCompare($tenant, $user);
|
||||||
|
|
||||||
if (! ($result['ok'] ?? false)) {
|
if (! ($result['ok'] ?? false)) {
|
||||||
|
$reasonCode = is_string($result['reason_code'] ?? null) ? (string) $result['reason_code'] : 'unknown';
|
||||||
|
|
||||||
|
$message = match ($reasonCode) {
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_ROLLOUT_DISABLED => 'Full-content baseline compare is currently disabled for controlled rollout.',
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE => 'The assigned baseline profile is not active.',
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_NO_ACTIVE_SNAPSHOT,
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_NO_CONSUMABLE_SNAPSHOT => 'No complete baseline snapshot is currently available for compare.',
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_SNAPSHOT_BUILDING => 'The latest baseline capture is still building. Compare will be available after it completes.',
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_SNAPSHOT_INCOMPLETE => 'The latest baseline capture is incomplete. Capture a new baseline before comparing.',
|
||||||
|
\App\Support\Baselines\BaselineReasonCodes::COMPARE_SNAPSHOT_SUPERSEDED => 'A newer complete baseline snapshot is current. Compare uses the latest complete baseline only.',
|
||||||
|
default => 'Reason: '.$reasonCode,
|
||||||
|
};
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Cannot start comparison')
|
->title('Cannot start comparison')
|
||||||
->body('Reason: '.($result['reason_code'] ?? 'unknown'))
|
->body($message)
|
||||||
->danger()
|
->danger()
|
||||||
->send();
|
->send();
|
||||||
|
|
||||||
|
|||||||
@ -8,7 +8,9 @@
|
|||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\UserTenantPreference;
|
use App\Models\UserTenantPreference;
|
||||||
use App\Services\Tenants\TenantOperabilityService;
|
use App\Services\Tenants\TenantOperabilityService;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
use App\Support\Tenants\TenantLifecyclePresentation;
|
use App\Support\Tenants\TenantLifecyclePresentation;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Pages\Page;
|
use Filament\Pages\Page;
|
||||||
@ -68,7 +70,22 @@ public function selectTenant(int $tenantId): void
|
|||||||
abort(403);
|
abort(403);
|
||||||
}
|
}
|
||||||
|
|
||||||
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
$workspaceContext = app(WorkspaceContext::class);
|
||||||
|
$workspaceId = $workspaceContext->currentWorkspaceId(request());
|
||||||
|
$tenant = null;
|
||||||
|
|
||||||
|
if ($workspaceId === null) {
|
||||||
|
$tenant = Tenant::query()->whereKey($tenantId)->first();
|
||||||
|
|
||||||
|
if ($tenant instanceof Tenant) {
|
||||||
|
$workspace = $tenant->workspace;
|
||||||
|
|
||||||
|
if ($workspace !== null && $user->canAccessTenant($tenant)) {
|
||||||
|
$workspaceContext->setCurrentWorkspace($workspace, $user, request());
|
||||||
|
$workspaceId = (int) $workspace->getKey();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if ($workspaceId === null) {
|
if ($workspaceId === null) {
|
||||||
$this->redirect(route('filament.admin.pages.choose-workspace'));
|
$this->redirect(route('filament.admin.pages.choose-workspace'));
|
||||||
@ -76,10 +93,12 @@ public function selectTenant(int $tenantId): void
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$tenant = Tenant::query()
|
if (! $tenant instanceof Tenant || (int) $tenant->workspace_id !== $workspaceId) {
|
||||||
->where('workspace_id', $workspaceId)
|
$tenant = Tenant::query()
|
||||||
->whereKey($tenantId)
|
->where('workspace_id', $workspaceId)
|
||||||
->first();
|
->whereKey($tenantId)
|
||||||
|
->first();
|
||||||
|
}
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
abort(404);
|
abort(404);
|
||||||
@ -89,13 +108,21 @@ public function selectTenant(int $tenantId): void
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! app(TenantOperabilityService::class)->canSelectAsContext($tenant)) {
|
$outcome = app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::SelectorEligibility,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: $workspaceId,
|
||||||
|
lane: TenantInteractionLane::StandardActiveOperating,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (! $outcome->allowed) {
|
||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->persistLastTenant($user, $tenant);
|
$this->persistLastTenant($user, $tenant);
|
||||||
|
|
||||||
if (! app(WorkspaceContext::class)->rememberTenantContext($tenant, request())) {
|
if (! $workspaceContext->rememberTenantContext($tenant, request())) {
|
||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -34,6 +34,7 @@
|
|||||||
use Filament\Tables\Contracts\HasTable;
|
use Filament\Tables\Contracts\HasTable;
|
||||||
use Filament\Tables\Filters\SelectFilter;
|
use Filament\Tables\Filters\SelectFilter;
|
||||||
use Filament\Tables\Table;
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Contracts\View\View;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
use UnitEnum;
|
use UnitEnum;
|
||||||
@ -42,8 +43,6 @@ class AuditLog extends Page implements HasTable
|
|||||||
{
|
{
|
||||||
use InteractsWithTable;
|
use InteractsWithTable;
|
||||||
|
|
||||||
public ?int $selectedAuditLogId = null;
|
|
||||||
|
|
||||||
protected static bool $isDiscovered = false;
|
protected static bool $isDiscovered = false;
|
||||||
|
|
||||||
protected static bool $shouldRegisterNavigation = false;
|
protected static bool $shouldRegisterNavigation = false;
|
||||||
@ -82,14 +81,15 @@ public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
|||||||
public function mount(): void
|
public function mount(): void
|
||||||
{
|
{
|
||||||
$this->authorizePageAccess();
|
$this->authorizePageAccess();
|
||||||
$this->selectedAuditLogId = is_numeric(request()->query('event')) ? (int) request()->query('event') : null;
|
$requestedEventId = is_numeric(request()->query('event')) ? (int) request()->query('event') : null;
|
||||||
|
|
||||||
app(CanonicalAdminTenantFilterState::class)->sync($this->getTableFiltersSessionKey(), request: request());
|
app(CanonicalAdminTenantFilterState::class)->sync($this->getTableFiltersSessionKey(), request: request());
|
||||||
|
|
||||||
$this->mountInteractsWithTable();
|
$this->mountInteractsWithTable();
|
||||||
|
|
||||||
if ($this->selectedAuditLogId !== null) {
|
if ($requestedEventId !== null) {
|
||||||
$this->selectedAuditLog();
|
$this->resolveAuditLog($requestedEventId);
|
||||||
|
$this->mountTableAction('inspect', (string) $requestedEventId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -98,31 +98,10 @@ public function mount(): void
|
|||||||
*/
|
*/
|
||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
$actions = app(OperateHubShell::class)->headerActions(
|
return app(OperateHubShell::class)->headerActions(
|
||||||
scopeActionName: 'operate_hub_scope_audit_log',
|
scopeActionName: 'operate_hub_scope_audit_log',
|
||||||
returnActionName: 'operate_hub_return_audit_log',
|
returnActionName: 'operate_hub_return_audit_log',
|
||||||
);
|
);
|
||||||
|
|
||||||
if ($this->selectedAuditLog() instanceof AuditLogModel) {
|
|
||||||
$actions[] = Action::make('clear_selected_audit_event')
|
|
||||||
->label('Close details')
|
|
||||||
->color('gray')
|
|
||||||
->action(function (): void {
|
|
||||||
$this->clearSelectedAuditLog();
|
|
||||||
});
|
|
||||||
|
|
||||||
$relatedLink = $this->selectedAuditLink();
|
|
||||||
|
|
||||||
if (is_array($relatedLink)) {
|
|
||||||
$actions[] = Action::make('open_selected_audit_target')
|
|
||||||
->label($relatedLink['label'])
|
|
||||||
->icon('heroicon-o-arrow-top-right-on-square')
|
|
||||||
->color('gray')
|
|
||||||
->url($relatedLink['url']);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $actions;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function table(Table $table): Table
|
public function table(Table $table): Table
|
||||||
@ -195,9 +174,16 @@ public function table(Table $table): Table
|
|||||||
->label('Inspect event')
|
->label('Inspect event')
|
||||||
->icon('heroicon-o-eye')
|
->icon('heroicon-o-eye')
|
||||||
->color('gray')
|
->color('gray')
|
||||||
->action(function (AuditLogModel $record): void {
|
->slideOver()
|
||||||
$this->selectedAuditLogId = (int) $record->getKey();
|
->stickyModalHeader()
|
||||||
}),
|
->modalSubmitAction(false)
|
||||||
|
->modalCancelAction(fn (Action $action): Action => $action->label('Close details'))
|
||||||
|
->modalHeading(fn (AuditLogModel $record): string => $record->summaryText())
|
||||||
|
->modalDescription(fn (AuditLogModel $record): ?string => $record->recorded_at?->toDayDateTimeString())
|
||||||
|
->modalContent(fn (AuditLogModel $record): View => view('filament.pages.monitoring.partials.audit-log-inspect-event', [
|
||||||
|
'selectedAudit' => $record,
|
||||||
|
'selectedAuditLink' => $this->auditTargetLink($record),
|
||||||
|
])),
|
||||||
])
|
])
|
||||||
->bulkActions([])
|
->bulkActions([])
|
||||||
->emptyStateHeading('No audit events match this view')
|
->emptyStateHeading('No audit events match this view')
|
||||||
@ -209,48 +195,11 @@ public function table(Table $table): Table
|
|||||||
->icon('heroicon-o-x-mark')
|
->icon('heroicon-o-x-mark')
|
||||||
->color('gray')
|
->color('gray')
|
||||||
->action(function (): void {
|
->action(function (): void {
|
||||||
$this->selectedAuditLogId = null;
|
|
||||||
$this->resetTable();
|
$this->resetTable();
|
||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function clearSelectedAuditLog(): void
|
|
||||||
{
|
|
||||||
$this->selectedAuditLogId = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function selectedAuditLog(): ?AuditLogModel
|
|
||||||
{
|
|
||||||
if (! is_numeric($this->selectedAuditLogId)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$record = $this->auditBaseQuery()
|
|
||||||
->whereKey((int) $this->selectedAuditLogId)
|
|
||||||
->first();
|
|
||||||
|
|
||||||
if (! $record instanceof AuditLogModel) {
|
|
||||||
throw new NotFoundHttpException;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $record;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array{label: string, url: string}|null
|
|
||||||
*/
|
|
||||||
public function selectedAuditLink(): ?array
|
|
||||||
{
|
|
||||||
$record = $this->selectedAuditLog();
|
|
||||||
|
|
||||||
if (! $record instanceof AuditLogModel) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return app(RelatedNavigationResolver::class)->auditTargetLink($record);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<int, Tenant>
|
* @return array<int, Tenant>
|
||||||
*/
|
*/
|
||||||
@ -323,6 +272,27 @@ private function auditBaseQuery(): Builder
|
|||||||
->latestFirst();
|
->latestFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function resolveAuditLog(int $auditLogId): AuditLogModel
|
||||||
|
{
|
||||||
|
$record = $this->auditBaseQuery()
|
||||||
|
->whereKey($auditLogId)
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if (! $record instanceof AuditLogModel) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $record;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{label: string, url: string}|null
|
||||||
|
*/
|
||||||
|
private function auditTargetLink(AuditLogModel $record): ?array
|
||||||
|
{
|
||||||
|
return app(RelatedNavigationResolver::class)->auditTargetLink($record);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<string, string>
|
* @return array<string, string>
|
||||||
*/
|
*/
|
||||||
|
|||||||
136
app/Filament/Pages/Monitoring/EvidenceOverview.php
Normal file
136
app/Filament/Pages/Monitoring/EvidenceOverview.php
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Pages\Monitoring;
|
||||||
|
|
||||||
|
use App\Filament\Resources\EvidenceSnapshotResource;
|
||||||
|
use App\Models\EvidenceSnapshot;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Pages\Page;
|
||||||
|
use Illuminate\Auth\AuthenticationException;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class EvidenceOverview extends Page
|
||||||
|
{
|
||||||
|
protected static bool $isDiscovered = false;
|
||||||
|
|
||||||
|
protected static bool $shouldRegisterNavigation = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-shield-check';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Monitoring';
|
||||||
|
|
||||||
|
protected static ?string $title = 'Evidence Overview';
|
||||||
|
|
||||||
|
protected string $view = 'filament.pages.monitoring.evidence-overview';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var list<array<string, mixed>>
|
||||||
|
*/
|
||||||
|
public array $rows = [];
|
||||||
|
|
||||||
|
public ?int $tenantFilter = null;
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forPage(ActionSurfaceProfile::ListOnlyReadOnly)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'The overview header exposes a clear-filters action when a tenant prefilter is active.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ClickableRow->value)
|
||||||
|
->exempt(ActionSurfaceSlot::ListRowMoreMenu, 'The overview exposes a single drill-down link per row without a More menu.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'The overview does not expose bulk actions.')
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'The empty state explains the current scope and offers a clear-filters CTA.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function mount(): void
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
throw new AuthenticationException;
|
||||||
|
}
|
||||||
|
|
||||||
|
$workspaceContext = app(WorkspaceContext::class);
|
||||||
|
$workspace = $workspaceContext->currentWorkspaceForMemberOrFail($user, request());
|
||||||
|
$workspaceId = (int) $workspace->getKey();
|
||||||
|
|
||||||
|
$accessibleTenants = $user->tenants()
|
||||||
|
->where('tenants.workspace_id', $workspaceId)
|
||||||
|
->orderBy('tenants.name')
|
||||||
|
->get()
|
||||||
|
->filter(fn (Tenant $tenant): bool => (int) $tenant->workspace_id === $workspaceId && $user->can('evidence.view', $tenant))
|
||||||
|
->values();
|
||||||
|
|
||||||
|
$this->tenantFilter = is_numeric(request()->query('tenant_id')) ? (int) request()->query('tenant_id') : null;
|
||||||
|
|
||||||
|
$tenantIds = $accessibleTenants->pluck('id')->map(static fn (mixed $id): int => (int) $id)->all();
|
||||||
|
|
||||||
|
$query = EvidenceSnapshot::query()
|
||||||
|
->with('tenant')
|
||||||
|
->where('workspace_id', $workspaceId)
|
||||||
|
->whereIn('tenant_id', $tenantIds)
|
||||||
|
->where('status', 'active')
|
||||||
|
->latest('generated_at');
|
||||||
|
|
||||||
|
if ($this->tenantFilter !== null) {
|
||||||
|
$query->where('tenant_id', $this->tenantFilter);
|
||||||
|
}
|
||||||
|
|
||||||
|
$snapshots = $query->get()->unique('tenant_id')->values();
|
||||||
|
|
||||||
|
$this->rows = $snapshots->map(function (EvidenceSnapshot $snapshot): array {
|
||||||
|
$truth = app(ArtifactTruthPresenter::class)->forEvidenceSnapshot($snapshot);
|
||||||
|
$freshnessSpec = BadgeCatalog::spec(BadgeDomain::GovernanceArtifactFreshness, $truth->freshnessState);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'tenant_name' => $snapshot->tenant?->name ?? 'Unknown tenant',
|
||||||
|
'tenant_id' => (int) $snapshot->tenant_id,
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
'completeness_state' => (string) $snapshot->completeness_state,
|
||||||
|
'generated_at' => $snapshot->generated_at?->toDateTimeString(),
|
||||||
|
'missing_dimensions' => (int) (($snapshot->summary['missing_dimensions'] ?? 0)),
|
||||||
|
'stale_dimensions' => (int) (($snapshot->summary['stale_dimensions'] ?? 0)),
|
||||||
|
'artifact_truth' => [
|
||||||
|
'label' => $truth->primaryLabel,
|
||||||
|
'color' => $truth->primaryBadgeSpec()->color,
|
||||||
|
'icon' => $truth->primaryBadgeSpec()->icon,
|
||||||
|
'explanation' => $truth->primaryExplanation,
|
||||||
|
],
|
||||||
|
'freshness' => [
|
||||||
|
'label' => $freshnessSpec->label,
|
||||||
|
'color' => $freshnessSpec->color,
|
||||||
|
'icon' => $freshnessSpec->icon,
|
||||||
|
],
|
||||||
|
'next_step' => $truth->nextStepText(),
|
||||||
|
'view_url' => $snapshot->tenant
|
||||||
|
? EvidenceSnapshotResource::getUrl('view', ['record' => $snapshot], tenant: $snapshot->tenant)
|
||||||
|
: null,
|
||||||
|
];
|
||||||
|
})->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<Action>
|
||||||
|
*/
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Action::make('clear_filters')
|
||||||
|
->label('Clear filters')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->tenantFilter !== null)
|
||||||
|
->url(route('admin.evidence.overview')),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
503
app/Filament/Pages/Monitoring/FindingExceptionsQueue.php
Normal file
503
app/Filament/Pages/Monitoring/FindingExceptionsQueue.php
Normal file
@ -0,0 +1,503 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Pages\Monitoring;
|
||||||
|
|
||||||
|
use App\Filament\Resources\FindingExceptionResource;
|
||||||
|
use App\Filament\Resources\FindingResource;
|
||||||
|
use App\Models\FindingException;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Models\Workspace;
|
||||||
|
use App\Services\Auth\WorkspaceCapabilityResolver;
|
||||||
|
use App\Services\Findings\FindingExceptionService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Badges\BadgeRenderer;
|
||||||
|
use App\Support\Filament\FilterOptionCatalog;
|
||||||
|
use App\Support\Filament\TablePaginationProfiles;
|
||||||
|
use App\Support\OperateHub\OperateHubShell;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDefaults;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\DateTimePicker;
|
||||||
|
use Filament\Forms\Components\Textarea;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Pages\Page;
|
||||||
|
use Filament\Tables\Columns\TextColumn;
|
||||||
|
use Filament\Tables\Concerns\InteractsWithTable;
|
||||||
|
use Filament\Tables\Contracts\HasTable;
|
||||||
|
use Filament\Tables\Filters\SelectFilter;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class FindingExceptionsQueue extends Page implements HasTable
|
||||||
|
{
|
||||||
|
use InteractsWithTable;
|
||||||
|
|
||||||
|
public ?int $selectedFindingExceptionId = null;
|
||||||
|
|
||||||
|
protected static bool $isDiscovered = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-shield-exclamation';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Monitoring';
|
||||||
|
|
||||||
|
protected static ?string $navigationLabel = 'Finding exceptions';
|
||||||
|
|
||||||
|
protected static ?string $slug = 'finding-exceptions/queue';
|
||||||
|
|
||||||
|
protected static ?string $title = 'Finding Exceptions Queue';
|
||||||
|
|
||||||
|
protected string $view = 'filament.pages.monitoring.finding-exceptions-queue';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<int, Tenant>|null
|
||||||
|
*/
|
||||||
|
private ?array $authorizedTenants = null;
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forPage(ActionSurfaceProfile::RunLog)
|
||||||
|
->withDefaults(new ActionSurfaceDefaults(
|
||||||
|
moreGroupLabel: 'More',
|
||||||
|
exportIsDefaultBulkActionForReadOnly: false,
|
||||||
|
))
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'Header actions keep workspace approval scope visible and expose selected exception review actions.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ViewAction->value)
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'Exception decisions are reviewed one record at a time in v1 and do not expose bulk actions.')
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'Empty state explains when the approval queue is empty and keeps navigation back to tenant findings available.')
|
||||||
|
->satisfy(ActionSurfaceSlot::DetailHeader, 'Selected exception detail exposes approve, reject, and related-record navigation actions in the page header.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canAccess(): bool
|
||||||
|
{
|
||||||
|
if (Filament::getCurrentPanel()?->getId() !== 'admin') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
||||||
|
|
||||||
|
if (! is_int($workspaceId)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$workspace = Workspace::query()->whereKey($workspaceId)->first();
|
||||||
|
|
||||||
|
if (! $workspace instanceof Workspace) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var WorkspaceCapabilityResolver $resolver */
|
||||||
|
$resolver = app(WorkspaceCapabilityResolver::class);
|
||||||
|
|
||||||
|
return $resolver->isMember($user, $workspace)
|
||||||
|
&& $resolver->can($user, $workspace, Capabilities::FINDING_EXCEPTION_APPROVE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function mount(): void
|
||||||
|
{
|
||||||
|
$this->selectedFindingExceptionId = is_numeric(request()->query('exception')) ? (int) request()->query('exception') : null;
|
||||||
|
$this->mountInteractsWithTable();
|
||||||
|
$this->applyRequestedTenantPrefilter();
|
||||||
|
|
||||||
|
if ($this->selectedFindingExceptionId !== null) {
|
||||||
|
$this->selectedFindingException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
$actions = app(OperateHubShell::class)->headerActions(
|
||||||
|
scopeActionName: 'operate_hub_scope_finding_exceptions',
|
||||||
|
returnActionName: 'operate_hub_return_finding_exceptions',
|
||||||
|
);
|
||||||
|
|
||||||
|
$actions[] = Action::make('clear_filters')
|
||||||
|
->label('Clear filters')
|
||||||
|
->icon('heroicon-o-x-mark')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->hasActiveQueueFilters())
|
||||||
|
->action(function (): void {
|
||||||
|
$this->removeTableFilter('tenant_id');
|
||||||
|
$this->removeTableFilter('status');
|
||||||
|
$this->removeTableFilter('current_validity_state');
|
||||||
|
$this->selectedFindingExceptionId = null;
|
||||||
|
$this->resetTable();
|
||||||
|
});
|
||||||
|
|
||||||
|
$actions[] = Action::make('view_tenant_register')
|
||||||
|
->label('View tenant register')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->filteredTenant() instanceof Tenant)
|
||||||
|
->url(function (): ?string {
|
||||||
|
$tenant = $this->filteredTenant();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FindingExceptionResource::getUrl('index', panel: 'tenant', tenant: $tenant);
|
||||||
|
});
|
||||||
|
|
||||||
|
$actions[] = Action::make('clear_selected_exception')
|
||||||
|
->label('Close details')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->selectedFindingExceptionId !== null)
|
||||||
|
->action(function (): void {
|
||||||
|
$this->selectedFindingExceptionId = null;
|
||||||
|
});
|
||||||
|
|
||||||
|
$actions[] = Action::make('open_selected_exception')
|
||||||
|
->label('Open tenant detail')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->selectedFindingExceptionId !== null)
|
||||||
|
->url(fn (): ?string => $this->selectedExceptionUrl());
|
||||||
|
|
||||||
|
$actions[] = Action::make('open_selected_finding')
|
||||||
|
->label('Open finding')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->selectedFindingExceptionId !== null)
|
||||||
|
->url(fn (): ?string => $this->selectedFindingUrl());
|
||||||
|
|
||||||
|
$actions[] = Action::make('approve_selected_exception')
|
||||||
|
->label('Approve exception')
|
||||||
|
->color('success')
|
||||||
|
->visible(fn (): bool => $this->selectedFindingException()?->isPending() ?? false)
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
DateTimePicker::make('effective_from')
|
||||||
|
->label('Effective from')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
DateTimePicker::make('expires_at')
|
||||||
|
->label('Expires at')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
Textarea::make('approval_reason')
|
||||||
|
->label('Approval reason')
|
||||||
|
->rows(3)
|
||||||
|
->maxLength(2000),
|
||||||
|
])
|
||||||
|
->action(function (array $data, FindingExceptionService $service): void {
|
||||||
|
$record = $this->selectedFindingException();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $user instanceof User) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$wasRenewalRequest = $record->isPendingRenewal();
|
||||||
|
$updated = $service->approve($record, $user, $data);
|
||||||
|
$this->selectedFindingExceptionId = (int) $updated->getKey();
|
||||||
|
$this->resetTable();
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title($wasRenewalRequest ? 'Exception renewed' : 'Exception approved')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
});
|
||||||
|
|
||||||
|
$actions[] = Action::make('reject_selected_exception')
|
||||||
|
->label('Reject exception')
|
||||||
|
->color('danger')
|
||||||
|
->visible(fn (): bool => $this->selectedFindingException()?->isPending() ?? false)
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Textarea::make('rejection_reason')
|
||||||
|
->label('Rejection reason')
|
||||||
|
->rows(3)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
])
|
||||||
|
->action(function (array $data, FindingExceptionService $service): void {
|
||||||
|
$record = $this->selectedFindingException();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $user instanceof User) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$wasRenewalRequest = $record->isPendingRenewal();
|
||||||
|
$updated = $service->reject($record, $user, $data);
|
||||||
|
$this->selectedFindingExceptionId = (int) $updated->getKey();
|
||||||
|
$this->resetTable();
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title($wasRenewalRequest ? 'Renewal rejected' : 'Exception rejected')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
});
|
||||||
|
|
||||||
|
return $actions;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->query(fn (): Builder => $this->queueBaseQuery())
|
||||||
|
->defaultSort('requested_at', 'asc')
|
||||||
|
->paginated(TablePaginationProfiles::customPage())
|
||||||
|
->persistFiltersInSession()
|
||||||
|
->persistSearchInSession()
|
||||||
|
->persistSortInSession()
|
||||||
|
->columns([
|
||||||
|
TextColumn::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingExceptionStatus)),
|
||||||
|
TextColumn::make('current_validity_state')
|
||||||
|
->label('Validity')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingRiskGovernanceValidity)),
|
||||||
|
TextColumn::make('tenant.name')
|
||||||
|
->label('Tenant')
|
||||||
|
->searchable(),
|
||||||
|
TextColumn::make('finding_summary')
|
||||||
|
->label('Finding')
|
||||||
|
->state(fn (FindingException $record): string => $record->finding?->resolvedSubjectDisplayName() ?: 'Finding #'.$record->finding_id)
|
||||||
|
->searchable(),
|
||||||
|
TextColumn::make('requester.name')
|
||||||
|
->label('Requested by')
|
||||||
|
->placeholder('—'),
|
||||||
|
TextColumn::make('owner.name')
|
||||||
|
->label('Owner')
|
||||||
|
->placeholder('—'),
|
||||||
|
TextColumn::make('review_due_at')
|
||||||
|
->label('Review due')
|
||||||
|
->dateTime()
|
||||||
|
->placeholder('—')
|
||||||
|
->sortable(),
|
||||||
|
TextColumn::make('expires_at')
|
||||||
|
->label('Expires')
|
||||||
|
->dateTime()
|
||||||
|
->placeholder('—')
|
||||||
|
->sortable(),
|
||||||
|
TextColumn::make('requested_at')
|
||||||
|
->label('Requested')
|
||||||
|
->dateTime()
|
||||||
|
->sortable(),
|
||||||
|
])
|
||||||
|
->filters([
|
||||||
|
SelectFilter::make('tenant_id')
|
||||||
|
->label('Tenant')
|
||||||
|
->options(fn (): array => $this->tenantFilterOptions())
|
||||||
|
->searchable(),
|
||||||
|
SelectFilter::make('status')
|
||||||
|
->options(FilterOptionCatalog::findingExceptionStatuses()),
|
||||||
|
SelectFilter::make('current_validity_state')
|
||||||
|
->label('Validity')
|
||||||
|
->options(FilterOptionCatalog::findingExceptionValidityStates()),
|
||||||
|
])
|
||||||
|
->actions([
|
||||||
|
Action::make('inspect_exception')
|
||||||
|
->label('Inspect exception')
|
||||||
|
->icon('heroicon-o-eye')
|
||||||
|
->color('gray')
|
||||||
|
->action(function (FindingException $record): void {
|
||||||
|
$this->selectedFindingExceptionId = (int) $record->getKey();
|
||||||
|
}),
|
||||||
|
])
|
||||||
|
->bulkActions([])
|
||||||
|
->emptyStateHeading('No exceptions match this queue')
|
||||||
|
->emptyStateDescription('Adjust the current tenant or lifecycle filters to review governed exceptions in this workspace.')
|
||||||
|
->emptyStateIcon('heroicon-o-shield-check')
|
||||||
|
->emptyStateActions([
|
||||||
|
Action::make('clear_filters')
|
||||||
|
->label('Clear filters')
|
||||||
|
->icon('heroicon-o-x-mark')
|
||||||
|
->color('gray')
|
||||||
|
->action(function (): void {
|
||||||
|
$this->removeTableFilter('tenant_id');
|
||||||
|
$this->removeTableFilter('status');
|
||||||
|
$this->removeTableFilter('current_validity_state');
|
||||||
|
$this->selectedFindingExceptionId = null;
|
||||||
|
$this->resetTable();
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function selectedFindingException(): ?FindingException
|
||||||
|
{
|
||||||
|
if (! is_int($this->selectedFindingExceptionId)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$record = $this->queueBaseQuery()
|
||||||
|
->whereKey($this->selectedFindingExceptionId)
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $record;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function selectedExceptionUrl(): ?string
|
||||||
|
{
|
||||||
|
$record = $this->selectedFindingException();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $record->tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FindingExceptionResource::getUrl('view', ['record' => $record], panel: 'tenant', tenant: $record->tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function selectedFindingUrl(): ?string
|
||||||
|
{
|
||||||
|
$record = $this->selectedFindingException();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $record->finding || ! $record->tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FindingResource::getUrl('view', ['record' => $record->finding], panel: 'tenant', tenant: $record->tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, Tenant>
|
||||||
|
*/
|
||||||
|
public function authorizedTenants(): array
|
||||||
|
{
|
||||||
|
if ($this->authorizedTenants !== null) {
|
||||||
|
return $this->authorizedTenants;
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
return $this->authorizedTenants = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
||||||
|
|
||||||
|
if (! is_int($workspaceId)) {
|
||||||
|
return $this->authorizedTenants = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenants = $user->tenants()
|
||||||
|
->where('tenants.workspace_id', $workspaceId)
|
||||||
|
->orderBy('tenants.name')
|
||||||
|
->get();
|
||||||
|
|
||||||
|
return $this->authorizedTenants = $tenants
|
||||||
|
->filter(fn (Tenant $tenant): bool => (int) $tenant->workspace_id === $workspaceId)
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function queueBaseQuery(): Builder
|
||||||
|
{
|
||||||
|
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
||||||
|
$tenantIds = array_values(array_map(
|
||||||
|
static fn (Tenant $tenant): int => (int) $tenant->getKey(),
|
||||||
|
$this->authorizedTenants(),
|
||||||
|
));
|
||||||
|
|
||||||
|
return FindingException::query()
|
||||||
|
->with([
|
||||||
|
'tenant',
|
||||||
|
'requester',
|
||||||
|
'owner',
|
||||||
|
'approver',
|
||||||
|
'finding' => fn ($query) => $query->withSubjectDisplayName(),
|
||||||
|
'decisions.actor',
|
||||||
|
'evidenceReferences',
|
||||||
|
])
|
||||||
|
->where('workspace_id', (int) $workspaceId)
|
||||||
|
->whereIn('tenant_id', $tenantIds === [] ? [-1] : $tenantIds);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private function tenantFilterOptions(): array
|
||||||
|
{
|
||||||
|
return Collection::make($this->authorizedTenants())
|
||||||
|
->mapWithKeys(static fn (Tenant $tenant): array => [
|
||||||
|
(string) $tenant->getKey() => $tenant->name,
|
||||||
|
])
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function applyRequestedTenantPrefilter(): void
|
||||||
|
{
|
||||||
|
$requestedTenant = request()->query('tenant');
|
||||||
|
|
||||||
|
if (! is_string($requestedTenant) && ! is_numeric($requestedTenant)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->authorizedTenants() as $tenant) {
|
||||||
|
if ((string) $tenant->getKey() !== (string) $requestedTenant && (string) $tenant->external_id !== (string) $requestedTenant) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->tableFilters['tenant_id']['value'] = (string) $tenant->getKey();
|
||||||
|
$this->tableDeferredFilters['tenant_id']['value'] = (string) $tenant->getKey();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function filteredTenant(): ?Tenant
|
||||||
|
{
|
||||||
|
$tenantId = $this->currentTenantFilterId();
|
||||||
|
|
||||||
|
if (! is_int($tenantId)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->authorizedTenants() as $tenant) {
|
||||||
|
if ((int) $tenant->getKey() === $tenantId) {
|
||||||
|
return $tenant;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function currentTenantFilterId(): ?int
|
||||||
|
{
|
||||||
|
$tenantFilter = data_get($this->tableFilters, 'tenant_id.value');
|
||||||
|
|
||||||
|
if (! is_numeric($tenantFilter)) {
|
||||||
|
$tenantFilter = data_get(session()->get($this->getTableFiltersSessionKey(), []), 'tenant_id.value');
|
||||||
|
}
|
||||||
|
|
||||||
|
return is_numeric($tenantFilter) ? (int) $tenantFilter : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function hasActiveQueueFilters(): bool
|
||||||
|
{
|
||||||
|
return $this->currentTenantFilterId() !== null
|
||||||
|
|| is_string(data_get($this->tableFilters, 'status.value'))
|
||||||
|
|| is_string(data_get($this->tableFilters, 'current_validity_state.value'));
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -13,6 +13,7 @@
|
|||||||
use App\Support\OperateHub\OperateHubShell;
|
use App\Support\OperateHub\OperateHubShell;
|
||||||
use App\Support\OperationRunOutcome;
|
use App\Support\OperationRunOutcome;
|
||||||
use App\Support\OperationRunStatus;
|
use App\Support\OperationRunStatus;
|
||||||
|
use App\Support\Operations\OperationLifecyclePolicy;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use BackedEnum;
|
use BackedEnum;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
@ -165,6 +166,68 @@ public function table(Table $table): Table
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{likely_stale:int,reconciled:int}
|
||||||
|
*/
|
||||||
|
public function lifecycleVisibilitySummary(): array
|
||||||
|
{
|
||||||
|
$baseQuery = $this->scopedSummaryQuery();
|
||||||
|
|
||||||
|
if (! $baseQuery instanceof Builder) {
|
||||||
|
return [
|
||||||
|
'likely_stale' => 0,
|
||||||
|
'reconciled' => 0,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
$reconciled = (clone $baseQuery)
|
||||||
|
->whereNotNull('context->reconciliation->reconciled_at')
|
||||||
|
->count();
|
||||||
|
|
||||||
|
$policy = app(OperationLifecyclePolicy::class);
|
||||||
|
$likelyStale = (clone $baseQuery)
|
||||||
|
->whereIn('status', [
|
||||||
|
OperationRunStatus::Queued->value,
|
||||||
|
OperationRunStatus::Running->value,
|
||||||
|
])
|
||||||
|
->where(function (Builder $query) use ($policy): void {
|
||||||
|
foreach ($policy->coveredTypeNames() as $type) {
|
||||||
|
$query->orWhere(function (Builder $typeQuery) use ($policy, $type): void {
|
||||||
|
$typeQuery
|
||||||
|
->where('type', $type)
|
||||||
|
->where(function (Builder $stateQuery) use ($policy, $type): void {
|
||||||
|
$stateQuery
|
||||||
|
->where(function (Builder $queuedQuery) use ($policy, $type): void {
|
||||||
|
$queuedQuery
|
||||||
|
->where('status', OperationRunStatus::Queued->value)
|
||||||
|
->whereNull('started_at')
|
||||||
|
->where('created_at', '<=', now()->subSeconds($policy->queuedStaleAfterSeconds($type)));
|
||||||
|
})
|
||||||
|
->orWhere(function (Builder $runningQuery) use ($policy, $type): void {
|
||||||
|
$runningQuery
|
||||||
|
->where('status', OperationRunStatus::Running->value)
|
||||||
|
->where(function (Builder $startedAtQuery) use ($policy, $type): void {
|
||||||
|
$startedAtQuery
|
||||||
|
->where('started_at', '<=', now()->subSeconds($policy->runningStaleAfterSeconds($type)))
|
||||||
|
->orWhere(function (Builder $fallbackQuery) use ($policy, $type): void {
|
||||||
|
$fallbackQuery
|
||||||
|
->whereNull('started_at')
|
||||||
|
->where('created_at', '<=', now()->subSeconds($policy->runningStaleAfterSeconds($type)));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
->count();
|
||||||
|
|
||||||
|
return [
|
||||||
|
'likely_stale' => $likelyStale,
|
||||||
|
'reconciled' => $reconciled,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
private function applyActiveTab(Builder $query): Builder
|
private function applyActiveTab(Builder $query): Builder
|
||||||
{
|
{
|
||||||
return match ($this->activeTab) {
|
return match ($this->activeTab) {
|
||||||
@ -172,6 +235,9 @@ private function applyActiveTab(Builder $query): Builder
|
|||||||
OperationRunStatus::Queued->value,
|
OperationRunStatus::Queued->value,
|
||||||
OperationRunStatus::Running->value,
|
OperationRunStatus::Running->value,
|
||||||
]),
|
]),
|
||||||
|
'blocked' => $query
|
||||||
|
->where('status', OperationRunStatus::Completed->value)
|
||||||
|
->where('outcome', OperationRunOutcome::Blocked->value),
|
||||||
'succeeded' => $query
|
'succeeded' => $query
|
||||||
->where('status', OperationRunStatus::Completed->value)
|
->where('status', OperationRunStatus::Completed->value)
|
||||||
->where('outcome', OperationRunOutcome::Succeeded->value),
|
->where('outcome', OperationRunOutcome::Succeeded->value),
|
||||||
@ -184,4 +250,26 @@ private function applyActiveTab(Builder $query): Builder
|
|||||||
default => $query,
|
default => $query,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function scopedSummaryQuery(): ?Builder
|
||||||
|
{
|
||||||
|
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
||||||
|
|
||||||
|
if (! $workspaceId) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenantFilter = data_get($this->tableFilters, 'tenant_id.value');
|
||||||
|
|
||||||
|
if (! is_numeric($tenantFilter)) {
|
||||||
|
$tenantFilter = data_get(session()->get($this->getTableFiltersSessionKey(), []), 'tenant_id.value');
|
||||||
|
}
|
||||||
|
|
||||||
|
return OperationRun::query()
|
||||||
|
->where('workspace_id', (int) $workspaceId)
|
||||||
|
->when(
|
||||||
|
is_numeric($tenantFilter),
|
||||||
|
fn (Builder $query): Builder => $query->where('tenant_id', (int) $tenantFilter),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -11,6 +11,7 @@
|
|||||||
use App\Services\Auth\CapabilityResolver;
|
use App\Services\Auth\CapabilityResolver;
|
||||||
use App\Services\Auth\WorkspaceCapabilityResolver;
|
use App\Services\Auth\WorkspaceCapabilityResolver;
|
||||||
use App\Services\Baselines\BaselineEvidenceCaptureResumeService;
|
use App\Services\Baselines\BaselineEvidenceCaptureResumeService;
|
||||||
|
use App\Services\Tenants\TenantOperabilityService;
|
||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
use App\Support\Navigation\CanonicalNavigationContext;
|
use App\Support\Navigation\CanonicalNavigationContext;
|
||||||
use App\Support\OperateHub\OperateHubShell;
|
use App\Support\OperateHub\OperateHubShell;
|
||||||
@ -18,8 +19,11 @@
|
|||||||
use App\Support\OpsUx\OperationUxPresenter;
|
use App\Support\OpsUx\OperationUxPresenter;
|
||||||
use App\Support\OpsUx\OpsUxBrowserEvents;
|
use App\Support\OpsUx\OpsUxBrowserEvents;
|
||||||
use App\Support\OpsUx\RunDetailPolling;
|
use App\Support\OpsUx\RunDetailPolling;
|
||||||
|
use App\Support\ReasonTranslation\ReasonPresenter;
|
||||||
use App\Support\RedactionIntegrity;
|
use App\Support\RedactionIntegrity;
|
||||||
use App\Support\Tenants\ReferencedTenantLifecyclePresentation;
|
use App\Support\Tenants\ReferencedTenantLifecyclePresentation;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
use Filament\Actions\ActionGroup;
|
use Filament\Actions\ActionGroup;
|
||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
@ -103,14 +107,7 @@ protected function getHeaderActions(): array
|
|||||||
return $actions;
|
return $actions;
|
||||||
}
|
}
|
||||||
|
|
||||||
$user = auth()->user();
|
$related = OperationRunLinks::related($this->run, $this->relatedLinksTenant());
|
||||||
$tenant = $this->run->tenant;
|
|
||||||
|
|
||||||
if ($tenant instanceof Tenant && (! $user instanceof User || ! app(CapabilityResolver::class)->isMember($user, $tenant))) {
|
|
||||||
$tenant = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$related = OperationRunLinks::related($this->run, $tenant);
|
|
||||||
|
|
||||||
$relatedActions = [];
|
$relatedActions = [];
|
||||||
|
|
||||||
@ -164,6 +161,62 @@ public function redactionIntegrityNote(): ?string
|
|||||||
return isset($this->run) ? RedactionIntegrity::noteForRun($this->run) : null;
|
return isset($this->run) ? RedactionIntegrity::noteForRun($this->run) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{tone: string, title: string, body: string}|null
|
||||||
|
*/
|
||||||
|
public function blockedExecutionBanner(): ?array
|
||||||
|
{
|
||||||
|
if (! isset($this->run) || (string) $this->run->outcome !== 'blocked') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(ReasonPresenter::class)->forOperationRun($this->run, 'run_detail');
|
||||||
|
$lines = $reasonEnvelope?->toBodyLines() ?? [
|
||||||
|
OperationUxPresenter::surfaceFailureDetail($this->run) ?? 'The queued run was refused before side effects could begin.',
|
||||||
|
OperationUxPresenter::surfaceGuidance($this->run) ?? 'Review the blocked prerequisite before retrying.',
|
||||||
|
];
|
||||||
|
|
||||||
|
return [
|
||||||
|
'tone' => 'amber',
|
||||||
|
'title' => 'Blocked by prerequisite',
|
||||||
|
'body' => implode(' ', $lines),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{tone: string, title: string, body: string}|null
|
||||||
|
*/
|
||||||
|
public function lifecycleBanner(): ?array
|
||||||
|
{
|
||||||
|
if (! isset($this->run)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$attention = OperationUxPresenter::lifecycleAttentionSummary($this->run);
|
||||||
|
|
||||||
|
if ($attention === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$detail = OperationUxPresenter::surfaceFailureDetail($this->run) ?? 'Lifecycle truth needs operator review.';
|
||||||
|
$guidance = OperationUxPresenter::surfaceGuidance($this->run);
|
||||||
|
$body = $guidance !== null ? $detail.' '.$guidance : $detail;
|
||||||
|
|
||||||
|
return match ($this->run->freshnessState()->value) {
|
||||||
|
'likely_stale' => [
|
||||||
|
'tone' => 'amber',
|
||||||
|
'title' => 'Likely stale run',
|
||||||
|
'body' => $body,
|
||||||
|
],
|
||||||
|
'reconciled_failed' => [
|
||||||
|
'tone' => 'rose',
|
||||||
|
'title' => 'Automatically reconciled',
|
||||||
|
'body' => $body,
|
||||||
|
],
|
||||||
|
default => null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array{tone: string, title: string, body: string}|null
|
* @return array{tone: string, title: string, body: string}|null
|
||||||
*/
|
*/
|
||||||
@ -372,4 +425,30 @@ private function canResumeCapture(): bool
|
|||||||
return $resolver->isMember($user, $workspace)
|
return $resolver->isMember($user, $workspace)
|
||||||
&& $resolver->can($user, $workspace, Capabilities::WORKSPACE_BASELINES_MANAGE);
|
&& $resolver->can($user, $workspace, Capabilities::WORKSPACE_BASELINES_MANAGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function relatedLinksTenant(): ?Tenant
|
||||||
|
{
|
||||||
|
if (! isset($this->run)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = auth()->user();
|
||||||
|
$tenant = $this->run->tenant;
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! app(CapabilityResolver::class)->isMember($user, $tenant)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::SelectorEligibility,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: (int) ($this->run->workspace_id ?? 0),
|
||||||
|
lane: TenantInteractionLane::StandardActiveOperating,
|
||||||
|
)->allowed ? $tenant : null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
328
app/Filament/Pages/Reviews/ReviewRegister.php
Normal file
328
app/Filament/Pages/Reviews/ReviewRegister.php
Normal file
@ -0,0 +1,328 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Pages\Reviews;
|
||||||
|
|
||||||
|
use App\Filament\Resources\TenantReviewResource;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\TenantReview;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Models\Workspace;
|
||||||
|
use App\Services\TenantReviews\TenantReviewRegisterService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Badges\BadgeRenderer;
|
||||||
|
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
||||||
|
use App\Support\Filament\FilterPresets;
|
||||||
|
use App\Support\Filament\TablePaginationProfiles;
|
||||||
|
use App\Support\TenantReviewCompletenessState;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Pages\Page;
|
||||||
|
use Filament\Tables\Columns\TextColumn;
|
||||||
|
use Filament\Tables\Concerns\InteractsWithTable;
|
||||||
|
use Filament\Tables\Contracts\HasTable;
|
||||||
|
use Filament\Tables\Filters\SelectFilter;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class ReviewRegister extends Page implements HasTable
|
||||||
|
{
|
||||||
|
use InteractsWithTable;
|
||||||
|
|
||||||
|
protected static bool $isDiscovered = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-document-magnifying-glass';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Reporting';
|
||||||
|
|
||||||
|
protected static ?string $navigationLabel = 'Reviews';
|
||||||
|
|
||||||
|
protected static ?string $title = 'Review Register';
|
||||||
|
|
||||||
|
protected static ?string $slug = 'reviews';
|
||||||
|
|
||||||
|
protected string $view = 'filament.pages.reviews.review-register';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<int, Tenant>|null
|
||||||
|
*/
|
||||||
|
private ?array $authorizedTenants = null;
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forPage(ActionSurfaceProfile::RunLog)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'Header actions provide a single Clear filters action for the canonical review register.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ClickableRow->value)
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'The review register does not expose bulk actions in the first slice.')
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'The empty state keeps exactly one Clear filters CTA.')
|
||||||
|
->exempt(ActionSurfaceSlot::DetailHeader, 'Row navigation returns to the tenant-scoped review detail rather than opening an inline canonical detail panel.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function mount(): void
|
||||||
|
{
|
||||||
|
$this->authorizePageAccess();
|
||||||
|
|
||||||
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
||||||
|
$this->getTableFiltersSessionKey(),
|
||||||
|
['status', 'published_state', 'completeness_state'],
|
||||||
|
request(),
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->applyRequestedTenantPrefilter();
|
||||||
|
$this->mountInteractsWithTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Action::make('clear_filters')
|
||||||
|
->label('Clear filters')
|
||||||
|
->icon('heroicon-o-x-mark')
|
||||||
|
->color('gray')
|
||||||
|
->visible(fn (): bool => $this->hasActiveFilters())
|
||||||
|
->action(function (): void {
|
||||||
|
$this->resetTable();
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->query(fn (): Builder => $this->registerQuery())
|
||||||
|
->defaultSort('generated_at', 'desc')
|
||||||
|
->paginated(TablePaginationProfiles::customPage())
|
||||||
|
->persistFiltersInSession()
|
||||||
|
->persistSearchInSession()
|
||||||
|
->persistSortInSession()
|
||||||
|
->recordUrl(fn (TenantReview $record): string => TenantReviewResource::tenantScopedUrl('view', ['record' => $record], $record->tenant, 'tenant'))
|
||||||
|
->columns([
|
||||||
|
TextColumn::make('tenant.name')->label('Tenant')->searchable(),
|
||||||
|
TextColumn::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewStatus)),
|
||||||
|
TextColumn::make('artifact_truth')
|
||||||
|
->label('Artifact truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->primaryLabel)
|
||||||
|
->color(fn (TenantReview $record): string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->primaryBadgeSpec()->color)
|
||||||
|
->icon(fn (TenantReview $record): ?string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->primaryBadgeSpec()->icon)
|
||||||
|
->iconColor(fn (TenantReview $record): ?string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->primaryBadgeSpec()->iconColor)
|
||||||
|
->description(fn (TenantReview $record): ?string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->primaryExplanation)
|
||||||
|
->wrap(),
|
||||||
|
TextColumn::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewCompleteness)),
|
||||||
|
TextColumn::make('generated_at')->dateTime()->placeholder('—')->sortable(),
|
||||||
|
TextColumn::make('published_at')->dateTime()->placeholder('—')->sortable(),
|
||||||
|
TextColumn::make('publication_truth')
|
||||||
|
->label('Publication')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
app(ArtifactTruthPresenter::class)->forTenantReview($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->label)
|
||||||
|
->color(fn (TenantReview $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
app(ArtifactTruthPresenter::class)->forTenantReview($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->color)
|
||||||
|
->icon(fn (TenantReview $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
app(ArtifactTruthPresenter::class)->forTenantReview($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->icon)
|
||||||
|
->iconColor(fn (TenantReview $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
app(ArtifactTruthPresenter::class)->forTenantReview($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->iconColor),
|
||||||
|
TextColumn::make('artifact_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => app(ArtifactTruthPresenter::class)->forTenantReview($record)->nextStepText())
|
||||||
|
->wrap(),
|
||||||
|
])
|
||||||
|
->filters([
|
||||||
|
SelectFilter::make('tenant_id')
|
||||||
|
->label('Tenant')
|
||||||
|
->options(fn (): array => $this->tenantFilterOptions())
|
||||||
|
->default(fn (): ?string => $this->defaultTenantFilter())
|
||||||
|
->searchable(),
|
||||||
|
SelectFilter::make('status')
|
||||||
|
->options([
|
||||||
|
'draft' => 'Draft',
|
||||||
|
'ready' => 'Ready',
|
||||||
|
'published' => 'Published',
|
||||||
|
'archived' => 'Archived',
|
||||||
|
'superseded' => 'Superseded',
|
||||||
|
'failed' => 'Failed',
|
||||||
|
]),
|
||||||
|
SelectFilter::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->options(BadgeCatalog::options(BadgeDomain::TenantReviewCompleteness, TenantReviewCompletenessState::values())),
|
||||||
|
SelectFilter::make('published_state')
|
||||||
|
->label('Published state')
|
||||||
|
->options([
|
||||||
|
'published' => 'Published',
|
||||||
|
'unpublished' => 'Not published',
|
||||||
|
])
|
||||||
|
->query(function (Builder $query, array $data): Builder {
|
||||||
|
return match ($data['value'] ?? null) {
|
||||||
|
'published' => $query->whereNotNull('published_at'),
|
||||||
|
'unpublished' => $query->whereNull('published_at'),
|
||||||
|
default => $query,
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
FilterPresets::dateRange('review_date', 'Review date', 'generated_at'),
|
||||||
|
])
|
||||||
|
->actions([
|
||||||
|
Action::make('view_review')
|
||||||
|
->label('View review')
|
||||||
|
->url(fn (TenantReview $record): string => TenantReviewResource::tenantScopedUrl('view', ['record' => $record], $record->tenant, 'tenant')),
|
||||||
|
Action::make('export_executive_pack')
|
||||||
|
->label('Export executive pack')
|
||||||
|
->icon('heroicon-o-arrow-down-tray')
|
||||||
|
->visible(fn (TenantReview $record): bool => auth()->user() instanceof User
|
||||||
|
&& auth()->user()->can(Capabilities::TENANT_REVIEW_MANAGE, $record->tenant)
|
||||||
|
&& in_array($record->status, ['ready', 'published'], true))
|
||||||
|
->action(fn (TenantReview $record): mixed => TenantReviewResource::executeExport($record)),
|
||||||
|
])
|
||||||
|
->bulkActions([])
|
||||||
|
->emptyStateHeading('No review records match this view')
|
||||||
|
->emptyStateDescription('Clear the current filters to return to the full review register for your entitled tenants.')
|
||||||
|
->emptyStateActions([
|
||||||
|
Action::make('clear_filters_empty')
|
||||||
|
->label('Clear filters')
|
||||||
|
->icon('heroicon-o-x-mark')
|
||||||
|
->color('gray')
|
||||||
|
->action(fn (): mixed => $this->resetTable()),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, Tenant>
|
||||||
|
*/
|
||||||
|
public function authorizedTenants(): array
|
||||||
|
{
|
||||||
|
if ($this->authorizedTenants !== null) {
|
||||||
|
return $this->authorizedTenants;
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = auth()->user();
|
||||||
|
$workspace = $this->workspace();
|
||||||
|
|
||||||
|
if (! $user instanceof User || ! $workspace instanceof Workspace) {
|
||||||
|
return $this->authorizedTenants = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->authorizedTenants = app(TenantReviewRegisterService::class)->authorizedTenants($user, $workspace);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function authorizePageAccess(): void
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
$workspace = $this->workspace();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $workspace instanceof Workspace) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
|
||||||
|
$service = app(TenantReviewRegisterService::class);
|
||||||
|
|
||||||
|
if (! $service->canAccessWorkspace($user, $workspace)) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->authorizedTenants() === []) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function registerQuery(): Builder
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
$workspace = $this->workspace();
|
||||||
|
|
||||||
|
if (! $user instanceof User || ! $workspace instanceof Workspace) {
|
||||||
|
return TenantReview::query()->whereRaw('1 = 0');
|
||||||
|
}
|
||||||
|
|
||||||
|
return app(TenantReviewRegisterService::class)->query($user, $workspace);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private function tenantFilterOptions(): array
|
||||||
|
{
|
||||||
|
return collect($this->authorizedTenants())
|
||||||
|
->mapWithKeys(static fn (Tenant $tenant): array => [
|
||||||
|
(string) $tenant->getKey() => $tenant->name,
|
||||||
|
])
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function defaultTenantFilter(): ?string
|
||||||
|
{
|
||||||
|
$tenantId = app(WorkspaceContext::class)->lastTenantId(request());
|
||||||
|
|
||||||
|
return is_int($tenantId) && array_key_exists($tenantId, $this->authorizedTenants())
|
||||||
|
? (string) $tenantId
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function applyRequestedTenantPrefilter(): void
|
||||||
|
{
|
||||||
|
$requestedTenant = request()->query('tenant');
|
||||||
|
|
||||||
|
if (! is_string($requestedTenant) && ! is_numeric($requestedTenant)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->authorizedTenants() as $tenant) {
|
||||||
|
if ((string) $tenant->getKey() !== (string) $requestedTenant && (string) $tenant->external_id !== (string) $requestedTenant) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->tableFilters['tenant_id']['value'] = (string) $tenant->getKey();
|
||||||
|
$this->tableDeferredFilters['tenant_id']['value'] = (string) $tenant->getKey();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function hasActiveFilters(): bool
|
||||||
|
{
|
||||||
|
$filters = array_filter((array) $this->tableFilters);
|
||||||
|
|
||||||
|
return $filters !== [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function workspace(): ?Workspace
|
||||||
|
{
|
||||||
|
$workspaceId = app(WorkspaceContext::class)->currentWorkspaceId(request());
|
||||||
|
|
||||||
|
return is_numeric($workspaceId)
|
||||||
|
? Workspace::query()->whereKey((int) $workspaceId)->first()
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -12,6 +12,8 @@
|
|||||||
use App\Services\Intune\TenantRequiredPermissionsViewModelBuilder;
|
use App\Services\Intune\TenantRequiredPermissionsViewModelBuilder;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use Filament\Pages\Page;
|
use Filament\Pages\Page;
|
||||||
|
use Livewire\Attributes\Locked;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
|
||||||
class TenantRequiredPermissions extends Page
|
class TenantRequiredPermissions extends Page
|
||||||
{
|
{
|
||||||
@ -41,7 +43,8 @@ class TenantRequiredPermissions extends Page
|
|||||||
*/
|
*/
|
||||||
public array $viewModel = [];
|
public array $viewModel = [];
|
||||||
|
|
||||||
public ?Tenant $scopedTenant = null;
|
#[Locked]
|
||||||
|
public ?int $scopedTenantId = null;
|
||||||
|
|
||||||
public static function canAccess(): bool
|
public static function canAccess(): bool
|
||||||
{
|
{
|
||||||
@ -50,7 +53,7 @@ public static function canAccess(): bool
|
|||||||
|
|
||||||
public function currentTenant(): ?Tenant
|
public function currentTenant(): ?Tenant
|
||||||
{
|
{
|
||||||
return $this->scopedTenant;
|
return $this->trustedScopedTenant();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function mount(): void
|
public function mount(): void
|
||||||
@ -61,7 +64,7 @@ public function mount(): void
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->scopedTenant = $tenant;
|
$this->scopedTenantId = (int) $tenant->getKey();
|
||||||
$this->heading = $tenant->getFilamentName();
|
$this->heading = $tenant->getFilamentName();
|
||||||
$this->subheading = 'Required permissions';
|
$this->subheading = 'Required permissions';
|
||||||
|
|
||||||
@ -143,7 +146,7 @@ public function resetFilters(): void
|
|||||||
|
|
||||||
private function refreshViewModel(): void
|
private function refreshViewModel(): void
|
||||||
{
|
{
|
||||||
$tenant = $this->scopedTenant;
|
$tenant = $this->trustedScopedTenant();
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
$this->viewModel = [];
|
$this->viewModel = [];
|
||||||
@ -172,7 +175,7 @@ private function refreshViewModel(): void
|
|||||||
|
|
||||||
public function reRunVerificationUrl(): string
|
public function reRunVerificationUrl(): string
|
||||||
{
|
{
|
||||||
$tenant = $this->scopedTenant;
|
$tenant = $this->trustedScopedTenant();
|
||||||
|
|
||||||
if ($tenant instanceof Tenant) {
|
if ($tenant instanceof Tenant) {
|
||||||
return TenantResource::getUrl('view', ['record' => $tenant]);
|
return TenantResource::getUrl('view', ['record' => $tenant]);
|
||||||
@ -183,7 +186,7 @@ public function reRunVerificationUrl(): string
|
|||||||
|
|
||||||
public function manageProviderConnectionUrl(): ?string
|
public function manageProviderConnectionUrl(): ?string
|
||||||
{
|
{
|
||||||
$tenant = $this->scopedTenant;
|
$tenant = $this->trustedScopedTenant();
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
return null;
|
return null;
|
||||||
@ -234,4 +237,47 @@ private static function hasScopedTenantAccess(?Tenant $tenant): bool
|
|||||||
|
|
||||||
return $user->canAccessTenant($tenant);
|
return $user->canAccessTenant($tenant);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function trustedScopedTenant(): ?Tenant
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$workspaceContext = app(WorkspaceContext::class);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$workspace = $workspaceContext->currentWorkspaceForMemberOrFail($user, request());
|
||||||
|
} catch (NotFoundHttpException) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$routeTenant = static::resolveScopedTenant();
|
||||||
|
|
||||||
|
if ($routeTenant instanceof Tenant) {
|
||||||
|
try {
|
||||||
|
return $workspaceContext->ensureTenantAccessibleInCurrentWorkspace($routeTenant, $user, request());
|
||||||
|
} catch (NotFoundHttpException) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->scopedTenantId === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = Tenant::query()->withTrashed()->whereKey($this->scopedTenantId)->first();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
return $workspaceContext->ensureTenantAccessibleInCurrentWorkspace($tenant, $user, request());
|
||||||
|
} catch (NotFoundHttpException) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -39,6 +39,7 @@
|
|||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
use App\Support\Badges\BadgeCatalog;
|
use App\Support\Badges\BadgeCatalog;
|
||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Livewire\TrustedState\TrustedStateResolver;
|
||||||
use App\Support\Onboarding\OnboardingCheckpoint;
|
use App\Support\Onboarding\OnboardingCheckpoint;
|
||||||
use App\Support\Onboarding\OnboardingDraftStage;
|
use App\Support\Onboarding\OnboardingDraftStage;
|
||||||
use App\Support\Onboarding\OnboardingLifecycleState;
|
use App\Support\Onboarding\OnboardingLifecycleState;
|
||||||
@ -51,7 +52,9 @@
|
|||||||
use App\Support\Providers\ProviderConsentStatus;
|
use App\Support\Providers\ProviderConsentStatus;
|
||||||
use App\Support\Providers\ProviderReasonCodes;
|
use App\Support\Providers\ProviderReasonCodes;
|
||||||
use App\Support\Providers\ProviderVerificationStatus;
|
use App\Support\Providers\ProviderVerificationStatus;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
use App\Support\Tenants\TenantLifecyclePresentation;
|
use App\Support\Tenants\TenantLifecyclePresentation;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
use App\Support\Verification\VerificationAssistViewModelBuilder;
|
use App\Support\Verification\VerificationAssistViewModelBuilder;
|
||||||
use App\Support\Verification\VerificationCheckStatus;
|
use App\Support\Verification\VerificationCheckStatus;
|
||||||
use App\Support\Verification\VerificationReportOverall;
|
use App\Support\Verification\VerificationReportOverall;
|
||||||
@ -86,6 +89,7 @@
|
|||||||
use Illuminate\Support\Facades\DB;
|
use Illuminate\Support\Facades\DB;
|
||||||
use Illuminate\Validation\ValidationException;
|
use Illuminate\Validation\ValidationException;
|
||||||
use InvalidArgumentException;
|
use InvalidArgumentException;
|
||||||
|
use Livewire\Attributes\Locked;
|
||||||
use RuntimeException;
|
use RuntimeException;
|
||||||
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
|
||||||
@ -121,8 +125,14 @@ protected function getLayoutData(): array
|
|||||||
|
|
||||||
public ?Tenant $managedTenant = null;
|
public ?Tenant $managedTenant = null;
|
||||||
|
|
||||||
|
#[Locked]
|
||||||
|
public ?int $managedTenantId = null;
|
||||||
|
|
||||||
public ?TenantOnboardingSession $onboardingSession = null;
|
public ?TenantOnboardingSession $onboardingSession = null;
|
||||||
|
|
||||||
|
#[Locked]
|
||||||
|
public ?int $onboardingSessionId = null;
|
||||||
|
|
||||||
public ?int $onboardingSessionVersion = null;
|
public ?int $onboardingSessionVersion = null;
|
||||||
|
|
||||||
public ?int $selectedProviderConnectionId = null;
|
public ?int $selectedProviderConnectionId = null;
|
||||||
@ -149,6 +159,8 @@ protected function getLayoutData(): array
|
|||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
$actions = [];
|
$actions = [];
|
||||||
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
if (isset($this->workspace)) {
|
if (isset($this->workspace)) {
|
||||||
$actions[] = Action::make('back_to_workspace')
|
$actions[] = Action::make('back_to_workspace')
|
||||||
@ -168,10 +180,10 @@ protected function getHeaderActions(): array
|
|||||||
$actions[] = Action::make('view_linked_tenant')
|
$actions[] = Action::make('view_linked_tenant')
|
||||||
->label($this->linkedTenantActionLabel())
|
->label($this->linkedTenantActionLabel())
|
||||||
->color('gray')
|
->color('gray')
|
||||||
->url(TenantResource::getUrl('view', ['record' => $this->managedTenant]));
|
->url($tenant instanceof Tenant ? TenantResource::getUrl('view', ['record' => $tenant]) : null);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->canResumeDraft($this->onboardingSession)) {
|
if ($this->canResumeDraft($draft)) {
|
||||||
$actions[] = Action::make('cancel_onboarding_draft')
|
$actions[] = Action::make('cancel_onboarding_draft')
|
||||||
->label('Cancel draft')
|
->label('Cancel draft')
|
||||||
->color('danger')
|
->color('danger')
|
||||||
@ -182,7 +194,7 @@ protected function getHeaderActions(): array
|
|||||||
->action(fn () => $this->cancelOnboardingDraft());
|
->action(fn () => $this->cancelOnboardingDraft());
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->canDeleteDraft($this->onboardingSession)) {
|
if ($this->canDeleteDraft($draft)) {
|
||||||
$actions[] = Action::make('delete_onboarding_draft_header')
|
$actions[] = Action::make('delete_onboarding_draft_header')
|
||||||
->label('Delete draft')
|
->label('Delete draft')
|
||||||
->color('danger')
|
->color('danger')
|
||||||
@ -200,27 +212,36 @@ protected function getHeaderActions(): array
|
|||||||
private function canViewLinkedTenant(): bool
|
private function canViewLinkedTenant(): bool
|
||||||
{
|
{
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
if (! $user instanceof User || ! $this->managedTenant instanceof Tenant) {
|
if (! $user instanceof User || ! $tenant instanceof Tenant) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! $user->canAccessTenant($this->managedTenant)) {
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return app(TenantOperabilityService::class)->canViewTenantSurface($this->managedTenant);
|
return app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::TenantBoundViewability,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: (int) $this->workspace->getKey(),
|
||||||
|
lane: TenantInteractionLane::AdministrativeManagement,
|
||||||
|
)->allowed;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function linkedTenantActionLabel(): string
|
private function linkedTenantActionLabel(): string
|
||||||
{
|
{
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
return 'View tenant';
|
return 'View tenant';
|
||||||
}
|
}
|
||||||
|
|
||||||
return sprintf(
|
return sprintf(
|
||||||
'View tenant (%s)',
|
'View tenant (%s)',
|
||||||
TenantLifecyclePresentation::fromTenant($this->managedTenant)->label,
|
TenantLifecyclePresentation::fromTenant($tenant)->label,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -704,7 +725,7 @@ private function loadOnboardingDraft(User $user, TenantOnboardingSession|int|str
|
|||||||
$tenant = $draft->tenant;
|
$tenant = $draft->tenant;
|
||||||
|
|
||||||
if ($tenant instanceof Tenant && (int) $tenant->workspace_id === (int) $this->workspace->getKey()) {
|
if ($tenant instanceof Tenant && (int) $tenant->workspace_id === (int) $this->workspace->getKey()) {
|
||||||
$this->managedTenant = $tenant;
|
$this->setManagedTenant($tenant);
|
||||||
}
|
}
|
||||||
|
|
||||||
$providerConnectionId = $draft->state['provider_connection_id'] ?? null;
|
$providerConnectionId = $draft->state['provider_connection_id'] ?? null;
|
||||||
@ -793,7 +814,9 @@ private function draftPickerSchema(): array
|
|||||||
*/
|
*/
|
||||||
private function resumeContextSchema(): array
|
private function resumeContextSchema(): array
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -806,19 +829,19 @@ private function resumeContextSchema(): array
|
|||||||
->schema([
|
->schema([
|
||||||
Text::make('Tenant')
|
Text::make('Tenant')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => $this->draftTitle($this->onboardingSession))
|
Text::make(fn () => $this->draftTitle($this->currentOnboardingSessionRecord() ?? $draft))
|
||||||
->weight(FontWeight::SemiBold),
|
->weight(FontWeight::SemiBold),
|
||||||
Text::make('Current stage')
|
Text::make('Current stage')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => $this->draftStageLabel($this->onboardingSession))
|
Text::make(fn () => $this->draftStageLabel($this->currentOnboardingSessionRecord() ?? $draft))
|
||||||
->badge()
|
->badge()
|
||||||
->color(fn (): string => $this->draftStageColor($this->onboardingSession)),
|
->color(fn () => $this->draftStageColor($this->currentOnboardingSessionRecord() ?? $draft)),
|
||||||
Text::make('Started by')
|
Text::make('Started by')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => $this->onboardingSession?->startedByUser?->name ?? 'Unknown'),
|
Text::make(fn () => ($this->currentOnboardingSessionRecord() ?? $draft)?->startedByUser?->name ?? 'Unknown'),
|
||||||
Text::make('Last updated by')
|
Text::make('Last updated by')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => $this->onboardingSession?->updatedByUser?->name ?? 'Unknown'),
|
Text::make(fn () => ($this->currentOnboardingSessionRecord() ?? $draft)?->updatedByUser?->name ?? 'Unknown'),
|
||||||
]),
|
]),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@ -828,11 +851,13 @@ private function resumeContextSchema(): array
|
|||||||
*/
|
*/
|
||||||
private function nonResumableSummarySchema(): array
|
private function nonResumableSummarySchema(): array
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
$statusLabel = $this->onboardingSession->status()->label();
|
$statusLabel = $draft->status()->label();
|
||||||
|
|
||||||
return [
|
return [
|
||||||
Callout::make("This onboarding draft is {$statusLabel}.")
|
Callout::make("This onboarding draft is {$statusLabel}.")
|
||||||
@ -847,16 +872,16 @@ private function nonResumableSummarySchema(): array
|
|||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => $statusLabel)
|
Text::make(fn (): string => $statusLabel)
|
||||||
->badge()
|
->badge()
|
||||||
->color(fn (): string => $this->draftStatusColor($this->onboardingSession)),
|
->color(fn () => $this->draftStatusColor($this->currentOnboardingSessionRecord() ?? $draft)),
|
||||||
Text::make('Primary domain')
|
Text::make('Primary domain')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => (string) (($this->onboardingSession?->state['primary_domain'] ?? null) ?: '—')),
|
Text::make(fn () => (string) ((($this->currentOnboardingSessionRecord() ?? $draft)?->state['primary_domain'] ?? null) ?: '—')),
|
||||||
Text::make('Environment')
|
Text::make('Environment')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => (string) (($this->onboardingSession?->state['environment'] ?? null) ?: '—')),
|
Text::make(fn () => (string) ((($this->currentOnboardingSessionRecord() ?? $draft)?->state['environment'] ?? null) ?: '—')),
|
||||||
Text::make('Notes')
|
Text::make('Notes')
|
||||||
->color('gray'),
|
->color('gray'),
|
||||||
Text::make(fn (): string => (string) (($this->onboardingSession?->state['notes'] ?? null) ?: '—')),
|
Text::make(fn () => (string) ((($this->currentOnboardingSessionRecord() ?? $draft)?->state['notes'] ?? null) ?: '—')),
|
||||||
]),
|
]),
|
||||||
SchemaActions::make([
|
SchemaActions::make([
|
||||||
Action::make('back_to_workspace_summary')
|
Action::make('back_to_workspace_summary')
|
||||||
@ -874,7 +899,7 @@ private function nonResumableSummarySchema(): array
|
|||||||
->modalHeading('Delete onboarding draft')
|
->modalHeading('Delete onboarding draft')
|
||||||
->modalDescription('This permanently deletes the onboarding draft record. The linked tenant record, if any, is not deleted.')
|
->modalDescription('This permanently deletes the onboarding draft record. The linked tenant record, if any, is not deleted.')
|
||||||
->modalSubmitActionLabel('Delete draft')
|
->modalSubmitActionLabel('Delete draft')
|
||||||
->visible(fn (): bool => $this->canDeleteDraft($this->onboardingSession))
|
->visible(fn (): bool => $this->canDeleteDraft($this->currentOnboardingSessionRecord() ?? $draft))
|
||||||
->action(fn () => $this->deleteOnboardingDraft()),
|
->action(fn () => $this->deleteOnboardingDraft()),
|
||||||
]),
|
]),
|
||||||
];
|
];
|
||||||
@ -884,7 +909,7 @@ private function startNewOnboardingDraft(): void
|
|||||||
{
|
{
|
||||||
$this->showDraftPicker = false;
|
$this->showDraftPicker = false;
|
||||||
$this->showStartState = true;
|
$this->showStartState = true;
|
||||||
$this->managedTenant = null;
|
$this->setManagedTenant(null);
|
||||||
$this->setOnboardingSession(null);
|
$this->setOnboardingSession(null);
|
||||||
$this->selectedProviderConnectionId = null;
|
$this->selectedProviderConnectionId = null;
|
||||||
$this->selectedBootstrapOperationTypes = [];
|
$this->selectedBootstrapOperationTypes = [];
|
||||||
@ -936,9 +961,20 @@ private function cancelOnboardingDraft(): void
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->authorize('cancel', $this->onboardingSession);
|
$this->authorizeWorkspaceMember($user);
|
||||||
|
|
||||||
if (! $this->canResumeDraft($this->onboardingSession)) {
|
$draft = app(TrustedStateResolver::class)->resolveOnboardingDraft(
|
||||||
|
$this->onboardingSessionId ?? $this->onboardingSession,
|
||||||
|
$user,
|
||||||
|
$this->workspace,
|
||||||
|
app(OnboardingDraftResolver::class),
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->setOnboardingSession($draft);
|
||||||
|
|
||||||
|
$this->authorize('cancel', $draft);
|
||||||
|
|
||||||
|
if (! $this->canResumeDraft($draft)) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Draft is not resumable')
|
->title('Draft is not resumable')
|
||||||
->warning()
|
->warning()
|
||||||
@ -999,8 +1035,7 @@ private function cancelOnboardingDraft(): void
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
$this->managedTenant = $normalizedTenant;
|
$this->setManagedTenant($normalizedTenant);
|
||||||
$this->onboardingSession->setRelation('tenant', $normalizedTenant);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
@ -1023,9 +1058,20 @@ private function deleteOnboardingDraft(): void
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->authorize('cancel', $this->onboardingSession);
|
$this->authorizeWorkspaceMember($user);
|
||||||
|
|
||||||
if (! $this->canDeleteDraft($this->onboardingSession)) {
|
$draft = app(TrustedStateResolver::class)->resolveOnboardingDraft(
|
||||||
|
$this->onboardingSessionId ?? $this->onboardingSession,
|
||||||
|
$user,
|
||||||
|
$this->workspace,
|
||||||
|
app(OnboardingDraftResolver::class),
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->setOnboardingSession($draft);
|
||||||
|
|
||||||
|
$this->authorize('cancel', $draft);
|
||||||
|
|
||||||
|
if (! $this->canDeleteDraft($draft)) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Draft cannot be deleted')
|
->title('Draft cannot be deleted')
|
||||||
->warning()
|
->warning()
|
||||||
@ -1034,7 +1080,6 @@ private function deleteOnboardingDraft(): void
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$draft = $this->onboardingSession;
|
|
||||||
$draftId = (int) $draft->getKey();
|
$draftId = (int) $draft->getKey();
|
||||||
$draftTitle = $this->draftTitle($draft);
|
$draftTitle = $this->draftTitle($draft);
|
||||||
$draftStatus = $draft->status()->value;
|
$draftStatus = $draft->status()->value;
|
||||||
@ -1062,7 +1107,7 @@ private function deleteOnboardingDraft(): void
|
|||||||
targetLabel: $draftTitle,
|
targetLabel: $draftTitle,
|
||||||
);
|
);
|
||||||
|
|
||||||
$this->managedTenant = null;
|
$this->setManagedTenant(null);
|
||||||
$this->setOnboardingSession(null);
|
$this->setOnboardingSession(null);
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
@ -1075,8 +1120,10 @@ private function deleteOnboardingDraft(): void
|
|||||||
|
|
||||||
private function showsNonResumableSummary(): bool
|
private function showsNonResumableSummary(): bool
|
||||||
{
|
{
|
||||||
return $this->onboardingSession instanceof TenantOnboardingSession
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
&& ! $this->canResumeDraft($this->onboardingSession);
|
|
||||||
|
return $draft instanceof TenantOnboardingSession
|
||||||
|
&& ! $this->canResumeDraft($draft);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function canDeleteDraft(?TenantOnboardingSession $draft): bool
|
private function canDeleteDraft(?TenantOnboardingSession $draft): bool
|
||||||
@ -1109,11 +1156,13 @@ private function onboardingEntryActionDescriptor(int $resumableDraftCount): \App
|
|||||||
|
|
||||||
private function shouldShowDraftLandingAction(): bool
|
private function shouldShowDraftLandingAction(): bool
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! $this->canResumeDraft($this->onboardingSession)) {
|
if (! $this->canResumeDraft($draft)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1211,14 +1260,95 @@ private function expectedDraftVersion(): ?int
|
|||||||
private function setOnboardingSession(?TenantOnboardingSession $draft): void
|
private function setOnboardingSession(?TenantOnboardingSession $draft): void
|
||||||
{
|
{
|
||||||
$this->onboardingSession = $draft;
|
$this->onboardingSession = $draft;
|
||||||
|
$this->onboardingSessionId = $draft instanceof TenantOnboardingSession
|
||||||
|
? (int) $draft->getKey()
|
||||||
|
: null;
|
||||||
$this->onboardingSessionVersion = $draft instanceof TenantOnboardingSession
|
$this->onboardingSessionVersion = $draft instanceof TenantOnboardingSession
|
||||||
? $draft->expectedVersion()
|
? $draft->expectedVersion()
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
|
if ($draft instanceof TenantOnboardingSession && $draft->tenant instanceof Tenant) {
|
||||||
|
$this->setManagedTenant($draft->tenant);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($draft instanceof TenantOnboardingSession && $draft->tenant_id !== null) {
|
||||||
|
$this->managedTenantId = (int) $draft->tenant_id;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->setManagedTenant(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function setManagedTenant(?Tenant $tenant): void
|
||||||
|
{
|
||||||
|
$this->managedTenant = $tenant;
|
||||||
|
$this->managedTenantId = $tenant instanceof Tenant
|
||||||
|
? (int) $tenant->getKey()
|
||||||
|
: null;
|
||||||
|
|
||||||
|
if ($this->onboardingSession instanceof TenantOnboardingSession && $tenant instanceof Tenant) {
|
||||||
|
$this->onboardingSession->setRelation('tenant', $tenant);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function currentOnboardingSessionRecord(): ?TenantOnboardingSession
|
||||||
|
{
|
||||||
|
if ($this->onboardingSession instanceof TenantOnboardingSession
|
||||||
|
&& $this->onboardingSessionId !== null
|
||||||
|
&& (int) $this->onboardingSession->getKey() === $this->onboardingSessionId) {
|
||||||
|
return $this->onboardingSession;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->onboardingSessionId === null) {
|
||||||
|
return $this->onboardingSession;
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = TenantOnboardingSession::query()
|
||||||
|
->with(['tenant', 'startedByUser', 'updatedByUser'])
|
||||||
|
->whereKey($this->onboardingSessionId);
|
||||||
|
|
||||||
|
if (isset($this->workspace)) {
|
||||||
|
$query->where('workspace_id', (int) $this->workspace->getKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
return $query->first();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function currentManagedTenantRecord(): ?Tenant
|
||||||
|
{
|
||||||
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if ($draft instanceof TenantOnboardingSession && $draft->tenant instanceof Tenant) {
|
||||||
|
return $draft->tenant;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->managedTenant instanceof Tenant
|
||||||
|
&& $this->managedTenantId !== null
|
||||||
|
&& (int) $this->managedTenant->getKey() === $this->managedTenantId) {
|
||||||
|
return $this->managedTenant;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->managedTenantId === null) {
|
||||||
|
return $this->managedTenant;
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = Tenant::query()->withTrashed()->whereKey($this->managedTenantId);
|
||||||
|
|
||||||
|
if (isset($this->workspace)) {
|
||||||
|
$query->where('workspace_id', (int) $this->workspace->getKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
return $query->first();
|
||||||
}
|
}
|
||||||
|
|
||||||
private function refreshOnboardingDraftFromBackend(): void
|
private function refreshOnboardingDraftFromBackend(): void
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1229,15 +1359,11 @@ private function refreshOnboardingDraftFromBackend(): void
|
|||||||
}
|
}
|
||||||
|
|
||||||
$this->setOnboardingSession(app(OnboardingDraftResolver::class)->resolve(
|
$this->setOnboardingSession(app(OnboardingDraftResolver::class)->resolve(
|
||||||
$this->onboardingSession,
|
$draft,
|
||||||
$user,
|
$user,
|
||||||
$this->workspace,
|
$this->workspace,
|
||||||
));
|
));
|
||||||
|
|
||||||
if ($this->onboardingSession->tenant instanceof Tenant) {
|
|
||||||
$this->managedTenant = $this->onboardingSession->tenant;
|
|
||||||
}
|
|
||||||
|
|
||||||
$providerConnectionId = $this->onboardingSession->state['provider_connection_id'] ?? null;
|
$providerConnectionId = $this->onboardingSession->state['provider_connection_id'] ?? null;
|
||||||
$this->selectedProviderConnectionId = $this->resolvePersistedProviderConnectionId($providerConnectionId);
|
$this->selectedProviderConnectionId = $this->resolvePersistedProviderConnectionId($providerConnectionId);
|
||||||
$this->initializeWizardData();
|
$this->initializeWizardData();
|
||||||
@ -1267,11 +1393,13 @@ private function handleImmutableDraft(string $title = 'This onboarding draft is
|
|||||||
|
|
||||||
private function lifecycleState(): OnboardingLifecycleState
|
private function lifecycleState(): OnboardingLifecycleState
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return OnboardingLifecycleState::Draft;
|
return OnboardingLifecycleState::Draft;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->lifecycleService()->snapshot($this->onboardingSession)['lifecycle_state'];
|
return $this->lifecycleService()->snapshot($draft)['lifecycle_state'];
|
||||||
}
|
}
|
||||||
|
|
||||||
private function lifecycleStateLabel(): string
|
private function lifecycleStateLabel(): string
|
||||||
@ -1294,30 +1422,38 @@ private function lifecycleStateColor(): string
|
|||||||
|
|
||||||
private function currentCheckpointLabel(): string
|
private function currentCheckpointLabel(): string
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return OnboardingCheckpoint::Identify->label();
|
return OnboardingCheckpoint::Identify->label();
|
||||||
}
|
}
|
||||||
|
|
||||||
return ($this->lifecycleService()->snapshot($this->onboardingSession)['current_checkpoint'] ?? OnboardingCheckpoint::Identify)?->label()
|
return ($this->lifecycleService()->snapshot($draft)['current_checkpoint'] ?? OnboardingCheckpoint::Identify)?->label()
|
||||||
?? OnboardingCheckpoint::Identify->label();
|
?? OnboardingCheckpoint::Identify->label();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function shouldPollCheckpointLifecycle(): bool
|
public function shouldPollCheckpointLifecycle(): bool
|
||||||
{
|
{
|
||||||
return $this->onboardingSession instanceof TenantOnboardingSession
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
&& $this->lifecycleService()->hasActiveCheckpoint($this->onboardingSession);
|
|
||||||
|
return $draft instanceof TenantOnboardingSession
|
||||||
|
&& $this->lifecycleService()->hasActiveCheckpoint($draft);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function refreshCheckpointLifecycle(): void
|
public function refreshCheckpointLifecycle(): void
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->setOnboardingSession($this->lifecycleService()->syncPersistedLifecycle($this->onboardingSession));
|
$this->setOnboardingSession($this->lifecycleService()->syncPersistedLifecycle($draft));
|
||||||
|
|
||||||
if ($this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
$this->managedTenant->refresh();
|
|
||||||
|
if ($tenant instanceof Tenant) {
|
||||||
|
$this->setManagedTenant($tenant->fresh());
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->initializeWizardData();
|
$this->initializeWizardData();
|
||||||
@ -1343,20 +1479,24 @@ private function initializeWizardData(): void
|
|||||||
$this->data['new_connection']['is_default'] ??= true;
|
$this->data['new_connection']['is_default'] ??= true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
$this->data['entra_tenant_id'] ??= (string) $this->managedTenant->tenant_id;
|
|
||||||
$this->data['environment'] ??= (string) ($this->managedTenant->environment ?? 'other');
|
|
||||||
$this->data['name'] ??= (string) $this->managedTenant->name;
|
|
||||||
$this->data['primary_domain'] ??= (string) ($this->managedTenant->domain ?? '');
|
|
||||||
|
|
||||||
$notes = is_array($this->managedTenant->metadata) ? ($this->managedTenant->metadata['notes'] ?? null) : null;
|
if ($tenant instanceof Tenant) {
|
||||||
|
$this->data['entra_tenant_id'] ??= (string) $tenant->tenant_id;
|
||||||
|
$this->data['environment'] ??= (string) ($tenant->environment ?? 'other');
|
||||||
|
$this->data['name'] ??= (string) $tenant->name;
|
||||||
|
$this->data['primary_domain'] ??= (string) ($tenant->domain ?? '');
|
||||||
|
|
||||||
|
$notes = is_array($tenant->metadata) ? ($tenant->metadata['notes'] ?? null) : null;
|
||||||
if (is_string($notes) && trim($notes) !== '') {
|
if (is_string($notes) && trim($notes) !== '') {
|
||||||
$this->data['notes'] ??= trim($notes);
|
$this->data['notes'] ??= trim($notes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
$state = is_array($this->onboardingSession->state) ? $this->onboardingSession->state : [];
|
|
||||||
|
if ($draft instanceof TenantOnboardingSession) {
|
||||||
|
$state = is_array($draft->state) ? $draft->state : [];
|
||||||
|
|
||||||
if (isset($state['entra_tenant_id']) && is_string($state['entra_tenant_id']) && trim($state['entra_tenant_id']) !== '') {
|
if (isset($state['entra_tenant_id']) && is_string($state['entra_tenant_id']) && trim($state['entra_tenant_id']) !== '') {
|
||||||
$this->data['entra_tenant_id'] ??= trim($state['entra_tenant_id']);
|
$this->data['entra_tenant_id'] ??= trim($state['entra_tenant_id']);
|
||||||
@ -1386,13 +1526,13 @@ private function initializeWizardData(): void
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$providerConnectionId = $this->resolvePersistedProviderConnectionId($this->onboardingSession->state['provider_connection_id'] ?? null);
|
$providerConnectionId = $this->resolvePersistedProviderConnectionId($draft->state['provider_connection_id'] ?? null);
|
||||||
if ($providerConnectionId !== null) {
|
if ($providerConnectionId !== null) {
|
||||||
$this->data['provider_connection_id'] = $providerConnectionId;
|
$this->data['provider_connection_id'] = $providerConnectionId;
|
||||||
$this->selectedProviderConnectionId = $providerConnectionId;
|
$this->selectedProviderConnectionId = $providerConnectionId;
|
||||||
}
|
}
|
||||||
|
|
||||||
$types = $this->onboardingSession->state['bootstrap_operation_types'] ?? null;
|
$types = $draft->state['bootstrap_operation_types'] ?? null;
|
||||||
if (is_array($types)) {
|
if (is_array($types)) {
|
||||||
$this->data['bootstrap_operation_types'] = array_values(array_filter($types, static fn ($v): bool => is_string($v) && $v !== ''));
|
$this->data['bootstrap_operation_types'] = array_values(array_filter($types, static fn ($v): bool => is_string($v) && $v !== ''));
|
||||||
}
|
}
|
||||||
@ -1410,7 +1550,7 @@ private function initializeWizardData(): void
|
|||||||
private function computeWizardStartStep(): int
|
private function computeWizardStartStep(): int
|
||||||
{
|
{
|
||||||
return app(OnboardingDraftStageResolver::class)
|
return app(OnboardingDraftStageResolver::class)
|
||||||
->resolve($this->onboardingSession)
|
->resolve($this->currentOnboardingSessionRecord())
|
||||||
->wizardStep();
|
->wizardStep();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1419,13 +1559,15 @@ private function computeWizardStartStep(): int
|
|||||||
*/
|
*/
|
||||||
private function providerConnectionOptions(): array
|
private function providerConnectionOptions(): array
|
||||||
{
|
{
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
return ProviderConnection::query()
|
return ProviderConnection::query()
|
||||||
->where('workspace_id', (int) $this->workspace->getKey())
|
->where('workspace_id', (int) $this->workspace->getKey())
|
||||||
->where('tenant_id', $this->managedTenant->getKey())
|
->where('tenant_id', $tenant->getKey())
|
||||||
->orderByDesc('is_default')
|
->orderByDesc('is_default')
|
||||||
->orderBy('display_name')
|
->orderBy('display_name')
|
||||||
->pluck('display_name', 'id')
|
->pluck('display_name', 'id')
|
||||||
@ -1442,11 +1584,13 @@ private function verificationStatusLabel(): string
|
|||||||
|
|
||||||
private function verificationStatus(): string
|
private function verificationStatus(): string
|
||||||
{
|
{
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$draft = $this->currentOnboardingSessionRecord();
|
||||||
|
|
||||||
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
return 'not_started';
|
return 'not_started';
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->lifecycleService()->verificationStatus($this->onboardingSession, $this->selectedProviderConnectionId);
|
return $this->lifecycleService()->verificationStatus($draft, $this->selectedProviderConnectionId);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function verificationStatusFromRunOutcome(OperationRun $run): string
|
private function verificationStatusFromRunOutcome(OperationRun $run): string
|
||||||
@ -1943,6 +2087,19 @@ private function authorizeEditableDraft(User $user): void
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$expectedVersion = $this->expectedDraftVersion();
|
||||||
|
|
||||||
|
$this->setOnboardingSession(app(TrustedStateResolver::class)->resolveOnboardingDraft(
|
||||||
|
$this->onboardingSessionId ?? $this->onboardingSession,
|
||||||
|
$user,
|
||||||
|
$this->workspace,
|
||||||
|
app(OnboardingDraftResolver::class),
|
||||||
|
));
|
||||||
|
|
||||||
|
if ($expectedVersion !== null) {
|
||||||
|
$this->onboardingSessionVersion = $expectedVersion;
|
||||||
|
}
|
||||||
|
|
||||||
$this->authorize('update', $this->onboardingSession);
|
$this->authorize('update', $this->onboardingSession);
|
||||||
|
|
||||||
if (! $this->canResumeDraft($this->onboardingSession)) {
|
if (! $this->canResumeDraft($this->onboardingSession)) {
|
||||||
@ -1950,17 +2107,56 @@ private function authorizeEditableDraft(User $user): void
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function trustedManagedTenantForUser(User $user): Tenant
|
||||||
|
{
|
||||||
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = $tenant->fresh();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = app(WorkspaceContext::class)->ensureTenantAccessibleInCurrentWorkspace($tenant, $user, request());
|
||||||
|
|
||||||
|
$this->setManagedTenant($tenant);
|
||||||
|
|
||||||
|
return $tenant;
|
||||||
|
}
|
||||||
|
|
||||||
private function canResumeDraft(?TenantOnboardingSession $draft): bool
|
private function canResumeDraft(?TenantOnboardingSession $draft): bool
|
||||||
{
|
{
|
||||||
return $draft instanceof TenantOnboardingSession
|
if (! $draft instanceof TenantOnboardingSession) {
|
||||||
&& $this->lifecycleService()->canResumeDraft($draft);
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $draft->tenant instanceof Tenant) {
|
||||||
|
return $this->lifecycleService()->canResumeDraft($draft);
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = $this->currentUser();
|
||||||
|
|
||||||
|
return app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $draft->tenant,
|
||||||
|
question: TenantOperabilityQuestion::ResumeOnboardingEligibility,
|
||||||
|
actor: $user instanceof User ? $user : null,
|
||||||
|
workspaceId: isset($this->workspace) ? (int) $this->workspace->getKey() : null,
|
||||||
|
lane: TenantInteractionLane::OnboardingWorkflow,
|
||||||
|
onboardingDraft: $draft,
|
||||||
|
)->allowed;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function authorizeWorkspaceMember(User $user): void
|
private function authorizeWorkspaceMember(User $user): void
|
||||||
{
|
{
|
||||||
if (! app(WorkspaceContext::class)->isMember($user, $this->workspace)) {
|
$this->workspace = app(TrustedStateResolver::class)->currentWorkspaceForMember(
|
||||||
abort(404);
|
$user,
|
||||||
}
|
app(WorkspaceContext::class),
|
||||||
|
request(),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function resolveWorkspaceIdForUnboundTenant(Tenant $tenant): ?int
|
private function resolveWorkspaceIdForUnboundTenant(Tenant $tenant): ?int
|
||||||
@ -2157,7 +2353,7 @@ public function identifyManagedTenant(array $data): void
|
|||||||
resourceId: (string) $tenant->getKey(),
|
resourceId: (string) $tenant->getKey(),
|
||||||
);
|
);
|
||||||
|
|
||||||
$this->managedTenant = $tenant;
|
$this->setManagedTenant($tenant);
|
||||||
$this->setOnboardingSession($session);
|
$this->setOnboardingSession($session);
|
||||||
});
|
});
|
||||||
} catch (OnboardingDraftConflictException) {
|
} catch (OnboardingDraftConflictException) {
|
||||||
@ -2196,13 +2392,11 @@ public function selectProviderConnection(int $providerConnectionId): void
|
|||||||
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_CONNECTION_VIEW);
|
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_CONNECTION_VIEW);
|
||||||
$this->authorizeEditableDraft($user);
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->trustedManagedTenantForUser($user);
|
||||||
abort(404);
|
|
||||||
}
|
|
||||||
|
|
||||||
$connection = ProviderConnection::query()
|
$connection = ProviderConnection::query()
|
||||||
->where('workspace_id', (int) $this->workspace->getKey())
|
->where('workspace_id', (int) $this->workspace->getKey())
|
||||||
->where('tenant_id', $this->managedTenant->getKey())
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
->whereKey($providerConnectionId)
|
->whereKey($providerConnectionId)
|
||||||
->first();
|
->first();
|
||||||
|
|
||||||
@ -2248,7 +2442,7 @@ public function selectProviderConnection(int $providerConnectionId): void
|
|||||||
context: [
|
context: [
|
||||||
'metadata' => [
|
'metadata' => [
|
||||||
'workspace_id' => (int) $this->workspace->getKey(),
|
'workspace_id' => (int) $this->workspace->getKey(),
|
||||||
'tenant_db_id' => (int) $this->managedTenant->getKey(),
|
'tenant_db_id' => (int) $tenant->getKey(),
|
||||||
'provider_connection_id' => (int) $connection->getKey(),
|
'provider_connection_id' => (int) $connection->getKey(),
|
||||||
'onboarding_session_id' => $this->onboardingSession?->getKey(),
|
'onboarding_session_id' => $this->onboardingSession?->getKey(),
|
||||||
],
|
],
|
||||||
@ -2281,11 +2475,7 @@ public function createProviderConnection(array $data): void
|
|||||||
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_CONNECTION_MANAGE);
|
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_CONNECTION_MANAGE);
|
||||||
$this->authorizeEditableDraft($user);
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->trustedManagedTenantForUser($user)->fresh();
|
||||||
abort(404);
|
|
||||||
}
|
|
||||||
|
|
||||||
$tenant = $this->managedTenant->fresh();
|
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
abort(404);
|
abort(404);
|
||||||
@ -2474,7 +2664,7 @@ public function createProviderConnection(array $data): void
|
|||||||
context: [
|
context: [
|
||||||
'metadata' => [
|
'metadata' => [
|
||||||
'workspace_id' => (int) $this->workspace->getKey(),
|
'workspace_id' => (int) $this->workspace->getKey(),
|
||||||
'tenant_db_id' => (int) $this->managedTenant->getKey(),
|
'tenant_db_id' => (int) $tenant->getKey(),
|
||||||
'provider_connection_id' => (int) $connection->getKey(),
|
'provider_connection_id' => (int) $connection->getKey(),
|
||||||
'onboarding_session_id' => $this->onboardingSession?->getKey(),
|
'onboarding_session_id' => $this->onboardingSession?->getKey(),
|
||||||
],
|
],
|
||||||
@ -2516,7 +2706,9 @@ public function startVerification(): void
|
|||||||
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_VERIFICATION_START);
|
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_VERIFICATION_START);
|
||||||
$this->authorizeEditableDraft($user);
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
try {
|
||||||
|
$tenant = $this->trustedManagedTenantForUser($user)->fresh();
|
||||||
|
} catch (\Symfony\Component\HttpKernel\Exception\NotFoundHttpException) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Identify a managed tenant first')
|
->title('Identify a managed tenant first')
|
||||||
->warning()
|
->warning()
|
||||||
@ -2525,8 +2717,6 @@ public function startVerification(): void
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$tenant = $this->managedTenant->fresh();
|
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
@ -2692,9 +2882,12 @@ public function startVerification(): void
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Verification blocked')
|
->title('Verification blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions($actions)
|
->actions($actions)
|
||||||
->send();
|
->send();
|
||||||
@ -2727,6 +2920,15 @@ public function startVerification(): void
|
|||||||
|
|
||||||
public function refreshVerificationStatus(): void
|
public function refreshVerificationStatus(): void
|
||||||
{
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->authorizeWorkspaceMember($user);
|
||||||
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
$this->refreshCheckpointLifecycle();
|
$this->refreshCheckpointLifecycle();
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
@ -2749,11 +2951,7 @@ public function startBootstrap(array $operationTypes): void
|
|||||||
$this->authorizeWorkspaceMember($user);
|
$this->authorizeWorkspaceMember($user);
|
||||||
$this->authorizeEditableDraft($user);
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->trustedManagedTenantForUser($user)->fresh();
|
||||||
abort(404);
|
|
||||||
}
|
|
||||||
|
|
||||||
$tenant = $this->managedTenant->fresh();
|
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
abort(404);
|
abort(404);
|
||||||
@ -3054,6 +3252,19 @@ private function canCompleteOnboarding(): bool
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$user = $this->currentUser();
|
||||||
|
|
||||||
|
if (! app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $this->managedTenant,
|
||||||
|
question: TenantOperabilityQuestion::OnboardingCompletionEligibility,
|
||||||
|
actor: $user instanceof User ? $user : null,
|
||||||
|
workspaceId: isset($this->workspace) ? (int) $this->workspace->getKey() : null,
|
||||||
|
lane: TenantInteractionLane::OnboardingWorkflow,
|
||||||
|
onboardingDraft: $this->onboardingSession,
|
||||||
|
)->allowed) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (! $this->resolveSelectedProviderConnection($this->managedTenant)) {
|
if (! $this->resolveSelectedProviderConnection($this->managedTenant)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -3075,23 +3286,27 @@ private function canCompleteOnboarding(): bool
|
|||||||
|
|
||||||
private function completionSummaryTenantLine(): string
|
private function completionSummaryTenantLine(): string
|
||||||
{
|
{
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
return '—';
|
return '—';
|
||||||
}
|
}
|
||||||
|
|
||||||
$name = $this->managedTenant->name ?? '—';
|
$name = $tenant->name ?? '—';
|
||||||
$tenantId = $this->managedTenant->graphTenantId();
|
$tenantId = $tenant->graphTenantId();
|
||||||
|
|
||||||
return $tenantId !== null ? "{$name} ({$tenantId})" : $name;
|
return $tenantId !== null ? "{$name} ({$tenantId})" : $name;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function completionSummaryConnectionLabel(): string
|
private function completionSummaryConnectionLabel(): string
|
||||||
{
|
{
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
$tenant = $this->currentManagedTenantRecord();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
return '—';
|
return '—';
|
||||||
}
|
}
|
||||||
|
|
||||||
$connection = $this->resolveSelectedProviderConnection($this->managedTenant);
|
$connection = $this->resolveSelectedProviderConnection($tenant);
|
||||||
|
|
||||||
if (! $connection instanceof ProviderConnection) {
|
if (! $connection instanceof ProviderConnection) {
|
||||||
return 'Not configured';
|
return 'Not configured';
|
||||||
@ -3219,12 +3434,29 @@ public function completeOnboarding(): void
|
|||||||
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_ACTIVATE);
|
$this->authorizeWorkspaceMutation($user, Capabilities::WORKSPACE_MANAGED_TENANT_ONBOARD_ACTIVATE);
|
||||||
$this->authorizeEditableDraft($user);
|
$this->authorizeEditableDraft($user);
|
||||||
|
|
||||||
if (! $this->managedTenant instanceof Tenant) {
|
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! $this->onboardingSession instanceof TenantOnboardingSession) {
|
$tenant = $this->trustedManagedTenantForUser($user);
|
||||||
abort(404);
|
|
||||||
|
$completionOutcome = app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::OnboardingCompletionEligibility,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: (int) $this->workspace->getKey(),
|
||||||
|
lane: TenantInteractionLane::OnboardingWorkflow,
|
||||||
|
onboardingDraft: $this->onboardingSession,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (! $completionOutcome->allowed) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Onboarding unavailable')
|
||||||
|
->body('This tenant can no longer be completed from the current onboarding workflow state.')
|
||||||
|
->warning()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$run = $this->verificationRun();
|
$run = $this->verificationRun();
|
||||||
@ -3260,7 +3492,7 @@ public function completeOnboarding(): void
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$tenant = $this->managedTenant->fresh();
|
$tenant = $tenant->fresh();
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
abort(404);
|
abort(404);
|
||||||
|
|||||||
@ -10,6 +10,9 @@
|
|||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\Workspace;
|
use App\Models\Workspace;
|
||||||
use App\Services\Tenants\TenantOperabilityService;
|
use App\Services\Tenants\TenantOperabilityService;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use Filament\Pages\Page;
|
use Filament\Pages\Page;
|
||||||
use Illuminate\Database\Eloquent\Collection;
|
use Illuminate\Database\Eloquent\Collection;
|
||||||
|
|
||||||
@ -64,7 +67,15 @@ public function getTenants(): Collection
|
|||||||
->where('workspace_id', $this->workspace->getKey())
|
->where('workspace_id', $this->workspace->getKey())
|
||||||
->orderBy('name')
|
->orderBy('name')
|
||||||
->get()
|
->get()
|
||||||
->filter(fn (Tenant $tenant): bool => app(TenantOperabilityService::class)->canViewTenantSurface($tenant))
|
->filter(function (Tenant $tenant) use ($user): bool {
|
||||||
|
return app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::AdministrativeDiscoverability,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: app(WorkspaceContext::class)->currentWorkspaceId(request()),
|
||||||
|
lane: TenantInteractionLane::AdministrativeManagement,
|
||||||
|
)->allowed;
|
||||||
|
})
|
||||||
->values();
|
->values();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -3,6 +3,7 @@
|
|||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
use App\Exceptions\InvalidPolicyTypeException;
|
use App\Exceptions\InvalidPolicyTypeException;
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\BackupScheduleResource\Pages;
|
use App\Filament\Resources\BackupScheduleResource\Pages;
|
||||||
use App\Filament\Resources\BackupScheduleResource\RelationManagers\BackupScheduleOperationRunsRelationManager;
|
use App\Filament\Resources\BackupScheduleResource\RelationManagers\BackupScheduleOperationRunsRelationManager;
|
||||||
@ -64,6 +65,7 @@
|
|||||||
|
|
||||||
class BackupScheduleResource extends Resource
|
class BackupScheduleResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
protected static ?string $model = BackupSchedule::class;
|
protected static ?string $model = BackupSchedule::class;
|
||||||
@ -581,6 +583,8 @@ public static function table(Table $table): Table
|
|||||||
->color('danger')
|
->color('danger')
|
||||||
->visible(fn (BackupSchedule $record): bool => ! $record->trashed())
|
->visible(fn (BackupSchedule $record): bool => ! $record->trashed())
|
||||||
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
||||||
|
$record = static::resolveProtectedScheduleRecordOrFail($record);
|
||||||
|
|
||||||
Gate::authorize('delete', $record);
|
Gate::authorize('delete', $record);
|
||||||
|
|
||||||
if ($record->trashed()) {
|
if ($record->trashed()) {
|
||||||
@ -622,6 +626,8 @@ public static function table(Table $table): Table
|
|||||||
->color('success')
|
->color('success')
|
||||||
->visible(fn (BackupSchedule $record): bool => $record->trashed())
|
->visible(fn (BackupSchedule $record): bool => $record->trashed())
|
||||||
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
||||||
|
$record = static::resolveProtectedScheduleRecordOrFail($record);
|
||||||
|
|
||||||
Gate::authorize('restore', $record);
|
Gate::authorize('restore', $record);
|
||||||
|
|
||||||
if (! $record->trashed()) {
|
if (! $record->trashed()) {
|
||||||
@ -662,6 +668,8 @@ public static function table(Table $table): Table
|
|||||||
->color('danger')
|
->color('danger')
|
||||||
->visible(fn (BackupSchedule $record): bool => $record->trashed())
|
->visible(fn (BackupSchedule $record): bool => $record->trashed())
|
||||||
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
->action(function (BackupSchedule $record, AuditLogger $auditLogger): void {
|
||||||
|
$record = static::resolveProtectedScheduleRecordOrFail($record);
|
||||||
|
|
||||||
Gate::authorize('forceDelete', $record);
|
Gate::authorize('forceDelete', $record);
|
||||||
|
|
||||||
if (! $record->trashed()) {
|
if (! $record->trashed()) {
|
||||||
@ -919,17 +927,32 @@ public static function table(Table $table): Table
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanelOrFail()->getKey();
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
|
||||||
->where('tenant_id', $tenantId)
|
|
||||||
->orderByDesc('is_enabled')
|
->orderByDesc('is_enabled')
|
||||||
->orderBy('next_run_at');
|
->orderBy('next_run_at');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getRecordRouteBindingEloquentQuery(): Builder
|
public static function getRecordRouteBindingEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
return static::getEloquentQuery()->withTrashed();
|
return static::scopeTenantOwnedQuery(parent::getEloquentQuery()->withTrashed())
|
||||||
|
->orderByDesc('is_enabled')
|
||||||
|
->orderBy('next_run_at');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string $key): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($key, parent::getEloquentQuery()->withTrashed());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static function resolveProtectedScheduleRecordOrFail(BackupSchedule|int|string $record): BackupSchedule
|
||||||
|
{
|
||||||
|
$resolvedRecord = static::resolveScopedRecordOrFail($record instanceof Model ? $record->getKey() : $record);
|
||||||
|
|
||||||
|
if (! $resolvedRecord instanceof BackupSchedule) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedRecord;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getRelations(): array
|
public static function getRelations(): array
|
||||||
|
|||||||
@ -5,7 +5,6 @@
|
|||||||
use App\Filament\Resources\BackupScheduleResource;
|
use App\Filament\Resources\BackupScheduleResource;
|
||||||
use Filament\Resources\Pages\EditRecord;
|
use Filament\Resources\Pages\EditRecord;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
|
||||||
|
|
||||||
class EditBackupSchedule extends EditRecord
|
class EditBackupSchedule extends EditRecord
|
||||||
{
|
{
|
||||||
@ -13,15 +12,7 @@ class EditBackupSchedule extends EditRecord
|
|||||||
|
|
||||||
protected function resolveRecord(int|string $key): Model
|
protected function resolveRecord(int|string $key): Model
|
||||||
{
|
{
|
||||||
$record = BackupScheduleResource::getEloquentQuery()
|
return BackupScheduleResource::resolveScopedRecordOrFail($key);
|
||||||
->withTrashed()
|
|
||||||
->find($key);
|
|
||||||
|
|
||||||
if ($record === null) {
|
|
||||||
throw (new ModelNotFoundException)->setModel(BackupScheduleResource::getModel(), [$key]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $record;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function mutateFormDataBeforeSave(array $data): array
|
protected function mutateFormDataBeforeSave(array $data): array
|
||||||
|
|||||||
@ -5,18 +5,32 @@
|
|||||||
use App\Filament\Resources\BackupScheduleResource;
|
use App\Filament\Resources\BackupScheduleResource;
|
||||||
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
||||||
use Filament\Resources\Pages\ListRecords;
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
||||||
|
|
||||||
class ListBackupSchedules extends ListRecords
|
class ListBackupSchedules extends ListRecords
|
||||||
{
|
{
|
||||||
protected static string $resource = BackupScheduleResource::class;
|
protected static string $resource = BackupScheduleResource::class;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true && filled($context['recordKey'] ?? null) && in_array($name, ['archive', 'restore', 'forceDelete'], true)) {
|
||||||
|
try {
|
||||||
|
BackupScheduleResource::resolveScopedRecordOrFail($context['recordKey']);
|
||||||
|
} catch (ModelNotFoundException) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
public function mount(): void
|
public function mount(): void
|
||||||
{
|
{
|
||||||
app(CanonicalAdminTenantFilterState::class)->sync(
|
$this->syncCanonicalAdminTenantFilterState();
|
||||||
$this->getTableFiltersSessionKey(),
|
|
||||||
request: request(),
|
|
||||||
tenantFilterName: null,
|
|
||||||
);
|
|
||||||
|
|
||||||
parent::mount();
|
parent::mount();
|
||||||
}
|
}
|
||||||
@ -40,4 +54,14 @@ private function tableHasRecords(): bool
|
|||||||
{
|
{
|
||||||
return $this->getTableRecords()->count() > 0;
|
return $this->getTableRecords()->count() > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function syncCanonicalAdminTenantFilterState(): void
|
||||||
|
{
|
||||||
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
||||||
|
$this->getTableFiltersSessionKey(),
|
||||||
|
tenantSensitiveFilters: [],
|
||||||
|
request: request(),
|
||||||
|
tenantFilterName: null,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,6 +12,7 @@
|
|||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use Closure;
|
||||||
use Filament\Actions;
|
use Filament\Actions;
|
||||||
use Filament\Resources\RelationManagers\RelationManager;
|
use Filament\Resources\RelationManagers\RelationManager;
|
||||||
use Filament\Tables;
|
use Filament\Tables;
|
||||||
@ -24,6 +25,19 @@ class BackupScheduleOperationRunsRelationManager extends RelationManager
|
|||||||
|
|
||||||
protected static ?string $title = 'Executions';
|
protected static ?string $title = 'Executions';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true && $name === 'view' && filled($context['recordKey'] ?? null)) {
|
||||||
|
$this->resolveOwnerScopedOperationRun($context['recordKey']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
{
|
{
|
||||||
return ActionSurfaceDeclaration::forRelationManager(ActionSurfaceProfile::RelationManager)
|
return ActionSurfaceDeclaration::forRelationManager(ActionSurfaceProfile::RelationManager)
|
||||||
@ -48,7 +62,7 @@ public function table(Table $table): Table
|
|||||||
|
|
||||||
Tables\Columns\TextColumn::make('type')
|
Tables\Columns\TextColumn::make('type')
|
||||||
->label('Type')
|
->label('Type')
|
||||||
->formatStateUsing([OperationCatalog::class, 'label']),
|
->formatStateUsing(Closure::fromCallable([self::class, 'formatOperationType'])),
|
||||||
|
|
||||||
Tables\Columns\TextColumn::make('status')
|
Tables\Columns\TextColumn::make('status')
|
||||||
->badge()
|
->badge()
|
||||||
@ -87,6 +101,7 @@ public function table(Table $table): Table
|
|||||||
->label('View')
|
->label('View')
|
||||||
->icon('heroicon-o-eye')
|
->icon('heroicon-o-eye')
|
||||||
->url(function (OperationRun $record): string {
|
->url(function (OperationRun $record): string {
|
||||||
|
$record = $this->resolveOwnerScopedOperationRun($record);
|
||||||
$tenant = Tenant::currentOrFail();
|
$tenant = Tenant::currentOrFail();
|
||||||
|
|
||||||
return OperationRunLinks::view($record, $tenant);
|
return OperationRunLinks::view($record, $tenant);
|
||||||
@ -97,4 +112,32 @@ public function table(Table $table): Table
|
|||||||
->emptyStateHeading('No schedule runs yet')
|
->emptyStateHeading('No schedule runs yet')
|
||||||
->emptyStateDescription('Operation history will appear here after this schedule has been enqueued.');
|
->emptyStateDescription('Operation history will appear here after this schedule has been enqueued.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function resolveOwnerScopedOperationRun(mixed $record): OperationRun
|
||||||
|
{
|
||||||
|
$recordId = $record instanceof OperationRun
|
||||||
|
? (int) $record->getKey()
|
||||||
|
: (is_numeric($record) ? (int) $record : 0);
|
||||||
|
|
||||||
|
if ($recordId <= 0) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolvedRecord = $this->getOwnerRecord()
|
||||||
|
->operationRuns()
|
||||||
|
->where('tenant_id', Tenant::currentOrFail()->getKey())
|
||||||
|
->whereKey($recordId)
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if (! $resolvedRecord instanceof OperationRun) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedRecord;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function formatOperationType(?string $state): string
|
||||||
|
{
|
||||||
|
return OperationCatalog::label($state);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\BackupSetResource\Pages;
|
use App\Filament\Resources\BackupSetResource\Pages;
|
||||||
use App\Filament\Resources\BackupSetResource\RelationManagers\BackupItemsRelationManager;
|
use App\Filament\Resources\BackupSetResource\RelationManagers\BackupItemsRelationManager;
|
||||||
@ -56,6 +57,7 @@
|
|||||||
|
|
||||||
class BackupSetResource extends Resource
|
class BackupSetResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
protected static ?string $model = BackupSet::class;
|
protected static ?string $model = BackupSet::class;
|
||||||
@ -120,13 +122,12 @@ public static function canCreate(): bool
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
return static::getTenantOwnedEloquentQuery();
|
||||||
|
}
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
public static function resolveScopedRecordOrFail(int|string $key): \Illuminate\Database\Eloquent\Model
|
||||||
return parent::getEloquentQuery()->whereRaw('1 = 0');
|
{
|
||||||
}
|
return static::resolveTenantOwnedRecordOrFail($key, parent::getEloquentQuery()->withTrashed());
|
||||||
|
|
||||||
return parent::getEloquentQuery()->where('tenant_id', (int) $tenant->getKey());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function form(Schema $schema): Schema
|
public static function form(Schema $schema): Schema
|
||||||
|
|||||||
@ -17,6 +17,7 @@
|
|||||||
use Filament\Actions\ActionGroup;
|
use Filament\Actions\ActionGroup;
|
||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewBackupSet extends ViewRecord
|
class ViewBackupSet extends ViewRecord
|
||||||
{
|
{
|
||||||
@ -24,6 +25,11 @@ class ViewBackupSet extends ViewRecord
|
|||||||
|
|
||||||
protected static string $resource = BackupSetResource::class;
|
protected static string $resource = BackupSetResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return BackupSetResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
$actions = [
|
$actions = [
|
||||||
|
|||||||
@ -43,6 +43,27 @@ public function closeAddPoliciesModal(): void
|
|||||||
$this->unmountAction();
|
$this->unmountAction();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true) {
|
||||||
|
$backupSet = $this->getOwnerRecord();
|
||||||
|
|
||||||
|
if ($name === 'remove' && filled($context['recordKey'] ?? null)) {
|
||||||
|
$this->resolveOwnerScopedBackupItemId($backupSet, $context['recordKey']);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($name === 'bulk_remove' && ($context['bulk'] ?? false) === true) {
|
||||||
|
$this->resolveOwnerScopedBackupItemIdsFromKeys($backupSet, $this->selectedTableRecords);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
public function table(Table $table): Table
|
public function table(Table $table): Table
|
||||||
{
|
{
|
||||||
$refreshTable = Actions\Action::make('refreshTable')
|
$refreshTable = Actions\Action::make('refreshTable')
|
||||||
@ -77,7 +98,7 @@ public function table(Table $table): Table
|
|||||||
->color('danger')
|
->color('danger')
|
||||||
->icon('heroicon-o-x-mark')
|
->icon('heroicon-o-x-mark')
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->action(function (BackupItem $record): void {
|
->action(function (mixed $record): void {
|
||||||
$backupSet = $this->getOwnerRecord();
|
$backupSet = $this->getOwnerRecord();
|
||||||
|
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
@ -94,7 +115,7 @@ public function table(Table $table): Table
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$backupItemIds = [(int) $record->getKey()];
|
$backupItemIds = [$this->resolveOwnerScopedBackupItemId($backupSet, $record)];
|
||||||
|
|
||||||
/** @var OperationRunService $opService */
|
/** @var OperationRunService $opService */
|
||||||
$opService = app(OperationRunService::class);
|
$opService = app(OperationRunService::class);
|
||||||
@ -173,14 +194,7 @@ public function table(Table $table): Table
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
$backupItemIds = $records
|
$backupItemIds = $this->resolveOwnerScopedBackupItemIdsFromKeys($backupSet, $this->selectedTableRecords);
|
||||||
->pluck('id')
|
|
||||||
->map(fn (mixed $value): int => (int) $value)
|
|
||||||
->filter(fn (int $value): bool => $value > 0)
|
|
||||||
->unique()
|
|
||||||
->sort()
|
|
||||||
->values()
|
|
||||||
->all();
|
|
||||||
|
|
||||||
if ($backupItemIds === []) {
|
if ($backupItemIds === []) {
|
||||||
return;
|
return;
|
||||||
@ -434,4 +448,68 @@ private static function applyRestoreModeFilter(Builder $query, mixed $value): Bu
|
|||||||
|
|
||||||
return $query->whereIn('policy_type', $types);
|
return $query->whereIn('policy_type', $types);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function resolveOwnerScopedBackupItemId(\App\Models\BackupSet $backupSet, mixed $record): int
|
||||||
|
{
|
||||||
|
$recordId = $this->normalizeBackupItemKey($record);
|
||||||
|
|
||||||
|
if ($recordId <= 0) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolvedId = $backupSet->items()
|
||||||
|
->where('tenant_id', (int) $backupSet->tenant_id)
|
||||||
|
->whereKey($recordId)
|
||||||
|
->value('id');
|
||||||
|
|
||||||
|
if (! is_numeric($resolvedId) || (int) $resolvedId <= 0) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (int) $resolvedId;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, int>
|
||||||
|
*/
|
||||||
|
private function resolveOwnerScopedBackupItemIdsFromKeys(\App\Models\BackupSet $backupSet, array $recordKeys): array
|
||||||
|
{
|
||||||
|
$requestedIds = collect($recordKeys)
|
||||||
|
->map(fn (mixed $record): int => $this->normalizeBackupItemKey($record))
|
||||||
|
->filter(fn (int $value): bool => $value > 0)
|
||||||
|
->unique()
|
||||||
|
->sort()
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
|
||||||
|
if ($requestedIds === []) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolvedIds = $backupSet->items()
|
||||||
|
->where('tenant_id', (int) $backupSet->tenant_id)
|
||||||
|
->whereIn('id', $requestedIds)
|
||||||
|
->pluck('id')
|
||||||
|
->map(fn (mixed $value): int => (int) $value)
|
||||||
|
->filter(fn (int $value): bool => $value > 0)
|
||||||
|
->unique()
|
||||||
|
->sort()
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
|
||||||
|
if (count($resolvedIds) !== count($requestedIds)) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedIds;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function normalizeBackupItemKey(mixed $record): int
|
||||||
|
{
|
||||||
|
if ($record instanceof BackupItem) {
|
||||||
|
return (int) $record->getKey();
|
||||||
|
}
|
||||||
|
|
||||||
|
return is_numeric($record) ? (int) $record : 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,19 +6,28 @@
|
|||||||
|
|
||||||
use App\Filament\Resources\BaselineProfileResource\Pages;
|
use App\Filament\Resources\BaselineProfileResource\Pages;
|
||||||
use App\Models\BaselineProfile;
|
use App\Models\BaselineProfile;
|
||||||
|
use App\Models\BaselineSnapshot;
|
||||||
|
use App\Models\BaselineTenantAssignment;
|
||||||
|
use App\Models\Tenant;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\Workspace;
|
use App\Models\Workspace;
|
||||||
use App\Services\Audit\WorkspaceAuditLogger;
|
use App\Services\Audit\WorkspaceAuditLogger;
|
||||||
|
use App\Services\Auth\CapabilityResolver;
|
||||||
|
use App\Services\Baselines\BaselineSnapshotTruthResolver;
|
||||||
use App\Support\Audit\AuditActionId;
|
use App\Support\Audit\AuditActionId;
|
||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
use App\Support\Badges\BadgeRenderer;
|
use App\Support\Badges\BadgeRenderer;
|
||||||
use App\Support\Baselines\BaselineCaptureMode;
|
use App\Support\Baselines\BaselineCaptureMode;
|
||||||
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
||||||
use App\Support\Baselines\BaselineProfileStatus;
|
use App\Support\Baselines\BaselineProfileStatus;
|
||||||
|
use App\Support\Baselines\BaselineReasonCodes;
|
||||||
use App\Support\Filament\FilterOptionCatalog;
|
use App\Support\Filament\FilterOptionCatalog;
|
||||||
use App\Support\Inventory\InventoryPolicyTypeMeta;
|
use App\Support\Inventory\InventoryPolicyTypeMeta;
|
||||||
use App\Support\Rbac\WorkspaceUiEnforcement;
|
use App\Support\Rbac\WorkspaceUiEnforcement;
|
||||||
|
use App\Support\ReasonTranslation\ReasonPresenter;
|
||||||
|
use App\Support\ReasonTranslation\ReasonResolutionEnvelope;
|
||||||
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
@ -288,15 +297,32 @@ public static function infolist(Schema $schema): Schema
|
|||||||
->placeholder('None'),
|
->placeholder('None'),
|
||||||
])
|
])
|
||||||
->columnSpanFull(),
|
->columnSpanFull(),
|
||||||
|
Section::make('Baseline truth')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('current_snapshot_truth')
|
||||||
|
->label('Current snapshot')
|
||||||
|
->state(fn (BaselineProfile $record): string => self::currentSnapshotLabel($record)),
|
||||||
|
TextEntry::make('latest_attempted_snapshot_truth')
|
||||||
|
->label('Latest attempt')
|
||||||
|
->state(fn (BaselineProfile $record): string => self::latestAttemptedSnapshotLabel($record)),
|
||||||
|
TextEntry::make('compare_readiness')
|
||||||
|
->label('Compare readiness')
|
||||||
|
->badge()
|
||||||
|
->state(fn (BaselineProfile $record): string => self::compareReadinessLabel($record))
|
||||||
|
->color(fn (BaselineProfile $record): string => self::compareReadinessColor($record))
|
||||||
|
->icon(fn (BaselineProfile $record): ?string => self::compareReadinessIcon($record)),
|
||||||
|
TextEntry::make('baseline_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->state(fn (BaselineProfile $record): string => self::profileNextStep($record))
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(2)
|
||||||
|
->columnSpanFull(),
|
||||||
Section::make('Metadata')
|
Section::make('Metadata')
|
||||||
->schema([
|
->schema([
|
||||||
TextEntry::make('createdByUser.name')
|
TextEntry::make('createdByUser.name')
|
||||||
->label('Created by')
|
->label('Created by')
|
||||||
->placeholder('—'),
|
->placeholder('—'),
|
||||||
TextEntry::make('activeSnapshot.captured_at')
|
|
||||||
->label('Last snapshot')
|
|
||||||
->dateTime()
|
|
||||||
->placeholder('No snapshot yet'),
|
|
||||||
TextEntry::make('created_at')
|
TextEntry::make('created_at')
|
||||||
->dateTime(),
|
->dateTime(),
|
||||||
TextEntry::make('updated_at')
|
TextEntry::make('updated_at')
|
||||||
@ -355,10 +381,27 @@ public static function table(Table $table): Table
|
|||||||
TextColumn::make('tenant_assignments_count')
|
TextColumn::make('tenant_assignments_count')
|
||||||
->label('Assigned tenants')
|
->label('Assigned tenants')
|
||||||
->counts('tenantAssignments'),
|
->counts('tenantAssignments'),
|
||||||
TextColumn::make('activeSnapshot.captured_at')
|
TextColumn::make('current_snapshot_truth')
|
||||||
->label('Last snapshot')
|
->label('Current snapshot')
|
||||||
->dateTime()
|
->getStateUsing(fn (BaselineProfile $record): string => self::currentSnapshotLabel($record))
|
||||||
->placeholder('No snapshot'),
|
->description(fn (BaselineProfile $record): ?string => self::currentSnapshotDescription($record))
|
||||||
|
->wrap(),
|
||||||
|
TextColumn::make('latest_attempted_snapshot_truth')
|
||||||
|
->label('Latest attempt')
|
||||||
|
->getStateUsing(fn (BaselineProfile $record): string => self::latestAttemptedSnapshotLabel($record))
|
||||||
|
->description(fn (BaselineProfile $record): ?string => self::latestAttemptedSnapshotDescription($record))
|
||||||
|
->wrap(),
|
||||||
|
TextColumn::make('compare_readiness')
|
||||||
|
->label('Compare readiness')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (BaselineProfile $record): string => self::compareReadinessLabel($record))
|
||||||
|
->color(fn (BaselineProfile $record): string => self::compareReadinessColor($record))
|
||||||
|
->icon(fn (BaselineProfile $record): ?string => self::compareReadinessIcon($record))
|
||||||
|
->wrap(),
|
||||||
|
TextColumn::make('baseline_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->getStateUsing(fn (BaselineProfile $record): string => self::profileNextStep($record))
|
||||||
|
->wrap(),
|
||||||
TextColumn::make('created_at')
|
TextColumn::make('created_at')
|
||||||
->dateTime()
|
->dateTime()
|
||||||
->sortable()
|
->sortable()
|
||||||
@ -545,4 +588,167 @@ private static function archiveTableAction(?Workspace $workspace): Action
|
|||||||
|
|
||||||
return $action;
|
return $action;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static function currentSnapshotLabel(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
$snapshot = self::effectiveSnapshot($profile);
|
||||||
|
|
||||||
|
if (! $snapshot instanceof BaselineSnapshot) {
|
||||||
|
return 'No complete snapshot';
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::snapshotReference($snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentSnapshotDescription(BaselineProfile $profile): ?string
|
||||||
|
{
|
||||||
|
$snapshot = self::effectiveSnapshot($profile);
|
||||||
|
|
||||||
|
if (! $snapshot instanceof BaselineSnapshot) {
|
||||||
|
return self::compareAvailabilityEnvelope($profile)?->shortExplanation;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $snapshot->captured_at?->toDayDateTimeString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function latestAttemptedSnapshotLabel(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
$latestAttempt = self::latestAttemptedSnapshot($profile);
|
||||||
|
|
||||||
|
if (! $latestAttempt instanceof BaselineSnapshot) {
|
||||||
|
return 'No capture attempts yet';
|
||||||
|
}
|
||||||
|
|
||||||
|
$effectiveSnapshot = self::effectiveSnapshot($profile);
|
||||||
|
|
||||||
|
if ($effectiveSnapshot instanceof BaselineSnapshot && (int) $effectiveSnapshot->getKey() === (int) $latestAttempt->getKey()) {
|
||||||
|
return 'Matches current snapshot';
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::snapshotReference($latestAttempt);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function latestAttemptedSnapshotDescription(BaselineProfile $profile): ?string
|
||||||
|
{
|
||||||
|
$latestAttempt = self::latestAttemptedSnapshot($profile);
|
||||||
|
|
||||||
|
if (! $latestAttempt instanceof BaselineSnapshot) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$effectiveSnapshot = self::effectiveSnapshot($profile);
|
||||||
|
|
||||||
|
if ($effectiveSnapshot instanceof BaselineSnapshot && (int) $effectiveSnapshot->getKey() === (int) $latestAttempt->getKey()) {
|
||||||
|
return 'No newer attempt is pending.';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $latestAttempt->captured_at?->toDayDateTimeString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compareReadinessLabel(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
return self::compareAvailabilityEnvelope($profile)?->operatorLabel ?? 'Ready';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compareReadinessColor(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
return match (self::compareAvailabilityReason($profile)) {
|
||||||
|
null => 'success',
|
||||||
|
BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE => 'gray',
|
||||||
|
default => 'warning',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compareReadinessIcon(BaselineProfile $profile): ?string
|
||||||
|
{
|
||||||
|
return match (self::compareAvailabilityReason($profile)) {
|
||||||
|
null => 'heroicon-m-check-badge',
|
||||||
|
BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE => 'heroicon-m-pause-circle',
|
||||||
|
default => 'heroicon-m-exclamation-triangle',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function profileNextStep(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
return self::compareAvailabilityEnvelope($profile)?->guidanceText() ?? 'No action needed.';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function effectiveSnapshot(BaselineProfile $profile): ?BaselineSnapshot
|
||||||
|
{
|
||||||
|
return app(BaselineSnapshotTruthResolver::class)->resolveEffectiveSnapshot($profile);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function latestAttemptedSnapshot(BaselineProfile $profile): ?BaselineSnapshot
|
||||||
|
{
|
||||||
|
return app(BaselineSnapshotTruthResolver::class)->resolveLatestAttemptedSnapshot($profile);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compareAvailabilityReason(BaselineProfile $profile): ?string
|
||||||
|
{
|
||||||
|
$status = $profile->status instanceof BaselineProfileStatus
|
||||||
|
? $profile->status
|
||||||
|
: BaselineProfileStatus::tryFrom((string) $profile->status);
|
||||||
|
|
||||||
|
if ($status !== BaselineProfileStatus::Active) {
|
||||||
|
return BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE;
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolution = app(BaselineSnapshotTruthResolver::class)->resolveCompareSnapshot($profile);
|
||||||
|
$reasonCode = $resolution['reason_code'] ?? null;
|
||||||
|
|
||||||
|
if (is_string($reasonCode) && trim($reasonCode) !== '') {
|
||||||
|
return trim($reasonCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! self::hasEligibleCompareTarget($profile)) {
|
||||||
|
return BaselineReasonCodes::COMPARE_NO_ELIGIBLE_TARGET;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compareAvailabilityEnvelope(BaselineProfile $profile): ?ReasonResolutionEnvelope
|
||||||
|
{
|
||||||
|
$reasonCode = self::compareAvailabilityReason($profile);
|
||||||
|
|
||||||
|
if (! is_string($reasonCode)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return app(ReasonPresenter::class)->forArtifactTruth($reasonCode, 'artifact_truth');
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function snapshotReference(BaselineSnapshot $snapshot): string
|
||||||
|
{
|
||||||
|
$lifecycleLabel = BadgeCatalog::spec(BadgeDomain::BaselineSnapshotLifecycle, $snapshot->lifecycleState()->value)->label;
|
||||||
|
|
||||||
|
return sprintf('Snapshot #%d (%s)', (int) $snapshot->getKey(), $lifecycleLabel);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hasEligibleCompareTarget(BaselineProfile $profile): bool
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenantIds = BaselineTenantAssignment::query()
|
||||||
|
->where('workspace_id', (int) $profile->workspace_id)
|
||||||
|
->where('baseline_profile_id', (int) $profile->getKey())
|
||||||
|
->pluck('tenant_id')
|
||||||
|
->all();
|
||||||
|
|
||||||
|
if ($tenantIds === []) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolver = app(CapabilityResolver::class);
|
||||||
|
|
||||||
|
return Tenant::query()
|
||||||
|
->where('workspace_id', (int) $profile->workspace_id)
|
||||||
|
->whereIn('id', $tenantIds)
|
||||||
|
->get(['id'])
|
||||||
|
->contains(fn (Tenant $tenant): bool => $resolver->can($user, $tenant, Capabilities::TENANT_SYNC));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -183,7 +183,7 @@ private function compareNowAction(): Action
|
|||||||
|
|
||||||
$modalDescription = $captureMode === BaselineCaptureMode::FullContent
|
$modalDescription = $captureMode === BaselineCaptureMode::FullContent
|
||||||
? 'Select the target tenant. This will refresh content evidence on demand (redacted) before comparing.'
|
? 'Select the target tenant. This will refresh content evidence on demand (redacted) before comparing.'
|
||||||
: 'Select the target tenant to compare its current inventory against the active baseline snapshot.';
|
: 'Select the target tenant to compare its current inventory against the effective current baseline snapshot.';
|
||||||
|
|
||||||
return Action::make('compareNow')
|
return Action::make('compareNow')
|
||||||
->label($label)
|
->label($label)
|
||||||
@ -198,7 +198,7 @@ private function compareNowAction(): Action
|
|||||||
->required()
|
->required()
|
||||||
->searchable(),
|
->searchable(),
|
||||||
])
|
])
|
||||||
->disabled(fn (): bool => $this->getEligibleCompareTenantOptions() === [])
|
->disabled(fn (): bool => $this->getEligibleCompareTenantOptions() === [] || ! $this->profileHasConsumableSnapshot())
|
||||||
->action(function (array $data): void {
|
->action(function (array $data): void {
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
|
|
||||||
@ -256,7 +256,11 @@ private function compareNowAction(): Action
|
|||||||
$message = match ($reasonCode) {
|
$message = match ($reasonCode) {
|
||||||
BaselineReasonCodes::COMPARE_ROLLOUT_DISABLED => 'Full-content baseline compare is currently disabled for controlled rollout.',
|
BaselineReasonCodes::COMPARE_ROLLOUT_DISABLED => 'Full-content baseline compare is currently disabled for controlled rollout.',
|
||||||
BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE => 'This baseline profile is not active.',
|
BaselineReasonCodes::COMPARE_PROFILE_NOT_ACTIVE => 'This baseline profile is not active.',
|
||||||
BaselineReasonCodes::COMPARE_NO_ACTIVE_SNAPSHOT => 'This baseline profile has no active snapshot.',
|
BaselineReasonCodes::COMPARE_NO_ACTIVE_SNAPSHOT,
|
||||||
|
BaselineReasonCodes::COMPARE_NO_CONSUMABLE_SNAPSHOT => 'This baseline profile has no complete snapshot available for compare yet.',
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_BUILDING => 'The latest baseline capture is still building. Compare will be available after it completes.',
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_INCOMPLETE => 'The latest baseline capture is incomplete. Capture a new baseline before comparing.',
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_SUPERSEDED => 'A newer complete snapshot is current. Compare uses the latest complete baseline only.',
|
||||||
default => 'Reason: '.str_replace('.', ' ', $reasonCode),
|
default => 'Reason: '.str_replace('.', ' ', $reasonCode),
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -395,4 +399,12 @@ private function hasManageCapability(): bool
|
|||||||
return $resolver->isMember($user, $workspace)
|
return $resolver->isMember($user, $workspace)
|
||||||
&& $resolver->can($user, $workspace, Capabilities::WORKSPACE_BASELINES_MANAGE);
|
&& $resolver->can($user, $workspace, Capabilities::WORKSPACE_BASELINES_MANAGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function profileHasConsumableSnapshot(): bool
|
||||||
|
{
|
||||||
|
/** @var BaselineProfile $profile */
|
||||||
|
$profile = $this->getRecord();
|
||||||
|
|
||||||
|
return $profile->resolveCurrentConsumableSnapshot() !== null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -9,7 +9,12 @@
|
|||||||
use App\Models\BaselineSnapshot;
|
use App\Models\BaselineSnapshot;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\Workspace;
|
use App\Models\Workspace;
|
||||||
|
use App\Services\Baselines\BaselineSnapshotTruthResolver;
|
||||||
|
use App\Services\Baselines\SnapshotRendering\FidelityState;
|
||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Baselines\BaselineSnapshotLifecycleState;
|
||||||
use App\Support\Filament\FilterPresets;
|
use App\Support\Filament\FilterPresets;
|
||||||
use App\Support\Navigation\CrossResourceNavigationMatrix;
|
use App\Support\Navigation\CrossResourceNavigationMatrix;
|
||||||
use App\Support\Navigation\RelatedContextEntry;
|
use App\Support\Navigation\RelatedContextEntry;
|
||||||
@ -18,6 +23,8 @@
|
|||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthEnvelope;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use BackedEnum;
|
use BackedEnum;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
@ -165,15 +172,39 @@ public static function table(Table $table): Table
|
|||||||
->label('Captured')
|
->label('Captured')
|
||||||
->since()
|
->since()
|
||||||
->sortable(),
|
->sortable(),
|
||||||
|
TextColumn::make('artifact_truth')
|
||||||
|
->label('Artifact truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(static fn (BaselineSnapshot $record): string => self::truthEnvelope($record)->primaryLabel)
|
||||||
|
->color(static fn (BaselineSnapshot $record): string => self::truthEnvelope($record)->primaryBadgeSpec()->color)
|
||||||
|
->icon(static fn (BaselineSnapshot $record): ?string => self::truthEnvelope($record)->primaryBadgeSpec()->icon)
|
||||||
|
->iconColor(static fn (BaselineSnapshot $record): ?string => self::truthEnvelope($record)->primaryBadgeSpec()->iconColor)
|
||||||
|
->description(static fn (BaselineSnapshot $record): ?string => self::truthEnvelope($record)->primaryExplanation)
|
||||||
|
->wrap(),
|
||||||
|
TextColumn::make('lifecycle_state')
|
||||||
|
->label('Lifecycle')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(static fn (BaselineSnapshot $record): string => self::lifecycleSpec($record)->label)
|
||||||
|
->color(static fn (BaselineSnapshot $record): string => self::lifecycleSpec($record)->color)
|
||||||
|
->icon(static fn (BaselineSnapshot $record): ?string => self::lifecycleSpec($record)->icon)
|
||||||
|
->iconColor(static fn (BaselineSnapshot $record): ?string => self::lifecycleSpec($record)->iconColor)
|
||||||
|
->sortable(),
|
||||||
|
TextColumn::make('current_truth')
|
||||||
|
->label('Current truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(static fn (BaselineSnapshot $record): string => self::currentTruthLabel($record))
|
||||||
|
->color(static fn (BaselineSnapshot $record): string => self::currentTruthColor($record))
|
||||||
|
->icon(static fn (BaselineSnapshot $record): ?string => self::currentTruthIcon($record))
|
||||||
|
->description(static fn (BaselineSnapshot $record): ?string => self::currentTruthDescription($record))
|
||||||
|
->wrap(),
|
||||||
TextColumn::make('fidelity_summary')
|
TextColumn::make('fidelity_summary')
|
||||||
->label('Fidelity')
|
->label('Fidelity')
|
||||||
->getStateUsing(static fn (BaselineSnapshot $record): string => self::fidelitySummary($record))
|
->getStateUsing(static fn (BaselineSnapshot $record): string => self::fidelitySummary($record))
|
||||||
->wrap(),
|
->wrap(),
|
||||||
TextColumn::make('snapshot_state')
|
TextColumn::make('artifact_next_step')
|
||||||
->label('State')
|
->label('Next step')
|
||||||
->badge()
|
->getStateUsing(static fn (BaselineSnapshot $record): string => self::truthEnvelope($record)->nextStepText())
|
||||||
->getStateUsing(static fn (BaselineSnapshot $record): string => self::stateLabel($record))
|
->wrap(),
|
||||||
->color(static fn (BaselineSnapshot $record): string => self::hasGaps($record) ? 'warning' : 'success'),
|
|
||||||
])
|
])
|
||||||
->recordUrl(static fn (BaselineSnapshot $record): ?string => static::canView($record)
|
->recordUrl(static fn (BaselineSnapshot $record): ?string => static::canView($record)
|
||||||
? static::getUrl('view', ['record' => $record])
|
? static::getUrl('view', ['record' => $record])
|
||||||
@ -183,10 +214,10 @@ public static function table(Table $table): Table
|
|||||||
->label('Baseline')
|
->label('Baseline')
|
||||||
->options(static::baselineProfileOptions())
|
->options(static::baselineProfileOptions())
|
||||||
->searchable(),
|
->searchable(),
|
||||||
SelectFilter::make('snapshot_state')
|
SelectFilter::make('lifecycle_state')
|
||||||
->label('State')
|
->label('Lifecycle')
|
||||||
->options(static::snapshotStateOptions())
|
->options(static::lifecycleOptions())
|
||||||
->query(fn (Builder $query, array $data): Builder => static::applySnapshotStateFilter($query, $data['value'] ?? null)),
|
->query(fn (Builder $query, array $data): Builder => static::applyLifecycleFilter($query, $data['value'] ?? null)),
|
||||||
FilterPresets::dateRange('captured_at', 'Captured', 'captured_at'),
|
FilterPresets::dateRange('captured_at', 'Captured', 'captured_at'),
|
||||||
])
|
])
|
||||||
->actions([
|
->actions([
|
||||||
@ -247,12 +278,9 @@ private static function baselineProfileOptions(): array
|
|||||||
/**
|
/**
|
||||||
* @return array<string, string>
|
* @return array<string, string>
|
||||||
*/
|
*/
|
||||||
private static function snapshotStateOptions(): array
|
private static function lifecycleOptions(): array
|
||||||
{
|
{
|
||||||
return [
|
return BadgeCatalog::options(BadgeDomain::BaselineSnapshotLifecycle, BaselineSnapshotLifecycleState::values());
|
||||||
'complete' => 'Complete',
|
|
||||||
'with_gaps' => 'Captured with gaps',
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function resolveWorkspace(): ?Workspace
|
public static function resolveWorkspace(): ?Workspace
|
||||||
@ -290,7 +318,13 @@ private static function fidelitySummary(BaselineSnapshot $snapshot): string
|
|||||||
{
|
{
|
||||||
$counts = self::fidelityCounts($snapshot);
|
$counts = self::fidelityCounts($snapshot);
|
||||||
|
|
||||||
return sprintf('Content %d, Meta %d', (int) ($counts['content'] ?? 0), (int) ($counts['meta'] ?? 0));
|
return sprintf(
|
||||||
|
'%s %d, %s %d',
|
||||||
|
BadgeCatalog::spec(BadgeDomain::BaselineSnapshotFidelity, FidelityState::Full->value)->label,
|
||||||
|
(int) ($counts['content'] ?? 0),
|
||||||
|
BadgeCatalog::spec(BadgeDomain::BaselineSnapshotFidelity, FidelityState::ReferenceOnly->value)->label,
|
||||||
|
(int) ($counts['meta'] ?? 0),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function gapsCount(BaselineSnapshot $snapshot): int
|
private static function gapsCount(BaselineSnapshot $snapshot): int
|
||||||
@ -298,6 +332,17 @@ private static function gapsCount(BaselineSnapshot $snapshot): int
|
|||||||
$summary = self::summary($snapshot);
|
$summary = self::summary($snapshot);
|
||||||
$gaps = $summary['gaps'] ?? null;
|
$gaps = $summary['gaps'] ?? null;
|
||||||
$gaps = is_array($gaps) ? $gaps : [];
|
$gaps = is_array($gaps) ? $gaps : [];
|
||||||
|
$byReason = is_array($gaps['by_reason'] ?? null) ? $gaps['by_reason'] : [];
|
||||||
|
|
||||||
|
if ($byReason !== []) {
|
||||||
|
return array_sum(array_map(
|
||||||
|
static fn (mixed $count, string $reason): int => in_array($reason, ['meta_fallback'], true) || ! is_numeric($count)
|
||||||
|
? 0
|
||||||
|
: (int) $count,
|
||||||
|
$byReason,
|
||||||
|
array_keys($byReason),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
$count = $gaps['count'] ?? 0;
|
$count = $gaps['count'] ?? 0;
|
||||||
|
|
||||||
@ -309,32 +354,86 @@ private static function hasGaps(BaselineSnapshot $snapshot): bool
|
|||||||
return self::gapsCount($snapshot) > 0;
|
return self::gapsCount($snapshot) > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function stateLabel(BaselineSnapshot $snapshot): string
|
private static function lifecycleSpec(BaselineSnapshot $snapshot): \App\Support\Badges\BadgeSpec
|
||||||
{
|
{
|
||||||
return self::hasGaps($snapshot) ? 'Captured with gaps' : 'Complete';
|
return BadgeCatalog::spec(BadgeDomain::BaselineSnapshotLifecycle, $snapshot->lifecycleState()->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function applySnapshotStateFilter(Builder $query, mixed $value): Builder
|
private static function applyLifecycleFilter(Builder $query, mixed $value): Builder
|
||||||
{
|
{
|
||||||
if (! is_string($value) || trim($value) === '') {
|
if (! is_string($value) || trim($value) === '') {
|
||||||
return $query;
|
return $query;
|
||||||
}
|
}
|
||||||
|
|
||||||
$gapCountExpression = self::gapCountExpression($query);
|
return $query->where('lifecycle_state', trim($value));
|
||||||
|
|
||||||
return match ($value) {
|
|
||||||
'complete' => $query->whereRaw("{$gapCountExpression} = 0"),
|
|
||||||
'with_gaps' => $query->whereRaw("{$gapCountExpression} > 0"),
|
|
||||||
default => $query,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function gapCountExpression(Builder $query): string
|
private static function gapCountExpression(Builder $query): string
|
||||||
{
|
{
|
||||||
return match ($query->getConnection()->getDriverName()) {
|
return match ($query->getConnection()->getDriverName()) {
|
||||||
'sqlite' => "COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.count') AS INTEGER), 0)",
|
'sqlite' => "COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.missing_evidence') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.missing_role_definition_version_reference') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.invalid_subject') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.duplicate_subject') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.policy_not_found') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.throttled') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.capture_failed') AS INTEGER), 0) + COALESCE(CAST(json_extract(summary_jsonb, '$.gaps.by_reason.budget_exhausted') AS INTEGER), 0)",
|
||||||
'pgsql' => "COALESCE((summary_jsonb #>> '{gaps,count}')::int, 0)",
|
'pgsql' => "COALESCE((summary_jsonb #>> '{gaps,by_reason,missing_evidence}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,missing_role_definition_version_reference}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,invalid_subject}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,duplicate_subject}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,policy_not_found}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,throttled}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,capture_failed}')::int, 0) + COALESCE((summary_jsonb #>> '{gaps,by_reason,budget_exhausted}')::int, 0)",
|
||||||
default => "COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.count') AS UNSIGNED), 0)",
|
default => "COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.missing_evidence') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.missing_role_definition_version_reference') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.invalid_subject') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.duplicate_subject') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.policy_not_found') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.throttled') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.capture_failed') AS UNSIGNED), 0) + COALESCE(CAST(JSON_EXTRACT(summary_jsonb, '$.gaps.by_reason.budget_exhausted') AS UNSIGNED), 0)",
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static function gapSpec(BaselineSnapshot $snapshot): \App\Support\Badges\BadgeSpec
|
||||||
|
{
|
||||||
|
return BadgeCatalog::spec(
|
||||||
|
BadgeDomain::BaselineSnapshotGapStatus,
|
||||||
|
self::hasGaps($snapshot) ? 'gaps_present' : 'clear',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function truthEnvelope(BaselineSnapshot $snapshot): ArtifactTruthEnvelope
|
||||||
|
{
|
||||||
|
return app(ArtifactTruthPresenter::class)->forBaselineSnapshot($snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentTruthLabel(BaselineSnapshot $snapshot): string
|
||||||
|
{
|
||||||
|
return match (self::currentTruthState($snapshot)) {
|
||||||
|
'current' => 'Current baseline',
|
||||||
|
'historical' => 'Historical trace',
|
||||||
|
default => 'Not compare input',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentTruthDescription(BaselineSnapshot $snapshot): ?string
|
||||||
|
{
|
||||||
|
return match (self::currentTruthState($snapshot)) {
|
||||||
|
'current' => 'Compare resolves to this snapshot as the current baseline truth.',
|
||||||
|
'historical' => 'A newer complete snapshot is now the current baseline truth for this profile.',
|
||||||
|
default => self::truthEnvelope($snapshot)->primaryExplanation,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentTruthColor(BaselineSnapshot $snapshot): string
|
||||||
|
{
|
||||||
|
return match (self::currentTruthState($snapshot)) {
|
||||||
|
'current' => 'success',
|
||||||
|
'historical' => 'gray',
|
||||||
|
default => 'warning',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentTruthIcon(BaselineSnapshot $snapshot): ?string
|
||||||
|
{
|
||||||
|
return match (self::currentTruthState($snapshot)) {
|
||||||
|
'current' => 'heroicon-m-check-badge',
|
||||||
|
'historical' => 'heroicon-m-clock',
|
||||||
|
default => 'heroicon-m-exclamation-triangle',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentTruthState(BaselineSnapshot $snapshot): string
|
||||||
|
{
|
||||||
|
if (! $snapshot->isConsumable()) {
|
||||||
|
return 'unusable';
|
||||||
|
}
|
||||||
|
|
||||||
|
return app(BaselineSnapshotTruthResolver::class)->isHistoricallySuperseded($snapshot)
|
||||||
|
? 'historical'
|
||||||
|
: 'current';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -35,6 +35,8 @@ public function mount(int|string $record): void
|
|||||||
$snapshot = $this->getRecord();
|
$snapshot = $this->getRecord();
|
||||||
|
|
||||||
if ($snapshot instanceof BaselineSnapshot) {
|
if ($snapshot instanceof BaselineSnapshot) {
|
||||||
|
$snapshot->loadMissing(['baselineProfile', 'items']);
|
||||||
|
|
||||||
$relatedContext = app(RelatedNavigationResolver::class)
|
$relatedContext = app(RelatedNavigationResolver::class)
|
||||||
->detailEntries(CrossResourceNavigationMatrix::SOURCE_BASELINE_SNAPSHOT, $snapshot);
|
->detailEntries(CrossResourceNavigationMatrix::SOURCE_BASELINE_SNAPSHOT, $snapshot);
|
||||||
|
|
||||||
|
|||||||
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Concerns\ScopesGlobalSearchToTenant;
|
use App\Filament\Concerns\ScopesGlobalSearchToTenant;
|
||||||
use App\Filament\Resources\EntraGroupResource\Pages;
|
use App\Filament\Resources\EntraGroupResource\Pages;
|
||||||
use App\Models\EntraGroup;
|
use App\Models\EntraGroup;
|
||||||
@ -9,7 +11,6 @@
|
|||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
use App\Support\Badges\BadgeRenderer;
|
use App\Support\Badges\BadgeRenderer;
|
||||||
use App\Support\Filament\TablePaginationProfiles;
|
use App\Support\Filament\TablePaginationProfiles;
|
||||||
use App\Support\OperateHub\OperateHubShell;
|
|
||||||
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
@ -33,12 +34,16 @@
|
|||||||
|
|
||||||
class EntraGroupResource extends Resource
|
class EntraGroupResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
|
use ResolvesPanelTenantContext;
|
||||||
use ScopesGlobalSearchToTenant;
|
use ScopesGlobalSearchToTenant;
|
||||||
|
|
||||||
protected static bool $isScopedToTenant = false;
|
protected static bool $isScopedToTenant = false;
|
||||||
|
|
||||||
protected static ?string $model = EntraGroup::class;
|
protected static ?string $model = EntraGroup::class;
|
||||||
|
|
||||||
|
protected static ?string $tenantOwnershipRelationshipName = 'tenant';
|
||||||
|
|
||||||
protected static ?string $recordTitleAttribute = 'display_name';
|
protected static ?string $recordTitleAttribute = 'display_name';
|
||||||
|
|
||||||
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-user-group';
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-user-group';
|
||||||
@ -188,17 +193,15 @@ public static function table(Table $table): Table
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenant = static::panelTenantContext();
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
|
||||||
->when(
|
|
||||||
$tenant instanceof Tenant,
|
|
||||||
fn (Builder $query): Builder => $query->where('tenant_id', (int) $tenant->getKey()),
|
|
||||||
fn (Builder $query): Builder => $query->whereRaw('1 = 0'),
|
|
||||||
)
|
|
||||||
->latest('id');
|
->latest('id');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string $key): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
public static function getGlobalSearchResultUrl(Model $record): string
|
public static function getGlobalSearchResultUrl(Model $record): string
|
||||||
{
|
{
|
||||||
$tenant = $record instanceof EntraGroup && $record->tenant instanceof Tenant
|
$tenant = $record instanceof EntraGroup && $record->tenant instanceof Tenant
|
||||||
@ -216,19 +219,6 @@ public static function getPages(): array
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function panelTenantContext(): ?Tenant
|
|
||||||
{
|
|
||||||
if (Filament::getCurrentPanel()?->getId() === 'admin') {
|
|
||||||
$tenant = app(OperateHubShell::class)->activeEntitledTenant(request());
|
|
||||||
|
|
||||||
return $tenant instanceof Tenant ? $tenant : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$tenant = Tenant::current();
|
|
||||||
|
|
||||||
return $tenant instanceof Tenant ? $tenant : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array<string, mixed> $parameters
|
* @param array<string, mixed> $parameters
|
||||||
*/
|
*/
|
||||||
|
|||||||
@ -8,11 +8,17 @@
|
|||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewEntraGroup extends ViewRecord
|
class ViewEntraGroup extends ViewRecord
|
||||||
{
|
{
|
||||||
protected static string $resource = EntraGroupResource::class;
|
protected static string $resource = EntraGroupResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return EntraGroupResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
protected function authorizeAccess(): void
|
protected function authorizeAccess(): void
|
||||||
{
|
{
|
||||||
$tenant = EntraGroupResource::panelTenantContext();
|
$tenant = EntraGroupResource::panelTenantContext();
|
||||||
|
|||||||
676
app/Filament/Resources/EvidenceSnapshotResource.php
Normal file
676
app/Filament/Resources/EvidenceSnapshotResource.php
Normal file
@ -0,0 +1,676 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
|
use App\Filament\Resources\EvidenceSnapshotResource\Pages;
|
||||||
|
use App\Models\EvidenceSnapshot;
|
||||||
|
use App\Models\EvidenceSnapshotItem;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\Evidence\EvidenceSnapshotService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Badges\BadgeRenderer;
|
||||||
|
use App\Support\Evidence\EvidenceCompletenessState;
|
||||||
|
use App\Support\Evidence\EvidenceSnapshotStatus;
|
||||||
|
use App\Support\OperationCatalog;
|
||||||
|
use App\Support\OperationRunLinks;
|
||||||
|
use App\Support\OperationRunOutcome;
|
||||||
|
use App\Support\OperationRunStatus;
|
||||||
|
use App\Support\Rbac\UiEnforcement;
|
||||||
|
use App\Support\Rbac\UiTooltips;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthEnvelope;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Infolists\Components\RepeatableEntry;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Infolists\Components\ViewEntry;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Panel;
|
||||||
|
use Filament\Resources\Pages\PageRegistration;
|
||||||
|
use Filament\Resources\Resource;
|
||||||
|
use Filament\Schemas\Components\Section;
|
||||||
|
use Filament\Schemas\Schema;
|
||||||
|
use Filament\Tables;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Routing\Route;
|
||||||
|
use Illuminate\Support\Arr;
|
||||||
|
use Illuminate\Support\Facades\Route as RouteFacade;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class EvidenceSnapshotResource extends Resource
|
||||||
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
|
protected static ?string $model = EvidenceSnapshot::class;
|
||||||
|
|
||||||
|
protected static ?string $slug = 'evidence';
|
||||||
|
|
||||||
|
protected static ?string $tenantOwnershipRelationshipName = 'tenant';
|
||||||
|
|
||||||
|
protected static bool $isGloballySearchable = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-shield-check';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Governance';
|
||||||
|
|
||||||
|
protected static ?string $navigationLabel = 'Evidence';
|
||||||
|
|
||||||
|
protected static ?int $navigationSort = 55;
|
||||||
|
|
||||||
|
public static function canViewAny(): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $user->can(Capabilities::EVIDENCE_VIEW, $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canView(Model $record): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant) || ! $user->can(Capabilities::EVIDENCE_VIEW, $tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ! $record instanceof EvidenceSnapshot
|
||||||
|
|| ((int) $record->tenant_id === (int) $tenant->getKey() && (int) $record->workspace_id === (int) $tenant->workspace_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forResource(ActionSurfaceProfile::CrudListAndView)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'Create snapshot is available from the list header.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ClickableRow->value)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'Empty state includes a Create snapshot CTA.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListRowMoreMenu, 'Evidence snapshots keep only primary View and Expire row actions.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'Evidence snapshots do not support bulk actions.')
|
||||||
|
->satisfy(ActionSurfaceSlot::DetailHeader, 'View page exposes Refresh evidence and Expire snapshot actions.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getEloquentQuery(): Builder
|
||||||
|
{
|
||||||
|
return static::getTenantOwnedEloquentQuery()->with(['tenant', 'initiator', 'operationRun', 'items']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string|null $record): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($record);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function form(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function infolist(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema->schema([
|
||||||
|
Section::make('Artifact truth')
|
||||||
|
->schema([
|
||||||
|
ViewEntry::make('artifact_truth')
|
||||||
|
->hiddenLabel()
|
||||||
|
->view('filament.infolists.entries.governance-artifact-truth')
|
||||||
|
->state(fn (EvidenceSnapshot $record): array => static::truthEnvelope($record)->toArray())
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columnSpanFull(),
|
||||||
|
Section::make('Snapshot')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceSnapshotStatus)),
|
||||||
|
TextEntry::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceCompleteness)),
|
||||||
|
TextEntry::make('tenant.name')->label('Tenant'),
|
||||||
|
TextEntry::make('generated_at')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('expires_at')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('operationRun.id')
|
||||||
|
->label('Operation run')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->url(fn (EvidenceSnapshot $record): ?string => $record->operation_run_id ? OperationRunLinks::tenantlessView((int) $record->operation_run_id) : null)
|
||||||
|
->openUrlInNewTab(),
|
||||||
|
TextEntry::make('fingerprint')->copyable()->placeholder('—')->columnSpanFull()->fontFamily('mono'),
|
||||||
|
TextEntry::make('previous_fingerprint')->copyable()->placeholder('—')->columnSpanFull()->fontFamily('mono'),
|
||||||
|
])
|
||||||
|
->columns(2),
|
||||||
|
Section::make('Summary')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('summary.finding_count')->label('Findings')->placeholder('—'),
|
||||||
|
TextEntry::make('summary.report_count')->label('Reports')->placeholder('—'),
|
||||||
|
TextEntry::make('summary.operation_count')->label('Operations')->placeholder('—'),
|
||||||
|
TextEntry::make('summary.missing_dimensions')->label(static::evidenceCompletenessCountLabel(EvidenceCompletenessState::Missing->value))->placeholder('—'),
|
||||||
|
TextEntry::make('summary.stale_dimensions')->label(static::evidenceCompletenessCountLabel(EvidenceCompletenessState::Stale->value))->placeholder('—'),
|
||||||
|
])
|
||||||
|
->columns(2),
|
||||||
|
Section::make('Evidence dimensions')
|
||||||
|
->schema([
|
||||||
|
RepeatableEntry::make('items')
|
||||||
|
->hiddenLabel()
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('dimension_key')->label('Dimension')
|
||||||
|
->formatStateUsing(fn (string $state): string => Str::headline($state)),
|
||||||
|
TextEntry::make('state')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceCompleteness)),
|
||||||
|
TextEntry::make('source_kind')->label('Source')
|
||||||
|
->formatStateUsing(fn (string $state): string => Str::headline($state)),
|
||||||
|
TextEntry::make('freshness_at')->dateTime()->placeholder('—'),
|
||||||
|
ViewEntry::make('summary_payload_highlights')
|
||||||
|
->label('Summary')
|
||||||
|
->view('filament.infolists.entries.evidence-dimension-summary')
|
||||||
|
->state(fn (EvidenceSnapshotItem $record): array => static::dimensionSummaryPresentation($record))
|
||||||
|
->columnSpanFull(),
|
||||||
|
ViewEntry::make('summary_payload_raw')
|
||||||
|
->label('Raw summary JSON')
|
||||||
|
->view('filament.infolists.entries.snapshot-json')
|
||||||
|
->state(fn (EvidenceSnapshotItem $record): array => is_array($record->summary_payload) ? $record->summary_payload : [])
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(4),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->defaultSort('created_at', 'desc')
|
||||||
|
->recordUrl(fn (EvidenceSnapshot $record): string => static::getUrl('view', ['record' => $record]))
|
||||||
|
->columns([
|
||||||
|
Tables\Columns\TextColumn::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceSnapshotStatus))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_truth')
|
||||||
|
->label('Artifact truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (EvidenceSnapshot $record): string => static::truthEnvelope($record)->primaryLabel)
|
||||||
|
->color(fn (EvidenceSnapshot $record): string => static::truthEnvelope($record)->primaryBadgeSpec()->color)
|
||||||
|
->icon(fn (EvidenceSnapshot $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->icon)
|
||||||
|
->iconColor(fn (EvidenceSnapshot $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->iconColor)
|
||||||
|
->description(fn (EvidenceSnapshot $record): ?string => static::truthEnvelope($record)->primaryExplanation)
|
||||||
|
->wrap(),
|
||||||
|
Tables\Columns\TextColumn::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('generated_at')->dateTime()->sortable()->placeholder('—'),
|
||||||
|
Tables\Columns\TextColumn::make('summary.finding_count')->label('Findings'),
|
||||||
|
Tables\Columns\TextColumn::make('summary.missing_dimensions')->label(static::evidenceCompletenessCountLabel(EvidenceCompletenessState::Missing->value)),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->getStateUsing(fn (EvidenceSnapshot $record): string => static::truthEnvelope($record)->nextStepText())
|
||||||
|
->wrap(),
|
||||||
|
])
|
||||||
|
->filters([
|
||||||
|
Tables\Filters\SelectFilter::make('status')
|
||||||
|
->options(BadgeCatalog::options(BadgeDomain::EvidenceSnapshotStatus, EvidenceSnapshotStatus::values())),
|
||||||
|
Tables\Filters\SelectFilter::make('completeness_state')
|
||||||
|
->options(BadgeCatalog::options(BadgeDomain::EvidenceCompleteness, EvidenceCompletenessState::values())),
|
||||||
|
])
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_snapshot')
|
||||||
|
->label('View snapshot')
|
||||||
|
->url(fn (EvidenceSnapshot $record): string => static::getUrl('view', ['record' => $record])),
|
||||||
|
UiEnforcement::forTableAction(
|
||||||
|
Actions\Action::make('expire')
|
||||||
|
->label('Expire snapshot')
|
||||||
|
->color('danger')
|
||||||
|
->hidden(fn (EvidenceSnapshot $record): bool => ! static::canExpireRecord($record))
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (EvidenceSnapshot $record): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
app(EvidenceSnapshotService::class)->expire($record, $user);
|
||||||
|
|
||||||
|
Notification::make()->success()->title('Snapshot expired')->send();
|
||||||
|
}),
|
||||||
|
fn (EvidenceSnapshot $record): EvidenceSnapshot => $record,
|
||||||
|
)
|
||||||
|
->preserveVisibility()
|
||||||
|
->requireCapability(Capabilities::EVIDENCE_MANAGE)
|
||||||
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
|
->apply(),
|
||||||
|
])
|
||||||
|
->bulkActions([])
|
||||||
|
->emptyStateHeading('No evidence snapshots yet')
|
||||||
|
->emptyStateDescription('Create the first snapshot to capture immutable evidence for this tenant.')
|
||||||
|
->emptyStateActions([
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('create_first_snapshot')
|
||||||
|
->label('Create first snapshot')
|
||||||
|
->icon('heroicon-o-plus')
|
||||||
|
->action(fn (): mixed => static::executeGeneration([])),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::EVIDENCE_MANAGE)
|
||||||
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
|
->apply(),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getPages(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'index' => Pages\ListEvidenceSnapshots::route('/'),
|
||||||
|
'view' => new PageRegistration(
|
||||||
|
page: Pages\ViewEvidenceSnapshot::class,
|
||||||
|
route: fn (Panel $panel): Route => RouteFacade::get('/{record}', Pages\ViewEvidenceSnapshot::class)
|
||||||
|
->whereNumber('record')
|
||||||
|
->middleware(Pages\ViewEvidenceSnapshot::getRouteMiddleware($panel))
|
||||||
|
->withoutMiddleware(Pages\ViewEvidenceSnapshot::getWithoutRouteMiddleware($panel)),
|
||||||
|
),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function dimensionSummaryPresentation(EvidenceSnapshotItem $item): array
|
||||||
|
{
|
||||||
|
$payload = is_array($item->summary_payload) ? $item->summary_payload : [];
|
||||||
|
|
||||||
|
return match ($item->dimension_key) {
|
||||||
|
'findings_summary' => static::findingsSummaryPresentation($payload),
|
||||||
|
'permission_posture' => static::permissionPosturePresentation($payload),
|
||||||
|
'entra_admin_roles' => static::entraAdminRolesPresentation($payload),
|
||||||
|
'baseline_drift_posture' => static::baselineDriftPosturePresentation($payload),
|
||||||
|
'operations_summary' => static::operationsSummaryPresentation($payload),
|
||||||
|
default => static::genericSummaryPresentation($payload),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function findingsSummaryPresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$count = (int) ($payload['count'] ?? 0);
|
||||||
|
$openCount = (int) ($payload['open_count'] ?? 0);
|
||||||
|
$severityCounts = is_array($payload['severity_counts'] ?? null) ? $payload['severity_counts'] : [];
|
||||||
|
$entries = is_array($payload['entries'] ?? null) ? $payload['entries'] : [];
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => sprintf('%d findings, %d open.', $count, $openCount),
|
||||||
|
'highlights' => [
|
||||||
|
['label' => 'Findings', 'value' => (string) $count],
|
||||||
|
['label' => 'Open findings', 'value' => (string) $openCount],
|
||||||
|
['label' => 'Critical', 'value' => (string) ((int) ($severityCounts['critical'] ?? 0))],
|
||||||
|
['label' => 'High', 'value' => (string) ((int) ($severityCounts['high'] ?? 0))],
|
||||||
|
['label' => 'Medium', 'value' => (string) ((int) ($severityCounts['medium'] ?? 0))],
|
||||||
|
['label' => 'Low', 'value' => (string) ((int) ($severityCounts['low'] ?? 0))],
|
||||||
|
],
|
||||||
|
'items' => collect($entries)
|
||||||
|
->map(fn (mixed $entry): ?string => is_array($entry) ? static::findingEntryLabel($entry) : null)
|
||||||
|
->filter()
|
||||||
|
->take(5)
|
||||||
|
->values()
|
||||||
|
->all(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function permissionPosturePresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$requiredCount = (int) ($payload['required_count'] ?? 0);
|
||||||
|
$grantedCount = (int) ($payload['granted_count'] ?? 0);
|
||||||
|
$postureScore = $payload['posture_score'] ?? null;
|
||||||
|
$reportPayload = is_array($payload['payload'] ?? null) ? $payload['payload'] : [];
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => sprintf('%d of %d required permissions granted.', $grantedCount, $requiredCount),
|
||||||
|
'highlights' => [
|
||||||
|
['label' => 'Granted permissions', 'value' => (string) $grantedCount],
|
||||||
|
['label' => 'Required permissions', 'value' => (string) $requiredCount],
|
||||||
|
['label' => 'Posture score', 'value' => $postureScore === null ? '—' : (string) $postureScore],
|
||||||
|
],
|
||||||
|
'items' => static::namedItemsFromArray(
|
||||||
|
Arr::get($reportPayload, 'missing_permissions', Arr::get($reportPayload, 'missing', [])),
|
||||||
|
'No missing permission details captured.'
|
||||||
|
),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function entraAdminRolesPresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$roleCount = (int) ($payload['role_count'] ?? 0);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => sprintf('%d privileged Entra roles captured.', $roleCount),
|
||||||
|
'highlights' => [
|
||||||
|
['label' => 'Role count', 'value' => (string) $roleCount],
|
||||||
|
],
|
||||||
|
'items' => static::namedItemsFromArray($payload['roles'] ?? [], 'No role details captured.'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function baselineDriftPosturePresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$driftCount = (int) ($payload['drift_count'] ?? 0);
|
||||||
|
$openDriftCount = (int) ($payload['open_drift_count'] ?? 0);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => sprintf('%d drift findings, %d still open.', $driftCount, $openDriftCount),
|
||||||
|
'highlights' => [
|
||||||
|
['label' => 'Drift findings', 'value' => (string) $driftCount],
|
||||||
|
['label' => 'Open drift findings', 'value' => (string) $openDriftCount],
|
||||||
|
],
|
||||||
|
'items' => [],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function operationsSummaryPresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$operationCount = (int) ($payload['operation_count'] ?? 0);
|
||||||
|
$failedCount = (int) ($payload['failed_count'] ?? 0);
|
||||||
|
$partialCount = (int) ($payload['partial_count'] ?? 0);
|
||||||
|
$entries = is_array($payload['entries'] ?? null) ? $payload['entries'] : [];
|
||||||
|
$actionSummary = $failedCount === 0 && $partialCount === 0
|
||||||
|
? 'No action needed.'
|
||||||
|
: sprintf('%d execution failures, %d need follow-up.', $failedCount, $partialCount);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => sprintf('%d operations in the last 30 days. %s', $operationCount, $actionSummary),
|
||||||
|
'highlights' => [
|
||||||
|
['label' => 'Operations', 'value' => (string) $operationCount],
|
||||||
|
['label' => 'Execution failures', 'value' => (string) $failedCount],
|
||||||
|
['label' => 'Needs follow-up', 'value' => (string) $partialCount],
|
||||||
|
],
|
||||||
|
'items' => collect($entries)
|
||||||
|
->map(fn (mixed $entry): ?string => is_array($entry) ? static::operationEntryLabel($entry) : null)
|
||||||
|
->filter()
|
||||||
|
->take(5)
|
||||||
|
->values()
|
||||||
|
->all(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $payload
|
||||||
|
* @return array{summary:?string,highlights:list<array{label:string,value:string}>,items:list<string>}
|
||||||
|
*/
|
||||||
|
private static function genericSummaryPresentation(array $payload): array
|
||||||
|
{
|
||||||
|
$highlights = collect($payload)
|
||||||
|
->reject(fn (mixed $value, string|int $key): bool => in_array((string) $key, ['entries', 'payload', 'roles'], true) || is_array($value))
|
||||||
|
->take(6)
|
||||||
|
->map(fn (mixed $value, string|int $key): array => [
|
||||||
|
'label' => Str::headline((string) $key),
|
||||||
|
'value' => static::stringifySummaryValue($value),
|
||||||
|
])
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => empty($highlights) ? 'No summary details captured.' : null,
|
||||||
|
'highlights' => $highlights,
|
||||||
|
'items' => [],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private static function namedItemsFromArray(mixed $items, string $emptyFallback): array
|
||||||
|
{
|
||||||
|
if (! is_array($items) || $items === []) {
|
||||||
|
return [$emptyFallback];
|
||||||
|
}
|
||||||
|
|
||||||
|
$labels = collect($items)
|
||||||
|
->map(function (mixed $item): ?string {
|
||||||
|
if (is_string($item)) {
|
||||||
|
return trim($item) !== '' ? $item : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! is_array($item)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (['display_name', 'displayName', 'name', 'title', 'id'] as $key) {
|
||||||
|
$value = $item[$key] ?? null;
|
||||||
|
|
||||||
|
if (is_string($value) && trim($value) !== '') {
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
})
|
||||||
|
->filter()
|
||||||
|
->take(5)
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
|
|
||||||
|
return $labels === [] ? [$emptyFallback] : $labels;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $entry
|
||||||
|
*/
|
||||||
|
private static function findingEntryLabel(array $entry): ?string
|
||||||
|
{
|
||||||
|
$title = $entry['title'] ?? null;
|
||||||
|
$severity = $entry['severity'] ?? null;
|
||||||
|
$status = $entry['status'] ?? null;
|
||||||
|
|
||||||
|
if (! is_string($title) || trim($title) === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$parts = [trim($title)];
|
||||||
|
|
||||||
|
if (is_string($severity) && trim($severity) !== '') {
|
||||||
|
$parts[] = Str::headline($severity);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_string($status) && trim($status) !== '') {
|
||||||
|
$parts[] = Str::headline($status);
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode(' · ', $parts);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $entry
|
||||||
|
*/
|
||||||
|
private static function operationEntryLabel(array $entry): ?string
|
||||||
|
{
|
||||||
|
$type = $entry['type'] ?? null;
|
||||||
|
|
||||||
|
if (! is_string($type) || trim($type) === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$parts = [static::operationTypeLabel($type)];
|
||||||
|
|
||||||
|
$stateLabel = static::operationEntryStateLabel($entry);
|
||||||
|
|
||||||
|
if ($stateLabel !== null) {
|
||||||
|
$parts[] = $stateLabel;
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode(' · ', $parts);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canExpireRecord(EvidenceSnapshot $record): bool
|
||||||
|
{
|
||||||
|
return (string) $record->status !== EvidenceSnapshotStatus::Expired->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function operationTypeLabel(string $type): string
|
||||||
|
{
|
||||||
|
$label = OperationCatalog::label($type);
|
||||||
|
|
||||||
|
return $label === 'Unknown operation' ? 'Operation' : $label;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $entry
|
||||||
|
*/
|
||||||
|
private static function operationEntryStateLabel(array $entry): ?string
|
||||||
|
{
|
||||||
|
$status = is_string($entry['status'] ?? null) ? trim((string) $entry['status']) : null;
|
||||||
|
$outcome = is_string($entry['outcome'] ?? null) ? trim((string) $entry['outcome']) : null;
|
||||||
|
|
||||||
|
return match ($status) {
|
||||||
|
OperationRunStatus::Queued->value => static::badgeLabel(BadgeDomain::OperationRunStatus, $status),
|
||||||
|
OperationRunStatus::Running->value => static::badgeLabel(BadgeDomain::OperationRunStatus, $status),
|
||||||
|
OperationRunStatus::Completed->value => match ($outcome) {
|
||||||
|
OperationRunOutcome::Pending->value => static::badgeLabel(BadgeDomain::OperationRunStatus, $status),
|
||||||
|
OperationRunOutcome::Succeeded->value,
|
||||||
|
OperationRunOutcome::PartiallySucceeded->value,
|
||||||
|
OperationRunOutcome::Blocked->value,
|
||||||
|
OperationRunOutcome::Failed->value,
|
||||||
|
OperationRunOutcome::Cancelled->value => static::badgeLabel(BadgeDomain::OperationRunOutcome, $outcome),
|
||||||
|
default => static::badgeLabel(BadgeDomain::OperationRunStatus, $status),
|
||||||
|
},
|
||||||
|
default => $outcome !== null ? static::badgeLabel(BadgeDomain::OperationRunOutcome, $outcome) : null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function evidenceCompletenessCountLabel(string $state): string
|
||||||
|
{
|
||||||
|
return BadgeCatalog::spec(BadgeDomain::EvidenceCompleteness, $state)->label;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function badgeLabel(BadgeDomain $domain, ?string $state): ?string
|
||||||
|
{
|
||||||
|
if ($state === null || trim($state) === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$label = BadgeCatalog::spec($domain, $state)->label;
|
||||||
|
|
||||||
|
return $label === 'Unknown' ? null : $label;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function truthEnvelope(EvidenceSnapshot $record): ArtifactTruthEnvelope
|
||||||
|
{
|
||||||
|
return app(ArtifactTruthPresenter::class)->forEvidenceSnapshot($record);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function stringifySummaryValue(mixed $value): string
|
||||||
|
{
|
||||||
|
return match (true) {
|
||||||
|
$value === null => '—',
|
||||||
|
is_bool($value) => $value ? 'Yes' : 'No',
|
||||||
|
is_scalar($value) => (string) $value,
|
||||||
|
default => '—',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
public static function executeGeneration(array $data): void
|
||||||
|
{
|
||||||
|
$tenant = Filament::getTenant();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
Notification::make()->danger()->title('Unable to create snapshot — missing context.')->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$snapshot = app(EvidenceSnapshotService::class)->generate(
|
||||||
|
tenant: $tenant,
|
||||||
|
user: $user,
|
||||||
|
allowStale: (bool) ($data['allow_stale'] ?? false),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (! $snapshot->wasRecentlyCreated) {
|
||||||
|
Notification::make()
|
||||||
|
->success()
|
||||||
|
->title('Snapshot already available')
|
||||||
|
->body('A matching active snapshot already exists. No new run was started.')
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_snapshot')
|
||||||
|
->label('View snapshot')
|
||||||
|
->url(static::getUrl('view', ['record' => $snapshot], tenant: $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->success()
|
||||||
|
->title('Create snapshot queued')
|
||||||
|
->body('The snapshot is being generated in the background.')
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_run')
|
||||||
|
->label('View run')
|
||||||
|
->url($snapshot->operation_run_id ? OperationRunLinks::tenantlessView((int) $snapshot->operation_run_id) : static::getUrl('view', ['record' => $snapshot], tenant: $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,40 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\EvidenceSnapshotResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\EvidenceSnapshotResource;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Rbac\UiEnforcement;
|
||||||
|
use Filament\Actions;
|
||||||
|
use Filament\Forms\Components\Toggle;
|
||||||
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
use Filament\Schemas\Components\Section;
|
||||||
|
|
||||||
|
class ListEvidenceSnapshots extends ListRecords
|
||||||
|
{
|
||||||
|
protected static string $resource = EvidenceSnapshotResource::class;
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('create_snapshot')
|
||||||
|
->label('Create snapshot')
|
||||||
|
->icon('heroicon-o-plus')
|
||||||
|
->action(fn (array $data): mixed => EvidenceSnapshotResource::executeGeneration($data))
|
||||||
|
->form([
|
||||||
|
Section::make('Snapshot options')
|
||||||
|
->schema([
|
||||||
|
Toggle::make('allow_stale')
|
||||||
|
->label('Allow stale dimensions')
|
||||||
|
->default(false),
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::EVIDENCE_MANAGE)
|
||||||
|
->apply(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,102 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\EvidenceSnapshotResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\EvidenceSnapshotResource;
|
||||||
|
use App\Filament\Resources\ReviewPackResource;
|
||||||
|
use App\Models\ReviewPack;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\Evidence\EvidenceSnapshotService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\OperationRunLinks;
|
||||||
|
use App\Support\Rbac\UiEnforcement;
|
||||||
|
use Filament\Actions;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
|
class ViewEvidenceSnapshot extends ViewRecord
|
||||||
|
{
|
||||||
|
protected static string $resource = EvidenceSnapshotResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return EvidenceSnapshotResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Actions\Action::make('view_run')
|
||||||
|
->label('View run')
|
||||||
|
->icon('heroicon-o-eye')
|
||||||
|
->color('gray')
|
||||||
|
->url(fn (): ?string => $this->record->operation_run_id ? OperationRunLinks::tenantlessView((int) $this->record->operation_run_id) : null)
|
||||||
|
->hidden(fn (): bool => ! is_numeric($this->record->operation_run_id)),
|
||||||
|
Actions\Action::make('view_review_pack')
|
||||||
|
->label('View review pack')
|
||||||
|
->icon('heroicon-o-document-text')
|
||||||
|
->color('gray')
|
||||||
|
->url(function (): ?string {
|
||||||
|
$pack = $this->latestReviewPack();
|
||||||
|
|
||||||
|
if (! $pack instanceof ReviewPack || ! $pack->tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ReviewPackResource::getUrl('view', ['record' => $pack], tenant: $pack->tenant);
|
||||||
|
})
|
||||||
|
->hidden(fn (): bool => ! $this->latestReviewPack() instanceof ReviewPack),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('refresh_snapshot')
|
||||||
|
->label('Refresh evidence')
|
||||||
|
->icon('heroicon-o-arrow-path')
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
app(EvidenceSnapshotService::class)->refresh($this->record, $user);
|
||||||
|
|
||||||
|
Notification::make()->success()->title('Refresh evidence queued')->send();
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::EVIDENCE_MANAGE)
|
||||||
|
->apply(),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('expire_snapshot')
|
||||||
|
->label('Expire snapshot')
|
||||||
|
->icon('heroicon-o-x-circle')
|
||||||
|
->color('danger')
|
||||||
|
->hidden(fn (): bool => ! EvidenceSnapshotResource::canExpireRecord($this->record))
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
app(EvidenceSnapshotService::class)->expire($this->record, $user);
|
||||||
|
$this->refreshFormData(['status', 'expires_at']);
|
||||||
|
|
||||||
|
Notification::make()->success()->title('Snapshot expired')->send();
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::EVIDENCE_MANAGE)
|
||||||
|
->apply(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function latestReviewPack(): ?ReviewPack
|
||||||
|
{
|
||||||
|
return $this->record->reviewPacks()
|
||||||
|
->latest('created_at')
|
||||||
|
->first();
|
||||||
|
}
|
||||||
|
}
|
||||||
488
app/Filament/Resources/FindingExceptionResource.php
Normal file
488
app/Filament/Resources/FindingExceptionResource.php
Normal file
@ -0,0 +1,488 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
|
use App\Filament\Resources\FindingExceptionResource\Pages;
|
||||||
|
use App\Models\FindingException;
|
||||||
|
use App\Models\FindingExceptionEvidenceReference;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\Findings\FindingExceptionService;
|
||||||
|
use App\Services\Findings\FindingRiskGovernanceResolver;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Badges\BadgeRenderer;
|
||||||
|
use App\Support\Filament\FilterOptionCatalog;
|
||||||
|
use App\Support\Filament\TablePaginationProfiles;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\DateTimePicker;
|
||||||
|
use Filament\Forms\Components\Repeater;
|
||||||
|
use Filament\Forms\Components\Select;
|
||||||
|
use Filament\Forms\Components\Textarea;
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\RepeatableEntry;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Resources\Resource;
|
||||||
|
use Filament\Schemas\Components\Section;
|
||||||
|
use Filament\Schemas\Schema;
|
||||||
|
use Filament\Tables;
|
||||||
|
use Filament\Tables\Filters\SelectFilter;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use InvalidArgumentException;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class FindingExceptionResource extends Resource
|
||||||
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
|
protected static ?string $model = FindingException::class;
|
||||||
|
|
||||||
|
protected static ?string $tenantOwnershipRelationshipName = 'tenant';
|
||||||
|
|
||||||
|
protected static bool $isGloballySearchable = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-shield-exclamation';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Governance';
|
||||||
|
|
||||||
|
protected static ?string $navigationLabel = 'Risk exceptions';
|
||||||
|
|
||||||
|
protected static ?int $navigationSort = 60;
|
||||||
|
|
||||||
|
public static function shouldRegisterNavigation(): bool
|
||||||
|
{
|
||||||
|
if (Filament::getCurrentPanel()?->getId() === 'admin') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::shouldRegisterNavigation();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canViewAny(): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $user->can(Capabilities::FINDING_EXCEPTION_VIEW, $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canView(Model $record): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant) || ! $user->can(Capabilities::FINDING_EXCEPTION_VIEW, $tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ! $record instanceof FindingException
|
||||||
|
|| ((int) $record->tenant_id === (int) $tenant->getKey() && (int) $record->workspace_id === (int) $tenant->workspace_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forResource(ActionSurfaceProfile::CrudListAndView)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'List header links back to findings where exception requests originate.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ClickableRow->value)
|
||||||
|
->exempt(ActionSurfaceSlot::ListRowMoreMenu, 'v1 keeps exception mutations direct and avoids a More menu.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'Exception decisions require per-record review and intentionally omit bulk actions in v1.')
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'Empty state explains that new requests start from finding detail.')
|
||||||
|
->satisfy(ActionSurfaceSlot::DetailHeader, 'Detail header exposes linked finding navigation plus state-aware renewal and revocation actions.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getEloquentQuery(): Builder
|
||||||
|
{
|
||||||
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
->with(static::relationshipsForView());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string|null $record): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($record, parent::getEloquentQuery()->with(static::relationshipsForView()));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function form(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function infolist(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema->schema([
|
||||||
|
Section::make('Exception')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingExceptionStatus)),
|
||||||
|
TextEntry::make('current_validity_state')
|
||||||
|
->label('Validity')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingRiskGovernanceValidity)),
|
||||||
|
TextEntry::make('governance_warning')
|
||||||
|
->label('Governance warning')
|
||||||
|
->state(fn (FindingException $record): ?string => static::governanceWarning($record))
|
||||||
|
->color(fn (FindingException $record): string => static::governanceWarningColor($record))
|
||||||
|
->columnSpanFull()
|
||||||
|
->visible(fn (FindingException $record): bool => static::governanceWarning($record) !== null),
|
||||||
|
TextEntry::make('tenant.name')->label('Tenant'),
|
||||||
|
TextEntry::make('finding_summary')
|
||||||
|
->label('Finding')
|
||||||
|
->state(fn (FindingException $record): string => static::findingSummary($record)),
|
||||||
|
TextEntry::make('requester.name')->label('Requested by')->placeholder('—'),
|
||||||
|
TextEntry::make('owner.name')->label('Owner')->placeholder('—'),
|
||||||
|
TextEntry::make('approver.name')->label('Approved by')->placeholder('—'),
|
||||||
|
TextEntry::make('requested_at')->label('Requested')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('approved_at')->label('Approved')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('review_due_at')->label('Review due')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('effective_from')->label('Effective from')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('expires_at')->label('Expires')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('request_reason')->label('Request reason')->columnSpanFull(),
|
||||||
|
TextEntry::make('approval_reason')->label('Approval reason')->placeholder('—')->columnSpanFull(),
|
||||||
|
TextEntry::make('rejection_reason')->label('Rejection reason')->placeholder('—')->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(2),
|
||||||
|
Section::make('Decision history')
|
||||||
|
->schema([
|
||||||
|
RepeatableEntry::make('decisions')
|
||||||
|
->hiddenLabel()
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('decision_type')->label('Decision'),
|
||||||
|
TextEntry::make('actor.name')->label('Actor')->placeholder('—'),
|
||||||
|
TextEntry::make('decided_at')->label('Decided')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('reason')->label('Reason')->placeholder('—')->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(3),
|
||||||
|
]),
|
||||||
|
Section::make('Evidence references')
|
||||||
|
->schema([
|
||||||
|
RepeatableEntry::make('evidenceReferences')
|
||||||
|
->hiddenLabel()
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('label')->label('Label'),
|
||||||
|
TextEntry::make('source_type')->label('Source'),
|
||||||
|
TextEntry::make('source_id')->label('Source ID')->placeholder('—'),
|
||||||
|
TextEntry::make('source_fingerprint')->label('Fingerprint')->placeholder('—'),
|
||||||
|
TextEntry::make('measured_at')->label('Measured')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('summary_payload')
|
||||||
|
->label('Summary')
|
||||||
|
->state(function (FindingExceptionEvidenceReference $record): ?string {
|
||||||
|
if ($record->summary_payload === [] || $record->summary_payload === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json_encode($record->summary_payload, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE) ?: null;
|
||||||
|
})
|
||||||
|
->placeholder('—')
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(2),
|
||||||
|
])
|
||||||
|
->visible(fn (FindingException $record): bool => $record->evidenceReferences->isNotEmpty()),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->defaultSort('requested_at', 'desc')
|
||||||
|
->paginated(TablePaginationProfiles::resource())
|
||||||
|
->persistFiltersInSession()
|
||||||
|
->persistSearchInSession()
|
||||||
|
->persistSortInSession()
|
||||||
|
->recordUrl(fn (FindingException $record): string => static::getUrl('view', ['record' => $record]))
|
||||||
|
->columns([
|
||||||
|
Tables\Columns\TextColumn::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('current_validity_state')
|
||||||
|
->label('Validity')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('finding_summary')
|
||||||
|
->label('Finding')
|
||||||
|
->state(fn (FindingException $record): string => static::findingSummary($record))
|
||||||
|
->searchable(),
|
||||||
|
Tables\Columns\TextColumn::make('governance_warning')
|
||||||
|
->label('Governance warning')
|
||||||
|
->state(fn (FindingException $record): ?string => static::governanceWarning($record))
|
||||||
|
->color(fn (FindingException $record): string => static::governanceWarningColor($record))
|
||||||
|
->wrap(),
|
||||||
|
Tables\Columns\TextColumn::make('requester.name')
|
||||||
|
->label('Requested by')
|
||||||
|
->placeholder('—'),
|
||||||
|
Tables\Columns\TextColumn::make('owner.name')
|
||||||
|
->label('Owner')
|
||||||
|
->placeholder('—'),
|
||||||
|
Tables\Columns\TextColumn::make('review_due_at')
|
||||||
|
->label('Review due')
|
||||||
|
->dateTime()
|
||||||
|
->placeholder('—')
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('requested_at')
|
||||||
|
->label('Requested')
|
||||||
|
->dateTime()
|
||||||
|
->sortable(),
|
||||||
|
])
|
||||||
|
->filters([
|
||||||
|
SelectFilter::make('status')
|
||||||
|
->options(FilterOptionCatalog::findingExceptionStatuses()),
|
||||||
|
SelectFilter::make('current_validity_state')
|
||||||
|
->label('Validity')
|
||||||
|
->options(FilterOptionCatalog::findingExceptionValidityStates()),
|
||||||
|
])
|
||||||
|
->actions([
|
||||||
|
Action::make('renew_exception')
|
||||||
|
->label('Renew exception')
|
||||||
|
->icon('heroicon-o-arrow-path')
|
||||||
|
->color('warning')
|
||||||
|
->visible(fn (FindingException $record): bool => static::canManageRecord($record) && $record->canBeRenewed())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Select::make('owner_user_id')
|
||||||
|
->label('Owner')
|
||||||
|
->required()
|
||||||
|
->options(fn (): array => static::tenantMemberOptions())
|
||||||
|
->searchable(),
|
||||||
|
Textarea::make('request_reason')
|
||||||
|
->label('Renewal reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
DateTimePicker::make('review_due_at')
|
||||||
|
->label('Review due at')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
DateTimePicker::make('expires_at')
|
||||||
|
->label('Requested expiry')
|
||||||
|
->seconds(false),
|
||||||
|
Repeater::make('evidence_references')
|
||||||
|
->label('Evidence references')
|
||||||
|
->schema([
|
||||||
|
TextInput::make('label')
|
||||||
|
->label('Label')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_type')
|
||||||
|
->label('Source type')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_id')
|
||||||
|
->label('Source ID')
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_fingerprint')
|
||||||
|
->label('Fingerprint')
|
||||||
|
->maxLength(255),
|
||||||
|
DateTimePicker::make('measured_at')
|
||||||
|
->label('Measured at')
|
||||||
|
->seconds(false),
|
||||||
|
])
|
||||||
|
->defaultItems(0)
|
||||||
|
->collapsed(),
|
||||||
|
])
|
||||||
|
->action(function (FindingException $record, array $data, FindingExceptionService $service): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User || ! $record->tenant instanceof Tenant) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$service->renew($record, $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request submitted')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
}),
|
||||||
|
Action::make('revoke_exception')
|
||||||
|
->label('Revoke exception')
|
||||||
|
->icon('heroicon-o-no-symbol')
|
||||||
|
->color('danger')
|
||||||
|
->visible(fn (FindingException $record): bool => static::canManageRecord($record) && $record->canBeRevoked())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Textarea::make('revocation_reason')
|
||||||
|
->label('Revocation reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
])
|
||||||
|
->action(function (FindingException $record, array $data, FindingExceptionService $service): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$service->revoke($record, $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revocation failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revoked')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
}),
|
||||||
|
])
|
||||||
|
->bulkActions([])
|
||||||
|
->emptyStateHeading('No exceptions match this view')
|
||||||
|
->emptyStateDescription('Exception requests are created from finding detail when a governed risk acceptance review is needed.')
|
||||||
|
->emptyStateIcon('heroicon-o-shield-exclamation')
|
||||||
|
->emptyStateActions([
|
||||||
|
Action::make('open_findings')
|
||||||
|
->label('Open findings')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->url(fn (): string => FindingResource::getUrl('index')),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getPages(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'index' => Pages\ListFindingExceptions::route('/'),
|
||||||
|
'view' => Pages\ViewFindingException::route('/{record}'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string|array<int|string, mixed>>
|
||||||
|
*/
|
||||||
|
private static function relationshipsForView(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'tenant',
|
||||||
|
'requester',
|
||||||
|
'owner',
|
||||||
|
'approver',
|
||||||
|
'currentDecision',
|
||||||
|
'decisions.actor',
|
||||||
|
'evidenceReferences',
|
||||||
|
'finding' => fn ($query) => $query->withSubjectDisplayName(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private static function tenantMemberOptions(): array
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return \App\Models\TenantMembership::query()
|
||||||
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
|
->join('users', 'users.id', '=', 'tenant_memberships.user_id')
|
||||||
|
->orderBy('users.name')
|
||||||
|
->pluck('users.name', 'users.id')
|
||||||
|
->mapWithKeys(fn (string $name, int|string $id): array => [(int) $id => $name])
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function findingSummary(FindingException $record): string
|
||||||
|
{
|
||||||
|
$summary = $record->finding?->resolvedSubjectDisplayName();
|
||||||
|
|
||||||
|
if (is_string($summary) && trim($summary) !== '') {
|
||||||
|
return trim($summary);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'Finding #'.$record->finding_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function canManageRecord(FindingException $record): bool
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
return $user instanceof User
|
||||||
|
&& $record->tenant instanceof Tenant
|
||||||
|
&& $user->canAccessTenant($record->tenant)
|
||||||
|
&& $user->can(Capabilities::FINDING_EXCEPTION_MANAGE, $record->tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function governanceWarning(FindingException $record): ?string
|
||||||
|
{
|
||||||
|
$finding = $record->relationLoaded('finding')
|
||||||
|
? $record->finding
|
||||||
|
: $record->finding()->withSubjectDisplayName()->first();
|
||||||
|
|
||||||
|
if (! $finding instanceof \App\Models\Finding) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return app(FindingRiskGovernanceResolver::class)->resolveWarningMessage($finding, $record);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function governanceWarningColor(FindingException $record): string
|
||||||
|
{
|
||||||
|
$finding = $record->relationLoaded('finding')
|
||||||
|
? $record->finding
|
||||||
|
: $record->finding()->withSubjectDisplayName()->first();
|
||||||
|
|
||||||
|
if ($finding instanceof \App\Models\Finding && $record->requiresFreshDecisionForFinding($finding)) {
|
||||||
|
return 'warning';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'danger';
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\FindingExceptionResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\FindingExceptionResource;
|
||||||
|
use App\Filament\Resources\FindingResource;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
|
||||||
|
class ListFindingExceptions extends ListRecords
|
||||||
|
{
|
||||||
|
protected static string $resource = FindingExceptionResource::class;
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Action::make('open_findings')
|
||||||
|
->label('Open findings')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->url(FindingResource::getUrl('index')),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,208 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\FindingExceptionResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\FindingExceptionResource;
|
||||||
|
use App\Filament\Resources\FindingResource;
|
||||||
|
use App\Models\FindingException;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\Findings\FindingExceptionService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Forms\Components\DateTimePicker;
|
||||||
|
use Filament\Forms\Components\Repeater;
|
||||||
|
use Filament\Forms\Components\Select;
|
||||||
|
use Filament\Forms\Components\Textarea;
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use InvalidArgumentException;
|
||||||
|
|
||||||
|
class ViewFindingException extends ViewRecord
|
||||||
|
{
|
||||||
|
protected static string $resource = FindingExceptionResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return FindingExceptionResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Action::make('open_finding')
|
||||||
|
->label('Open finding')
|
||||||
|
->icon('heroicon-o-arrow-top-right-on-square')
|
||||||
|
->color('gray')
|
||||||
|
->url(function (): ?string {
|
||||||
|
$record = $this->getRecord();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $record->finding || ! $record->tenant) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FindingResource::getUrl('view', ['record' => $record->finding], panel: 'tenant', tenant: $record->tenant);
|
||||||
|
}),
|
||||||
|
Action::make('renew_exception')
|
||||||
|
->label('Renew exception')
|
||||||
|
->icon('heroicon-o-arrow-path')
|
||||||
|
->color('warning')
|
||||||
|
->visible(fn (): bool => $this->canManageRecord() && $this->getRecord() instanceof FindingException && $this->getRecord()->canBeRenewed())
|
||||||
|
->fillForm(fn (): array => [
|
||||||
|
'owner_user_id' => $this->getRecord() instanceof FindingException ? $this->getRecord()->owner_user_id : null,
|
||||||
|
])
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Select::make('owner_user_id')
|
||||||
|
->label('Owner')
|
||||||
|
->required()
|
||||||
|
->options(fn (): array => FindingExceptionResource::canViewAny() ? $this->tenantMemberOptions() : [])
|
||||||
|
->searchable(),
|
||||||
|
Textarea::make('request_reason')
|
||||||
|
->label('Renewal reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
DateTimePicker::make('review_due_at')
|
||||||
|
->label('Review due at')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
DateTimePicker::make('expires_at')
|
||||||
|
->label('Requested expiry')
|
||||||
|
->seconds(false),
|
||||||
|
Repeater::make('evidence_references')
|
||||||
|
->label('Evidence references')
|
||||||
|
->schema([
|
||||||
|
TextInput::make('label')
|
||||||
|
->label('Label')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_type')
|
||||||
|
->label('Source type')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_id')
|
||||||
|
->label('Source ID')
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_fingerprint')
|
||||||
|
->label('Fingerprint')
|
||||||
|
->maxLength(255),
|
||||||
|
DateTimePicker::make('measured_at')
|
||||||
|
->label('Measured at')
|
||||||
|
->seconds(false),
|
||||||
|
])
|
||||||
|
->defaultItems(0)
|
||||||
|
->collapsed(),
|
||||||
|
])
|
||||||
|
->action(function (array $data, FindingExceptionService $service): void {
|
||||||
|
$record = $this->getRecord();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $user instanceof User) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$service->renew($record, $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request submitted')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
$this->refreshFormData(['status', 'current_validity_state', 'review_due_at']);
|
||||||
|
}),
|
||||||
|
Action::make('revoke_exception')
|
||||||
|
->label('Revoke exception')
|
||||||
|
->icon('heroicon-o-no-symbol')
|
||||||
|
->color('danger')
|
||||||
|
->visible(fn (): bool => $this->canManageRecord() && $this->getRecord() instanceof FindingException && $this->getRecord()->canBeRevoked())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Textarea::make('revocation_reason')
|
||||||
|
->label('Revocation reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
])
|
||||||
|
->action(function (array $data, FindingExceptionService $service): void {
|
||||||
|
$record = $this->getRecord();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException || ! $user instanceof User) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$service->revoke($record, $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revocation failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revoked')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
$this->refreshFormData(['status', 'current_validity_state', 'revocation_reason', 'revoked_at']);
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function tenantMemberOptions(): array
|
||||||
|
{
|
||||||
|
$record = $this->getRecord();
|
||||||
|
|
||||||
|
if (! $record instanceof FindingException) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = $record->tenant;
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return \App\Models\TenantMembership::query()
|
||||||
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
|
->join('users', 'users.id', '=', 'tenant_memberships.user_id')
|
||||||
|
->orderBy('users.name')
|
||||||
|
->pluck('users.name', 'users.id')
|
||||||
|
->mapWithKeys(fn (string $name, int|string $id): array => [(int) $id => $name])
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function canManageRecord(): bool
|
||||||
|
{
|
||||||
|
$record = $this->getRecord();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
return $record instanceof FindingException
|
||||||
|
&& $record->tenant instanceof Tenant
|
||||||
|
&& $user instanceof User
|
||||||
|
&& $user->canAccessTenant($record->tenant)
|
||||||
|
&& $user->can(Capabilities::FINDING_EXCEPTION_MANAGE, $record->tenant);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,14 +2,18 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\FindingResource\Pages;
|
use App\Filament\Resources\FindingResource\Pages;
|
||||||
use App\Models\Finding;
|
use App\Models\Finding;
|
||||||
|
use App\Models\FindingException;
|
||||||
use App\Models\PolicyVersion;
|
use App\Models\PolicyVersion;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Models\TenantMembership;
|
use App\Models\TenantMembership;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Services\Drift\DriftFindingDiffBuilder;
|
use App\Services\Drift\DriftFindingDiffBuilder;
|
||||||
|
use App\Services\Findings\FindingExceptionService;
|
||||||
|
use App\Services\Findings\FindingRiskGovernanceResolver;
|
||||||
use App\Services\Findings\FindingWorkflowService;
|
use App\Services\Findings\FindingWorkflowService;
|
||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
@ -34,6 +38,8 @@
|
|||||||
use Filament\Actions\BulkAction;
|
use Filament\Actions\BulkAction;
|
||||||
use Filament\Actions\BulkActionGroup;
|
use Filament\Actions\BulkActionGroup;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\DateTimePicker;
|
||||||
|
use Filament\Forms\Components\Repeater;
|
||||||
use Filament\Forms\Components\Select;
|
use Filament\Forms\Components\Select;
|
||||||
use Filament\Forms\Components\Textarea;
|
use Filament\Forms\Components\Textarea;
|
||||||
use Filament\Forms\Components\TextInput;
|
use Filament\Forms\Components\TextInput;
|
||||||
@ -55,6 +61,7 @@
|
|||||||
|
|
||||||
class FindingResource extends Resource
|
class FindingResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
protected static ?string $model = Finding::class;
|
protected static ?string $model = Finding::class;
|
||||||
@ -112,7 +119,8 @@ public static function canView(Model $record): bool
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($record instanceof Finding) {
|
if ($record instanceof Finding) {
|
||||||
return (int) $record->tenant_id === (int) $tenant->getKey();
|
return (int) $record->tenant_id === (int) $tenant->getKey()
|
||||||
|
&& (int) $record->workspace_id === (int) $tenant->workspace_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -220,6 +228,62 @@ public static function infolist(Schema $schema): Schema
|
|||||||
->columns(2)
|
->columns(2)
|
||||||
->columnSpanFull(),
|
->columnSpanFull(),
|
||||||
|
|
||||||
|
Section::make('Risk governance')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('finding_governance_status')
|
||||||
|
->label('Exception status')
|
||||||
|
->badge()
|
||||||
|
->state(fn (Finding $record): ?string => $record->findingException?->status)
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingExceptionStatus))
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('finding_governance_validity')
|
||||||
|
->label('Validity')
|
||||||
|
->badge()
|
||||||
|
->state(function (Finding $record): ?string {
|
||||||
|
if ($record->findingException instanceof FindingException) {
|
||||||
|
return $record->findingException->current_validity_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (string) $record->status === Finding::STATUS_RISK_ACCEPTED
|
||||||
|
? FindingException::VALIDITY_MISSING_SUPPORT
|
||||||
|
: null;
|
||||||
|
})
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::FindingRiskGovernanceValidity))
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('finding_governance_warning')
|
||||||
|
->label('Governance warning')
|
||||||
|
->state(fn (Finding $record): ?string => static::governanceWarning($record))
|
||||||
|
->color(fn (Finding $record): string => static::governanceWarningColor($record))
|
||||||
|
->columnSpanFull()
|
||||||
|
->visible(fn (Finding $record): bool => static::governanceWarning($record) !== null),
|
||||||
|
TextEntry::make('finding_governance_owner')
|
||||||
|
->label('Exception owner')
|
||||||
|
->state(fn (Finding $record): ?string => $record->findingException?->owner?->name)
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('finding_governance_approver')
|
||||||
|
->label('Approver')
|
||||||
|
->state(fn (Finding $record): ?string => $record->findingException?->approver?->name)
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('finding_governance_review_due')
|
||||||
|
->label('Review due')
|
||||||
|
->state(fn (Finding $record): mixed => $record->findingException?->review_due_at)
|
||||||
|
->dateTime()
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('finding_governance_expires')
|
||||||
|
->label('Expires')
|
||||||
|
->state(fn (Finding $record): mixed => $record->findingException?->expires_at)
|
||||||
|
->dateTime()
|
||||||
|
->placeholder('—'),
|
||||||
|
])
|
||||||
|
->columns(2)
|
||||||
|
->visible(fn (Finding $record): bool => $record->findingException instanceof FindingException || (string) $record->status === Finding::STATUS_RISK_ACCEPTED),
|
||||||
|
|
||||||
Section::make('Evidence')
|
Section::make('Evidence')
|
||||||
->schema([
|
->schema([
|
||||||
TextEntry::make('redaction_integrity_note')
|
TextEntry::make('redaction_integrity_note')
|
||||||
@ -752,6 +816,7 @@ public static function table(Table $table): Table
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
$workflow->triage($record, $tenant, $user);
|
$workflow->triage($record, $tenant, $user);
|
||||||
$triagedCount++;
|
$triagedCount++;
|
||||||
} catch (Throwable) {
|
} catch (Throwable) {
|
||||||
@ -832,6 +897,7 @@ public static function table(Table $table): Table
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
$workflow->assign($record, $tenant, $user, $assigneeUserId, $ownerUserId);
|
$workflow->assign($record, $tenant, $user, $assigneeUserId, $ownerUserId);
|
||||||
$assignedCount++;
|
$assignedCount++;
|
||||||
} catch (Throwable) {
|
} catch (Throwable) {
|
||||||
@ -906,6 +972,7 @@ public static function table(Table $table): Table
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
$workflow->resolve($record, $tenant, $user, $reason);
|
$workflow->resolve($record, $tenant, $user, $reason);
|
||||||
$resolvedCount++;
|
$resolvedCount++;
|
||||||
} catch (Throwable) {
|
} catch (Throwable) {
|
||||||
@ -980,6 +1047,7 @@ public static function table(Table $table): Table
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
$workflow->close($record, $tenant, $user, $reason);
|
$workflow->close($record, $tenant, $user, $reason);
|
||||||
$closedCount++;
|
$closedCount++;
|
||||||
} catch (Throwable) {
|
} catch (Throwable) {
|
||||||
@ -1007,79 +1075,6 @@ public static function table(Table $table): Table
|
|||||||
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
->apply(),
|
->apply(),
|
||||||
|
|
||||||
UiEnforcement::forBulkAction(
|
|
||||||
BulkAction::make('risk_accept_selected')
|
|
||||||
->label('Risk accept selected')
|
|
||||||
->icon('heroicon-o-shield-check')
|
|
||||||
->color('warning')
|
|
||||||
->requiresConfirmation()
|
|
||||||
->form([
|
|
||||||
Textarea::make('closed_reason')
|
|
||||||
->label('Risk acceptance reason')
|
|
||||||
->rows(3)
|
|
||||||
->required()
|
|
||||||
->maxLength(255),
|
|
||||||
])
|
|
||||||
->action(function (Collection $records, array $data, FindingWorkflowService $workflow): void {
|
|
||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
|
||||||
$user = auth()->user();
|
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$reason = (string) ($data['closed_reason'] ?? '');
|
|
||||||
|
|
||||||
$acceptedCount = 0;
|
|
||||||
$skippedCount = 0;
|
|
||||||
$failedCount = 0;
|
|
||||||
|
|
||||||
foreach ($records as $record) {
|
|
||||||
if (! $record instanceof Finding) {
|
|
||||||
$skippedCount++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((int) $record->tenant_id !== (int) $tenant->getKey()) {
|
|
||||||
$skippedCount++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (! $record->hasOpenStatus()) {
|
|
||||||
$skippedCount++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
$workflow->riskAccept($record, $tenant, $user, $reason);
|
|
||||||
$acceptedCount++;
|
|
||||||
} catch (Throwable) {
|
|
||||||
$failedCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$body = "Risk accepted {$acceptedCount} finding".($acceptedCount === 1 ? '' : 's').'.';
|
|
||||||
if ($skippedCount > 0) {
|
|
||||||
$body .= " Skipped {$skippedCount}.";
|
|
||||||
}
|
|
||||||
if ($failedCount > 0) {
|
|
||||||
$body .= " Failed {$failedCount}.";
|
|
||||||
}
|
|
||||||
|
|
||||||
Notification::make()
|
|
||||||
->title('Bulk risk accept completed')
|
|
||||||
->body($body)
|
|
||||||
->status($failedCount > 0 ? 'warning' : 'success')
|
|
||||||
->send();
|
|
||||||
})
|
|
||||||
->deselectRecordsAfterCompletion(),
|
|
||||||
)
|
|
||||||
->requireCapability(Capabilities::TENANT_FINDINGS_RISK_ACCEPT)
|
|
||||||
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
|
||||||
->apply(),
|
|
||||||
])->label('More'),
|
])->label('More'),
|
||||||
])
|
])
|
||||||
->emptyStateHeading('No findings match this view')
|
->emptyStateHeading('No findings match this view')
|
||||||
@ -1089,12 +1084,19 @@ public static function table(Table $table): Table
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanel()?->getKey();
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
->with(['assigneeUser', 'ownerUser', 'closedByUser', 'findingException.owner', 'findingException.approver', 'findingException.currentDecision'])
|
||||||
|
->withSubjectDisplayName();
|
||||||
|
}
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
public static function resolveScopedRecordOrFail(int|string $key): Model
|
||||||
->with(['assigneeUser', 'ownerUser', 'closedByUser'])
|
{
|
||||||
->withSubjectDisplayName()
|
return static::resolveTenantOwnedRecordOrFail(
|
||||||
->when($tenantId, fn (Builder $query) => $query->where('tenant_id', $tenantId));
|
$key,
|
||||||
|
parent::getEloquentQuery()
|
||||||
|
->with(['assigneeUser', 'ownerUser', 'closedByUser', 'findingException.owner', 'findingException.approver', 'findingException.currentDecision'])
|
||||||
|
->withSubjectDisplayName(),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1170,7 +1172,9 @@ public static function workflowActions(): array
|
|||||||
static::assignAction(),
|
static::assignAction(),
|
||||||
static::resolveAction(),
|
static::resolveAction(),
|
||||||
static::closeAction(),
|
static::closeAction(),
|
||||||
static::riskAcceptAction(),
|
static::requestExceptionAction(),
|
||||||
|
static::renewExceptionAction(),
|
||||||
|
static::revokeExceptionAction(),
|
||||||
static::reopenAction(),
|
static::reopenAction(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@ -1182,7 +1186,7 @@ public static function triageAction(): Actions\Action
|
|||||||
->label('Triage')
|
->label('Triage')
|
||||||
->icon('heroicon-o-check')
|
->icon('heroicon-o-check')
|
||||||
->color('gray')
|
->color('gray')
|
||||||
->visible(fn (Finding $record): bool => in_array((string) $record->status, [
|
->visible(fn (Finding $record): bool => in_array(static::freshWorkflowStatus($record), [
|
||||||
Finding::STATUS_NEW,
|
Finding::STATUS_NEW,
|
||||||
Finding::STATUS_REOPENED,
|
Finding::STATUS_REOPENED,
|
||||||
Finding::STATUS_ACKNOWLEDGED,
|
Finding::STATUS_ACKNOWLEDGED,
|
||||||
@ -1208,7 +1212,7 @@ public static function startProgressAction(): Actions\Action
|
|||||||
->label('Start progress')
|
->label('Start progress')
|
||||||
->icon('heroicon-o-play')
|
->icon('heroicon-o-play')
|
||||||
->color('info')
|
->color('info')
|
||||||
->visible(fn (Finding $record): bool => in_array((string) $record->status, [
|
->visible(fn (Finding $record): bool => in_array(static::freshWorkflowStatus($record), [
|
||||||
Finding::STATUS_TRIAGED,
|
Finding::STATUS_TRIAGED,
|
||||||
Finding::STATUS_ACKNOWLEDGED,
|
Finding::STATUS_ACKNOWLEDGED,
|
||||||
], true))
|
], true))
|
||||||
@ -1233,7 +1237,7 @@ public static function assignAction(): Actions\Action
|
|||||||
->label('Assign')
|
->label('Assign')
|
||||||
->icon('heroicon-o-user-plus')
|
->icon('heroicon-o-user-plus')
|
||||||
->color('gray')
|
->color('gray')
|
||||||
->visible(fn (Finding $record): bool => $record->hasOpenStatus())
|
->visible(fn (Finding $record): bool => static::freshWorkflowRecord($record)->hasOpenStatus())
|
||||||
->fillForm(fn (Finding $record): array => [
|
->fillForm(fn (Finding $record): array => [
|
||||||
'assignee_user_id' => $record->assignee_user_id,
|
'assignee_user_id' => $record->assignee_user_id,
|
||||||
'owner_user_id' => $record->owner_user_id,
|
'owner_user_id' => $record->owner_user_id,
|
||||||
@ -1277,7 +1281,7 @@ public static function resolveAction(): Actions\Action
|
|||||||
->label('Resolve')
|
->label('Resolve')
|
||||||
->icon('heroicon-o-check-badge')
|
->icon('heroicon-o-check-badge')
|
||||||
->color('success')
|
->color('success')
|
||||||
->visible(fn (Finding $record): bool => $record->hasOpenStatus())
|
->visible(fn (Finding $record): bool => static::freshWorkflowRecord($record)->hasOpenStatus())
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->form([
|
->form([
|
||||||
Textarea::make('resolved_reason')
|
Textarea::make('resolved_reason')
|
||||||
@ -1312,6 +1316,7 @@ public static function closeAction(): Actions\Action
|
|||||||
->label('Close')
|
->label('Close')
|
||||||
->icon('heroicon-o-x-circle')
|
->icon('heroicon-o-x-circle')
|
||||||
->color('danger')
|
->color('danger')
|
||||||
|
->visible(fn (Finding $record): bool => static::freshWorkflowRecord($record)->hasOpenStatus())
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->form([
|
->form([
|
||||||
Textarea::make('closed_reason')
|
Textarea::make('closed_reason')
|
||||||
@ -1339,36 +1344,153 @@ public static function closeAction(): Actions\Action
|
|||||||
->apply();
|
->apply();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function riskAcceptAction(): Actions\Action
|
public static function requestExceptionAction(): Actions\Action
|
||||||
{
|
{
|
||||||
return UiEnforcement::forAction(
|
return UiEnforcement::forAction(
|
||||||
Actions\Action::make('risk_accept')
|
Actions\Action::make('request_exception')
|
||||||
->label('Risk accept')
|
->label('Request exception')
|
||||||
->icon('heroicon-o-shield-check')
|
->icon('heroicon-o-shield-exclamation')
|
||||||
->color('warning')
|
->color('warning')
|
||||||
|
->visible(fn (Finding $record): bool => static::freshWorkflowRecord($record)->hasOpenStatus())
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->form([
|
->form([
|
||||||
Textarea::make('closed_reason')
|
Select::make('owner_user_id')
|
||||||
->label('Risk acceptance reason')
|
->label('Owner')
|
||||||
->rows(3)
|
|
||||||
->required()
|
->required()
|
||||||
->maxLength(255),
|
->options(fn (): array => static::tenantMemberOptions())
|
||||||
|
->searchable(),
|
||||||
|
Textarea::make('request_reason')
|
||||||
|
->label('Request reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
DateTimePicker::make('review_due_at')
|
||||||
|
->label('Review due at')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
DateTimePicker::make('expires_at')
|
||||||
|
->label('Expires at')
|
||||||
|
->seconds(false),
|
||||||
|
Repeater::make('evidence_references')
|
||||||
|
->label('Evidence references')
|
||||||
|
->schema([
|
||||||
|
TextInput::make('label')
|
||||||
|
->label('Label')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_type')
|
||||||
|
->label('Source type')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_id')
|
||||||
|
->label('Source ID')
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_fingerprint')
|
||||||
|
->label('Fingerprint')
|
||||||
|
->maxLength(255),
|
||||||
|
DateTimePicker::make('measured_at')
|
||||||
|
->label('Measured at')
|
||||||
|
->seconds(false),
|
||||||
|
])
|
||||||
|
->defaultItems(0)
|
||||||
|
->collapsed(),
|
||||||
])
|
])
|
||||||
->action(function (Finding $record, array $data, FindingWorkflowService $workflow): void {
|
->action(function (Finding $record, array $data, FindingExceptionService $service): void {
|
||||||
static::runWorkflowMutation(
|
static::runExceptionRequestMutation($record, $data, $service);
|
||||||
record: $record,
|
|
||||||
successTitle: 'Finding marked as risk accepted',
|
|
||||||
callback: fn (Finding $finding, Tenant $tenant, User $user): Finding => $workflow->riskAccept(
|
|
||||||
$finding,
|
|
||||||
$tenant,
|
|
||||||
$user,
|
|
||||||
(string) ($data['closed_reason'] ?? ''),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
->preserveVisibility()
|
->preserveVisibility()
|
||||||
->requireCapability(Capabilities::TENANT_FINDINGS_RISK_ACCEPT)
|
->requireCapability(Capabilities::FINDING_EXCEPTION_MANAGE)
|
||||||
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
|
->apply();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function renewExceptionAction(): Actions\Action
|
||||||
|
{
|
||||||
|
return UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('renew_exception')
|
||||||
|
->label('Renew exception')
|
||||||
|
->icon('heroicon-o-arrow-path')
|
||||||
|
->color('warning')
|
||||||
|
->visible(fn (Finding $record): bool => static::currentFindingException($record)?->canBeRenewed() ?? false)
|
||||||
|
->fillForm(fn (Finding $record): array => [
|
||||||
|
'owner_user_id' => static::currentFindingException($record)?->owner_user_id,
|
||||||
|
])
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Select::make('owner_user_id')
|
||||||
|
->label('Owner')
|
||||||
|
->required()
|
||||||
|
->options(fn (): array => static::tenantMemberOptions())
|
||||||
|
->searchable(),
|
||||||
|
Textarea::make('request_reason')
|
||||||
|
->label('Renewal reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
DateTimePicker::make('review_due_at')
|
||||||
|
->label('Review due at')
|
||||||
|
->required()
|
||||||
|
->seconds(false),
|
||||||
|
DateTimePicker::make('expires_at')
|
||||||
|
->label('Requested expiry')
|
||||||
|
->seconds(false),
|
||||||
|
Repeater::make('evidence_references')
|
||||||
|
->label('Evidence references')
|
||||||
|
->schema([
|
||||||
|
TextInput::make('label')
|
||||||
|
->label('Label')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_type')
|
||||||
|
->label('Source type')
|
||||||
|
->required()
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_id')
|
||||||
|
->label('Source ID')
|
||||||
|
->maxLength(255),
|
||||||
|
TextInput::make('source_fingerprint')
|
||||||
|
->label('Fingerprint')
|
||||||
|
->maxLength(255),
|
||||||
|
DateTimePicker::make('measured_at')
|
||||||
|
->label('Measured at')
|
||||||
|
->seconds(false),
|
||||||
|
])
|
||||||
|
->defaultItems(0)
|
||||||
|
->collapsed(),
|
||||||
|
])
|
||||||
|
->action(function (Finding $record, array $data, FindingExceptionService $service): void {
|
||||||
|
static::runExceptionRenewalMutation($record, $data, $service);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
->preserveVisibility()
|
||||||
|
->requireCapability(Capabilities::FINDING_EXCEPTION_MANAGE)
|
||||||
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
|
->apply();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function revokeExceptionAction(): Actions\Action
|
||||||
|
{
|
||||||
|
return UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('revoke_exception')
|
||||||
|
->label('Revoke exception')
|
||||||
|
->icon('heroicon-o-no-symbol')
|
||||||
|
->color('danger')
|
||||||
|
->visible(fn (Finding $record): bool => static::currentFindingException($record)?->canBeRevoked() ?? false)
|
||||||
|
->requiresConfirmation()
|
||||||
|
->form([
|
||||||
|
Textarea::make('revocation_reason')
|
||||||
|
->label('Revocation reason')
|
||||||
|
->rows(4)
|
||||||
|
->required()
|
||||||
|
->maxLength(2000),
|
||||||
|
])
|
||||||
|
->action(function (Finding $record, array $data, FindingExceptionService $service): void {
|
||||||
|
static::runExceptionRevocationMutation($record, $data, $service);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
->preserveVisibility()
|
||||||
|
->requireCapability(Capabilities::FINDING_EXCEPTION_MANAGE)
|
||||||
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
->tooltip(UiTooltips::INSUFFICIENT_PERMISSION)
|
||||||
->apply();
|
->apply();
|
||||||
}
|
}
|
||||||
@ -1381,7 +1503,7 @@ public static function reopenAction(): Actions\Action
|
|||||||
->icon('heroicon-o-arrow-uturn-left')
|
->icon('heroicon-o-arrow-uturn-left')
|
||||||
->color('warning')
|
->color('warning')
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (Finding $record): bool => Finding::isTerminalStatus((string) $record->status))
|
->visible(fn (Finding $record): bool => Finding::isTerminalStatus(static::freshWorkflowStatus($record)))
|
||||||
->action(function (Finding $record, FindingWorkflowService $workflow): void {
|
->action(function (Finding $record, FindingWorkflowService $workflow): void {
|
||||||
static::runWorkflowMutation(
|
static::runWorkflowMutation(
|
||||||
record: $record,
|
record: $record,
|
||||||
@ -1401,6 +1523,7 @@ public static function reopenAction(): Actions\Action
|
|||||||
*/
|
*/
|
||||||
private static function runWorkflowMutation(Finding $record, string $successTitle, callable $callback): void
|
private static function runWorkflowMutation(Finding $record, string $successTitle, callable $callback): void
|
||||||
{
|
{
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
|
|
||||||
@ -1417,6 +1540,15 @@ private static function runWorkflowMutation(Finding $record, string $successTitl
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((int) $record->workspace_id !== (int) $tenant->workspace_id) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Finding belongs to a different workspace')
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$callback($record, $tenant, $user);
|
$callback($record, $tenant, $user);
|
||||||
} catch (InvalidArgumentException $e) {
|
} catch (InvalidArgumentException $e) {
|
||||||
@ -1435,6 +1567,194 @@ private static function runWorkflowMutation(Finding $record, string $successTitl
|
|||||||
->send();
|
->send();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
private static function runExceptionRequestMutation(Finding $record, array $data, FindingExceptionService $service): void
|
||||||
|
{
|
||||||
|
$record = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$createdException = $service->request($record, $tenant, $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception request failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception request submitted')
|
||||||
|
->success()
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_exception')
|
||||||
|
->label('View exception')
|
||||||
|
->url(static::findingExceptionViewUrl($createdException, $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
private static function runExceptionRenewalMutation(Finding $record, array $data, FindingExceptionService $service): void
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$renewedException = $service->renew(static::resolveCurrentFindingExceptionOrFail($record), $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Renewal request submitted')
|
||||||
|
->success()
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_exception')
|
||||||
|
->label('View exception')
|
||||||
|
->url(static::findingExceptionViewUrl($renewedException, $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
private static function runExceptionRevocationMutation(Finding $record, array $data, FindingExceptionService $service): void
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$revokedException = $service->revoke(static::resolveCurrentFindingExceptionOrFail($record), $user, $data);
|
||||||
|
} catch (InvalidArgumentException $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revocation failed')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Exception revoked')
|
||||||
|
->success()
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_exception')
|
||||||
|
->label('View exception')
|
||||||
|
->url(static::findingExceptionViewUrl($revokedException, $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function freshWorkflowRecord(Finding $record): Finding
|
||||||
|
{
|
||||||
|
return static::resolveProtectedFindingRecordOrFail($record);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function freshWorkflowStatus(Finding $record): string
|
||||||
|
{
|
||||||
|
return (string) static::freshWorkflowRecord($record)->status;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function resolveProtectedFindingRecordOrFail(Finding|int|string $record): Finding
|
||||||
|
{
|
||||||
|
$resolvedRecord = static::resolveScopedRecordOrFail($record instanceof Model ? $record->getKey() : $record);
|
||||||
|
|
||||||
|
if (! $resolvedRecord instanceof Finding) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedRecord;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function currentFindingException(Finding $record): ?FindingException
|
||||||
|
{
|
||||||
|
$finding = static::resolveProtectedFindingRecordOrFail($record);
|
||||||
|
|
||||||
|
return static::resolvedFindingException($finding);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function resolvedFindingException(Finding $finding): ?FindingException
|
||||||
|
{
|
||||||
|
$exception = $finding->relationLoaded('findingException')
|
||||||
|
? $finding->findingException
|
||||||
|
: $finding->findingException()->with('currentDecision')->first();
|
||||||
|
|
||||||
|
if (! $exception instanceof FindingException) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$exception->loadMissing('currentDecision');
|
||||||
|
|
||||||
|
return $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function resolveCurrentFindingExceptionOrFail(Finding $record): FindingException
|
||||||
|
{
|
||||||
|
$exception = static::currentFindingException($record);
|
||||||
|
|
||||||
|
if (! $exception instanceof FindingException) {
|
||||||
|
throw new InvalidArgumentException('This finding does not have an exception to manage.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function findingExceptionViewUrl(\App\Models\FindingException $exception, Tenant $tenant): string
|
||||||
|
{
|
||||||
|
$panelId = Filament::getCurrentPanel()?->getId();
|
||||||
|
|
||||||
|
if ($panelId === 'admin') {
|
||||||
|
return FindingExceptionResource::getUrl('view', ['record' => $exception], panel: 'admin');
|
||||||
|
}
|
||||||
|
|
||||||
|
return FindingExceptionResource::getUrl('view', ['record' => $exception], panel: 'tenant', tenant: $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function governanceWarning(Finding $finding): ?string
|
||||||
|
{
|
||||||
|
return app(FindingRiskGovernanceResolver::class)
|
||||||
|
->resolveWarningMessage($finding, static::resolvedFindingException($finding));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function governanceWarningColor(Finding $finding): string
|
||||||
|
{
|
||||||
|
$exception = static::resolvedFindingException($finding);
|
||||||
|
|
||||||
|
if ($exception instanceof FindingException && $exception->requiresFreshDecisionForFinding($finding)) {
|
||||||
|
return 'warning';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'danger';
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<int, string>
|
* @return array<int, string>
|
||||||
*/
|
*/
|
||||||
|
|||||||
@ -23,6 +23,7 @@
|
|||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Resources\Pages\ListRecords;
|
use Filament\Resources\Pages\ListRecords;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
||||||
use Illuminate\Support\Arr;
|
use Illuminate\Support\Arr;
|
||||||
use Throwable;
|
use Throwable;
|
||||||
|
|
||||||
@ -32,14 +33,26 @@ class ListFindings extends ListRecords
|
|||||||
|
|
||||||
protected static string $resource = FindingResource::class;
|
protected static string $resource = FindingResource::class;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true && filled($context['recordKey'] ?? null) && in_array($name, ['triage', 'start_progress', 'assign', 'resolve', 'close', 'request_exception', 'reopen'], true)) {
|
||||||
|
try {
|
||||||
|
FindingResource::resolveScopedRecordOrFail($context['recordKey']);
|
||||||
|
} catch (ModelNotFoundException) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
public function mount(): void
|
public function mount(): void
|
||||||
{
|
{
|
||||||
app(CanonicalAdminTenantFilterState::class)->sync(
|
$this->syncCanonicalAdminTenantFilterState();
|
||||||
$this->getTableFiltersSessionKey(),
|
|
||||||
tenantSensitiveFilters: ['scope_key', 'run_ids'],
|
|
||||||
request: request(),
|
|
||||||
tenantFilterName: null,
|
|
||||||
);
|
|
||||||
|
|
||||||
parent::mount();
|
parent::mount();
|
||||||
}
|
}
|
||||||
@ -246,15 +259,7 @@ protected function getHeaderActions(): array
|
|||||||
|
|
||||||
protected function buildAllMatchingQuery(): Builder
|
protected function buildAllMatchingQuery(): Builder
|
||||||
{
|
{
|
||||||
$query = Finding::query();
|
$query = FindingResource::getEloquentQuery();
|
||||||
|
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanel()?->getKey();
|
|
||||||
|
|
||||||
if (! is_numeric($tenantId)) {
|
|
||||||
return $query->whereRaw('1 = 0');
|
|
||||||
}
|
|
||||||
|
|
||||||
$query->where('tenant_id', (int) $tenantId);
|
|
||||||
|
|
||||||
$query->where('status', Finding::STATUS_NEW);
|
$query->where('status', Finding::STATUS_NEW);
|
||||||
|
|
||||||
@ -304,6 +309,16 @@ protected function buildAllMatchingQuery(): Builder
|
|||||||
return $query;
|
return $query;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function syncCanonicalAdminTenantFilterState(): void
|
||||||
|
{
|
||||||
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
||||||
|
$this->getTableFiltersSessionKey(),
|
||||||
|
tenantSensitiveFilters: ['scope_key', 'run_ids'],
|
||||||
|
request: request(),
|
||||||
|
tenantFilterName: null,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private function filterIsActive(string $filterName): bool
|
private function filterIsActive(string $filterName): bool
|
||||||
{
|
{
|
||||||
$state = $this->getTableFilterState($filterName);
|
$state = $this->getTableFilterState($filterName);
|
||||||
|
|||||||
@ -8,11 +8,17 @@
|
|||||||
use Filament\Actions;
|
use Filament\Actions;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
use Illuminate\Contracts\Support\Htmlable;
|
use Illuminate\Contracts\Support\Htmlable;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewFinding extends ViewRecord
|
class ViewFinding extends ViewRecord
|
||||||
{
|
{
|
||||||
protected static string $resource = FindingResource::class;
|
protected static string $resource = FindingResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return FindingResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
|||||||
@ -3,6 +3,7 @@
|
|||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
use App\Filament\Clusters\Inventory\InventoryCluster;
|
use App\Filament\Clusters\Inventory\InventoryCluster;
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\InventoryItemResource\Pages;
|
use App\Filament\Resources\InventoryItemResource\Pages;
|
||||||
use App\Models\InventoryItem;
|
use App\Models\InventoryItem;
|
||||||
@ -38,6 +39,7 @@
|
|||||||
|
|
||||||
class InventoryItemResource extends Resource
|
class InventoryItemResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
protected static ?string $model = InventoryItem::class;
|
protected static ?string $model = InventoryItem::class;
|
||||||
@ -334,13 +336,15 @@ public static function table(Table $table): Table
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanel()?->getKey();
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
|
||||||
->when($tenantId, fn (Builder $query) => $query->where('tenant_id', $tenantId))
|
|
||||||
->with('lastSeenRun');
|
->with('lastSeenRun');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string $key): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($key, parent::getEloquentQuery()->with('lastSeenRun'));
|
||||||
|
}
|
||||||
|
|
||||||
public static function getPages(): array
|
public static function getPages(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
|||||||
@ -174,6 +174,8 @@ protected function getHeaderActions(): array
|
|||||||
],
|
],
|
||||||
context: array_merge($computed['selection'], [
|
context: array_merge($computed['selection'], [
|
||||||
'selection_hash' => $computed['selection_hash'],
|
'selection_hash' => $computed['selection_hash'],
|
||||||
|
'execution_authority_mode' => 'actor_bound',
|
||||||
|
'required_capability' => Capabilities::TENANT_INVENTORY_SYNC_RUN,
|
||||||
'target_scope' => [
|
'target_scope' => [
|
||||||
'entra_tenant_id' => $tenant->graphTenantId(),
|
'entra_tenant_id' => $tenant->graphTenantId(),
|
||||||
],
|
],
|
||||||
|
|||||||
@ -4,8 +4,14 @@
|
|||||||
|
|
||||||
use App\Filament\Resources\InventoryItemResource;
|
use App\Filament\Resources\InventoryItemResource;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewInventoryItem extends ViewRecord
|
class ViewInventoryItem extends ViewRecord
|
||||||
{
|
{
|
||||||
protected static string $resource = InventoryItemResource::class;
|
protected static string $resource = InventoryItemResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return InventoryItemResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -8,6 +8,7 @@
|
|||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\VerificationCheckAcknowledgement;
|
use App\Models\VerificationCheckAcknowledgement;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
use App\Support\Badges\BadgeRenderer;
|
use App\Support\Badges\BadgeRenderer;
|
||||||
use App\Support\Baselines\BaselineCompareReasonCode;
|
use App\Support\Baselines\BaselineCompareReasonCode;
|
||||||
@ -21,13 +22,17 @@
|
|||||||
use App\Support\OperationRunLinks;
|
use App\Support\OperationRunLinks;
|
||||||
use App\Support\OperationRunOutcome;
|
use App\Support\OperationRunOutcome;
|
||||||
use App\Support\OperationRunStatus;
|
use App\Support\OperationRunStatus;
|
||||||
|
use App\Support\OpsUx\OperationUxPresenter;
|
||||||
use App\Support\OpsUx\RunDurationInsights;
|
use App\Support\OpsUx\RunDurationInsights;
|
||||||
|
use App\Support\OpsUx\SummaryCountsNormalizer;
|
||||||
|
use App\Support\ReasonTranslation\ReasonPresenter;
|
||||||
use App\Support\Tenants\ReferencedTenantLifecyclePresentation;
|
use App\Support\Tenants\ReferencedTenantLifecyclePresentation;
|
||||||
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
use App\Support\Ui\ActionSurface\ActionSurfaceDefaults;
|
use App\Support\Ui\ActionSurface\ActionSurfaceDefaults;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use BackedEnum;
|
use BackedEnum;
|
||||||
use Filament\Actions;
|
use Filament\Actions;
|
||||||
@ -123,10 +128,11 @@ public static function table(Table $table): Table
|
|||||||
->columns([
|
->columns([
|
||||||
Tables\Columns\TextColumn::make('status')
|
Tables\Columns\TextColumn::make('status')
|
||||||
->badge()
|
->badge()
|
||||||
->formatStateUsing(BadgeRenderer::label(BadgeDomain::OperationRunStatus))
|
->formatStateUsing(fn (mixed $state, OperationRun $record): string => BadgeRenderer::spec(BadgeDomain::OperationRunStatus, static::statusBadgeState($record))->label)
|
||||||
->color(BadgeRenderer::color(BadgeDomain::OperationRunStatus))
|
->color(fn (mixed $state, OperationRun $record): string => BadgeRenderer::spec(BadgeDomain::OperationRunStatus, static::statusBadgeState($record))->color)
|
||||||
->icon(BadgeRenderer::icon(BadgeDomain::OperationRunStatus))
|
->icon(fn (mixed $state, OperationRun $record): ?string => BadgeRenderer::spec(BadgeDomain::OperationRunStatus, static::statusBadgeState($record))->icon)
|
||||||
->iconColor(BadgeRenderer::iconColor(BadgeDomain::OperationRunStatus)),
|
->iconColor(fn (mixed $state, OperationRun $record): ?string => BadgeRenderer::spec(BadgeDomain::OperationRunStatus, static::statusBadgeState($record))->iconColor)
|
||||||
|
->description(fn (OperationRun $record): ?string => OperationUxPresenter::lifecycleAttentionSummary($record)),
|
||||||
Tables\Columns\TextColumn::make('type')
|
Tables\Columns\TextColumn::make('type')
|
||||||
->label('Operation')
|
->label('Operation')
|
||||||
->formatStateUsing(fn (?string $state): string => OperationCatalog::label((string) $state))
|
->formatStateUsing(fn (?string $state): string => OperationCatalog::label((string) $state))
|
||||||
@ -149,10 +155,11 @@ public static function table(Table $table): Table
|
|||||||
}),
|
}),
|
||||||
Tables\Columns\TextColumn::make('outcome')
|
Tables\Columns\TextColumn::make('outcome')
|
||||||
->badge()
|
->badge()
|
||||||
->formatStateUsing(BadgeRenderer::label(BadgeDomain::OperationRunOutcome))
|
->formatStateUsing(fn (mixed $state, OperationRun $record): string => BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, static::outcomeBadgeState($record))->label)
|
||||||
->color(BadgeRenderer::color(BadgeDomain::OperationRunOutcome))
|
->color(fn (mixed $state, OperationRun $record): string => BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, static::outcomeBadgeState($record))->color)
|
||||||
->icon(BadgeRenderer::icon(BadgeDomain::OperationRunOutcome))
|
->icon(fn (mixed $state, OperationRun $record): ?string => BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, static::outcomeBadgeState($record))->icon)
|
||||||
->iconColor(BadgeRenderer::iconColor(BadgeDomain::OperationRunOutcome)),
|
->iconColor(fn (mixed $state, OperationRun $record): ?string => BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, static::outcomeBadgeState($record))->iconColor)
|
||||||
|
->description(fn (OperationRun $record): ?string => OperationUxPresenter::surfaceGuidance($record)),
|
||||||
])
|
])
|
||||||
->filters([
|
->filters([
|
||||||
Tables\Filters\SelectFilter::make('tenant_id')
|
Tables\Filters\SelectFilter::make('tenant_id')
|
||||||
@ -205,13 +212,9 @@ public static function table(Table $table): Table
|
|||||||
return FilterOptionCatalog::operationTypes(array_keys($types));
|
return FilterOptionCatalog::operationTypes(array_keys($types));
|
||||||
}),
|
}),
|
||||||
Tables\Filters\SelectFilter::make('status')
|
Tables\Filters\SelectFilter::make('status')
|
||||||
->options([
|
->options(BadgeCatalog::options(BadgeDomain::OperationRunStatus, OperationRunStatus::values())),
|
||||||
OperationRunStatus::Queued->value => 'Queued',
|
|
||||||
OperationRunStatus::Running->value => 'Running',
|
|
||||||
OperationRunStatus::Completed->value => 'Completed',
|
|
||||||
]),
|
|
||||||
Tables\Filters\SelectFilter::make('outcome')
|
Tables\Filters\SelectFilter::make('outcome')
|
||||||
->options(OperationRunOutcome::uiLabels(includeReserved: false)),
|
->options(BadgeCatalog::options(BadgeDomain::OperationRunOutcome, OperationRunOutcome::values(includeReserved: false))),
|
||||||
Tables\Filters\SelectFilter::make('initiator_name')
|
Tables\Filters\SelectFilter::make('initiator_name')
|
||||||
->label('Initiator')
|
->label('Initiator')
|
||||||
->options(function (): array {
|
->options(function (): array {
|
||||||
@ -251,13 +254,24 @@ private static function enterpriseDetailPage(OperationRun $record): \App\Support
|
|||||||
{
|
{
|
||||||
$factory = new \App\Support\Ui\EnterpriseDetail\EnterpriseDetailSectionFactory;
|
$factory = new \App\Support\Ui\EnterpriseDetail\EnterpriseDetailSectionFactory;
|
||||||
|
|
||||||
$statusSpec = BadgeRenderer::spec(BadgeDomain::OperationRunStatus, $record->status);
|
$statusSpec = BadgeRenderer::spec(BadgeDomain::OperationRunStatus, static::statusBadgeState($record));
|
||||||
$outcomeSpec = BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, $record->outcome);
|
$outcomeSpec = BadgeRenderer::spec(BadgeDomain::OperationRunOutcome, static::outcomeBadgeState($record));
|
||||||
$targetScope = static::targetScopeDisplay($record);
|
$targetScope = static::targetScopeDisplay($record);
|
||||||
$summaryLine = \App\Support\OpsUx\SummaryCountsNormalizer::renderSummaryLine(is_array($record->summary_counts) ? $record->summary_counts : []);
|
$summaryLine = \App\Support\OpsUx\SummaryCountsNormalizer::renderSummaryLine(is_array($record->summary_counts) ? $record->summary_counts : []);
|
||||||
$referencedTenantLifecycle = $record->tenant instanceof Tenant
|
$referencedTenantLifecycle = $record->tenant instanceof Tenant
|
||||||
? ReferencedTenantLifecyclePresentation::forOperationRun($record->tenant)
|
? ReferencedTenantLifecyclePresentation::forOperationRun($record->tenant)
|
||||||
: null;
|
: null;
|
||||||
|
$artifactTruth = $record->isGovernanceArtifactOperation()
|
||||||
|
? app(ArtifactTruthPresenter::class)->forOperationRun($record)
|
||||||
|
: null;
|
||||||
|
$artifactTruthBadge = $artifactTruth !== null
|
||||||
|
? $factory->statusBadge(
|
||||||
|
$artifactTruth->primaryBadgeSpec()->label,
|
||||||
|
$artifactTruth->primaryBadgeSpec()->color,
|
||||||
|
$artifactTruth->primaryBadgeSpec()->icon,
|
||||||
|
$artifactTruth->primaryBadgeSpec()->iconColor,
|
||||||
|
)
|
||||||
|
: null;
|
||||||
|
|
||||||
$builder = \App\Support\Ui\EnterpriseDetail\EnterpriseDetailBuilder::make('operation_run', 'workspace-context')
|
$builder = \App\Support\Ui\EnterpriseDetail\EnterpriseDetailBuilder::make('operation_run', 'workspace-context')
|
||||||
->header(new \App\Support\Ui\EnterpriseDetail\SummaryHeaderData(
|
->header(new \App\Support\Ui\EnterpriseDetail\SummaryHeaderData(
|
||||||
@ -287,6 +301,15 @@ private static function enterpriseDetailPage(OperationRun $record): \App\Support
|
|||||||
$factory->keyFact('Expected duration', RunDurationInsights::expectedHuman($record)),
|
$factory->keyFact('Expected duration', RunDurationInsights::expectedHuman($record)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
$factory->viewSection(
|
||||||
|
id: 'artifact_truth',
|
||||||
|
kind: 'current_status',
|
||||||
|
title: 'Artifact truth',
|
||||||
|
view: 'filament.infolists.entries.governance-artifact-truth',
|
||||||
|
viewData: ['artifactTruthState' => $artifactTruth?->toArray()],
|
||||||
|
visible: $record->isGovernanceArtifactOperation(),
|
||||||
|
description: 'Run lifecycle stays separate from whether the intended governance artifact was actually produced and usable.',
|
||||||
|
),
|
||||||
$factory->viewSection(
|
$factory->viewSection(
|
||||||
id: 'related_context',
|
id: 'related_context',
|
||||||
kind: 'related_context',
|
kind: 'related_context',
|
||||||
@ -304,6 +327,9 @@ private static function enterpriseDetailPage(OperationRun $record): \App\Support
|
|||||||
items: array_values(array_filter([
|
items: array_values(array_filter([
|
||||||
$factory->keyFact('Status', $statusSpec->label, badge: $factory->statusBadge($statusSpec->label, $statusSpec->color, $statusSpec->icon, $statusSpec->iconColor)),
|
$factory->keyFact('Status', $statusSpec->label, badge: $factory->statusBadge($statusSpec->label, $statusSpec->color, $statusSpec->icon, $statusSpec->iconColor)),
|
||||||
$factory->keyFact('Outcome', $outcomeSpec->label, badge: $factory->statusBadge($outcomeSpec->label, $outcomeSpec->color, $outcomeSpec->icon, $outcomeSpec->iconColor)),
|
$factory->keyFact('Outcome', $outcomeSpec->label, badge: $factory->statusBadge($outcomeSpec->label, $outcomeSpec->color, $outcomeSpec->icon, $outcomeSpec->iconColor)),
|
||||||
|
$artifactTruth !== null
|
||||||
|
? $factory->keyFact('Artifact truth', $artifactTruth->primaryLabel, badge: $artifactTruthBadge)
|
||||||
|
: null,
|
||||||
$referencedTenantLifecycle !== null
|
$referencedTenantLifecycle !== null
|
||||||
? $factory->keyFact(
|
? $factory->keyFact(
|
||||||
'Tenant lifecycle',
|
'Tenant lifecycle',
|
||||||
@ -322,7 +348,34 @@ private static function enterpriseDetailPage(OperationRun $record): \App\Support
|
|||||||
$referencedTenantLifecycle?->contextNote !== null
|
$referencedTenantLifecycle?->contextNote !== null
|
||||||
? $factory->keyFact('Viewer context', $referencedTenantLifecycle->contextNote)
|
? $factory->keyFact('Viewer context', $referencedTenantLifecycle->contextNote)
|
||||||
: null,
|
: null,
|
||||||
|
static::freshnessLabel($record) !== null
|
||||||
|
? $factory->keyFact('Freshness', (string) static::freshnessLabel($record))
|
||||||
|
: null,
|
||||||
|
static::reconciliationHeadline($record) !== null
|
||||||
|
? $factory->keyFact('Lifecycle truth', (string) static::reconciliationHeadline($record))
|
||||||
|
: null,
|
||||||
|
static::reconciledAtLabel($record) !== null
|
||||||
|
? $factory->keyFact('Reconciled at', (string) static::reconciledAtLabel($record))
|
||||||
|
: null,
|
||||||
|
static::reconciliationSourceLabel($record) !== null
|
||||||
|
? $factory->keyFact('Reconciled by', (string) static::reconciliationSourceLabel($record))
|
||||||
|
: null,
|
||||||
|
$artifactTruth !== null
|
||||||
|
? $factory->keyFact('Artifact next step', $artifactTruth->nextStepText())
|
||||||
|
: null,
|
||||||
|
OperationUxPresenter::surfaceGuidance($record) !== null
|
||||||
|
? $factory->keyFact('Next step', OperationUxPresenter::surfaceGuidance($record))
|
||||||
|
: null,
|
||||||
$summaryLine !== null ? $factory->keyFact('Counts', $summaryLine) : null,
|
$summaryLine !== null ? $factory->keyFact('Counts', $summaryLine) : null,
|
||||||
|
static::blockedExecutionReasonCode($record) !== null
|
||||||
|
? $factory->keyFact('Blocked reason', static::blockedExecutionReasonCode($record))
|
||||||
|
: null,
|
||||||
|
static::blockedExecutionDetail($record) !== null
|
||||||
|
? $factory->keyFact('Blocked detail', static::blockedExecutionDetail($record))
|
||||||
|
: null,
|
||||||
|
static::blockedExecutionSource($record) !== null
|
||||||
|
? $factory->keyFact('Blocked by', static::blockedExecutionSource($record))
|
||||||
|
: null,
|
||||||
RunDurationInsights::stuckGuidance($record) !== null ? $factory->keyFact('Guidance', RunDurationInsights::stuckGuidance($record)) : null,
|
RunDurationInsights::stuckGuidance($record) !== null ? $factory->keyFact('Guidance', RunDurationInsights::stuckGuidance($record)) : null,
|
||||||
])),
|
])),
|
||||||
),
|
),
|
||||||
@ -369,13 +422,26 @@ private static function enterpriseDetailPage(OperationRun $record): \App\Support
|
|||||||
$factory->viewSection(
|
$factory->viewSection(
|
||||||
id: 'failures',
|
id: 'failures',
|
||||||
kind: 'operational_context',
|
kind: 'operational_context',
|
||||||
title: 'Failures',
|
title: (string) $record->outcome === OperationRunOutcome::Blocked->value ? 'Blocked execution details' : 'Failures',
|
||||||
view: 'filament.infolists.entries.snapshot-json',
|
view: 'filament.infolists.entries.snapshot-json',
|
||||||
viewData: ['payload' => $record->failure_summary ?? []],
|
viewData: ['payload' => $record->failure_summary ?? []],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (static::reconciliationPayload($record) !== []) {
|
||||||
|
$builder->addSection(
|
||||||
|
$factory->viewSection(
|
||||||
|
id: 'reconciliation',
|
||||||
|
kind: 'operational_context',
|
||||||
|
title: 'Lifecycle reconciliation',
|
||||||
|
view: 'filament.infolists.entries.snapshot-json',
|
||||||
|
viewData: ['payload' => static::reconciliationPayload($record)],
|
||||||
|
description: 'Lifecycle reconciliation is diagnostic evidence showing when TenantPilot force-resolved the run.',
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if ((string) $record->type === 'baseline_compare') {
|
if ((string) $record->type === 'baseline_compare') {
|
||||||
$baselineCompareFacts = static::baselineCompareFacts($record, $factory);
|
$baselineCompareFacts = static::baselineCompareFacts($record, $factory);
|
||||||
$baselineCompareEvidence = static::baselineCompareEvidencePayload($record);
|
$baselineCompareEvidence = static::baselineCompareEvidencePayload($record);
|
||||||
@ -445,12 +511,68 @@ private static function summaryCountFacts(
|
|||||||
$counts = \App\Support\OpsUx\SummaryCountsNormalizer::normalize(is_array($record->summary_counts) ? $record->summary_counts : []);
|
$counts = \App\Support\OpsUx\SummaryCountsNormalizer::normalize(is_array($record->summary_counts) ? $record->summary_counts : []);
|
||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
static fn (string $key, int $value): array => $factory->keyFact(ucfirst(str_replace('_', ' ', $key)), $value),
|
static fn (string $key, int $value): array => $factory->keyFact(SummaryCountsNormalizer::label($key), $value),
|
||||||
array_keys($counts),
|
array_keys($counts),
|
||||||
array_values($counts),
|
array_values($counts),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static function blockedExecutionReasonCode(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
if ((string) $record->outcome !== OperationRunOutcome::Blocked->value) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(ReasonPresenter::class)->forOperationRun($record, 'run_detail');
|
||||||
|
|
||||||
|
if ($reasonEnvelope !== null) {
|
||||||
|
return $reasonEnvelope->operatorLabel;
|
||||||
|
}
|
||||||
|
|
||||||
|
$context = is_array($record->context) ? $record->context : [];
|
||||||
|
$reasonCode = data_get($context, 'execution_legitimacy.reason_code')
|
||||||
|
?? data_get($context, 'reason_code')
|
||||||
|
?? data_get($record->failure_summary, '0.reason_code');
|
||||||
|
|
||||||
|
return is_string($reasonCode) && trim($reasonCode) !== '' ? trim($reasonCode) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function blockedExecutionDetail(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
if ((string) $record->outcome !== OperationRunOutcome::Blocked->value) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(ReasonPresenter::class)->forOperationRun($record, 'run_detail');
|
||||||
|
|
||||||
|
if ($reasonEnvelope !== null) {
|
||||||
|
return $reasonEnvelope->shortExplanation;
|
||||||
|
}
|
||||||
|
|
||||||
|
$message = data_get($record->failure_summary, '0.message');
|
||||||
|
|
||||||
|
return is_string($message) && trim($message) !== '' ? trim($message) : 'Execution was refused before work began.';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function blockedExecutionSource(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
if ((string) $record->outcome !== OperationRunOutcome::Blocked->value) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$context = is_array($record->context) ? $record->context : [];
|
||||||
|
$blockedBy = $context['blocked_by'] ?? null;
|
||||||
|
|
||||||
|
if (! is_string($blockedBy) || trim($blockedBy) === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return match (trim($blockedBy)) {
|
||||||
|
'queued_execution_legitimacy' => 'Execution legitimacy revalidation',
|
||||||
|
default => ucfirst(str_replace('_', ' ', trim($blockedBy))),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return list<array<string, mixed>>
|
* @return list<array<string, mixed>>
|
||||||
*/
|
*/
|
||||||
@ -630,6 +752,82 @@ private static function contextPayload(OperationRun $record): array
|
|||||||
return $context;
|
return $context;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{status:string,freshness_state:string}
|
||||||
|
*/
|
||||||
|
private static function statusBadgeState(OperationRun $record): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'status' => (string) $record->status,
|
||||||
|
'freshness_state' => $record->freshnessState()->value,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{outcome:string,status:string,freshness_state:string}
|
||||||
|
*/
|
||||||
|
private static function outcomeBadgeState(OperationRun $record): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'outcome' => (string) $record->outcome,
|
||||||
|
'status' => (string) $record->status,
|
||||||
|
'freshness_state' => $record->freshnessState()->value,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function freshnessLabel(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
return match ($record->freshnessState()->value) {
|
||||||
|
'fresh_active' => 'Fresh activity',
|
||||||
|
'likely_stale' => 'Likely stale',
|
||||||
|
'reconciled_failed' => 'Automatically reconciled',
|
||||||
|
'terminal_normal' => 'Terminal truth confirmed',
|
||||||
|
default => null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function reconciliationHeadline(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
if (! $record->isLifecycleReconciled()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'TenantPilot force-resolved this run after normal lifecycle truth was lost.';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function reconciledAtLabel(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
$reconciledAt = data_get($record->reconciliation(), 'reconciled_at');
|
||||||
|
|
||||||
|
return is_string($reconciledAt) && trim($reconciledAt) !== '' ? trim($reconciledAt) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function reconciliationSourceLabel(OperationRun $record): ?string
|
||||||
|
{
|
||||||
|
$source = data_get($record->reconciliation(), 'source');
|
||||||
|
|
||||||
|
if (! is_string($source) || trim($source) === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return match (trim($source)) {
|
||||||
|
'failed_callback' => 'Direct failed() bridge',
|
||||||
|
'scheduled_reconciler' => 'Scheduled reconciler',
|
||||||
|
'adapter_reconciler' => 'Adapter reconciler',
|
||||||
|
default => ucfirst(str_replace('_', ' ', trim($source))),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function reconciliationPayload(OperationRun $record): array
|
||||||
|
{
|
||||||
|
$reconciliation = $record->reconciliation();
|
||||||
|
|
||||||
|
return $reconciliation;
|
||||||
|
}
|
||||||
|
|
||||||
private static function formatDetailTimestamp(mixed $value): string
|
private static function formatDetailTimestamp(mixed $value): string
|
||||||
{
|
{
|
||||||
if (! $value instanceof \Illuminate\Support\Carbon) {
|
if (! $value instanceof \Illuminate\Support\Carbon) {
|
||||||
|
|||||||
@ -2,7 +2,9 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
|
use App\Filament\Concerns\ScopesGlobalSearchToTenant;
|
||||||
use App\Filament\Resources\PolicyResource\Pages;
|
use App\Filament\Resources\PolicyResource\Pages;
|
||||||
use App\Filament\Resources\PolicyResource\RelationManagers\VersionsRelationManager;
|
use App\Filament\Resources\PolicyResource\RelationManagers\VersionsRelationManager;
|
||||||
use App\Jobs\BulkPolicyDeleteJob;
|
use App\Jobs\BulkPolicyDeleteJob;
|
||||||
@ -54,7 +56,9 @@
|
|||||||
|
|
||||||
class PolicyResource extends Resource
|
class PolicyResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
use ScopesGlobalSearchToTenant;
|
||||||
|
|
||||||
protected static ?string $model = Policy::class;
|
protected static ?string $model = Policy::class;
|
||||||
|
|
||||||
@ -1010,16 +1014,25 @@ public static function table(Table $table): Table
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanelOrFail()->getKey();
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
|
||||||
->when($tenantId, fn (Builder $query) => $query->where('tenant_id', $tenantId))
|
|
||||||
->withCount('versions')
|
->withCount('versions')
|
||||||
->with([
|
->with([
|
||||||
'versions' => fn ($query) => $query->orderByDesc('captured_at')->limit(1),
|
'versions' => fn ($query) => $query->orderByDesc('captured_at')->limit(1),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string $key): \Illuminate\Database\Eloquent\Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail(
|
||||||
|
$key,
|
||||||
|
parent::getEloquentQuery()
|
||||||
|
->withCount('versions')
|
||||||
|
->with([
|
||||||
|
'versions' => fn ($query) => $query->orderByDesc('captured_at')->limit(1),
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public static function getRelations(): array
|
public static function getRelations(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
|||||||
@ -3,12 +3,20 @@
|
|||||||
namespace App\Filament\Resources\PolicyResource\Pages;
|
namespace App\Filament\Resources\PolicyResource\Pages;
|
||||||
|
|
||||||
use App\Filament\Resources\PolicyResource;
|
use App\Filament\Resources\PolicyResource;
|
||||||
|
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
||||||
use Filament\Resources\Pages\ListRecords;
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
|
||||||
class ListPolicies extends ListRecords
|
class ListPolicies extends ListRecords
|
||||||
{
|
{
|
||||||
protected static string $resource = PolicyResource::class;
|
protected static string $resource = PolicyResource::class;
|
||||||
|
|
||||||
|
public function mount(): void
|
||||||
|
{
|
||||||
|
$this->syncCanonicalAdminTenantFilterState();
|
||||||
|
|
||||||
|
parent::mount();
|
||||||
|
}
|
||||||
|
|
||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
@ -22,4 +30,14 @@ protected function getTableEmptyStateActions(): array
|
|||||||
PolicyResource::makeSyncAction(),
|
PolicyResource::makeSyncAction(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function syncCanonicalAdminTenantFilterState(): void
|
||||||
|
{
|
||||||
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
||||||
|
$this->getTableFiltersSessionKey(),
|
||||||
|
tenantSensitiveFilters: [],
|
||||||
|
request: request(),
|
||||||
|
tenantFilterName: null,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -16,6 +16,7 @@
|
|||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
use Filament\Support\Enums\Width;
|
use Filament\Support\Enums\Width;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Support\Str;
|
||||||
|
|
||||||
class ViewPolicy extends ViewRecord
|
class ViewPolicy extends ViewRecord
|
||||||
@ -24,6 +25,11 @@ class ViewPolicy extends ViewRecord
|
|||||||
|
|
||||||
protected Width|string|null $maxContentWidth = Width::Full;
|
protected Width|string|null $maxContentWidth = Width::Full;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return PolicyResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
protected function getActions(): array
|
protected function getActions(): array
|
||||||
{
|
{
|
||||||
return [$this->makeCaptureSnapshotAction()];
|
return [$this->makeCaptureSnapshotAction()];
|
||||||
|
|||||||
@ -4,6 +4,7 @@
|
|||||||
|
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\RestoreRunResource;
|
use App\Filament\Resources\RestoreRunResource;
|
||||||
|
use App\Models\Policy;
|
||||||
use App\Models\PolicyVersion;
|
use App\Models\PolicyVersion;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
@ -31,6 +32,19 @@ class VersionsRelationManager extends RelationManager
|
|||||||
|
|
||||||
protected static string $relationship = 'versions';
|
protected static string $relationship = 'versions';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true && $name === 'restore_to_intune' && filled($context['recordKey'] ?? null)) {
|
||||||
|
$this->resolveOwnerScopedVersionRecord($this->getOwnerRecord(), $context['recordKey']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
{
|
{
|
||||||
return ActionSurfaceDeclaration::forRelationManager(ActionSurfaceProfile::RelationManager)
|
return ActionSurfaceDeclaration::forRelationManager(ActionSurfaceProfile::RelationManager)
|
||||||
@ -55,7 +69,8 @@ public function table(Table $table): Table
|
|||||||
->label('Preview only (dry-run)')
|
->label('Preview only (dry-run)')
|
||||||
->default(true),
|
->default(true),
|
||||||
])
|
])
|
||||||
->action(function (PolicyVersion $record, array $data, RestoreService $restoreService) {
|
->action(function (mixed $record, array $data, RestoreService $restoreService) {
|
||||||
|
$record = $this->resolveOwnerScopedVersionRecord($this->getOwnerRecord(), $record);
|
||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
|
|
||||||
@ -178,4 +193,26 @@ public function table(Table $table): Table
|
|||||||
->emptyStateHeading('No versions captured')
|
->emptyStateHeading('No versions captured')
|
||||||
->emptyStateDescription('Capture or sync this policy again to create version history entries.');
|
->emptyStateDescription('Capture or sync this policy again to create version history entries.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function resolveOwnerScopedVersionRecord(Policy $policy, mixed $record): PolicyVersion
|
||||||
|
{
|
||||||
|
$recordId = $record instanceof PolicyVersion
|
||||||
|
? (int) $record->getKey()
|
||||||
|
: (is_numeric($record) ? (int) $record : 0);
|
||||||
|
|
||||||
|
if ($recordId <= 0) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolvedRecord = $policy->versions()
|
||||||
|
->where('tenant_id', (int) $policy->tenant_id)
|
||||||
|
->whereKey($recordId)
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if (! $resolvedRecord instanceof PolicyVersion) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedRecord;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,7 +2,9 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
|
use App\Filament\Concerns\ScopesGlobalSearchToTenant;
|
||||||
use App\Filament\Resources\PolicyVersionResource\Pages;
|
use App\Filament\Resources\PolicyVersionResource\Pages;
|
||||||
use App\Jobs\BulkPolicyVersionForceDeleteJob;
|
use App\Jobs\BulkPolicyVersionForceDeleteJob;
|
||||||
use App\Jobs\BulkPolicyVersionPruneJob;
|
use App\Jobs\BulkPolicyVersionPruneJob;
|
||||||
@ -59,7 +61,9 @@
|
|||||||
|
|
||||||
class PolicyVersionResource extends Resource
|
class PolicyVersionResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
use ScopesGlobalSearchToTenant;
|
||||||
|
|
||||||
protected static ?string $model = PolicyVersion::class;
|
protected static ?string $model = PolicyVersion::class;
|
||||||
|
|
||||||
@ -893,7 +897,6 @@ public static function table(Table $table): Table
|
|||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenant = static::resolveTenantContextForCurrentPanelOrFail();
|
$tenant = static::resolveTenantContextForCurrentPanelOrFail();
|
||||||
$tenantId = $tenant->getKey();
|
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
|
|
||||||
$resolver = app(CapabilityResolver::class);
|
$resolver = app(CapabilityResolver::class);
|
||||||
@ -903,8 +906,7 @@ public static function getEloquentQuery(): Builder
|
|||||||
|| $resolver->can($user, $tenant, Capabilities::TENANT_FINDINGS_VIEW)
|
|| $resolver->can($user, $tenant, Capabilities::TENANT_FINDINGS_VIEW)
|
||||||
);
|
);
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
return static::getTenantOwnedEloquentQuery()
|
||||||
->when($tenantId, fn (Builder $query) => $query->where('tenant_id', $tenantId))
|
|
||||||
->when(! $canSeeBaselinePurposeEvidence, function (Builder $query): Builder {
|
->when(! $canSeeBaselinePurposeEvidence, function (Builder $query): Builder {
|
||||||
return $query->where(function (Builder $query): void {
|
return $query->where(function (Builder $query): void {
|
||||||
$query
|
$query
|
||||||
@ -918,6 +920,36 @@ public static function getEloquentQuery(): Builder
|
|||||||
->with('policy');
|
->with('policy');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string $key): \Illuminate\Database\Eloquent\Model
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanelOrFail();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
$resolver = app(CapabilityResolver::class);
|
||||||
|
$canSeeBaselinePurposeEvidence = $user instanceof User
|
||||||
|
&& (
|
||||||
|
$resolver->can($user, $tenant, Capabilities::TENANT_SYNC)
|
||||||
|
|| $resolver->can($user, $tenant, Capabilities::TENANT_FINDINGS_VIEW)
|
||||||
|
);
|
||||||
|
|
||||||
|
return static::resolveTenantOwnedRecordOrFail(
|
||||||
|
$key,
|
||||||
|
parent::getEloquentQuery()
|
||||||
|
->withTrashed()
|
||||||
|
->when(! $canSeeBaselinePurposeEvidence, function (Builder $query): Builder {
|
||||||
|
return $query->where(function (Builder $query): void {
|
||||||
|
$query
|
||||||
|
->whereNull('capture_purpose')
|
||||||
|
->orWhereNotIn('capture_purpose', [
|
||||||
|
PolicyVersionCapturePurpose::BaselineCapture->value,
|
||||||
|
PolicyVersionCapturePurpose::BaselineCompare->value,
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
->with('policy'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return list<array{
|
* @return list<array{
|
||||||
* key: string,
|
* key: string,
|
||||||
|
|||||||
@ -9,6 +9,7 @@
|
|||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
use Filament\Support\Enums\Width;
|
use Filament\Support\Enums\Width;
|
||||||
use Illuminate\Contracts\View\View;
|
use Illuminate\Contracts\View\View;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewPolicyVersion extends ViewRecord
|
class ViewPolicyVersion extends ViewRecord
|
||||||
{
|
{
|
||||||
@ -16,6 +17,11 @@ class ViewPolicyVersion extends ViewRecord
|
|||||||
|
|
||||||
protected Width|string|null $maxContentWidth = Width::Full;
|
protected Width|string|null $maxContentWidth = Width::Full;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return PolicyVersionResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
protected function getHeaderActions(): array
|
protected function getHeaderActions(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
|||||||
@ -824,9 +824,12 @@ public static function table(Table $table): Table
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Connection check blocked')
|
->title('Connection check blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Actions\Action::make('view_run')
|
Actions\Action::make('view_run')
|
||||||
@ -921,9 +924,12 @@ public static function table(Table $table): Table
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Inventory sync blocked')
|
->title('Inventory sync blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Actions\Action::make('view_run')
|
Actions\Action::make('view_run')
|
||||||
@ -1015,9 +1021,12 @@ public static function table(Table $table): Table
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Compliance snapshot blocked')
|
->title('Compliance snapshot blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Actions\Action::make('view_run')
|
Actions\Action::make('view_run')
|
||||||
|
|||||||
@ -278,9 +278,12 @@ protected function getHeaderActions(): array
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Connection check blocked')
|
->title('Connection check blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Action::make('view_run')
|
Action::make('view_run')
|
||||||
@ -647,9 +650,12 @@ protected function getHeaderActions(): array
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Inventory sync blocked')
|
->title('Inventory sync blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Action::make('view_run')
|
Action::make('view_run')
|
||||||
@ -758,9 +764,12 @@ protected function getHeaderActions(): array
|
|||||||
? (string) $result->run->context['reason_code']
|
? (string) $result->run->context['reason_code']
|
||||||
: 'unknown_error';
|
: 'unknown_error';
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Compliance snapshot blocked')
|
->title('Compliance snapshot blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions([
|
->actions([
|
||||||
Action::make('view_run')
|
Action::make('view_run')
|
||||||
|
|||||||
@ -4,6 +4,7 @@
|
|||||||
|
|
||||||
use App\Contracts\Hardening\WriteGateInterface;
|
use App\Contracts\Hardening\WriteGateInterface;
|
||||||
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
use App\Filament\Resources\RestoreRunResource\Pages;
|
use App\Filament\Resources\RestoreRunResource\Pages;
|
||||||
use App\Jobs\BulkRestoreRunDeleteJob;
|
use App\Jobs\BulkRestoreRunDeleteJob;
|
||||||
@ -66,6 +67,7 @@
|
|||||||
|
|
||||||
class RestoreRunResource extends Resource
|
class RestoreRunResource extends Resource
|
||||||
{
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
use ResolvesPanelTenantContext;
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
protected static ?string $model = RestoreRun::class;
|
protected static ?string $model = RestoreRun::class;
|
||||||
@ -242,18 +244,44 @@ public static function makeCreateAction(): Actions\CreateAction
|
|||||||
|
|
||||||
public static function getEloquentQuery(): Builder
|
public static function getEloquentQuery(): Builder
|
||||||
{
|
{
|
||||||
$tenantId = static::resolveTenantContextForCurrentPanel()?->getKey();
|
return static::scopeTenantOwnedQuery(parent::getEloquentQuery())
|
||||||
|
->with('backupSet');
|
||||||
|
}
|
||||||
|
|
||||||
return parent::getEloquentQuery()
|
public static function resolveScopedRecordOrFail(int|string $key): Model
|
||||||
->with('backupSet')
|
{
|
||||||
->when(
|
return static::resolveTenantOwnedRecordOrFail(
|
||||||
$tenantId !== null,
|
$key,
|
||||||
fn (Builder $query): Builder => $query->where('tenant_id', (int) $tenantId),
|
parent::getEloquentQuery()->withTrashed()->with('backupSet'),
|
||||||
)
|
);
|
||||||
->when(
|
}
|
||||||
$tenantId === null,
|
|
||||||
fn (Builder $query): Builder => $query->whereRaw('1 = 0'),
|
protected static function resolveProtectedRestoreRunRecordOrFail(RestoreRun|int|string $record): RestoreRun
|
||||||
);
|
{
|
||||||
|
$resolvedRecord = static::resolveScopedRecordOrFail($record instanceof Model ? $record->getKey() : $record);
|
||||||
|
|
||||||
|
if (! $resolvedRecord instanceof RestoreRun) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolvedRecord;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, int>
|
||||||
|
*/
|
||||||
|
protected static function resolveProtectedRestoreRunIds(Collection $records): array
|
||||||
|
{
|
||||||
|
return $records
|
||||||
|
->map(function (mixed $record): int {
|
||||||
|
$resolvedRecord = static::resolveProtectedRestoreRunRecordOrFail($record instanceof RestoreRun ? $record : (is_numeric($record) ? (int) $record : 0));
|
||||||
|
|
||||||
|
return (int) $resolvedRecord->getKey();
|
||||||
|
})
|
||||||
|
->filter(fn (int $value): bool => $value > 0)
|
||||||
|
->unique()
|
||||||
|
->values()
|
||||||
|
->all();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -796,10 +824,10 @@ public static function table(Table $table): Table
|
|||||||
->label('Total')
|
->label('Total')
|
||||||
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['total'] ?? 0)),
|
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['total'] ?? 0)),
|
||||||
Tables\Columns\TextColumn::make('summary_succeeded')
|
Tables\Columns\TextColumn::make('summary_succeeded')
|
||||||
->label('Succeeded')
|
->label('Applied')
|
||||||
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['succeeded'] ?? 0)),
|
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['succeeded'] ?? 0)),
|
||||||
Tables\Columns\TextColumn::make('summary_failed')
|
Tables\Columns\TextColumn::make('summary_failed')
|
||||||
->label('Failed')
|
->label('Failed items')
|
||||||
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['failed'] ?? 0)),
|
->state(fn (RestoreRun $record): int => (int) (($record->metadata ?? [])['failed'] ?? 0)),
|
||||||
Tables\Columns\TextColumn::make('started_at')->dateTime()->since()->sortable(),
|
Tables\Columns\TextColumn::make('started_at')->dateTime()->since()->sortable(),
|
||||||
Tables\Columns\TextColumn::make('completed_at')->dateTime()->since()->sortable(),
|
Tables\Columns\TextColumn::make('completed_at')->dateTime()->since()->sortable(),
|
||||||
@ -846,6 +874,8 @@ public static function table(Table $table): Table
|
|||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (RestoreRun $record): bool => $record->trashed())
|
->visible(fn (RestoreRun $record): bool => $record->trashed())
|
||||||
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
||||||
|
$record = static::resolveProtectedRestoreRunRecordOrFail($record);
|
||||||
|
|
||||||
$record->restore();
|
$record->restore();
|
||||||
|
|
||||||
if ($record->tenant) {
|
if ($record->tenant) {
|
||||||
@ -877,6 +907,8 @@ public static function table(Table $table): Table
|
|||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (RestoreRun $record): bool => ! $record->trashed())
|
->visible(fn (RestoreRun $record): bool => ! $record->trashed())
|
||||||
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
||||||
|
$record = static::resolveProtectedRestoreRunRecordOrFail($record);
|
||||||
|
|
||||||
if (! $record->isDeletable()) {
|
if (! $record->isDeletable()) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Restore run cannot be archived')
|
->title('Restore run cannot be archived')
|
||||||
@ -918,6 +950,8 @@ public static function table(Table $table): Table
|
|||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (RestoreRun $record): bool => $record->trashed())
|
->visible(fn (RestoreRun $record): bool => $record->trashed())
|
||||||
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
->action(function (RestoreRun $record, \App\Services\Intune\AuditLogger $auditLogger) {
|
||||||
|
$record = static::resolveProtectedRestoreRunRecordOrFail($record);
|
||||||
|
|
||||||
if ($record->tenant) {
|
if ($record->tenant) {
|
||||||
$auditLogger->log(
|
$auditLogger->log(
|
||||||
tenant: $record->tenant,
|
tenant: $record->tenant,
|
||||||
@ -978,7 +1012,7 @@ public static function table(Table $table): Table
|
|||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
$count = $records->count();
|
$count = $records->count();
|
||||||
$ids = $records->pluck('id')->toArray();
|
$ids = static::resolveProtectedRestoreRunIds($records);
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
return;
|
return;
|
||||||
@ -1048,7 +1082,7 @@ public static function table(Table $table): Table
|
|||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
$count = $records->count();
|
$count = $records->count();
|
||||||
$ids = $records->pluck('id')->toArray();
|
$ids = static::resolveProtectedRestoreRunIds($records);
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
return;
|
return;
|
||||||
@ -1138,7 +1172,7 @@ public static function table(Table $table): Table
|
|||||||
$tenant = static::resolveTenantContextForCurrentPanel();
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
$user = auth()->user();
|
$user = auth()->user();
|
||||||
$count = $records->count();
|
$count = $records->count();
|
||||||
$ids = $records->pluck('id')->toArray();
|
$ids = static::resolveProtectedRestoreRunIds($records);
|
||||||
|
|
||||||
if (! $tenant instanceof Tenant) {
|
if (! $tenant instanceof Tenant) {
|
||||||
return;
|
return;
|
||||||
@ -1227,7 +1261,7 @@ public static function infolist(Schema $schema): Schema
|
|||||||
$succeeded = (int) ($meta['succeeded'] ?? 0);
|
$succeeded = (int) ($meta['succeeded'] ?? 0);
|
||||||
$failed = (int) ($meta['failed'] ?? 0);
|
$failed = (int) ($meta['failed'] ?? 0);
|
||||||
|
|
||||||
return sprintf('Total: %d • Succeeded: %d • Failed: %d', $total, $succeeded, $failed);
|
return sprintf('Total: %d • Applied: %d • Failed items: %d', $total, $succeeded, $failed);
|
||||||
}),
|
}),
|
||||||
Infolists\Components\TextEntry::make('is_dry_run')
|
Infolists\Components\TextEntry::make('is_dry_run')
|
||||||
->label('Dry-run')
|
->label('Dry-run')
|
||||||
@ -1694,6 +1728,8 @@ public static function createRestoreRun(array $data): RestoreRun
|
|||||||
'restore_run_id' => (int) $restoreRun->getKey(),
|
'restore_run_id' => (int) $restoreRun->getKey(),
|
||||||
'backup_set_id' => (int) $backupSet->getKey(),
|
'backup_set_id' => (int) $backupSet->getKey(),
|
||||||
'is_dry_run' => (bool) ($restoreRun->is_dry_run ?? false),
|
'is_dry_run' => (bool) ($restoreRun->is_dry_run ?? false),
|
||||||
|
'execution_authority_mode' => 'actor_bound',
|
||||||
|
'required_capability' => Capabilities::TENANT_MANAGE,
|
||||||
],
|
],
|
||||||
initiator: $initiator,
|
initiator: $initiator,
|
||||||
);
|
);
|
||||||
@ -1925,6 +1961,7 @@ private static function rerunActionWithGate(): Actions\Action|BulkAction
|
|||||||
\App\Services\Intune\AuditLogger $auditLogger,
|
\App\Services\Intune\AuditLogger $auditLogger,
|
||||||
HasTable $livewire
|
HasTable $livewire
|
||||||
) {
|
) {
|
||||||
|
$record = static::resolveProtectedRestoreRunRecordOrFail($record);
|
||||||
$tenant = $record->tenant;
|
$tenant = $record->tenant;
|
||||||
$backupSet = $record->backupSet;
|
$backupSet = $record->backupSet;
|
||||||
|
|
||||||
@ -2092,6 +2129,8 @@ private static function rerunActionWithGate(): Actions\Action|BulkAction
|
|||||||
'restore_run_id' => (int) $newRun->getKey(),
|
'restore_run_id' => (int) $newRun->getKey(),
|
||||||
'backup_set_id' => (int) $backupSet->getKey(),
|
'backup_set_id' => (int) $backupSet->getKey(),
|
||||||
'is_dry_run' => (bool) ($newRun->is_dry_run ?? false),
|
'is_dry_run' => (bool) ($newRun->is_dry_run ?? false),
|
||||||
|
'execution_authority_mode' => 'actor_bound',
|
||||||
|
'required_capability' => Capabilities::TENANT_MANAGE,
|
||||||
],
|
],
|
||||||
initiator: $initiator,
|
initiator: $initiator,
|
||||||
);
|
);
|
||||||
|
|||||||
@ -3,12 +3,42 @@
|
|||||||
namespace App\Filament\Resources\RestoreRunResource\Pages;
|
namespace App\Filament\Resources\RestoreRunResource\Pages;
|
||||||
|
|
||||||
use App\Filament\Resources\RestoreRunResource;
|
use App\Filament\Resources\RestoreRunResource;
|
||||||
|
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
||||||
use Filament\Resources\Pages\ListRecords;
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
||||||
|
|
||||||
class ListRestoreRuns extends ListRecords
|
class ListRestoreRuns extends ListRecords
|
||||||
{
|
{
|
||||||
protected static string $resource = RestoreRunResource::class;
|
protected static string $resource = RestoreRunResource::class;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $arguments
|
||||||
|
* @param array<string, mixed> $context
|
||||||
|
*/
|
||||||
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
||||||
|
{
|
||||||
|
if (($context['table'] ?? false) === true && filled($context['recordKey'] ?? null) && in_array($name, ['archive', 'forceDelete', 'rerun'], true)) {
|
||||||
|
try {
|
||||||
|
RestoreRunResource::resolveScopedRecordOrFail($context['recordKey']);
|
||||||
|
} catch (ModelNotFoundException) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::mountAction($name, $arguments, $context);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function mount(): void
|
||||||
|
{
|
||||||
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
||||||
|
$this->getTableFiltersSessionKey(),
|
||||||
|
request: request(),
|
||||||
|
tenantFilterName: null,
|
||||||
|
);
|
||||||
|
|
||||||
|
parent::mount();
|
||||||
|
}
|
||||||
|
|
||||||
private function tableHasRecords(): bool
|
private function tableHasRecords(): bool
|
||||||
{
|
{
|
||||||
return $this->getTableRecords()->count() > 0;
|
return $this->getTableRecords()->count() > 0;
|
||||||
|
|||||||
@ -4,8 +4,14 @@
|
|||||||
|
|
||||||
use App\Filament\Resources\RestoreRunResource;
|
use App\Filament\Resources\RestoreRunResource;
|
||||||
use Filament\Resources\Pages\ViewRecord;
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
class ViewRestoreRun extends ViewRecord
|
class ViewRestoreRun extends ViewRecord
|
||||||
{
|
{
|
||||||
protected static string $resource = RestoreRunResource::class;
|
protected static string $resource = RestoreRunResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return RestoreRunResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,12 +2,15 @@
|
|||||||
|
|
||||||
namespace App\Filament\Resources;
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Exceptions\ReviewPackEvidenceResolutionException;
|
||||||
|
use App\Filament\Resources\EvidenceSnapshotResource as TenantEvidenceSnapshotResource;
|
||||||
use App\Filament\Resources\ReviewPackResource\Pages;
|
use App\Filament\Resources\ReviewPackResource\Pages;
|
||||||
use App\Models\ReviewPack;
|
use App\Models\ReviewPack;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Services\ReviewPackService;
|
use App\Services\ReviewPackService;
|
||||||
use App\Support\Auth\Capabilities;
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
use App\Support\Badges\BadgeDomain;
|
use App\Support\Badges\BadgeDomain;
|
||||||
use App\Support\Badges\BadgeRenderer;
|
use App\Support\Badges\BadgeRenderer;
|
||||||
use App\Support\OpsUx\OperationUxPresenter;
|
use App\Support\OpsUx\OperationUxPresenter;
|
||||||
@ -17,11 +20,14 @@
|
|||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthEnvelope;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
use BackedEnum;
|
use BackedEnum;
|
||||||
use Filament\Actions;
|
use Filament\Actions;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Forms\Components\Toggle;
|
use Filament\Forms\Components\Toggle;
|
||||||
use Filament\Infolists\Components\TextEntry;
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Infolists\Components\ViewEntry;
|
||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Resources\Resource;
|
use Filament\Resources\Resource;
|
||||||
use Filament\Schemas\Components\Section;
|
use Filament\Schemas\Components\Section;
|
||||||
@ -109,6 +115,15 @@ public static function infolist(Schema $schema): Schema
|
|||||||
{
|
{
|
||||||
return $schema
|
return $schema
|
||||||
->schema([
|
->schema([
|
||||||
|
Section::make('Artifact truth')
|
||||||
|
->schema([
|
||||||
|
ViewEntry::make('artifact_truth')
|
||||||
|
->hiddenLabel()
|
||||||
|
->view('filament.infolists.entries.governance-artifact-truth')
|
||||||
|
->state(fn (ReviewPack $record): array => static::truthEnvelope($record)->toArray())
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columnSpanFull(),
|
||||||
Section::make('Status')
|
Section::make('Status')
|
||||||
->schema([
|
->schema([
|
||||||
TextEntry::make('status')
|
TextEntry::make('status')
|
||||||
@ -164,6 +179,21 @@ public static function infolist(Schema $schema): Schema
|
|||||||
Section::make('Metadata')
|
Section::make('Metadata')
|
||||||
->schema([
|
->schema([
|
||||||
TextEntry::make('initiator.name')->label('Initiated by')->placeholder('—'),
|
TextEntry::make('initiator.name')->label('Initiated by')->placeholder('—'),
|
||||||
|
TextEntry::make('tenantReview.id')
|
||||||
|
->label('Tenant review')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->url(fn (ReviewPack $record): ?string => $record->tenantReview && $record->tenant
|
||||||
|
? TenantReviewResource::tenantScopedUrl('view', ['record' => $record->tenantReview], $record->tenant)
|
||||||
|
: null)
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('summary.review_status')
|
||||||
|
->label('Review status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewStatus))
|
||||||
|
->placeholder('—'),
|
||||||
TextEntry::make('operationRun.id')
|
TextEntry::make('operationRun.id')
|
||||||
->label('Operation run')
|
->label('Operation run')
|
||||||
->url(fn (ReviewPack $record): ?string => $record->operation_run_id
|
->url(fn (ReviewPack $record): ?string => $record->operation_run_id
|
||||||
@ -177,6 +207,33 @@ public static function infolist(Schema $schema): Schema
|
|||||||
])
|
])
|
||||||
->columns(2)
|
->columns(2)
|
||||||
->columnSpanFull(),
|
->columnSpanFull(),
|
||||||
|
|
||||||
|
Section::make('Evidence snapshot')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('summary.evidence_resolution.outcome')
|
||||||
|
->label('Resolution')
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('evidenceSnapshot.id')
|
||||||
|
->label('Snapshot')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->url(fn (ReviewPack $record): ?string => $record->evidenceSnapshot
|
||||||
|
? TenantEvidenceSnapshotResource::getUrl('view', ['record' => $record->evidenceSnapshot], tenant: $record->tenant)
|
||||||
|
: null),
|
||||||
|
TextEntry::make('evidenceSnapshot.completeness_state')
|
||||||
|
->label('Snapshot completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::EvidenceCompleteness))
|
||||||
|
->placeholder('—'),
|
||||||
|
TextEntry::make('summary.evidence_resolution.snapshot_fingerprint')
|
||||||
|
->label('Snapshot fingerprint')
|
||||||
|
->copyable()
|
||||||
|
->placeholder('—'),
|
||||||
|
])
|
||||||
|
->columns(2)
|
||||||
|
->columnSpanFull(),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -194,6 +251,15 @@ public static function table(Table $table): Table
|
|||||||
->icon(BadgeRenderer::icon(BadgeDomain::ReviewPackStatus))
|
->icon(BadgeRenderer::icon(BadgeDomain::ReviewPackStatus))
|
||||||
->iconColor(BadgeRenderer::iconColor(BadgeDomain::ReviewPackStatus))
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::ReviewPackStatus))
|
||||||
->sortable(),
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_truth')
|
||||||
|
->label('Artifact truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (ReviewPack $record): string => static::truthEnvelope($record)->primaryLabel)
|
||||||
|
->color(fn (ReviewPack $record): string => static::truthEnvelope($record)->primaryBadgeSpec()->color)
|
||||||
|
->icon(fn (ReviewPack $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->icon)
|
||||||
|
->iconColor(fn (ReviewPack $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->iconColor)
|
||||||
|
->description(fn (ReviewPack $record): ?string => static::truthEnvelope($record)->primaryExplanation)
|
||||||
|
->wrap(),
|
||||||
Tables\Columns\TextColumn::make('tenant.name')
|
Tables\Columns\TextColumn::make('tenant.name')
|
||||||
->label('Tenant')
|
->label('Tenant')
|
||||||
->searchable(),
|
->searchable(),
|
||||||
@ -201,6 +267,10 @@ public static function table(Table $table): Table
|
|||||||
->dateTime()
|
->dateTime()
|
||||||
->sortable()
|
->sortable()
|
||||||
->placeholder('—'),
|
->placeholder('—'),
|
||||||
|
Tables\Columns\TextColumn::make('tenantReview.id')
|
||||||
|
->label('Review')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->toggleable(isToggledHiddenByDefault: true),
|
||||||
Tables\Columns\TextColumn::make('expires_at')
|
Tables\Columns\TextColumn::make('expires_at')
|
||||||
->dateTime()
|
->dateTime()
|
||||||
->sortable()
|
->sortable()
|
||||||
@ -209,6 +279,29 @@ public static function table(Table $table): Table
|
|||||||
->label('Size')
|
->label('Size')
|
||||||
->formatStateUsing(fn ($state): string => $state ? Number::fileSize((int) $state) : '—')
|
->formatStateUsing(fn ($state): string => $state ? Number::fileSize((int) $state) : '—')
|
||||||
->sortable(),
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('publication_truth')
|
||||||
|
->label('Publication')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (ReviewPack $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->label)
|
||||||
|
->color(fn (ReviewPack $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->color)
|
||||||
|
->icon(fn (ReviewPack $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->icon)
|
||||||
|
->iconColor(fn (ReviewPack $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->iconColor),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->getStateUsing(fn (ReviewPack $record): string => static::truthEnvelope($record)->nextStepText())
|
||||||
|
->wrap(),
|
||||||
Tables\Columns\TextColumn::make('created_at')
|
Tables\Columns\TextColumn::make('created_at')
|
||||||
->label('Created')
|
->label('Created')
|
||||||
->since()
|
->since()
|
||||||
@ -304,6 +397,11 @@ public static function getPages(): array
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static function truthEnvelope(ReviewPack $record): ArtifactTruthEnvelope
|
||||||
|
{
|
||||||
|
return app(ArtifactTruthPresenter::class)->forReviewPack($record);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array<string, mixed> $data
|
* @param array<string, mixed> $data
|
||||||
*/
|
*/
|
||||||
@ -331,7 +429,23 @@ public static function executeGeneration(array $data): void
|
|||||||
'include_operations' => (bool) ($data['include_operations'] ?? true),
|
'include_operations' => (bool) ($data['include_operations'] ?? true),
|
||||||
];
|
];
|
||||||
|
|
||||||
$reviewPack = $service->generate($tenant, $user, $options);
|
try {
|
||||||
|
$reviewPack = $service->generate($tenant, $user, $options);
|
||||||
|
} catch (ReviewPackEvidenceResolutionException $exception) {
|
||||||
|
$reasons = $exception->result->reasons;
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->danger()
|
||||||
|
->title(match ($exception->result->outcome) {
|
||||||
|
'missing_snapshot' => 'Create snapshot required',
|
||||||
|
'snapshot_ineligible' => 'Snapshot is not eligible',
|
||||||
|
default => 'Unable to generate review pack',
|
||||||
|
})
|
||||||
|
->body($reasons === [] ? $exception->getMessage() : implode(' ', $reasons))
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (! $reviewPack->wasRecentlyCreated) {
|
if (! $reviewPack->wasRecentlyCreated) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
|
|||||||
@ -41,7 +41,10 @@
|
|||||||
use App\Support\Rbac\UiEnforcement;
|
use App\Support\Rbac\UiEnforcement;
|
||||||
use App\Support\Tenants\TenantActionDescriptor;
|
use App\Support\Tenants\TenantActionDescriptor;
|
||||||
use App\Support\Tenants\TenantActionSurface;
|
use App\Support\Tenants\TenantActionSurface;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
use App\Support\Tenants\TenantLifecyclePresentation;
|
use App\Support\Tenants\TenantLifecyclePresentation;
|
||||||
|
use App\Support\Tenants\TenantOperabilityOutcome;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
@ -228,7 +231,7 @@ public static function getGlobalSearchEloquentQuery(): Builder
|
|||||||
return static::getEloquentQuery()->whereRaw('1 = 0');
|
return static::getEloquentQuery()->whereRaw('1 = 0');
|
||||||
}
|
}
|
||||||
|
|
||||||
return static::tenantOperability()->applySelectableScope(
|
return static::tenantOperability()->applyAdministrativeDiscoverabilityScope(
|
||||||
static::getEloquentQuery(),
|
static::getEloquentQuery(),
|
||||||
(new Tenant)->getTable(),
|
(new Tenant)->getTable(),
|
||||||
);
|
);
|
||||||
@ -513,7 +516,7 @@ public static function table(Table $table): Table
|
|||||||
->icon('heroicon-o-check-badge')
|
->icon('heroicon-o-check-badge')
|
||||||
->color('primary')
|
->color('primary')
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (Tenant $record): bool => $record->isActive())
|
->visible(fn (Tenant $record): bool => static::verificationActionVisible($record))
|
||||||
->action(function (
|
->action(function (
|
||||||
Tenant $record,
|
Tenant $record,
|
||||||
StartVerification $verification,
|
StartVerification $verification,
|
||||||
@ -605,9 +608,12 @@ public static function table(Table $table): Table
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Verification blocked')
|
->title('Verification blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions($actions)
|
->actions($actions)
|
||||||
->send();
|
->send();
|
||||||
@ -905,8 +911,20 @@ public static function infolist(Schema $schema): Schema
|
|||||||
->visible(fn (Tenant $record): bool => filled($record->rbac_status)),
|
->visible(fn (Tenant $record): bool => filled($record->rbac_status)),
|
||||||
Section::make('RBAC Details')
|
Section::make('RBAC Details')
|
||||||
->schema([
|
->schema([
|
||||||
|
Infolists\Components\TextEntry::make('rbac_status_reason_label')
|
||||||
|
->label('Reason')
|
||||||
|
->state(fn (Tenant $record): ?string => app(\App\Support\ReasonTranslation\ReasonPresenter::class)
|
||||||
|
->primaryLabel(app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forRbacReason($record->rbac_status_reason, 'detail')))
|
||||||
|
->visible(fn (?string $state): bool => filled($state)),
|
||||||
|
Infolists\Components\TextEntry::make('rbac_status_reason_explanation')
|
||||||
|
->label('Explanation')
|
||||||
|
->state(fn (Tenant $record): ?string => app(\App\Support\ReasonTranslation\ReasonPresenter::class)
|
||||||
|
->shortExplanation(app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forRbacReason($record->rbac_status_reason, 'detail')))
|
||||||
|
->visible(fn (?string $state): bool => filled($state))
|
||||||
|
->columnSpanFull(),
|
||||||
Infolists\Components\TextEntry::make('rbac_status_reason')
|
Infolists\Components\TextEntry::make('rbac_status_reason')
|
||||||
->label('Reason'),
|
->label('Diagnostic code')
|
||||||
|
->copyable(),
|
||||||
Infolists\Components\TextEntry::make('rbac_role_definition_id')
|
Infolists\Components\TextEntry::make('rbac_role_definition_id')
|
||||||
->label('Role definition ID')
|
->label('Role definition ID')
|
||||||
->copyable(),
|
->copyable(),
|
||||||
@ -1081,6 +1099,26 @@ public static function relatedOnboardingDraftUrl(Tenant $tenant): ?string
|
|||||||
return route('admin.onboarding.draft', ['onboardingDraft' => (int) $draft->getKey()]);
|
return route('admin.onboarding.draft', ['onboardingDraft' => (int) $draft->getKey()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function verificationActionVisible(Tenant $tenant): bool
|
||||||
|
{
|
||||||
|
$outcome = static::verificationReadinessOutcome($tenant);
|
||||||
|
|
||||||
|
return $outcome->allowed || $outcome->isDeniedForCapability();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function verificationReadinessOutcome(Tenant $tenant): TenantOperabilityOutcome
|
||||||
|
{
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
return static::tenantOperability()->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::VerificationReadinessEligibility,
|
||||||
|
actor: $user instanceof User ? $user : null,
|
||||||
|
workspaceId: app(WorkspaceContext::class)->currentWorkspaceId(request()),
|
||||||
|
lane: TenantInteractionLane::AdministrativeManagement,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private static function tenantActionDescriptorForSurface(Tenant $tenant, TenantActionSurface $surface, string $key): ?TenantActionDescriptor
|
private static function tenantActionDescriptorForSurface(Tenant $tenant, TenantActionSurface $surface, string $key): ?TenantActionDescriptor
|
||||||
{
|
{
|
||||||
$descriptor = static::tenantActionCatalog($tenant, $surface)
|
$descriptor = static::tenantActionCatalog($tenant, $surface)
|
||||||
|
|||||||
@ -87,7 +87,7 @@ protected function getHeaderActions(): array
|
|||||||
->icon('heroicon-o-check-badge')
|
->icon('heroicon-o-check-badge')
|
||||||
->color('primary')
|
->color('primary')
|
||||||
->requiresConfirmation()
|
->requiresConfirmation()
|
||||||
->visible(fn (Tenant $record): bool => $record->isActive())
|
->visible(fn (Tenant $record): bool => TenantResource::verificationActionVisible($record))
|
||||||
->action(function (
|
->action(function (
|
||||||
Tenant $record,
|
Tenant $record,
|
||||||
StartVerification $verification,
|
StartVerification $verification,
|
||||||
@ -178,9 +178,12 @@ protected function getHeaderActions(): array
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Verification blocked')
|
->title('Verification blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions($actions)
|
->actions($actions)
|
||||||
->send();
|
->send();
|
||||||
|
|||||||
612
app/Filament/Resources/TenantReviewResource.php
Normal file
612
app/Filament/Resources/TenantReviewResource.php
Normal file
@ -0,0 +1,612 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources;
|
||||||
|
|
||||||
|
use App\Filament\Concerns\InteractsWithTenantOwnedRecords;
|
||||||
|
use App\Filament\Concerns\ResolvesPanelTenantContext;
|
||||||
|
use App\Filament\Resources\TenantReviewResource\Pages;
|
||||||
|
use App\Models\EvidenceSnapshot;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\TenantReview;
|
||||||
|
use App\Models\TenantReviewSection;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\ReviewPackService;
|
||||||
|
use App\Services\TenantReviews\TenantReviewService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\Badges\BadgeCatalog;
|
||||||
|
use App\Support\Badges\BadgeDomain;
|
||||||
|
use App\Support\Badges\BadgeRenderer;
|
||||||
|
use App\Support\OperationRunLinks;
|
||||||
|
use App\Support\OperationRunType;
|
||||||
|
use App\Support\OpsUx\OperationUxPresenter;
|
||||||
|
use App\Support\Rbac\UiEnforcement;
|
||||||
|
use App\Support\TenantReviewCompletenessState;
|
||||||
|
use App\Support\TenantReviewStatus;
|
||||||
|
use App\Support\Ui\ActionSurface\ActionSurfaceDeclaration;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceInspectAffordance;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceProfile;
|
||||||
|
use App\Support\Ui\ActionSurface\Enums\ActionSurfaceSlot;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthEnvelope;
|
||||||
|
use App\Support\Ui\GovernanceArtifactTruth\ArtifactTruthPresenter;
|
||||||
|
use BackedEnum;
|
||||||
|
use Filament\Actions;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\Select;
|
||||||
|
use Filament\Infolists\Components\RepeatableEntry;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Infolists\Components\ViewEntry;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Resources\Resource;
|
||||||
|
use Filament\Schemas\Components\Section;
|
||||||
|
use Filament\Schemas\Schema;
|
||||||
|
use Filament\Support\Enums\TextSize;
|
||||||
|
use Filament\Tables;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
use UnitEnum;
|
||||||
|
|
||||||
|
class TenantReviewResource extends Resource
|
||||||
|
{
|
||||||
|
use InteractsWithTenantOwnedRecords;
|
||||||
|
use ResolvesPanelTenantContext;
|
||||||
|
|
||||||
|
protected static bool $isDiscovered = false;
|
||||||
|
|
||||||
|
protected static ?string $model = TenantReview::class;
|
||||||
|
|
||||||
|
protected static ?string $slug = 'reviews';
|
||||||
|
|
||||||
|
protected static ?string $tenantOwnershipRelationshipName = 'tenant';
|
||||||
|
|
||||||
|
protected static bool $isGloballySearchable = false;
|
||||||
|
|
||||||
|
protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-document-magnifying-glass';
|
||||||
|
|
||||||
|
protected static string|UnitEnum|null $navigationGroup = 'Reporting';
|
||||||
|
|
||||||
|
protected static ?string $navigationLabel = 'Reviews';
|
||||||
|
|
||||||
|
protected static ?int $navigationSort = 45;
|
||||||
|
|
||||||
|
public static function shouldRegisterNavigation(): bool
|
||||||
|
{
|
||||||
|
return Filament::getCurrentPanel()?->getId() === 'tenant';
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canViewAny(): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $user->can(Capabilities::TENANT_REVIEW_VIEW, $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function canView(Model $record): bool
|
||||||
|
{
|
||||||
|
$tenant = static::resolveTenantContextForCurrentPanel();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User || ! $record instanceof TenantReview) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((int) $record->tenant_id !== (int) $tenant->getKey()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $user->can('view', $record);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function actionSurfaceDeclaration(): ActionSurfaceDeclaration
|
||||||
|
{
|
||||||
|
return ActionSurfaceDeclaration::forResource(ActionSurfaceProfile::CrudListAndView)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListHeader, 'Create review is available from the review library header.')
|
||||||
|
->satisfy(ActionSurfaceSlot::InspectAffordance, ActionSurfaceInspectAffordance::ClickableRow->value)
|
||||||
|
->satisfy(ActionSurfaceSlot::ListEmptyState, 'Empty state includes exactly one Create first review CTA.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListBulkMoreGroup, 'Tenant reviews do not expose bulk actions in the first slice.')
|
||||||
|
->exempt(ActionSurfaceSlot::ListRowMoreMenu, 'Primary row actions stay limited to View review and Export executive pack.')
|
||||||
|
->satisfy(ActionSurfaceSlot::DetailHeader, 'Detail exposes Refresh review, Publish review, Export executive pack, Archive review, and Create next review as applicable.');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getEloquentQuery(): Builder
|
||||||
|
{
|
||||||
|
return static::getTenantOwnedEloquentQuery()
|
||||||
|
->with(['tenant', 'evidenceSnapshot', 'operationRun', 'initiator', 'publisher', 'currentExportReviewPack', 'sections'])
|
||||||
|
->latest('generated_at')
|
||||||
|
->latest('id');
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function resolveScopedRecordOrFail(int|string|null $record): Model
|
||||||
|
{
|
||||||
|
return static::resolveTenantOwnedRecordOrFail($record);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function form(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function infolist(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema->schema([
|
||||||
|
Section::make('Artifact truth')
|
||||||
|
->schema([
|
||||||
|
ViewEntry::make('artifact_truth')
|
||||||
|
->hiddenLabel()
|
||||||
|
->view('filament.infolists.entries.governance-artifact-truth')
|
||||||
|
->state(fn (TenantReview $record): array => static::truthEnvelope($record)->toArray())
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columnSpanFull(),
|
||||||
|
Section::make('Review')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewStatus)),
|
||||||
|
TextEntry::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewCompleteness)),
|
||||||
|
TextEntry::make('tenant.name')->label('Tenant'),
|
||||||
|
TextEntry::make('generated_at')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('published_at')->dateTime()->placeholder('—'),
|
||||||
|
TextEntry::make('evidenceSnapshot.id')
|
||||||
|
->label('Evidence snapshot')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->url(fn (TenantReview $record): ?string => $record->evidenceSnapshot
|
||||||
|
? EvidenceSnapshotResource::getUrl('view', ['record' => $record->evidenceSnapshot], tenant: $record->tenant)
|
||||||
|
: null),
|
||||||
|
TextEntry::make('currentExportReviewPack.id')
|
||||||
|
->label('Current export')
|
||||||
|
->formatStateUsing(fn (?int $state): string => $state ? '#'.$state : '—')
|
||||||
|
->url(fn (TenantReview $record): ?string => $record->currentExportReviewPack
|
||||||
|
? ReviewPackResource::getUrl('view', ['record' => $record->currentExportReviewPack], tenant: $record->tenant)
|
||||||
|
: null),
|
||||||
|
TextEntry::make('fingerprint')
|
||||||
|
->copyable()
|
||||||
|
->placeholder('—')
|
||||||
|
->columnSpanFull()
|
||||||
|
->fontFamily('mono')
|
||||||
|
->size(TextSize::ExtraSmall),
|
||||||
|
])
|
||||||
|
->columns(2)
|
||||||
|
->columnSpanFull(),
|
||||||
|
Section::make('Executive posture')
|
||||||
|
->schema([
|
||||||
|
ViewEntry::make('review_summary')
|
||||||
|
->hiddenLabel()
|
||||||
|
->view('filament.infolists.entries.tenant-review-summary')
|
||||||
|
->state(fn (TenantReview $record): array => static::summaryPresentation($record))
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columnSpanFull(),
|
||||||
|
Section::make('Sections')
|
||||||
|
->schema([
|
||||||
|
RepeatableEntry::make('sections')
|
||||||
|
->hiddenLabel()
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('title'),
|
||||||
|
TextEntry::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewCompleteness)),
|
||||||
|
TextEntry::make('measured_at')->dateTime()->placeholder('—'),
|
||||||
|
Section::make('Details')
|
||||||
|
->schema([
|
||||||
|
ViewEntry::make('section_payload')
|
||||||
|
->hiddenLabel()
|
||||||
|
->view('filament.infolists.entries.tenant-review-section')
|
||||||
|
->state(fn (TenantReviewSection $record): array => static::sectionPresentation($record))
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->collapsible()
|
||||||
|
->collapsed()
|
||||||
|
->columnSpanFull(),
|
||||||
|
])
|
||||||
|
->columns(3),
|
||||||
|
])
|
||||||
|
->columnSpanFull(),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->defaultSort('generated_at', 'desc')
|
||||||
|
->persistFiltersInSession()
|
||||||
|
->persistSearchInSession()
|
||||||
|
->persistSortInSession()
|
||||||
|
->recordUrl(fn (TenantReview $record): string => static::tenantScopedUrl('view', ['record' => $record], $record->tenant))
|
||||||
|
->columns([
|
||||||
|
Tables\Columns\TextColumn::make('status')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewStatus))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewStatus))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewStatus))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewStatus))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_truth')
|
||||||
|
->label('Artifact truth')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => static::truthEnvelope($record)->primaryLabel)
|
||||||
|
->color(fn (TenantReview $record): string => static::truthEnvelope($record)->primaryBadgeSpec()->color)
|
||||||
|
->icon(fn (TenantReview $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->icon)
|
||||||
|
->iconColor(fn (TenantReview $record): ?string => static::truthEnvelope($record)->primaryBadgeSpec()->iconColor)
|
||||||
|
->description(fn (TenantReview $record): ?string => static::truthEnvelope($record)->primaryExplanation)
|
||||||
|
->wrap(),
|
||||||
|
Tables\Columns\TextColumn::make('completeness_state')
|
||||||
|
->label('Completeness')
|
||||||
|
->badge()
|
||||||
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->color(BadgeRenderer::color(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->icon(BadgeRenderer::icon(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::TenantReviewCompleteness))
|
||||||
|
->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('generated_at')->dateTime()->placeholder('—')->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('published_at')->dateTime()->placeholder('—')->sortable(),
|
||||||
|
Tables\Columns\TextColumn::make('summary.finding_count')->label('Findings'),
|
||||||
|
Tables\Columns\TextColumn::make('summary.section_state_counts.missing')->label(static::reviewCompletenessCountLabel(TenantReviewCompletenessState::Missing->value)),
|
||||||
|
Tables\Columns\TextColumn::make('publication_truth')
|
||||||
|
->label('Publication')
|
||||||
|
->badge()
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->label)
|
||||||
|
->color(fn (TenantReview $record): string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->color)
|
||||||
|
->icon(fn (TenantReview $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->icon)
|
||||||
|
->iconColor(fn (TenantReview $record): ?string => BadgeCatalog::spec(
|
||||||
|
BadgeDomain::GovernanceArtifactPublicationReadiness,
|
||||||
|
static::truthEnvelope($record)->publicationReadiness ?? 'internal_only',
|
||||||
|
)->iconColor),
|
||||||
|
Tables\Columns\IconColumn::make('summary.has_ready_export')
|
||||||
|
->label('Export')
|
||||||
|
->boolean(),
|
||||||
|
Tables\Columns\TextColumn::make('artifact_next_step')
|
||||||
|
->label('Next step')
|
||||||
|
->getStateUsing(fn (TenantReview $record): string => static::truthEnvelope($record)->nextStepText())
|
||||||
|
->wrap(),
|
||||||
|
Tables\Columns\TextColumn::make('fingerprint')
|
||||||
|
->toggleable(isToggledHiddenByDefault: true)
|
||||||
|
->searchable(),
|
||||||
|
])
|
||||||
|
->filters([
|
||||||
|
Tables\Filters\SelectFilter::make('status')
|
||||||
|
->options(collect(TenantReviewStatus::cases())
|
||||||
|
->mapWithKeys(fn (TenantReviewStatus $status): array => [$status->value => Str::headline($status->value)])
|
||||||
|
->all()),
|
||||||
|
Tables\Filters\SelectFilter::make('completeness_state')
|
||||||
|
->options(BadgeCatalog::options(BadgeDomain::TenantReviewCompleteness, TenantReviewCompletenessState::values())),
|
||||||
|
\App\Support\Filament\FilterPresets::dateRange('review_date', 'Review date', 'generated_at'),
|
||||||
|
])
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_review')
|
||||||
|
->label('View review')
|
||||||
|
->url(fn (TenantReview $record): string => static::tenantScopedUrl('view', ['record' => $record], $record->tenant)),
|
||||||
|
UiEnforcement::forTableAction(
|
||||||
|
Actions\Action::make('export_executive_pack')
|
||||||
|
->label('Export executive pack')
|
||||||
|
->icon('heroicon-o-arrow-down-tray')
|
||||||
|
->visible(fn (TenantReview $record): bool => in_array($record->status, [
|
||||||
|
TenantReviewStatus::Ready->value,
|
||||||
|
TenantReviewStatus::Published->value,
|
||||||
|
], true))
|
||||||
|
->action(fn (TenantReview $record): mixed => static::executeExport($record)),
|
||||||
|
fn (TenantReview $record): TenantReview => $record,
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
])
|
||||||
|
->bulkActions([])
|
||||||
|
->emptyStateHeading('No tenant reviews yet')
|
||||||
|
->emptyStateDescription('Create the first review from an anchored evidence snapshot to start the recurring review history for this tenant.')
|
||||||
|
->emptyStateActions([
|
||||||
|
static::makeCreateReviewAction(
|
||||||
|
name: 'create_first_review',
|
||||||
|
label: 'Create first review',
|
||||||
|
icon: 'heroicon-o-plus',
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getPages(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'index' => Pages\ListTenantReviews::route('/'),
|
||||||
|
'view' => Pages\ViewTenantReview::route('/{record}'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function makeCreateReviewAction(
|
||||||
|
string $name = 'create_review',
|
||||||
|
string $label = 'Create review',
|
||||||
|
string $icon = 'heroicon-o-plus',
|
||||||
|
): Actions\Action {
|
||||||
|
return UiEnforcement::forAction(
|
||||||
|
Actions\Action::make($name)
|
||||||
|
->label($label)
|
||||||
|
->icon($icon)
|
||||||
|
->form([
|
||||||
|
Section::make('Evidence basis')
|
||||||
|
->schema([
|
||||||
|
Select::make('evidence_snapshot_id')
|
||||||
|
->label('Evidence snapshot')
|
||||||
|
->required()
|
||||||
|
->options(fn (): array => static::evidenceSnapshotOptions())
|
||||||
|
->searchable()
|
||||||
|
->helperText('Choose the anchored evidence snapshot for this review.'),
|
||||||
|
]),
|
||||||
|
])
|
||||||
|
->action(fn (array $data): mixed => static::executeCreateReview($data)),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->apply();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
public static function executeCreateReview(array $data): void
|
||||||
|
{
|
||||||
|
$tenant = Filament::getTenant();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant || ! $user instanceof User) {
|
||||||
|
Notification::make()->danger()->title('Unable to create review — missing context.')->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->can(Capabilities::TENANT_REVIEW_MANAGE, $tenant)) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
$snapshotId = $data['evidence_snapshot_id'] ?? null;
|
||||||
|
$snapshot = is_numeric($snapshotId)
|
||||||
|
? EvidenceSnapshot::query()
|
||||||
|
->whereKey((int) $snapshotId)
|
||||||
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
|
->first()
|
||||||
|
: null;
|
||||||
|
|
||||||
|
if (! $snapshot instanceof EvidenceSnapshot) {
|
||||||
|
Notification::make()->danger()->title('Select a valid evidence snapshot.')->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$review = app(TenantReviewService::class)->create($tenant, $snapshot, $user);
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
Notification::make()->danger()->title('Unable to create review')->body($throwable->getMessage())->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $review->wasRecentlyCreated) {
|
||||||
|
Notification::make()
|
||||||
|
->success()
|
||||||
|
->title('Review already available')
|
||||||
|
->body('A matching mutable review already exists for this evidence basis.')
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_review')
|
||||||
|
->label('View review')
|
||||||
|
->url(static::tenantScopedUrl('view', ['record' => $review], $tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$toast = OperationUxPresenter::queuedToast(OperationRunType::TenantReviewCompose->value)
|
||||||
|
->body('The review is being composed in the background.');
|
||||||
|
|
||||||
|
if ($review->operation_run_id) {
|
||||||
|
$toast->actions([
|
||||||
|
Actions\Action::make('view_run')
|
||||||
|
->label('View run')
|
||||||
|
->url(OperationRunLinks::tenantlessView((int) $review->operation_run_id)),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$toast->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function executeExport(TenantReview $review): void
|
||||||
|
{
|
||||||
|
$review->loadMissing(['tenant', 'currentExportReviewPack']);
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User || ! $review->tenant instanceof Tenant) {
|
||||||
|
Notification::make()->danger()->title('Unable to export review — missing context.')->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($review->tenant)) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->can('export', $review)) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
$service = app(ReviewPackService::class);
|
||||||
|
|
||||||
|
if ($service->checkActiveRunForReview($review)) {
|
||||||
|
OperationUxPresenter::alreadyQueuedToast(OperationRunType::ReviewPackGenerate->value)
|
||||||
|
->body('An executive pack export is already queued or running for this review.')
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$pack = $service->generateFromReview($review, $user, [
|
||||||
|
'include_pii' => true,
|
||||||
|
'include_operations' => true,
|
||||||
|
]);
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
Notification::make()->danger()->title('Unable to export executive pack')->body($throwable->getMessage())->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $pack->wasRecentlyCreated) {
|
||||||
|
Notification::make()
|
||||||
|
->success()
|
||||||
|
->title('Executive pack already available')
|
||||||
|
->body('A matching executive pack already exists for this review.')
|
||||||
|
->actions([
|
||||||
|
Actions\Action::make('view_pack')
|
||||||
|
->label('View pack')
|
||||||
|
->url(ReviewPackResource::getUrl('view', ['record' => $pack], tenant: $review->tenant)),
|
||||||
|
])
|
||||||
|
->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
OperationUxPresenter::queuedToast(OperationRunType::ReviewPackGenerate->value)
|
||||||
|
->body('The executive pack is being generated in the background.')
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $parameters
|
||||||
|
*/
|
||||||
|
public static function tenantScopedUrl(
|
||||||
|
string $page = 'index',
|
||||||
|
array $parameters = [],
|
||||||
|
?Tenant $tenant = null,
|
||||||
|
?string $panel = null,
|
||||||
|
): string {
|
||||||
|
$panelId = $panel ?? 'tenant';
|
||||||
|
|
||||||
|
return static::getUrl($page, $parameters, panel: $panelId, tenant: $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private static function evidenceSnapshotOptions(): array
|
||||||
|
{
|
||||||
|
$tenant = Filament::getTenant();
|
||||||
|
|
||||||
|
if (! $tenant instanceof Tenant) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return EvidenceSnapshot::query()
|
||||||
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
|
->whereNotNull('generated_at')
|
||||||
|
->orderByDesc('generated_at')
|
||||||
|
->orderByDesc('id')
|
||||||
|
->get()
|
||||||
|
->mapWithKeys(static fn (EvidenceSnapshot $snapshot): array => [
|
||||||
|
(string) $snapshot->getKey() => sprintf(
|
||||||
|
'#%d · %s · %s',
|
||||||
|
(int) $snapshot->getKey(),
|
||||||
|
BadgeCatalog::spec(BadgeDomain::EvidenceCompleteness, $snapshot->completeness_state)->label,
|
||||||
|
$snapshot->generated_at?->format('Y-m-d H:i') ?? 'Pending'
|
||||||
|
),
|
||||||
|
])
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function reviewCompletenessCountLabel(string $state): string
|
||||||
|
{
|
||||||
|
return BadgeCatalog::spec(BadgeDomain::TenantReviewCompleteness, $state)->label;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function summaryPresentation(TenantReview $record): array
|
||||||
|
{
|
||||||
|
$summary = is_array($record->summary) ? $record->summary : [];
|
||||||
|
|
||||||
|
return [
|
||||||
|
'highlights' => is_array($summary['highlights'] ?? null) ? $summary['highlights'] : [],
|
||||||
|
'next_actions' => is_array($summary['recommended_next_actions'] ?? null) ? $summary['recommended_next_actions'] : [],
|
||||||
|
'publish_blockers' => is_array($summary['publish_blockers'] ?? null) ? $summary['publish_blockers'] : [],
|
||||||
|
'metrics' => [
|
||||||
|
['label' => 'Findings', 'value' => (string) ($summary['finding_count'] ?? 0)],
|
||||||
|
['label' => 'Reports', 'value' => (string) ($summary['report_count'] ?? 0)],
|
||||||
|
['label' => 'Operations', 'value' => (string) ($summary['operation_count'] ?? 0)],
|
||||||
|
['label' => 'Sections', 'value' => (string) ($summary['section_count'] ?? 0)],
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function sectionPresentation(TenantReviewSection $section): array
|
||||||
|
{
|
||||||
|
$summary = is_array($section->summary_payload) ? $section->summary_payload : [];
|
||||||
|
$render = is_array($section->render_payload) ? $section->render_payload : [];
|
||||||
|
$review = $section->tenantReview;
|
||||||
|
$tenant = $section->tenant;
|
||||||
|
|
||||||
|
return [
|
||||||
|
'summary' => collect($summary)->map(function (mixed $value, string $key): ?array {
|
||||||
|
if (is_array($value) || $value === null || $value === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'label' => Str::headline($key),
|
||||||
|
'value' => (string) $value,
|
||||||
|
];
|
||||||
|
})->filter()->values()->all(),
|
||||||
|
'highlights' => is_array($render['highlights'] ?? null) ? $render['highlights'] : [],
|
||||||
|
'entries' => is_array($render['entries'] ?? null) ? $render['entries'] : [],
|
||||||
|
'disclosure' => is_string($render['disclosure'] ?? null) ? $render['disclosure'] : null,
|
||||||
|
'next_actions' => is_array($render['next_actions'] ?? null) ? $render['next_actions'] : [],
|
||||||
|
'empty_state' => is_string($render['empty_state'] ?? null) ? $render['empty_state'] : null,
|
||||||
|
'links' => [],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function truthEnvelope(TenantReview $record): ArtifactTruthEnvelope
|
||||||
|
{
|
||||||
|
return app(ArtifactTruthPresenter::class)->forTenantReview($record);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,20 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\TenantReviewResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\TenantReviewResource;
|
||||||
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
|
||||||
|
class ListTenantReviews extends ListRecords
|
||||||
|
{
|
||||||
|
protected static string $resource = TenantReviewResource::class;
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
TenantReviewResource::makeCreateReviewAction(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,205 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\TenantReviewResource\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\TenantReviewResource;
|
||||||
|
use App\Models\Tenant;
|
||||||
|
use App\Models\TenantReview;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\TenantReviews\TenantReviewLifecycleService;
|
||||||
|
use App\Services\TenantReviews\TenantReviewService;
|
||||||
|
use App\Support\Auth\Capabilities;
|
||||||
|
use App\Support\OperationRunLinks;
|
||||||
|
use App\Support\Rbac\UiEnforcement;
|
||||||
|
use App\Support\TenantReviewStatus;
|
||||||
|
use Filament\Actions;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Filament\Resources\Pages\ViewRecord;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
|
class ViewTenantReview extends ViewRecord
|
||||||
|
{
|
||||||
|
protected static string $resource = TenantReviewResource::class;
|
||||||
|
|
||||||
|
protected function resolveRecord(int|string $key): Model
|
||||||
|
{
|
||||||
|
return TenantReviewResource::resolveScopedRecordOrFail($key);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function authorizeAccess(): void
|
||||||
|
{
|
||||||
|
$tenant = TenantReviewResource::panelTenantContext();
|
||||||
|
$record = $this->getRecord();
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User || ! $tenant instanceof Tenant || ! $record instanceof TenantReview) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((int) $record->tenant_id !== (int) $tenant->getKey()) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->canAccessTenant($tenant)) {
|
||||||
|
abort(404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $user->can('view', $record)) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
Actions\Action::make('view_run')
|
||||||
|
->label('View run')
|
||||||
|
->icon('heroicon-o-eye')
|
||||||
|
->color('gray')
|
||||||
|
->hidden(fn (): bool => ! is_numeric($this->record->operation_run_id))
|
||||||
|
->url(fn (): ?string => $this->record->operation_run_id
|
||||||
|
? OperationRunLinks::tenantlessView((int) $this->record->operation_run_id)
|
||||||
|
: null),
|
||||||
|
Actions\Action::make('view_export')
|
||||||
|
->label('View executive pack')
|
||||||
|
->icon('heroicon-o-document-arrow-down')
|
||||||
|
->color('gray')
|
||||||
|
->hidden(fn (): bool => ! $this->record->currentExportReviewPack)
|
||||||
|
->url(fn (): ?string => $this->record->currentExportReviewPack
|
||||||
|
? \App\Filament\Resources\ReviewPackResource::getUrl('view', ['record' => $this->record->currentExportReviewPack], tenant: $this->record->tenant)
|
||||||
|
: null),
|
||||||
|
Actions\Action::make('view_evidence')
|
||||||
|
->label('View evidence snapshot')
|
||||||
|
->icon('heroicon-o-shield-check')
|
||||||
|
->color('gray')
|
||||||
|
->hidden(fn (): bool => ! $this->record->evidenceSnapshot)
|
||||||
|
->url(fn (): ?string => $this->record->evidenceSnapshot
|
||||||
|
? \App\Filament\Resources\EvidenceSnapshotResource::getUrl('view', ['record' => $this->record->evidenceSnapshot], tenant: $this->record->tenant)
|
||||||
|
: null),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('refresh_review')
|
||||||
|
->label('Refresh review')
|
||||||
|
->icon('heroicon-o-arrow-path')
|
||||||
|
->hidden(fn (): bool => ! $this->record->isMutable())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
app(TenantReviewService::class)->refresh($this->record, $user);
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
Notification::make()->danger()->title('Unable to refresh review')->body($throwable->getMessage())->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Notification::make()->success()->title('Refresh review queued')->send();
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('publish_review')
|
||||||
|
->label('Publish review')
|
||||||
|
->icon('heroicon-o-check-badge')
|
||||||
|
->hidden(fn (): bool => ! $this->record->isMutable())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
app(TenantReviewLifecycleService::class)->publish($this->record, $user);
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
Notification::make()->danger()->title('Unable to publish review')->body($throwable->getMessage())->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->refreshFormData(['status', 'published_at', 'published_by_user_id', 'summary']);
|
||||||
|
Notification::make()->success()->title('Review published')->send();
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('export_executive_pack')
|
||||||
|
->label('Export executive pack')
|
||||||
|
->icon('heroicon-o-arrow-down-tray')
|
||||||
|
->hidden(fn (): bool => ! in_array($this->record->status, [
|
||||||
|
TenantReviewStatus::Ready->value,
|
||||||
|
TenantReviewStatus::Published->value,
|
||||||
|
], true))
|
||||||
|
->action(fn (): mixed => TenantReviewResource::executeExport($this->record)),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
Actions\ActionGroup::make([
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('create_next_review')
|
||||||
|
->label('Create next review')
|
||||||
|
->icon('heroicon-o-document-duplicate')
|
||||||
|
->hidden(fn (): bool => ! $this->record->isPublished())
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$nextReview = app(TenantReviewLifecycleService::class)->createNextReview($this->record, $user);
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
Notification::make()->danger()->title('Unable to create next review')->body($throwable->getMessage())->send();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->redirect(TenantReviewResource::tenantScopedUrl('view', ['record' => $nextReview], $nextReview->tenant));
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
UiEnforcement::forAction(
|
||||||
|
Actions\Action::make('archive_review')
|
||||||
|
->label('Archive review')
|
||||||
|
->icon('heroicon-o-archive-box')
|
||||||
|
->color('danger')
|
||||||
|
->hidden(fn (): bool => $this->record->statusEnum()->isTerminal())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->action(function (): void {
|
||||||
|
$user = auth()->user();
|
||||||
|
|
||||||
|
if (! $user instanceof User) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
app(TenantReviewLifecycleService::class)->archive($this->record, $user);
|
||||||
|
$this->refreshFormData(['status', 'archived_at']);
|
||||||
|
|
||||||
|
Notification::make()->success()->title('Review archived')->send();
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
->requireCapability(Capabilities::TENANT_REVIEW_MANAGE)
|
||||||
|
->preserveVisibility()
|
||||||
|
->apply(),
|
||||||
|
])
|
||||||
|
->label('More')
|
||||||
|
->icon('heroicon-m-ellipsis-vertical')
|
||||||
|
->color('gray'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -107,10 +107,7 @@ protected function getHeaderActions(): array
|
|||||||
->icon('heroicon-o-magnifying-glass')
|
->icon('heroicon-o-magnifying-glass')
|
||||||
->form($this->findingsScopeForm())
|
->form($this->findingsScopeForm())
|
||||||
->action(function (array $data, FindingsLifecycleBackfillRunbookService $runbookService): void {
|
->action(function (array $data, FindingsLifecycleBackfillRunbookService $runbookService): void {
|
||||||
$scope = FindingsLifecycleBackfillScope::fromArray([
|
$scope = $this->trustedFindingsScopeFromFormData($data, app(AllowedTenantUniverse::class));
|
||||||
'mode' => $data['scope_mode'] ?? null,
|
|
||||||
'tenant_id' => $data['tenant_id'] ?? null,
|
|
||||||
]);
|
|
||||||
|
|
||||||
$this->findingsScopeMode = $scope->mode;
|
$this->findingsScopeMode = $scope->mode;
|
||||||
$this->findingsTenantId = $scope->tenantId;
|
$this->findingsTenantId = $scope->tenantId;
|
||||||
@ -142,9 +139,7 @@ protected function getHeaderActions(): array
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$scope = $this->findingsScopeMode === FindingsLifecycleBackfillScope::MODE_SINGLE_TENANT
|
$scope = $this->trustedFindingsScopeFromState(app(AllowedTenantUniverse::class));
|
||||||
? FindingsLifecycleBackfillScope::singleTenant((int) $this->findingsTenantId)
|
|
||||||
: FindingsLifecycleBackfillScope::allTenants();
|
|
||||||
|
|
||||||
$user = auth('platform')->user();
|
$user = auth('platform')->user();
|
||||||
|
|
||||||
@ -286,4 +281,34 @@ private function lastRunForType(string $type): ?OperationRun
|
|||||||
->latest('id')
|
->latest('id')
|
||||||
->first();
|
->first();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
*/
|
||||||
|
private function trustedFindingsScopeFromFormData(array $data, AllowedTenantUniverse $allowedTenantUniverse): FindingsLifecycleBackfillScope
|
||||||
|
{
|
||||||
|
$scope = FindingsLifecycleBackfillScope::fromArray([
|
||||||
|
'mode' => $data['scope_mode'] ?? null,
|
||||||
|
'tenant_id' => $data['tenant_id'] ?? null,
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (! $scope->isSingleTenant()) {
|
||||||
|
return $scope;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = $allowedTenantUniverse->resolveAllowedOrFail($scope->tenantId);
|
||||||
|
|
||||||
|
return FindingsLifecycleBackfillScope::singleTenant((int) $tenant->getKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
private function trustedFindingsScopeFromState(AllowedTenantUniverse $allowedTenantUniverse): FindingsLifecycleBackfillScope
|
||||||
|
{
|
||||||
|
if ($this->findingsScopeMode !== FindingsLifecycleBackfillScope::MODE_SINGLE_TENANT) {
|
||||||
|
return FindingsLifecycleBackfillScope::allTenants();
|
||||||
|
}
|
||||||
|
|
||||||
|
$tenant = $allowedTenantUniverse->resolveAllowedOrFail($this->findingsTenantId);
|
||||||
|
|
||||||
|
return FindingsLifecycleBackfillScope::singleTenant((int) $tenant->getKey());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -22,6 +22,8 @@ protected function getViewData(): array
|
|||||||
|
|
||||||
$empty = [
|
$empty = [
|
||||||
'hasAssignment' => false,
|
'hasAssignment' => false,
|
||||||
|
'state' => 'no_assignment',
|
||||||
|
'message' => null,
|
||||||
'profileName' => null,
|
'profileName' => null,
|
||||||
'findingsCount' => 0,
|
'findingsCount' => 0,
|
||||||
'highCount' => 0,
|
'highCount' => 0,
|
||||||
@ -43,6 +45,8 @@ protected function getViewData(): array
|
|||||||
|
|
||||||
return [
|
return [
|
||||||
'hasAssignment' => true,
|
'hasAssignment' => true,
|
||||||
|
'state' => $stats->state,
|
||||||
|
'message' => $stats->message,
|
||||||
'profileName' => $stats->profileName,
|
'profileName' => $stats->profileName,
|
||||||
'findingsCount' => $stats->findingsCount ?? 0,
|
'findingsCount' => $stats->findingsCount ?? 0,
|
||||||
'highCount' => $stats->severityCounts['high'] ?? 0,
|
'highCount' => $stats->severityCounts['high'] ?? 0,
|
||||||
|
|||||||
@ -11,6 +11,7 @@
|
|||||||
use App\Support\OperationCatalog;
|
use App\Support\OperationCatalog;
|
||||||
use App\Support\OperationRunLinks;
|
use App\Support\OperationRunLinks;
|
||||||
use App\Support\OpsUx\ActiveRuns;
|
use App\Support\OpsUx\ActiveRuns;
|
||||||
|
use App\Support\OpsUx\OperationUxPresenter;
|
||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Tables\Columns\TextColumn;
|
use Filament\Tables\Columns\TextColumn;
|
||||||
use Filament\Tables\Table;
|
use Filament\Tables\Table;
|
||||||
@ -57,7 +58,8 @@ public function table(Table $table): Table
|
|||||||
->formatStateUsing(BadgeRenderer::label(BadgeDomain::OperationRunOutcome))
|
->formatStateUsing(BadgeRenderer::label(BadgeDomain::OperationRunOutcome))
|
||||||
->color(BadgeRenderer::color(BadgeDomain::OperationRunOutcome))
|
->color(BadgeRenderer::color(BadgeDomain::OperationRunOutcome))
|
||||||
->icon(BadgeRenderer::icon(BadgeDomain::OperationRunOutcome))
|
->icon(BadgeRenderer::icon(BadgeDomain::OperationRunOutcome))
|
||||||
->iconColor(BadgeRenderer::iconColor(BadgeDomain::OperationRunOutcome)),
|
->iconColor(BadgeRenderer::iconColor(BadgeDomain::OperationRunOutcome))
|
||||||
|
->description(fn (OperationRun $record): ?string => OperationUxPresenter::surfaceGuidance($record)),
|
||||||
TextColumn::make('created_at')
|
TextColumn::make('created_at')
|
||||||
->label('Started')
|
->label('Started')
|
||||||
->sortable()
|
->sortable()
|
||||||
|
|||||||
@ -44,8 +44,10 @@ protected function getViewData(): array
|
|||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'shouldShow' => $hasWarnings && $runUrl !== null,
|
'shouldShow' => ($hasWarnings && $runUrl !== null) || $stats->state === 'no_snapshot',
|
||||||
'runUrl' => $runUrl,
|
'runUrl' => $runUrl,
|
||||||
|
'state' => $stats->state,
|
||||||
|
'message' => $stats->message,
|
||||||
'coverageStatus' => $coverageStatus,
|
'coverageStatus' => $coverageStatus,
|
||||||
'fidelity' => $stats->fidelity,
|
'fidelity' => $stats->fidelity,
|
||||||
'uncoveredTypesCount' => $stats->uncoveredTypesCount ?? count($uncoveredTypes),
|
'uncoveredTypesCount' => $stats->uncoveredTypesCount ?? count($uncoveredTypes),
|
||||||
|
|||||||
@ -131,6 +131,7 @@ protected function getViewData(): array
|
|||||||
$canManage = $isTenantMember && $user->can(Capabilities::REVIEW_PACK_MANAGE, $tenant);
|
$canManage = $isTenantMember && $user->can(Capabilities::REVIEW_PACK_MANAGE, $tenant);
|
||||||
|
|
||||||
$latestPack = ReviewPack::query()
|
$latestPack = ReviewPack::query()
|
||||||
|
->with('tenantReview')
|
||||||
->where('tenant_id', (int) $tenant->getKey())
|
->where('tenant_id', (int) $tenant->getKey())
|
||||||
->orderByDesc('created_at')
|
->orderByDesc('created_at')
|
||||||
->orderByDesc('id')
|
->orderByDesc('id')
|
||||||
@ -146,6 +147,7 @@ protected function getViewData(): array
|
|||||||
'canManage' => $canManage,
|
'canManage' => $canManage,
|
||||||
'downloadUrl' => null,
|
'downloadUrl' => null,
|
||||||
'failedReason' => null,
|
'failedReason' => null,
|
||||||
|
'reviewUrl' => null,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -158,6 +160,11 @@ protected function getViewData(): array
|
|||||||
$downloadUrl = $service->generateDownloadUrl($latestPack);
|
$downloadUrl = $service->generateDownloadUrl($latestPack);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$reviewUrl = null;
|
||||||
|
if ($latestPack->tenantReview && $canView) {
|
||||||
|
$reviewUrl = \App\Filament\Resources\TenantReviewResource::tenantScopedUrl('view', ['record' => $latestPack->tenantReview], $tenant);
|
||||||
|
}
|
||||||
|
|
||||||
$failedReason = null;
|
$failedReason = null;
|
||||||
if ($statusEnum === ReviewPackStatus::Failed && $latestPack->operationRun) {
|
if ($statusEnum === ReviewPackStatus::Failed && $latestPack->operationRun) {
|
||||||
$opContext = is_array($latestPack->operationRun->context) ? $latestPack->operationRun->context : [];
|
$opContext = is_array($latestPack->operationRun->context) ? $latestPack->operationRun->context : [];
|
||||||
@ -173,6 +180,7 @@ protected function getViewData(): array
|
|||||||
'canManage' => $canManage,
|
'canManage' => $canManage,
|
||||||
'downloadUrl' => $downloadUrl,
|
'downloadUrl' => $downloadUrl,
|
||||||
'failedReason' => $failedReason,
|
'failedReason' => $failedReason,
|
||||||
|
'reviewUrl' => $reviewUrl,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -200,6 +208,7 @@ private function emptyState(): array
|
|||||||
'canManage' => false,
|
'canManage' => false,
|
||||||
'downloadUrl' => null,
|
'downloadUrl' => null,
|
||||||
'failedReason' => null,
|
'failedReason' => null,
|
||||||
|
'reviewUrl' => null,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -134,9 +134,12 @@ public function startVerification(StartVerification $verification): void
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$reasonEnvelope = app(\App\Support\ReasonTranslation\ReasonPresenter::class)->forOperationRun($result->run, 'notification');
|
||||||
|
$bodyLines = $reasonEnvelope?->toBodyLines() ?? ['Blocked by provider configuration.'];
|
||||||
|
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title('Verification blocked')
|
->title('Verification blocked')
|
||||||
->body("Blocked by provider configuration ({$reasonCode}).")
|
->body(implode("\n", $bodyLines))
|
||||||
->warning()
|
->warning()
|
||||||
->actions($actions)
|
->actions($actions)
|
||||||
->send();
|
->send();
|
||||||
|
|||||||
@ -23,6 +23,7 @@ class WorkspaceRecentOperations extends Widget
|
|||||||
* status_color: string,
|
* status_color: string,
|
||||||
* outcome_label: string,
|
* outcome_label: string,
|
||||||
* outcome_color: string,
|
* outcome_color: string,
|
||||||
|
* guidance: ?string,
|
||||||
* started_at: string,
|
* started_at: string,
|
||||||
* url: string
|
* url: string
|
||||||
* }>
|
* }>
|
||||||
@ -48,6 +49,7 @@ class WorkspaceRecentOperations extends Widget
|
|||||||
* status_color: string,
|
* status_color: string,
|
||||||
* outcome_label: string,
|
* outcome_label: string,
|
||||||
* outcome_color: string,
|
* outcome_color: string,
|
||||||
|
* guidance: ?string,
|
||||||
* started_at: string,
|
* started_at: string,
|
||||||
* url: string
|
* url: string
|
||||||
* }> $operations
|
* }> $operations
|
||||||
|
|||||||
@ -9,6 +9,8 @@
|
|||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\UserTenantPreference;
|
use App\Models\UserTenantPreference;
|
||||||
use App\Services\Tenants\TenantOperabilityService;
|
use App\Services\Tenants\TenantOperabilityService;
|
||||||
|
use App\Support\Tenants\TenantInteractionLane;
|
||||||
|
use App\Support\Tenants\TenantOperabilityQuestion;
|
||||||
use App\Support\Workspaces\WorkspaceContext;
|
use App\Support\Workspaces\WorkspaceContext;
|
||||||
use Illuminate\Http\RedirectResponse;
|
use Illuminate\Http\RedirectResponse;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
@ -47,7 +49,15 @@ public function __invoke(Request $request): RedirectResponse
|
|||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! app(TenantOperabilityService::class)->canSelectAsContext($tenant)) {
|
$outcome = app(TenantOperabilityService::class)->outcomeFor(
|
||||||
|
tenant: $tenant,
|
||||||
|
question: TenantOperabilityQuestion::SelectorEligibility,
|
||||||
|
actor: $user,
|
||||||
|
workspaceId: $workspaceId,
|
||||||
|
lane: TenantInteractionLane::StandardActiveOperating,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (! $outcome->allowed) {
|
||||||
abort(404);
|
abort(404);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -177,7 +177,7 @@ private function isWorkspaceOptionalPath(Request $request, string $path): bool
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($path === '/livewire/update') {
|
if ($this->isLivewireUpdatePath($path)) {
|
||||||
$refererPath = parse_url((string) $request->headers->get('referer', ''), PHP_URL_PATH) ?? '';
|
$refererPath = parse_url((string) $request->headers->get('referer', ''), PHP_URL_PATH) ?? '';
|
||||||
$refererPath = '/'.ltrim((string) $refererPath, '/');
|
$refererPath = '/'.ltrim((string) $refererPath, '/');
|
||||||
|
|
||||||
@ -193,6 +193,11 @@ private function isWorkspaceOptionalPath(Request $request, string $path): bool
|
|||||||
return preg_match('#^/admin/operations/[^/]+$#', $path) === 1;
|
return preg_match('#^/admin/operations/[^/]+$#', $path) === 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function isLivewireUpdatePath(string $path): bool
|
||||||
|
{
|
||||||
|
return preg_match('#^/livewire(?:-[^/]+)?/update$#', $path) === 1;
|
||||||
|
}
|
||||||
|
|
||||||
private function isChooserFirstPath(string $path): bool
|
private function isChooserFirstPath(string $path): bool
|
||||||
{
|
{
|
||||||
return in_array($path, ['/admin', '/admin/choose-tenant'], true);
|
return in_array($path, ['/admin', '/admin/choose-tenant'], true);
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
use App\Jobs\Operations\BackupSetRestoreWorkerJob;
|
use App\Jobs\Operations\BackupSetRestoreWorkerJob;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
@ -11,11 +12,18 @@
|
|||||||
use Illuminate\Queue\InteractsWithQueue;
|
use Illuminate\Queue\InteractsWithQueue;
|
||||||
use Illuminate\Queue\SerializesModels;
|
use Illuminate\Queue\SerializesModels;
|
||||||
use RuntimeException;
|
use RuntimeException;
|
||||||
use Throwable;
|
|
||||||
|
|
||||||
class BulkBackupSetRestoreJob implements ShouldQueue
|
class BulkBackupSetRestoreJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 300;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public int $bulkRunId = 0;
|
public int $bulkRunId = 0;
|
||||||
|
|
||||||
@ -68,32 +76,6 @@ public function handle(OperationRunService $runs): void
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function failed(Throwable $e): void
|
|
||||||
{
|
|
||||||
$run = $this->operationRun;
|
|
||||||
|
|
||||||
if (! $run instanceof OperationRun && $this->bulkRunId > 0) {
|
|
||||||
$run = OperationRun::query()->find($this->bulkRunId);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (! $run instanceof OperationRun) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @var OperationRunService $runs */
|
|
||||||
$runs = app(OperationRunService::class);
|
|
||||||
|
|
||||||
$runs->updateRun(
|
|
||||||
$run,
|
|
||||||
status: 'completed',
|
|
||||||
outcome: 'failed',
|
|
||||||
failures: [[
|
|
||||||
'code' => 'bulk_job.failed',
|
|
||||||
'message' => $e->getMessage(),
|
|
||||||
]],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function resolveOperationRun(): OperationRun
|
private function resolveOperationRun(): OperationRun
|
||||||
{
|
{
|
||||||
if ($this->operationRun instanceof OperationRun) {
|
if ($this->operationRun instanceof OperationRun) {
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Operations\PolicyBulkDeleteWorkerJob;
|
use App\Jobs\Operations\PolicyBulkDeleteWorkerJob;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
@ -32,6 +33,11 @@ public function __construct(
|
|||||||
$this->operationRun = $operationRun;
|
$this->operationRun = $operationRun;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
|
}
|
||||||
|
|
||||||
public function handle(OperationRunService $runs): void
|
public function handle(OperationRunService $runs): void
|
||||||
{
|
{
|
||||||
if (! $this->operationRun instanceof OperationRun) {
|
if (! $this->operationRun instanceof OperationRun) {
|
||||||
|
|||||||
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Operations\TenantSyncWorkerJob;
|
use App\Jobs\Operations\TenantSyncWorkerJob;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
@ -14,7 +16,15 @@
|
|||||||
|
|
||||||
class BulkTenantSyncJob implements ShouldQueue
|
class BulkTenantSyncJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 180;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
@ -32,6 +42,11 @@ public function __construct(
|
|||||||
$this->operationRun = $operationRun;
|
$this->operationRun = $operationRun;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
|
}
|
||||||
|
|
||||||
public function handle(OperationRunService $runs): void
|
public function handle(OperationRunService $runs): void
|
||||||
{
|
{
|
||||||
if (! $this->operationRun instanceof OperationRun) {
|
if (! $this->operationRun instanceof OperationRun) {
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\BaselineProfile;
|
use App\Models\BaselineProfile;
|
||||||
use App\Models\BaselineSnapshot;
|
use App\Models\BaselineSnapshot;
|
||||||
@ -12,6 +13,7 @@
|
|||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Services\Baselines\BaselineContentCapturePhase;
|
use App\Services\Baselines\BaselineContentCapturePhase;
|
||||||
use App\Services\Baselines\BaselineSnapshotIdentity;
|
use App\Services\Baselines\BaselineSnapshotIdentity;
|
||||||
|
use App\Services\Baselines\BaselineSnapshotItemNormalizer;
|
||||||
use App\Services\Baselines\CurrentStateHashResolver;
|
use App\Services\Baselines\CurrentStateHashResolver;
|
||||||
use App\Services\Baselines\Evidence\ResolvedEvidence;
|
use App\Services\Baselines\Evidence\ResolvedEvidence;
|
||||||
use App\Services\Baselines\InventoryMetaContract;
|
use App\Services\Baselines\InventoryMetaContract;
|
||||||
@ -20,7 +22,9 @@
|
|||||||
use App\Support\Baselines\BaselineCaptureMode;
|
use App\Support\Baselines\BaselineCaptureMode;
|
||||||
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
||||||
use App\Support\Baselines\BaselineProfileStatus;
|
use App\Support\Baselines\BaselineProfileStatus;
|
||||||
|
use App\Support\Baselines\BaselineReasonCodes;
|
||||||
use App\Support\Baselines\BaselineScope;
|
use App\Support\Baselines\BaselineScope;
|
||||||
|
use App\Support\Baselines\BaselineSnapshotLifecycleState;
|
||||||
use App\Support\Baselines\PolicyVersionCapturePurpose;
|
use App\Support\Baselines\PolicyVersionCapturePurpose;
|
||||||
use App\Support\Inventory\InventoryPolicyTypeMeta;
|
use App\Support\Inventory\InventoryPolicyTypeMeta;
|
||||||
use App\Support\OperationRunOutcome;
|
use App\Support\OperationRunOutcome;
|
||||||
@ -32,10 +36,19 @@
|
|||||||
use Illuminate\Queue\InteractsWithQueue;
|
use Illuminate\Queue\InteractsWithQueue;
|
||||||
use Illuminate\Queue\SerializesModels;
|
use Illuminate\Queue\SerializesModels;
|
||||||
use RuntimeException;
|
use RuntimeException;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
class CaptureBaselineSnapshotJob implements ShouldQueue
|
class CaptureBaselineSnapshotJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 300;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
@ -60,10 +73,12 @@ public function handle(
|
|||||||
OperationRunService $operationRunService,
|
OperationRunService $operationRunService,
|
||||||
?CurrentStateHashResolver $hashResolver = null,
|
?CurrentStateHashResolver $hashResolver = null,
|
||||||
?BaselineContentCapturePhase $contentCapturePhase = null,
|
?BaselineContentCapturePhase $contentCapturePhase = null,
|
||||||
|
?BaselineSnapshotItemNormalizer $snapshotItemNormalizer = null,
|
||||||
?BaselineFullContentRolloutGate $rolloutGate = null,
|
?BaselineFullContentRolloutGate $rolloutGate = null,
|
||||||
): void {
|
): void {
|
||||||
$hashResolver ??= app(CurrentStateHashResolver::class);
|
$hashResolver ??= app(CurrentStateHashResolver::class);
|
||||||
$contentCapturePhase ??= app(BaselineContentCapturePhase::class);
|
$contentCapturePhase ??= app(BaselineContentCapturePhase::class);
|
||||||
|
$snapshotItemNormalizer ??= app(BaselineSnapshotItemNormalizer::class);
|
||||||
$rolloutGate ??= app(BaselineFullContentRolloutGate::class);
|
$rolloutGate ??= app(BaselineFullContentRolloutGate::class);
|
||||||
|
|
||||||
if (! $this->operationRun instanceof OperationRun) {
|
if (! $this->operationRun instanceof OperationRun) {
|
||||||
@ -183,7 +198,12 @@ public function handle(
|
|||||||
gaps: $captureGaps,
|
gaps: $captureGaps,
|
||||||
);
|
);
|
||||||
|
|
||||||
$items = $snapshotItems['items'] ?? [];
|
$normalizedItems = $snapshotItemNormalizer->deduplicate($snapshotItems['items'] ?? []);
|
||||||
|
$items = $normalizedItems['items'];
|
||||||
|
|
||||||
|
if (($normalizedItems['duplicates'] ?? 0) > 0) {
|
||||||
|
$captureGaps['duplicate_subject_reference'] = ($captureGaps['duplicate_subject_reference'] ?? 0) + (int) $normalizedItems['duplicates'];
|
||||||
|
}
|
||||||
|
|
||||||
$identityHash = $identity->computeIdentity($items);
|
$identityHash = $identity->computeIdentity($items);
|
||||||
|
|
||||||
@ -200,16 +220,17 @@ public function handle(
|
|||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
$snapshot = $this->findOrCreateSnapshot(
|
$snapshotResult = $this->captureSnapshotArtifact(
|
||||||
$profile,
|
$profile,
|
||||||
$identityHash,
|
$identityHash,
|
||||||
$items,
|
$items,
|
||||||
$snapshotSummary,
|
$snapshotSummary,
|
||||||
);
|
);
|
||||||
|
|
||||||
$wasNewSnapshot = $snapshot->wasRecentlyCreated;
|
$snapshot = $snapshotResult['snapshot'];
|
||||||
|
$wasNewSnapshot = $snapshotResult['was_new_snapshot'];
|
||||||
|
|
||||||
if ($profile->status === BaselineProfileStatus::Active) {
|
if ($profile->status === BaselineProfileStatus::Active && $snapshot->isConsumable()) {
|
||||||
$profile->update(['active_snapshot_id' => $snapshot->getKey()]);
|
$profile->update(['active_snapshot_id' => $snapshot->getKey()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -250,6 +271,7 @@ public function handle(
|
|||||||
'snapshot_identity_hash' => $identityHash,
|
'snapshot_identity_hash' => $identityHash,
|
||||||
'was_new_snapshot' => $wasNewSnapshot,
|
'was_new_snapshot' => $wasNewSnapshot,
|
||||||
'items_captured' => $snapshotItems['items_count'],
|
'items_captured' => $snapshotItems['items_count'],
|
||||||
|
'snapshot_lifecycle' => $snapshot->lifecycleState()->value,
|
||||||
];
|
];
|
||||||
$this->operationRun->update(['context' => $updatedContext]);
|
$this->operationRun->update(['context' => $updatedContext]);
|
||||||
|
|
||||||
@ -500,29 +522,151 @@ private function buildSnapshotItems(
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
private function findOrCreateSnapshot(
|
/**
|
||||||
|
* @param array<int, array<string, mixed>> $snapshotItems
|
||||||
|
* @param array<string, mixed> $summaryJsonb
|
||||||
|
* @return array{snapshot: BaselineSnapshot, was_new_snapshot: bool}
|
||||||
|
*/
|
||||||
|
private function captureSnapshotArtifact(
|
||||||
BaselineProfile $profile,
|
BaselineProfile $profile,
|
||||||
string $identityHash,
|
string $identityHash,
|
||||||
array $snapshotItems,
|
array $snapshotItems,
|
||||||
array $summaryJsonb,
|
array $summaryJsonb,
|
||||||
): BaselineSnapshot {
|
): array {
|
||||||
|
$existing = $this->findExistingConsumableSnapshot($profile, $identityHash);
|
||||||
|
|
||||||
|
if ($existing instanceof BaselineSnapshot) {
|
||||||
|
$this->rememberSnapshotOnRun(
|
||||||
|
snapshot: $existing,
|
||||||
|
identityHash: $identityHash,
|
||||||
|
wasNewSnapshot: false,
|
||||||
|
expectedItems: count($snapshotItems),
|
||||||
|
persistedItems: count($snapshotItems),
|
||||||
|
);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'snapshot' => $existing,
|
||||||
|
'was_new_snapshot' => false,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
$expectedItems = count($snapshotItems);
|
||||||
|
$snapshot = $this->createBuildingSnapshot($profile, $identityHash, $summaryJsonb, $expectedItems);
|
||||||
|
|
||||||
|
$this->rememberSnapshotOnRun(
|
||||||
|
snapshot: $snapshot,
|
||||||
|
identityHash: $identityHash,
|
||||||
|
wasNewSnapshot: true,
|
||||||
|
expectedItems: $expectedItems,
|
||||||
|
persistedItems: 0,
|
||||||
|
);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$persistedItems = $this->persistSnapshotItems($snapshot, $snapshotItems);
|
||||||
|
|
||||||
|
if ($persistedItems !== $expectedItems) {
|
||||||
|
throw new RuntimeException('Baseline snapshot completion proof failed.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$snapshot->markComplete($identityHash, [
|
||||||
|
'expected_identity_hash' => $identityHash,
|
||||||
|
'expected_items' => $expectedItems,
|
||||||
|
'persisted_items' => $persistedItems,
|
||||||
|
'producer_run_id' => (int) $this->operationRun->getKey(),
|
||||||
|
'was_empty_capture' => $expectedItems === 0,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$snapshot->refresh();
|
||||||
|
|
||||||
|
$this->rememberSnapshotOnRun(
|
||||||
|
snapshot: $snapshot,
|
||||||
|
identityHash: $identityHash,
|
||||||
|
wasNewSnapshot: true,
|
||||||
|
expectedItems: $expectedItems,
|
||||||
|
persistedItems: $persistedItems,
|
||||||
|
);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'snapshot' => $snapshot,
|
||||||
|
'was_new_snapshot' => true,
|
||||||
|
];
|
||||||
|
} catch (Throwable $exception) {
|
||||||
|
$persistedItems = (int) BaselineSnapshotItem::query()
|
||||||
|
->where('baseline_snapshot_id', (int) $snapshot->getKey())
|
||||||
|
->count();
|
||||||
|
|
||||||
|
$reasonCode = $exception instanceof RuntimeException
|
||||||
|
&& $exception->getMessage() === 'Baseline snapshot completion proof failed.'
|
||||||
|
? BaselineReasonCodes::SNAPSHOT_COMPLETION_PROOF_FAILED
|
||||||
|
: BaselineReasonCodes::SNAPSHOT_CAPTURE_FAILED;
|
||||||
|
|
||||||
|
$snapshot->markIncomplete($reasonCode, [
|
||||||
|
'expected_identity_hash' => $identityHash,
|
||||||
|
'expected_items' => $expectedItems,
|
||||||
|
'persisted_items' => $persistedItems,
|
||||||
|
'producer_run_id' => (int) $this->operationRun->getKey(),
|
||||||
|
'was_empty_capture' => $expectedItems === 0,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$snapshot->refresh();
|
||||||
|
|
||||||
|
$this->rememberSnapshotOnRun(
|
||||||
|
snapshot: $snapshot,
|
||||||
|
identityHash: $identityHash,
|
||||||
|
wasNewSnapshot: true,
|
||||||
|
expectedItems: $expectedItems,
|
||||||
|
persistedItems: $persistedItems,
|
||||||
|
reasonCode: $reasonCode,
|
||||||
|
);
|
||||||
|
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function findExistingConsumableSnapshot(BaselineProfile $profile, string $identityHash): ?BaselineSnapshot
|
||||||
|
{
|
||||||
$existing = BaselineSnapshot::query()
|
$existing = BaselineSnapshot::query()
|
||||||
->where('workspace_id', $profile->workspace_id)
|
->where('workspace_id', $profile->workspace_id)
|
||||||
->where('baseline_profile_id', $profile->getKey())
|
->where('baseline_profile_id', $profile->getKey())
|
||||||
->where('snapshot_identity_hash', $identityHash)
|
->where('snapshot_identity_hash', $identityHash)
|
||||||
|
->where('lifecycle_state', BaselineSnapshotLifecycleState::Complete->value)
|
||||||
->first();
|
->first();
|
||||||
|
|
||||||
if ($existing instanceof BaselineSnapshot) {
|
return $existing instanceof BaselineSnapshot ? $existing : null;
|
||||||
return $existing;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
$snapshot = BaselineSnapshot::create([
|
/**
|
||||||
|
* @param array<string, mixed> $summaryJsonb
|
||||||
|
*/
|
||||||
|
private function createBuildingSnapshot(
|
||||||
|
BaselineProfile $profile,
|
||||||
|
string $identityHash,
|
||||||
|
array $summaryJsonb,
|
||||||
|
int $expectedItems,
|
||||||
|
): BaselineSnapshot {
|
||||||
|
return BaselineSnapshot::create([
|
||||||
'workspace_id' => (int) $profile->workspace_id,
|
'workspace_id' => (int) $profile->workspace_id,
|
||||||
'baseline_profile_id' => (int) $profile->getKey(),
|
'baseline_profile_id' => (int) $profile->getKey(),
|
||||||
'snapshot_identity_hash' => $identityHash,
|
'snapshot_identity_hash' => $this->temporarySnapshotIdentityHash($profile),
|
||||||
'captured_at' => now(),
|
'captured_at' => now(),
|
||||||
|
'lifecycle_state' => BaselineSnapshotLifecycleState::Building->value,
|
||||||
'summary_jsonb' => $summaryJsonb,
|
'summary_jsonb' => $summaryJsonb,
|
||||||
|
'completion_meta_jsonb' => [
|
||||||
|
'expected_identity_hash' => $identityHash,
|
||||||
|
'expected_items' => $expectedItems,
|
||||||
|
'persisted_items' => 0,
|
||||||
|
'producer_run_id' => (int) $this->operationRun->getKey(),
|
||||||
|
'was_empty_capture' => $expectedItems === 0,
|
||||||
|
],
|
||||||
]);
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, array<string, mixed>> $snapshotItems
|
||||||
|
*/
|
||||||
|
private function persistSnapshotItems(BaselineSnapshot $snapshot, array $snapshotItems): int
|
||||||
|
{
|
||||||
|
$persistedItems = 0;
|
||||||
|
|
||||||
foreach (array_chunk($snapshotItems, 100) as $chunk) {
|
foreach (array_chunk($snapshotItems, 100) as $chunk) {
|
||||||
$rows = array_map(
|
$rows = array_map(
|
||||||
@ -541,9 +685,56 @@ private function findOrCreateSnapshot(
|
|||||||
);
|
);
|
||||||
|
|
||||||
BaselineSnapshotItem::insert($rows);
|
BaselineSnapshotItem::insert($rows);
|
||||||
|
$persistedItems += count($rows);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $snapshot;
|
return $persistedItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function temporarySnapshotIdentityHash(BaselineProfile $profile): string
|
||||||
|
{
|
||||||
|
return hash(
|
||||||
|
'sha256',
|
||||||
|
implode('|', [
|
||||||
|
'building',
|
||||||
|
(string) $profile->getKey(),
|
||||||
|
(string) $this->operationRun->getKey(),
|
||||||
|
(string) microtime(true),
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function rememberSnapshotOnRun(
|
||||||
|
BaselineSnapshot $snapshot,
|
||||||
|
string $identityHash,
|
||||||
|
bool $wasNewSnapshot,
|
||||||
|
int $expectedItems,
|
||||||
|
int $persistedItems,
|
||||||
|
?string $reasonCode = null,
|
||||||
|
): void {
|
||||||
|
$context = is_array($this->operationRun->context) ? $this->operationRun->context : [];
|
||||||
|
$context['baseline_snapshot_id'] = (int) $snapshot->getKey();
|
||||||
|
$context['result'] = array_merge(
|
||||||
|
is_array($context['result'] ?? null) ? $context['result'] : [],
|
||||||
|
[
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
'snapshot_identity_hash' => $identityHash,
|
||||||
|
'was_new_snapshot' => $wasNewSnapshot,
|
||||||
|
'items_captured' => $persistedItems,
|
||||||
|
'snapshot_lifecycle' => $snapshot->lifecycleState()->value,
|
||||||
|
'expected_items' => $expectedItems,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
if (is_string($reasonCode) && $reasonCode !== '') {
|
||||||
|
$context['reason_code'] = $reasonCode;
|
||||||
|
$context['result']['snapshot_reason_code'] = $reasonCode;
|
||||||
|
} else {
|
||||||
|
unset($context['reason_code'], $context['result']['snapshot_reason_code']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->operationRun->update(['context' => $context]);
|
||||||
|
$this->operationRun->refresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -4,6 +4,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\BaselineProfile;
|
use App\Models\BaselineProfile;
|
||||||
use App\Models\BaselineSnapshot;
|
use App\Models\BaselineSnapshot;
|
||||||
@ -18,6 +19,7 @@
|
|||||||
use App\Services\Baselines\BaselineAutoCloseService;
|
use App\Services\Baselines\BaselineAutoCloseService;
|
||||||
use App\Services\Baselines\BaselineContentCapturePhase;
|
use App\Services\Baselines\BaselineContentCapturePhase;
|
||||||
use App\Services\Baselines\BaselineSnapshotIdentity;
|
use App\Services\Baselines\BaselineSnapshotIdentity;
|
||||||
|
use App\Services\Baselines\BaselineSnapshotTruthResolver;
|
||||||
use App\Services\Baselines\CurrentStateHashResolver;
|
use App\Services\Baselines\CurrentStateHashResolver;
|
||||||
use App\Services\Baselines\Evidence\BaselinePolicyVersionResolver;
|
use App\Services\Baselines\Evidence\BaselinePolicyVersionResolver;
|
||||||
use App\Services\Baselines\Evidence\ContentEvidenceProvider;
|
use App\Services\Baselines\Evidence\ContentEvidenceProvider;
|
||||||
@ -29,6 +31,7 @@
|
|||||||
use App\Services\Drift\Normalizers\ScopeTagsNormalizer;
|
use App\Services\Drift\Normalizers\ScopeTagsNormalizer;
|
||||||
use App\Services\Drift\Normalizers\SettingsNormalizer;
|
use App\Services\Drift\Normalizers\SettingsNormalizer;
|
||||||
use App\Services\Findings\FindingSlaPolicy;
|
use App\Services\Findings\FindingSlaPolicy;
|
||||||
|
use App\Services\Findings\FindingWorkflowService;
|
||||||
use App\Services\Intune\AuditLogger;
|
use App\Services\Intune\AuditLogger;
|
||||||
use App\Services\Intune\IntuneRoleDefinitionNormalizer;
|
use App\Services\Intune\IntuneRoleDefinitionNormalizer;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
@ -36,6 +39,7 @@
|
|||||||
use App\Support\Baselines\BaselineCaptureMode;
|
use App\Support\Baselines\BaselineCaptureMode;
|
||||||
use App\Support\Baselines\BaselineCompareReasonCode;
|
use App\Support\Baselines\BaselineCompareReasonCode;
|
||||||
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
use App\Support\Baselines\BaselineFullContentRolloutGate;
|
||||||
|
use App\Support\Baselines\BaselineReasonCodes;
|
||||||
use App\Support\Baselines\BaselineScope;
|
use App\Support\Baselines\BaselineScope;
|
||||||
use App\Support\Baselines\BaselineSubjectKey;
|
use App\Support\Baselines\BaselineSubjectKey;
|
||||||
use App\Support\Baselines\PolicyVersionCapturePurpose;
|
use App\Support\Baselines\PolicyVersionCapturePurpose;
|
||||||
@ -53,7 +57,15 @@
|
|||||||
|
|
||||||
class CompareBaselineToTenantJob implements ShouldQueue
|
class CompareBaselineToTenantJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 300;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array<int, string>
|
* @var array<int, string>
|
||||||
@ -83,6 +95,7 @@ public function handle(
|
|||||||
?SettingsResolver $settingsResolver = null,
|
?SettingsResolver $settingsResolver = null,
|
||||||
?BaselineAutoCloseService $baselineAutoCloseService = null,
|
?BaselineAutoCloseService $baselineAutoCloseService = null,
|
||||||
?CurrentStateHashResolver $hashResolver = null,
|
?CurrentStateHashResolver $hashResolver = null,
|
||||||
|
?BaselineSnapshotTruthResolver $snapshotTruthResolver = null,
|
||||||
?MetaEvidenceProvider $metaEvidenceProvider = null,
|
?MetaEvidenceProvider $metaEvidenceProvider = null,
|
||||||
?BaselineContentCapturePhase $contentCapturePhase = null,
|
?BaselineContentCapturePhase $contentCapturePhase = null,
|
||||||
?BaselineFullContentRolloutGate $rolloutGate = null,
|
?BaselineFullContentRolloutGate $rolloutGate = null,
|
||||||
@ -91,6 +104,7 @@ public function handle(
|
|||||||
$settingsResolver ??= app(SettingsResolver::class);
|
$settingsResolver ??= app(SettingsResolver::class);
|
||||||
$baselineAutoCloseService ??= app(BaselineAutoCloseService::class);
|
$baselineAutoCloseService ??= app(BaselineAutoCloseService::class);
|
||||||
$hashResolver ??= app(CurrentStateHashResolver::class);
|
$hashResolver ??= app(CurrentStateHashResolver::class);
|
||||||
|
$snapshotTruthResolver ??= app(BaselineSnapshotTruthResolver::class);
|
||||||
$metaEvidenceProvider ??= app(MetaEvidenceProvider::class);
|
$metaEvidenceProvider ??= app(MetaEvidenceProvider::class);
|
||||||
$contentCapturePhase ??= app(BaselineContentCapturePhase::class);
|
$contentCapturePhase ??= app(BaselineContentCapturePhase::class);
|
||||||
$rolloutGate ??= app(BaselineFullContentRolloutGate::class);
|
$rolloutGate ??= app(BaselineFullContentRolloutGate::class);
|
||||||
@ -277,12 +291,51 @@ public function handle(
|
|||||||
->where('workspace_id', (int) $profile->workspace_id)
|
->where('workspace_id', (int) $profile->workspace_id)
|
||||||
->where('baseline_profile_id', (int) $profile->getKey())
|
->where('baseline_profile_id', (int) $profile->getKey())
|
||||||
->whereKey($snapshotId)
|
->whereKey($snapshotId)
|
||||||
->first(['id', 'captured_at']);
|
->first();
|
||||||
|
|
||||||
if (! $snapshot instanceof BaselineSnapshot) {
|
if (! $snapshot instanceof BaselineSnapshot) {
|
||||||
throw new RuntimeException("BaselineSnapshot #{$snapshotId} not found.");
|
throw new RuntimeException("BaselineSnapshot #{$snapshotId} not found.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$snapshotResolution = $snapshotTruthResolver->resolveCompareSnapshot($profile, $snapshot);
|
||||||
|
|
||||||
|
if (! ($snapshotResolution['ok'] ?? false)) {
|
||||||
|
$reasonCode = is_string($snapshotResolution['reason_code'] ?? null)
|
||||||
|
? (string) $snapshotResolution['reason_code']
|
||||||
|
: BaselineReasonCodes::COMPARE_NO_CONSUMABLE_SNAPSHOT;
|
||||||
|
|
||||||
|
$context = is_array($this->operationRun->context) ? $this->operationRun->context : [];
|
||||||
|
$context['baseline_compare'] = array_merge(
|
||||||
|
is_array($context['baseline_compare'] ?? null) ? $context['baseline_compare'] : [],
|
||||||
|
[
|
||||||
|
'reason_code' => $reasonCode,
|
||||||
|
'effective_snapshot_id' => $snapshotResolution['effective_snapshot']?->getKey(),
|
||||||
|
'latest_attempted_snapshot_id' => $snapshotResolution['latest_attempted_snapshot']?->getKey(),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
$context['result'] = array_merge(
|
||||||
|
is_array($context['result'] ?? null) ? $context['result'] : [],
|
||||||
|
[
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->operationRun->update(['context' => $context]);
|
||||||
|
$this->operationRun->refresh();
|
||||||
|
|
||||||
|
$operationRunService->finalizeBlockedRun(
|
||||||
|
run: $this->operationRun,
|
||||||
|
reasonCode: $reasonCode,
|
||||||
|
message: $this->snapshotBlockedMessage($reasonCode),
|
||||||
|
);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var BaselineSnapshot $snapshot */
|
||||||
|
$snapshot = $snapshotResolution['snapshot'];
|
||||||
|
$snapshotId = (int) $snapshot->getKey();
|
||||||
|
|
||||||
$since = $snapshot->captured_at instanceof \DateTimeInterface
|
$since = $snapshot->captured_at instanceof \DateTimeInterface
|
||||||
? CarbonImmutable::instance($snapshot->captured_at)
|
? CarbonImmutable::instance($snapshot->captured_at)
|
||||||
: null;
|
: null;
|
||||||
@ -1003,6 +1056,17 @@ private function resolveLatestInventorySyncRun(Tenant $tenant): ?OperationRun
|
|||||||
return $run instanceof OperationRun ? $run : null;
|
return $run instanceof OperationRun ? $run : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function snapshotBlockedMessage(string $reasonCode): string
|
||||||
|
{
|
||||||
|
return match ($reasonCode) {
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_BUILDING => 'The selected baseline snapshot is still building and cannot be used for compare yet.',
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_INCOMPLETE => 'The selected baseline snapshot is incomplete and cannot be used for compare.',
|
||||||
|
BaselineReasonCodes::COMPARE_SNAPSHOT_SUPERSEDED => 'A newer complete baseline snapshot is now current, so this historical snapshot is blocked from compare.',
|
||||||
|
BaselineReasonCodes::COMPARE_INVALID_SNAPSHOT => 'The selected baseline snapshot is no longer available.',
|
||||||
|
default => 'No consumable baseline snapshot is currently available for compare.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compare baseline items vs current inventory and produce drift results.
|
* Compare baseline items vs current inventory and produce drift results.
|
||||||
*
|
*
|
||||||
@ -2130,20 +2194,14 @@ private function upsertFindings(
|
|||||||
: null;
|
: null;
|
||||||
|
|
||||||
if ($resolvedAt === null || $observedAt->greaterThan($resolvedAt)) {
|
if ($resolvedAt === null || $observedAt->greaterThan($resolvedAt)) {
|
||||||
$severity = (string) $driftItem['severity'];
|
$finding->save();
|
||||||
$slaDays = $slaPolicy->daysForSeverity($severity, $tenant);
|
|
||||||
|
|
||||||
$finding->forceFill([
|
app(FindingWorkflowService::class)->reopenBySystem(
|
||||||
'status' => Finding::STATUS_REOPENED,
|
finding: $finding,
|
||||||
'reopened_at' => $observedAt,
|
tenant: $tenant,
|
||||||
'resolved_at' => null,
|
reopenedAt: $observedAt,
|
||||||
'resolved_reason' => null,
|
operationRunId: (int) $this->operationRun->getKey(),
|
||||||
'closed_at' => null,
|
);
|
||||||
'closed_reason' => null,
|
|
||||||
'closed_by_user_id' => null,
|
|
||||||
'sla_days' => $slaDays,
|
|
||||||
'due_at' => $slaPolicy->dueAtForSeverity($severity, $tenant, $observedAt),
|
|
||||||
]);
|
|
||||||
|
|
||||||
$reopenedCount++;
|
$reopenedCount++;
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
85
app/Jobs/ComposeTenantReviewJob.php
Normal file
85
app/Jobs/ComposeTenantReviewJob.php
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
|
use App\Models\OperationRun;
|
||||||
|
use App\Models\TenantReview;
|
||||||
|
use App\Services\OperationRunService;
|
||||||
|
use App\Services\TenantReviews\TenantReviewService;
|
||||||
|
use App\Support\OperationRunOutcome;
|
||||||
|
use App\Support\OperationRunStatus;
|
||||||
|
use App\Support\TenantReviewStatus;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Foundation\Queue\Queueable;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
class ComposeTenantReviewJob implements ShouldQueue
|
||||||
|
{
|
||||||
|
use BridgesFailedOperationRun;
|
||||||
|
use Queueable;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
public int $tenantReviewId,
|
||||||
|
public int $operationRunId,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function handle(TenantReviewService $service, OperationRunService $operationRuns): void
|
||||||
|
{
|
||||||
|
$review = TenantReview::query()->with(['tenant', 'evidenceSnapshot.items'])->find($this->tenantReviewId);
|
||||||
|
$operationRun = OperationRun::query()->find($this->operationRunId);
|
||||||
|
|
||||||
|
if (! $review instanceof TenantReview || ! $operationRun instanceof OperationRun || ! $review->tenant) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$operationRuns->updateRun($operationRun, OperationRunStatus::Running->value, OperationRunOutcome::Pending->value);
|
||||||
|
$review->update(['status' => TenantReviewStatus::Draft->value]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$review = $service->compose($review);
|
||||||
|
|
||||||
|
$summary = is_array($review->summary) ? $review->summary : [];
|
||||||
|
|
||||||
|
$operationRuns->updateRun(
|
||||||
|
$operationRun,
|
||||||
|
status: OperationRunStatus::Completed->value,
|
||||||
|
outcome: OperationRunOutcome::Succeeded->value,
|
||||||
|
summaryCounts: [
|
||||||
|
'created' => 1,
|
||||||
|
'finding_count' => (int) ($summary['finding_count'] ?? 0),
|
||||||
|
'report_count' => (int) ($summary['report_count'] ?? 0),
|
||||||
|
'operation_count' => (int) ($summary['operation_count'] ?? 0),
|
||||||
|
'errors_recorded' => 0,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
} catch (Throwable $throwable) {
|
||||||
|
$review->update([
|
||||||
|
'status' => TenantReviewStatus::Failed->value,
|
||||||
|
'summary' => array_merge(is_array($review->summary) ? $review->summary : [], [
|
||||||
|
'error' => $throwable->getMessage(),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
|
||||||
|
$operationRuns->updateRun(
|
||||||
|
$operationRun,
|
||||||
|
status: OperationRunStatus::Completed->value,
|
||||||
|
outcome: OperationRunOutcome::Failed->value,
|
||||||
|
failures: [
|
||||||
|
[
|
||||||
|
'code' => 'tenant_review_compose.failed',
|
||||||
|
'message' => $throwable->getMessage(),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
throw $throwable;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
58
app/Jobs/Concerns/BridgesFailedOperationRun.php
Normal file
58
app/Jobs/Concerns/BridgesFailedOperationRun.php
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Jobs\Concerns;
|
||||||
|
|
||||||
|
use App\Models\OperationRun;
|
||||||
|
use App\Services\OperationRunService;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
trait BridgesFailedOperationRun
|
||||||
|
{
|
||||||
|
public function failed(Throwable $exception): void
|
||||||
|
{
|
||||||
|
$operationRun = $this->failedBridgeOperationRun();
|
||||||
|
|
||||||
|
if (! $operationRun instanceof OperationRun) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
app(OperationRunService::class)->bridgeFailedJobFailure($operationRun, $exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function failedBridgeOperationRun(): ?OperationRun
|
||||||
|
{
|
||||||
|
if (property_exists($this, 'operationRun') && $this->operationRun instanceof OperationRun) {
|
||||||
|
return $this->operationRun;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (property_exists($this, 'run') && $this->run instanceof OperationRun) {
|
||||||
|
return $this->run;
|
||||||
|
}
|
||||||
|
|
||||||
|
$candidateIds = [];
|
||||||
|
|
||||||
|
foreach (['operationRunId', 'bulkRunId', 'runId'] as $property) {
|
||||||
|
if (! property_exists($this, $property)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$value = $this->{$property};
|
||||||
|
|
||||||
|
if (is_numeric($value) && (int) $value > 0) {
|
||||||
|
$candidateIds[] = (int) $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (array_values(array_unique($candidateIds)) as $candidateId) {
|
||||||
|
$operationRun = OperationRun::query()->find($candidateId);
|
||||||
|
|
||||||
|
if ($operationRun instanceof OperationRun) {
|
||||||
|
return $operationRun;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -20,6 +20,10 @@ class EntraGroupSyncJob implements ShouldQueue
|
|||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
|
|||||||
@ -4,6 +4,8 @@
|
|||||||
|
|
||||||
use App\Contracts\Hardening\WriteGateInterface;
|
use App\Contracts\Hardening\WriteGateInterface;
|
||||||
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Listeners\SyncRestoreRunToOperationRun;
|
use App\Listeners\SyncRestoreRunToOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\RestoreRun;
|
use App\Models\RestoreRun;
|
||||||
@ -23,6 +25,10 @@ class ExecuteRestoreRunJob implements ShouldQueue
|
|||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 420;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
@ -34,6 +40,14 @@ public function __construct(
|
|||||||
$this->operationRun = $operationRun;
|
$this->operationRun = $operationRun;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, object>
|
||||||
|
*/
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
|
}
|
||||||
|
|
||||||
public function handle(RestoreService $restoreService, AuditLogger $auditLogger): void
|
public function handle(RestoreService $restoreService, AuditLogger $auditLogger): void
|
||||||
{
|
{
|
||||||
if (! $this->operationRun) {
|
if (! $this->operationRun) {
|
||||||
|
|||||||
122
app/Jobs/GenerateEvidenceSnapshotJob.php
Normal file
122
app/Jobs/GenerateEvidenceSnapshotJob.php
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Models\EvidenceSnapshot;
|
||||||
|
use App\Models\OperationRun;
|
||||||
|
use App\Services\Evidence\EvidenceSnapshotService;
|
||||||
|
use App\Services\OperationRunService;
|
||||||
|
use App\Support\Evidence\EvidenceSnapshotStatus;
|
||||||
|
use App\Support\OperationRunOutcome;
|
||||||
|
use App\Support\OperationRunStatus;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Foundation\Queue\Queueable;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
class GenerateEvidenceSnapshotJob implements ShouldQueue
|
||||||
|
{
|
||||||
|
use Queueable;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
public int $snapshotId,
|
||||||
|
public int $operationRunId,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function handle(EvidenceSnapshotService $service, OperationRunService $operationRuns): void
|
||||||
|
{
|
||||||
|
$snapshot = EvidenceSnapshot::query()->with('tenant')->find($this->snapshotId);
|
||||||
|
$operationRun = OperationRun::query()->find($this->operationRunId);
|
||||||
|
|
||||||
|
if (! $snapshot instanceof EvidenceSnapshot || ! $operationRun instanceof OperationRun || ! $snapshot->tenant) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$operationRuns->updateRun($operationRun, OperationRunStatus::Running->value, OperationRunOutcome::Pending->value);
|
||||||
|
$snapshot->update(['status' => EvidenceSnapshotStatus::Generating->value]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$payload = $service->buildSnapshotPayload($snapshot->tenant);
|
||||||
|
$previousActive = EvidenceSnapshot::query()
|
||||||
|
->where('tenant_id', (int) $snapshot->tenant_id)
|
||||||
|
->where('workspace_id', (int) $snapshot->workspace_id)
|
||||||
|
->where('status', EvidenceSnapshotStatus::Active->value)
|
||||||
|
->whereKeyNot((int) $snapshot->getKey())
|
||||||
|
->first();
|
||||||
|
|
||||||
|
$snapshot->items()->delete();
|
||||||
|
|
||||||
|
foreach ($payload['items'] as $item) {
|
||||||
|
$snapshot->items()->create([
|
||||||
|
'tenant_id' => (int) $snapshot->tenant_id,
|
||||||
|
'workspace_id' => (int) $snapshot->workspace_id,
|
||||||
|
'dimension_key' => $item['dimension_key'],
|
||||||
|
'state' => $item['state'],
|
||||||
|
'required' => $item['required'],
|
||||||
|
'source_kind' => $item['source_kind'],
|
||||||
|
'source_record_type' => $item['source_record_type'],
|
||||||
|
'source_record_id' => $item['source_record_id'],
|
||||||
|
'source_fingerprint' => $item['source_fingerprint'],
|
||||||
|
'measured_at' => $item['measured_at'],
|
||||||
|
'freshness_at' => $item['freshness_at'],
|
||||||
|
'summary_payload' => $item['summary_payload'],
|
||||||
|
'sort_order' => $item['sort_order'],
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($previousActive instanceof EvidenceSnapshot && $previousActive->fingerprint !== $payload['fingerprint']) {
|
||||||
|
$previousActive->update([
|
||||||
|
'status' => EvidenceSnapshotStatus::Superseded->value,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$snapshot->update([
|
||||||
|
'fingerprint' => $payload['fingerprint'],
|
||||||
|
'previous_fingerprint' => $previousActive?->fingerprint,
|
||||||
|
'status' => EvidenceSnapshotStatus::Active->value,
|
||||||
|
'completeness_state' => $payload['completeness'],
|
||||||
|
'generated_at' => now(),
|
||||||
|
'summary' => $payload['summary'],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$operationRuns->updateRun(
|
||||||
|
$operationRun,
|
||||||
|
status: OperationRunStatus::Completed->value,
|
||||||
|
outcome: OperationRunOutcome::Succeeded->value,
|
||||||
|
summaryCounts: [
|
||||||
|
'created' => 1,
|
||||||
|
'finding_count' => (int) ($payload['summary']['finding_count'] ?? 0),
|
||||||
|
'report_count' => (int) ($payload['summary']['report_count'] ?? 0),
|
||||||
|
'operation_count' => (int) ($payload['summary']['operation_count'] ?? 0),
|
||||||
|
'errors_recorded' => 0,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
} catch (Throwable $throwable) {
|
||||||
|
$snapshot->update([
|
||||||
|
'status' => EvidenceSnapshotStatus::Failed->value,
|
||||||
|
'summary' => [
|
||||||
|
'error' => $throwable->getMessage(),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$operationRuns->updateRun(
|
||||||
|
$operationRun,
|
||||||
|
status: OperationRunStatus::Completed->value,
|
||||||
|
outcome: OperationRunOutcome::Failed->value,
|
||||||
|
failures: [
|
||||||
|
[
|
||||||
|
'code' => 'evidence_snapshot_generation.failed',
|
||||||
|
'message' => $throwable->getMessage(),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
throw $throwable;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -4,11 +4,12 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Models\EvidenceSnapshot;
|
||||||
use App\Models\Finding;
|
use App\Models\Finding;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\ReviewPack;
|
use App\Models\ReviewPack;
|
||||||
use App\Models\StoredReport;
|
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
|
use App\Models\TenantReview;
|
||||||
use App\Services\Intune\SecretClassificationService;
|
use App\Services\Intune\SecretClassificationService;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
use App\Services\ReviewPackService;
|
use App\Services\ReviewPackService;
|
||||||
@ -27,6 +28,10 @@ class GenerateReviewPackJob implements ShouldQueue
|
|||||||
{
|
{
|
||||||
use Queueable;
|
use Queueable;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
public int $reviewPackId,
|
public int $reviewPackId,
|
||||||
public int $operationRunId,
|
public int $operationRunId,
|
||||||
@ -34,7 +39,7 @@ public function __construct(
|
|||||||
|
|
||||||
public function handle(OperationRunService $operationRunService): void
|
public function handle(OperationRunService $operationRunService): void
|
||||||
{
|
{
|
||||||
$reviewPack = ReviewPack::query()->find($this->reviewPackId);
|
$reviewPack = ReviewPack::query()->with(['tenant', 'evidenceSnapshot.items', 'tenantReview.sections'])->find($this->reviewPackId);
|
||||||
$operationRun = OperationRun::query()->find($this->operationRunId);
|
$operationRun = OperationRun::query()->find($this->operationRunId);
|
||||||
|
|
||||||
if (! $reviewPack instanceof ReviewPack || ! $operationRun instanceof OperationRun) {
|
if (! $reviewPack instanceof ReviewPack || ! $operationRun instanceof OperationRun) {
|
||||||
@ -54,12 +59,20 @@ public function handle(OperationRunService $operationRunService): void
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$snapshot = $reviewPack->evidenceSnapshot;
|
||||||
|
|
||||||
|
if (! $snapshot instanceof EvidenceSnapshot) {
|
||||||
|
$this->markFailed($reviewPack, $operationRun, $operationRunService, 'missing_snapshot', 'Evidence snapshot not found');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Mark running via OperationRunService (auto-sets started_at)
|
// Mark running via OperationRunService (auto-sets started_at)
|
||||||
$operationRunService->updateRun($operationRun, OperationRunStatus::Running->value);
|
$operationRunService->updateRun($operationRun, OperationRunStatus::Running->value);
|
||||||
$reviewPack->update(['status' => ReviewPackStatus::Generating->value]);
|
$reviewPack->update(['status' => ReviewPackStatus::Generating->value]);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$this->executeGeneration($reviewPack, $operationRun, $tenant, $operationRunService);
|
$this->executeGeneration($reviewPack, $operationRun, $tenant, $snapshot, $operationRunService);
|
||||||
} catch (Throwable $e) {
|
} catch (Throwable $e) {
|
||||||
$this->markFailed($reviewPack, $operationRun, $operationRunService, 'generation_error', $e->getMessage());
|
$this->markFailed($reviewPack, $operationRun, $operationRunService, 'generation_error', $e->getMessage());
|
||||||
|
|
||||||
@ -67,60 +80,44 @@ public function handle(OperationRunService $operationRunService): void
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function executeGeneration(ReviewPack $reviewPack, OperationRun $operationRun, Tenant $tenant, OperationRunService $operationRunService): void
|
private function executeGeneration(ReviewPack $reviewPack, OperationRun $operationRun, Tenant $tenant, EvidenceSnapshot $snapshot, OperationRunService $operationRunService): void
|
||||||
{
|
{
|
||||||
|
$review = $reviewPack->tenantReview;
|
||||||
|
|
||||||
|
if ($review instanceof TenantReview) {
|
||||||
|
$this->executeReviewDerivedGeneration($reviewPack, $review, $operationRun, $tenant, $snapshot, $operationRunService);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$options = $reviewPack->options ?? [];
|
$options = $reviewPack->options ?? [];
|
||||||
$includePii = (bool) ($options['include_pii'] ?? true);
|
$includePii = (bool) ($options['include_pii'] ?? true);
|
||||||
$includeOperations = (bool) ($options['include_operations'] ?? true);
|
$includeOperations = (bool) ($options['include_operations'] ?? true);
|
||||||
$tenantId = (int) $tenant->getKey();
|
$items = $snapshot->items->keyBy('dimension_key');
|
||||||
|
$findingsPayload = $this->itemSummaryPayload($items->get('findings_summary'));
|
||||||
|
$permissionPosturePayload = $this->itemSummaryPayload($items->get('permission_posture'));
|
||||||
|
$entraRolesPayload = $this->itemSummaryPayload($items->get('entra_admin_roles'));
|
||||||
|
$operationsPayload = $this->itemSummaryPayload($items->get('operations_summary'));
|
||||||
|
$riskAcceptance = is_array($snapshot->summary['risk_acceptance'] ?? null)
|
||||||
|
? $snapshot->summary['risk_acceptance']
|
||||||
|
: (is_array($findingsPayload['risk_acceptance'] ?? null) ? $findingsPayload['risk_acceptance'] : []);
|
||||||
|
|
||||||
// 1. Collect StoredReports
|
$findings = collect(is_array($findingsPayload['entries'] ?? null) ? $findingsPayload['entries'] : []);
|
||||||
$storedReports = StoredReport::query()
|
$recentOperations = collect($includeOperations && is_array($operationsPayload['entries'] ?? null) ? $operationsPayload['entries'] : []);
|
||||||
->where('tenant_id', $tenantId)
|
$hardening = is_array($snapshot->summary['hardening'] ?? null) ? $snapshot->summary['hardening'] : [];
|
||||||
->whereIn('report_type', [
|
$dataFreshness = $this->computeDataFreshness($items);
|
||||||
StoredReport::REPORT_TYPE_PERMISSION_POSTURE,
|
|
||||||
StoredReport::REPORT_TYPE_ENTRA_ADMIN_ROLES,
|
|
||||||
])
|
|
||||||
->get()
|
|
||||||
->keyBy('report_type');
|
|
||||||
|
|
||||||
// 2. Collect open findings
|
|
||||||
$findings = Finding::query()
|
|
||||||
->where('tenant_id', $tenantId)
|
|
||||||
->whereIn('status', Finding::openStatusesForQuery())
|
|
||||||
->orderBy('severity')
|
|
||||||
->orderBy('created_at')
|
|
||||||
->get();
|
|
||||||
|
|
||||||
// 3. Collect tenant hardening fields
|
|
||||||
$hardening = [
|
|
||||||
'rbac_last_checked_at' => $tenant->rbac_last_checked_at?->toIso8601String(),
|
|
||||||
'rbac_last_setup_at' => $tenant->rbac_last_setup_at?->toIso8601String(),
|
|
||||||
'rbac_canary_results' => $tenant->rbac_canary_results,
|
|
||||||
'rbac_last_warnings' => $tenant->rbac_last_warnings,
|
|
||||||
'rbac_scope_mode' => $tenant->rbac_scope_mode,
|
|
||||||
];
|
|
||||||
|
|
||||||
// 4. Collect recent OperationRuns (30 days)
|
|
||||||
$recentOperations = $includeOperations
|
|
||||||
? OperationRun::query()
|
|
||||||
->where('tenant_id', $tenantId)
|
|
||||||
->where('created_at', '>=', now()->subDays(30))
|
|
||||||
->orderByDesc('created_at')
|
|
||||||
->get()
|
|
||||||
: collect();
|
|
||||||
|
|
||||||
// 5. Data freshness
|
|
||||||
$dataFreshness = $this->computeDataFreshness($storedReports, $findings, $tenant);
|
|
||||||
|
|
||||||
// 6. Build file map
|
// 6. Build file map
|
||||||
$fileMap = $this->buildFileMap(
|
$fileMap = $this->buildFileMap(
|
||||||
storedReports: $storedReports,
|
|
||||||
findings: $findings,
|
findings: $findings,
|
||||||
hardening: $hardening,
|
hardening: $hardening,
|
||||||
|
permissionPosture: is_array($permissionPosturePayload['payload'] ?? null) ? $permissionPosturePayload['payload'] : [],
|
||||||
|
entraAdminRoles: ['roles' => is_array($entraRolesPayload['roles'] ?? null) ? $entraRolesPayload['roles'] : []],
|
||||||
recentOperations: $recentOperations,
|
recentOperations: $recentOperations,
|
||||||
tenant: $tenant,
|
tenant: $tenant,
|
||||||
|
snapshot: $snapshot,
|
||||||
dataFreshness: $dataFreshness,
|
dataFreshness: $dataFreshness,
|
||||||
|
riskAcceptance: $riskAcceptance,
|
||||||
includePii: $includePii,
|
includePii: $includePii,
|
||||||
includeOperations: $includeOperations,
|
includeOperations: $includeOperations,
|
||||||
);
|
);
|
||||||
@ -154,16 +151,24 @@ private function executeGeneration(ReviewPack $reviewPack, OperationRun $operati
|
|||||||
|
|
||||||
// 11. Compute summary
|
// 11. Compute summary
|
||||||
$summary = [
|
$summary = [
|
||||||
'finding_count' => $findings->count(),
|
'finding_count' => (int) ($snapshot->summary['finding_count'] ?? $findings->count()),
|
||||||
'report_count' => $storedReports->count(),
|
'report_count' => (int) ($snapshot->summary['report_count'] ?? 0),
|
||||||
'operation_count' => $recentOperations->count(),
|
'operation_count' => $recentOperations->count(),
|
||||||
'data_freshness' => $dataFreshness,
|
'data_freshness' => $dataFreshness,
|
||||||
|
'risk_acceptance' => $riskAcceptance,
|
||||||
|
'evidence_resolution' => [
|
||||||
|
'outcome' => 'resolved',
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
'snapshot_fingerprint' => (string) $snapshot->fingerprint,
|
||||||
|
'completeness_state' => (string) $snapshot->completeness_state,
|
||||||
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
// 12. Update ReviewPack
|
// 12. Update ReviewPack
|
||||||
$retentionDays = (int) config('tenantpilot.review_pack.retention_days', 90);
|
$retentionDays = (int) config('tenantpilot.review_pack.retention_days', 90);
|
||||||
$reviewPack->update([
|
$reviewPack->update([
|
||||||
'status' => ReviewPackStatus::Ready->value,
|
'status' => ReviewPackStatus::Ready->value,
|
||||||
|
'evidence_snapshot_id' => (int) $snapshot->getKey(),
|
||||||
'fingerprint' => $fingerprint,
|
'fingerprint' => $fingerprint,
|
||||||
'sha256' => $sha256,
|
'sha256' => $sha256,
|
||||||
'file_size' => $fileSize,
|
'file_size' => $fileSize,
|
||||||
@ -183,18 +188,113 @@ private function executeGeneration(ReviewPack $reviewPack, OperationRun $operati
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function executeReviewDerivedGeneration(
|
||||||
|
ReviewPack $reviewPack,
|
||||||
|
TenantReview $review,
|
||||||
|
OperationRun $operationRun,
|
||||||
|
Tenant $tenant,
|
||||||
|
EvidenceSnapshot $snapshot,
|
||||||
|
OperationRunService $operationRunService,
|
||||||
|
): void {
|
||||||
|
$options = $reviewPack->options ?? [];
|
||||||
|
$includePii = (bool) ($options['include_pii'] ?? true);
|
||||||
|
$includeOperations = (bool) ($options['include_operations'] ?? true);
|
||||||
|
|
||||||
|
$fileMap = $this->buildReviewDerivedFileMap(
|
||||||
|
review: $review,
|
||||||
|
tenant: $tenant,
|
||||||
|
snapshot: $snapshot,
|
||||||
|
includePii: $includePii,
|
||||||
|
includeOperations: $includeOperations,
|
||||||
|
);
|
||||||
|
|
||||||
|
$tempFile = tempnam(sys_get_temp_dir(), 'review-pack-');
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->assembleZip($tempFile, $fileMap);
|
||||||
|
|
||||||
|
$sha256 = hash_file('sha256', $tempFile);
|
||||||
|
$fileSize = filesize($tempFile);
|
||||||
|
$filePath = sprintf(
|
||||||
|
'review-packs/%s/review-%d-%s.zip',
|
||||||
|
$tenant->external_id,
|
||||||
|
(int) $review->getKey(),
|
||||||
|
now()->format('Y-m-d-His'),
|
||||||
|
);
|
||||||
|
|
||||||
|
Storage::disk('exports')->put($filePath, file_get_contents($tempFile));
|
||||||
|
} finally {
|
||||||
|
if (file_exists($tempFile)) {
|
||||||
|
unlink($tempFile);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$fingerprint = app(ReviewPackService::class)->computeFingerprintForReview($review, $options);
|
||||||
|
$reviewSummary = is_array($review->summary) ? $review->summary : [];
|
||||||
|
$summary = [
|
||||||
|
'tenant_review_id' => (int) $review->getKey(),
|
||||||
|
'review_status' => (string) $review->status,
|
||||||
|
'review_completeness_state' => (string) $review->completeness_state,
|
||||||
|
'section_count' => $review->sections->count(),
|
||||||
|
'finding_count' => (int) ($reviewSummary['finding_count'] ?? 0),
|
||||||
|
'report_count' => (int) ($reviewSummary['report_count'] ?? 0),
|
||||||
|
'operation_count' => $includeOperations ? (int) ($reviewSummary['operation_count'] ?? 0) : 0,
|
||||||
|
'highlights' => is_array($reviewSummary['highlights'] ?? null) ? $reviewSummary['highlights'] : [],
|
||||||
|
'publish_blockers' => is_array($reviewSummary['publish_blockers'] ?? null) ? $reviewSummary['publish_blockers'] : [],
|
||||||
|
'evidence_resolution' => [
|
||||||
|
'outcome' => 'resolved',
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
'snapshot_fingerprint' => (string) $snapshot->fingerprint,
|
||||||
|
'completeness_state' => (string) $snapshot->completeness_state,
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$retentionDays = (int) config('tenantpilot.review_pack.retention_days', 90);
|
||||||
|
$reviewPack->update([
|
||||||
|
'status' => ReviewPackStatus::Ready->value,
|
||||||
|
'evidence_snapshot_id' => (int) $snapshot->getKey(),
|
||||||
|
'fingerprint' => $fingerprint,
|
||||||
|
'sha256' => $sha256,
|
||||||
|
'file_size' => $fileSize,
|
||||||
|
'file_path' => $filePath,
|
||||||
|
'file_disk' => 'exports',
|
||||||
|
'generated_at' => now(),
|
||||||
|
'expires_at' => now()->addDays($retentionDays),
|
||||||
|
'summary' => $summary,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$review->update([
|
||||||
|
'current_export_review_pack_id' => (int) $reviewPack->getKey(),
|
||||||
|
'summary' => array_merge($reviewSummary, [
|
||||||
|
'has_ready_export' => true,
|
||||||
|
'current_export_review_pack_id' => (int) $reviewPack->getKey(),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
|
||||||
|
$operationRunService->updateRun(
|
||||||
|
$operationRun,
|
||||||
|
status: OperationRunStatus::Completed->value,
|
||||||
|
outcome: OperationRunOutcome::Succeeded->value,
|
||||||
|
summaryCounts: [
|
||||||
|
'created' => 1,
|
||||||
|
'finding_count' => (int) ($summary['finding_count'] ?? 0),
|
||||||
|
'report_count' => (int) ($summary['report_count'] ?? 0),
|
||||||
|
'operation_count' => (int) ($summary['operation_count'] ?? 0),
|
||||||
|
'errors_recorded' => 0,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param \Illuminate\Support\Collection<string, StoredReport> $storedReports
|
|
||||||
* @param \Illuminate\Database\Eloquent\Collection<int, Finding> $findings
|
|
||||||
* @return array<string, ?string>
|
* @return array<string, ?string>
|
||||||
*/
|
*/
|
||||||
private function computeDataFreshness($storedReports, $findings, Tenant $tenant): array
|
private function computeDataFreshness($items): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'permission_posture' => $storedReports->get(StoredReport::REPORT_TYPE_PERMISSION_POSTURE)?->updated_at?->toIso8601String(),
|
'permission_posture' => $items->get('permission_posture')?->freshness_at?->toIso8601String(),
|
||||||
'entra_admin_roles' => $storedReports->get(StoredReport::REPORT_TYPE_ENTRA_ADMIN_ROLES)?->updated_at?->toIso8601String(),
|
'entra_admin_roles' => $items->get('entra_admin_roles')?->freshness_at?->toIso8601String(),
|
||||||
'findings' => $findings->max('updated_at')?->toIso8601String() ?? $findings->max('created_at')?->toIso8601String(),
|
'findings' => $items->get('findings_summary')?->freshness_at?->toIso8601String(),
|
||||||
'hardening' => $tenant->rbac_last_checked_at?->toIso8601String(),
|
'hardening' => $items->get('baseline_drift_posture')?->freshness_at?->toIso8601String(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -204,12 +304,15 @@ private function computeDataFreshness($storedReports, $findings, Tenant $tenant)
|
|||||||
* @return array<string, string>
|
* @return array<string, string>
|
||||||
*/
|
*/
|
||||||
private function buildFileMap(
|
private function buildFileMap(
|
||||||
$storedReports,
|
|
||||||
$findings,
|
$findings,
|
||||||
array $hardening,
|
array $hardening,
|
||||||
|
array $permissionPosture,
|
||||||
|
array $entraAdminRoles,
|
||||||
$recentOperations,
|
$recentOperations,
|
||||||
Tenant $tenant,
|
Tenant $tenant,
|
||||||
|
EvidenceSnapshot $snapshot,
|
||||||
array $dataFreshness,
|
array $dataFreshness,
|
||||||
|
array $riskAcceptance,
|
||||||
bool $includePii,
|
bool $includePii,
|
||||||
bool $includeOperations,
|
bool $includeOperations,
|
||||||
): array {
|
): array {
|
||||||
@ -227,6 +330,12 @@ private function buildFileMap(
|
|||||||
'tenant_id' => $tenant->external_id,
|
'tenant_id' => $tenant->external_id,
|
||||||
'tenant_name' => $includePii ? $tenant->name : '[REDACTED]',
|
'tenant_name' => $includePii ? $tenant->name : '[REDACTED]',
|
||||||
'generated_at' => now()->toIso8601String(),
|
'generated_at' => now()->toIso8601String(),
|
||||||
|
'evidence_snapshot' => [
|
||||||
|
'id' => (int) $snapshot->getKey(),
|
||||||
|
'fingerprint' => (string) $snapshot->fingerprint,
|
||||||
|
'completeness_state' => (string) $snapshot->completeness_state,
|
||||||
|
'generated_at' => $snapshot->generated_at?->toIso8601String(),
|
||||||
|
],
|
||||||
'redaction_integrity' => [
|
'redaction_integrity' => [
|
||||||
'protected_values_hidden' => true,
|
'protected_values_hidden' => true,
|
||||||
'note' => RedactionIntegrity::protectedValueNote(),
|
'note' => RedactionIntegrity::protectedValueNote(),
|
||||||
@ -241,16 +350,14 @@ private function buildFileMap(
|
|||||||
$files['operations.csv'] = $this->buildOperationsCsv($recentOperations, $includePii);
|
$files['operations.csv'] = $this->buildOperationsCsv($recentOperations, $includePii);
|
||||||
|
|
||||||
// reports/entra_admin_roles.json
|
// reports/entra_admin_roles.json
|
||||||
$entraReport = $storedReports->get(StoredReport::REPORT_TYPE_ENTRA_ADMIN_ROLES);
|
|
||||||
$files['reports/entra_admin_roles.json'] = json_encode(
|
$files['reports/entra_admin_roles.json'] = json_encode(
|
||||||
$entraReport ? $this->redactReportPayload($entraReport->payload ?? [], $includePii) : [],
|
$this->redactReportPayload($entraAdminRoles, $includePii),
|
||||||
JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR,
|
JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR,
|
||||||
);
|
);
|
||||||
|
|
||||||
// reports/permission_posture.json
|
// reports/permission_posture.json
|
||||||
$postureReport = $storedReports->get(StoredReport::REPORT_TYPE_PERMISSION_POSTURE);
|
|
||||||
$files['reports/permission_posture.json'] = json_encode(
|
$files['reports/permission_posture.json'] = json_encode(
|
||||||
$postureReport ? $this->redactReportPayload($postureReport->payload ?? [], $includePii) : [],
|
$this->redactReportPayload($permissionPosture, $includePii),
|
||||||
JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR,
|
JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR,
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -258,8 +365,10 @@ private function buildFileMap(
|
|||||||
$files['summary.json'] = json_encode([
|
$files['summary.json'] = json_encode([
|
||||||
'data_freshness' => $dataFreshness,
|
'data_freshness' => $dataFreshness,
|
||||||
'finding_count' => $findings->count(),
|
'finding_count' => $findings->count(),
|
||||||
'report_count' => $storedReports->count(),
|
'report_count' => count(array_filter([$permissionPosture, $entraAdminRoles], static fn (array $payload): bool => $payload !== [])),
|
||||||
'operation_count' => $recentOperations->count(),
|
'operation_count' => $recentOperations->count(),
|
||||||
|
'risk_acceptance' => $riskAcceptance,
|
||||||
|
'snapshot_id' => (int) $snapshot->getKey(),
|
||||||
], JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR);
|
], JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR);
|
||||||
|
|
||||||
return $files;
|
return $files;
|
||||||
@ -273,18 +382,33 @@ private function buildFileMap(
|
|||||||
private function buildFindingsCsv($findings, bool $includePii): string
|
private function buildFindingsCsv($findings, bool $includePii): string
|
||||||
{
|
{
|
||||||
$handle = fopen('php://temp', 'r+');
|
$handle = fopen('php://temp', 'r+');
|
||||||
fputcsv($handle, ['id', 'finding_type', 'severity', 'status', 'title', 'description', 'created_at', 'updated_at']);
|
$this->writeCsvRow($handle, ['id', 'finding_type', 'severity', 'status', 'title', 'description', 'created_at', 'updated_at']);
|
||||||
|
|
||||||
foreach ($findings as $finding) {
|
foreach ($findings as $finding) {
|
||||||
fputcsv($handle, [
|
$row = $finding instanceof Finding
|
||||||
$finding->id,
|
? [
|
||||||
$finding->finding_type,
|
$finding->id,
|
||||||
$finding->severity,
|
$finding->finding_type,
|
||||||
$finding->status,
|
$finding->severity,
|
||||||
$includePii ? ($finding->title ?? '') : '[REDACTED]',
|
$finding->status,
|
||||||
$includePii ? ($finding->description ?? '') : '[REDACTED]',
|
$includePii ? ($finding->title ?? '') : '[REDACTED]',
|
||||||
$finding->created_at?->toIso8601String(),
|
$includePii ? ($finding->description ?? '') : '[REDACTED]',
|
||||||
$finding->updated_at?->toIso8601String(),
|
$finding->created_at?->toIso8601String(),
|
||||||
|
$finding->updated_at?->toIso8601String(),
|
||||||
|
]
|
||||||
|
: [
|
||||||
|
$finding['id'] ?? '',
|
||||||
|
$finding['finding_type'] ?? '',
|
||||||
|
$finding['severity'] ?? '',
|
||||||
|
$finding['status'] ?? '',
|
||||||
|
$includePii ? ($finding['title'] ?? '') : '[REDACTED]',
|
||||||
|
$includePii ? ($finding['description'] ?? '') : '[REDACTED]',
|
||||||
|
$finding['created_at'] ?? '',
|
||||||
|
$finding['updated_at'] ?? '',
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->writeCsvRow($handle, [
|
||||||
|
...$row,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -301,17 +425,31 @@ private function buildFindingsCsv($findings, bool $includePii): string
|
|||||||
private function buildOperationsCsv($operations, bool $includePii): string
|
private function buildOperationsCsv($operations, bool $includePii): string
|
||||||
{
|
{
|
||||||
$handle = fopen('php://temp', 'r+');
|
$handle = fopen('php://temp', 'r+');
|
||||||
fputcsv($handle, ['id', 'type', 'status', 'outcome', 'initiator', 'started_at', 'completed_at']);
|
$this->writeCsvRow($handle, ['id', 'type', 'status', 'outcome', 'initiator', 'started_at', 'completed_at']);
|
||||||
|
|
||||||
foreach ($operations as $operation) {
|
foreach ($operations as $operation) {
|
||||||
fputcsv($handle, [
|
$row = $operation instanceof OperationRun
|
||||||
$operation->id,
|
? [
|
||||||
$operation->type,
|
$operation->id,
|
||||||
$operation->status,
|
$operation->type,
|
||||||
$operation->outcome,
|
$operation->status,
|
||||||
$includePii ? ($operation->user?->name ?? '') : '[REDACTED]',
|
$operation->outcome,
|
||||||
$operation->started_at?->toIso8601String(),
|
$includePii ? ($operation->user?->name ?? '') : '[REDACTED]',
|
||||||
$operation->completed_at?->toIso8601String(),
|
$operation->started_at?->toIso8601String(),
|
||||||
|
$operation->completed_at?->toIso8601String(),
|
||||||
|
]
|
||||||
|
: [
|
||||||
|
$operation['id'] ?? '',
|
||||||
|
$operation['type'] ?? '',
|
||||||
|
$operation['status'] ?? '',
|
||||||
|
$operation['outcome'] ?? '',
|
||||||
|
$includePii ? ($operation['initiator_name'] ?? '') : '[REDACTED]',
|
||||||
|
$operation['started_at'] ?? '',
|
||||||
|
$operation['completed_at'] ?? '',
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->writeCsvRow($handle, [
|
||||||
|
...$row,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -322,6 +460,15 @@ private function buildOperationsCsv($operations, bool $includePii): string
|
|||||||
return $content;
|
return $content;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param resource $handle
|
||||||
|
* @param array<int, mixed> $row
|
||||||
|
*/
|
||||||
|
private function writeCsvRow($handle, array $row): void
|
||||||
|
{
|
||||||
|
fputcsv($handle, $row, ',', '"', '\\');
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Redact PII from a report payload.
|
* Redact PII from a report payload.
|
||||||
*
|
*
|
||||||
@ -431,9 +578,98 @@ private function assembleZip(string $tempFile, array $fileMap): void
|
|||||||
$zip->close();
|
$zip->close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private function buildReviewDerivedFileMap(
|
||||||
|
TenantReview $review,
|
||||||
|
Tenant $tenant,
|
||||||
|
EvidenceSnapshot $snapshot,
|
||||||
|
bool $includePii,
|
||||||
|
bool $includeOperations,
|
||||||
|
): array {
|
||||||
|
$reviewSummary = is_array($review->summary) ? $review->summary : [];
|
||||||
|
|
||||||
|
$sections = $review->sections
|
||||||
|
->filter(fn (mixed $section): bool => $includeOperations || $section->section_key !== 'operations_health')
|
||||||
|
->values();
|
||||||
|
|
||||||
|
$files = [
|
||||||
|
'metadata.json' => json_encode([
|
||||||
|
'version' => '1.0',
|
||||||
|
'tenant_id' => $tenant->external_id,
|
||||||
|
'tenant_name' => $includePii ? $tenant->name : '[REDACTED]',
|
||||||
|
'generated_at' => now()->toIso8601String(),
|
||||||
|
'tenant_review' => [
|
||||||
|
'id' => (int) $review->getKey(),
|
||||||
|
'status' => (string) $review->status,
|
||||||
|
'completeness_state' => (string) $review->completeness_state,
|
||||||
|
'published_at' => $review->published_at?->toIso8601String(),
|
||||||
|
'fingerprint' => (string) $review->fingerprint,
|
||||||
|
],
|
||||||
|
'evidence_snapshot' => [
|
||||||
|
'id' => (int) $snapshot->getKey(),
|
||||||
|
'fingerprint' => (string) $snapshot->fingerprint,
|
||||||
|
'completeness_state' => (string) $snapshot->completeness_state,
|
||||||
|
'generated_at' => $snapshot->generated_at?->toIso8601String(),
|
||||||
|
],
|
||||||
|
'options' => [
|
||||||
|
'include_pii' => $includePii,
|
||||||
|
'include_operations' => $includeOperations,
|
||||||
|
],
|
||||||
|
'redaction_integrity' => [
|
||||||
|
'protected_values_hidden' => true,
|
||||||
|
'note' => RedactionIntegrity::protectedValueNote(),
|
||||||
|
],
|
||||||
|
], JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR),
|
||||||
|
'summary.json' => json_encode($this->redactReportPayload(array_merge([
|
||||||
|
'tenant_review_id' => (int) $review->getKey(),
|
||||||
|
'review_status' => (string) $review->status,
|
||||||
|
'review_completeness_state' => (string) $review->completeness_state,
|
||||||
|
], $reviewSummary), $includePii), JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR),
|
||||||
|
'sections.json' => json_encode($sections->map(function ($section) use ($includePii): array {
|
||||||
|
$summaryPayload = is_array($section->summary_payload) ? $section->summary_payload : [];
|
||||||
|
$renderPayload = is_array($section->render_payload) ? $section->render_payload : [];
|
||||||
|
|
||||||
|
return [
|
||||||
|
'section_key' => (string) $section->section_key,
|
||||||
|
'title' => (string) $section->title,
|
||||||
|
'sort_order' => (int) $section->sort_order,
|
||||||
|
'required' => (bool) $section->required,
|
||||||
|
'completeness_state' => (string) $section->completeness_state,
|
||||||
|
'summary_payload' => $this->redactReportPayload($summaryPayload, $includePii),
|
||||||
|
'render_payload' => $this->redactReportPayload($renderPayload, $includePii),
|
||||||
|
];
|
||||||
|
})->all(), JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR),
|
||||||
|
];
|
||||||
|
|
||||||
|
foreach ($sections as $section) {
|
||||||
|
$renderPayload = is_array($section->render_payload) ? $section->render_payload : [];
|
||||||
|
$summaryPayload = is_array($section->summary_payload) ? $section->summary_payload : [];
|
||||||
|
$filename = sprintf('sections/%02d-%s.json', (int) $section->sort_order, (string) $section->section_key);
|
||||||
|
|
||||||
|
$files[$filename] = json_encode([
|
||||||
|
'title' => (string) $section->title,
|
||||||
|
'completeness_state' => (string) $section->completeness_state,
|
||||||
|
'summary_payload' => $this->redactReportPayload($summaryPayload, $includePii),
|
||||||
|
'render_payload' => $this->redactReportPayload($renderPayload, $includePii),
|
||||||
|
], JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $files;
|
||||||
|
}
|
||||||
|
|
||||||
private function markFailed(ReviewPack $reviewPack, OperationRun $operationRun, OperationRunService $operationRunService, string $reasonCode, string $errorMessage): void
|
private function markFailed(ReviewPack $reviewPack, OperationRun $operationRun, OperationRunService $operationRunService, string $reasonCode, string $errorMessage): void
|
||||||
{
|
{
|
||||||
$reviewPack->update(['status' => ReviewPackStatus::Failed->value]);
|
$reviewPack->update([
|
||||||
|
'status' => ReviewPackStatus::Failed->value,
|
||||||
|
'summary' => array_merge($reviewPack->summary ?? [], [
|
||||||
|
'evidence_resolution' => array_merge($reviewPack->summary['evidence_resolution'] ?? [], [
|
||||||
|
'outcome' => $reasonCode,
|
||||||
|
'reasons' => [mb_substr($errorMessage, 0, 500)],
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
|
||||||
$operationRunService->updateRun(
|
$operationRunService->updateRun(
|
||||||
$operationRun,
|
$operationRun,
|
||||||
@ -444,4 +680,13 @@ private function markFailed(ReviewPack $reviewPack, OperationRun $operationRun,
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function itemSummaryPayload(mixed $item): array
|
||||||
|
{
|
||||||
|
if (! $item instanceof \App\Models\EvidenceSnapshotItem || ! is_array($item->summary_payload)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item->summary_payload;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
57
app/Jobs/Middleware/EnsureQueuedExecutionLegitimate.php
Normal file
57
app/Jobs/Middleware/EnsureQueuedExecutionLegitimate.php
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Jobs\Middleware;
|
||||||
|
|
||||||
|
use App\Models\OperationRun;
|
||||||
|
use App\Services\OperationRunService;
|
||||||
|
use App\Services\Operations\QueuedExecutionLegitimacyGate;
|
||||||
|
use Closure;
|
||||||
|
|
||||||
|
class EnsureQueuedExecutionLegitimate
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param mixed $job
|
||||||
|
* @param callable $next
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function handle($job, Closure $next)
|
||||||
|
{
|
||||||
|
$run = $this->resolveRun($job);
|
||||||
|
|
||||||
|
if (! $run instanceof OperationRun) {
|
||||||
|
return $next($job);
|
||||||
|
}
|
||||||
|
|
||||||
|
$decision = app(QueuedExecutionLegitimacyGate::class)->evaluate($run);
|
||||||
|
|
||||||
|
if (! $decision->allowed) {
|
||||||
|
app(OperationRunService::class)->finalizeExecutionLegitimacyBlockedRun($run, $decision);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $next($job);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param mixed $job
|
||||||
|
*/
|
||||||
|
private function resolveRun($job): ?OperationRun
|
||||||
|
{
|
||||||
|
if (method_exists($job, 'getOperationRun')) {
|
||||||
|
$run = $job->getOperationRun();
|
||||||
|
|
||||||
|
return $run instanceof OperationRun ? $run : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (property_exists($job, 'operationRun')) {
|
||||||
|
$run = $job->operationRun;
|
||||||
|
|
||||||
|
return $run instanceof OperationRun ? $run : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -17,14 +17,7 @@ class TrackOperationRun
|
|||||||
*/
|
*/
|
||||||
public function handle($job, Closure $next)
|
public function handle($job, Closure $next)
|
||||||
{
|
{
|
||||||
// Check if the job has an 'operationRun' property or method
|
$run = $this->resolveRun($job);
|
||||||
$run = null;
|
|
||||||
|
|
||||||
if (method_exists($job, 'getOperationRun')) {
|
|
||||||
$run = $job->getOperationRun();
|
|
||||||
} elseif (property_exists($job, 'operationRun')) {
|
|
||||||
$run = $job->operationRun;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (! $run instanceof OperationRun) {
|
if (! $run instanceof OperationRun) {
|
||||||
return $next($job);
|
return $next($job);
|
||||||
@ -33,19 +26,23 @@ public function handle($job, Closure $next)
|
|||||||
/** @var OperationRunService $service */
|
/** @var OperationRunService $service */
|
||||||
$service = app(OperationRunService::class);
|
$service = app(OperationRunService::class);
|
||||||
|
|
||||||
// Mark as running
|
$run->refresh();
|
||||||
$service->updateRun($run, 'running');
|
|
||||||
|
if ($run->status === 'completed') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($run->status !== 'running') {
|
||||||
|
$service->updateRun($run, 'running');
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$response = $next($job);
|
$response = $next($job);
|
||||||
|
|
||||||
// If the job was released back onto the queue (retry / delay), do not mark the run as completed.
|
|
||||||
if (property_exists($job, 'job') && $job->job && method_exists($job->job, 'isReleased') && $job->job->isReleased()) {
|
if (property_exists($job, 'job') && $job->job && method_exists($job->job, 'isReleased') && $job->job->isReleased()) {
|
||||||
return $response;
|
return $response;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the job didn't already mark it as completed/failed, we do it here.
|
|
||||||
// Re-fetch to check current status
|
|
||||||
$run->refresh();
|
$run->refresh();
|
||||||
|
|
||||||
if ($run->status === 'running') {
|
if ($run->status === 'running') {
|
||||||
@ -58,4 +55,24 @@ public function handle($job, Closure $next)
|
|||||||
throw $e;
|
throw $e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param mixed $job
|
||||||
|
*/
|
||||||
|
private function resolveRun($job): ?OperationRun
|
||||||
|
{
|
||||||
|
if (method_exists($job, 'getOperationRun')) {
|
||||||
|
$run = $job->getOperationRun();
|
||||||
|
|
||||||
|
return $run instanceof OperationRun ? $run : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (property_exists($job, 'operationRun')) {
|
||||||
|
$run = $job->operationRun;
|
||||||
|
|
||||||
|
return $run instanceof OperationRun ? $run : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs\Operations;
|
namespace App\Jobs\Operations;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
use Illuminate\Bus\Queueable;
|
use Illuminate\Bus\Queueable;
|
||||||
@ -37,7 +38,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [];
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(OperationRunService $runs): void
|
public function handle(OperationRunService $runs): void
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs\Operations;
|
namespace App\Jobs\Operations;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
use Illuminate\Bus\Queueable;
|
use Illuminate\Bus\Queueable;
|
||||||
@ -38,7 +39,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [];
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(OperationRunService $runs): void
|
public function handle(OperationRunService $runs): void
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs\Operations;
|
namespace App\Jobs\Operations;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\Policy;
|
use App\Models\Policy;
|
||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
@ -33,6 +34,11 @@ public function __construct(
|
|||||||
$this->operationRun = $operationRun;
|
$this->operationRun = $operationRun;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
|
}
|
||||||
|
|
||||||
public function handle(OperationRunService $runs, TargetScopeConcurrencyLimiter $limiter): void
|
public function handle(OperationRunService $runs, TargetScopeConcurrencyLimiter $limiter): void
|
||||||
{
|
{
|
||||||
if (! $this->operationRun instanceof OperationRun) {
|
if (! $this->operationRun instanceof OperationRun) {
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs\Operations;
|
namespace App\Jobs\Operations;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
@ -36,6 +37,11 @@ public function __construct(
|
|||||||
$this->operationRun = $operationRun;
|
$this->operationRun = $operationRun;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate];
|
||||||
|
}
|
||||||
|
|
||||||
public function handle(
|
public function handle(
|
||||||
OperationRunService $runs,
|
OperationRunService $runs,
|
||||||
TargetScopeConcurrencyLimiter $limiter,
|
TargetScopeConcurrencyLimiter $limiter,
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\ProviderConnection;
|
use App\Models\ProviderConnection;
|
||||||
@ -41,7 +42,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(
|
public function handle(
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\ProviderConnection;
|
use App\Models\ProviderConnection;
|
||||||
@ -52,7 +53,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(
|
public function handle(
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\ProviderConnection;
|
use App\Models\ProviderConnection;
|
||||||
@ -41,7 +42,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(
|
public function handle(
|
||||||
|
|||||||
@ -4,6 +4,8 @@
|
|||||||
|
|
||||||
use App\Contracts\Hardening\WriteGateInterface;
|
use App\Contracts\Hardening\WriteGateInterface;
|
||||||
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
use App\Exceptions\Hardening\ProviderAccessHardeningRequired;
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\RestoreRun;
|
use App\Models\RestoreRun;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
@ -12,6 +14,7 @@
|
|||||||
use App\Services\OperationRunService;
|
use App\Services\OperationRunService;
|
||||||
use App\Support\OperationRunOutcome;
|
use App\Support\OperationRunOutcome;
|
||||||
use App\Support\OperationRunStatus;
|
use App\Support\OperationRunStatus;
|
||||||
|
use App\Support\Operations\ExecutionAuthorityMode;
|
||||||
use App\Support\OpsUx\AssignmentJobFingerprint;
|
use App\Support\OpsUx\AssignmentJobFingerprint;
|
||||||
use App\Support\OpsUx\RunFailureSanitizer;
|
use App\Support\OpsUx\RunFailureSanitizer;
|
||||||
use Illuminate\Bus\Queueable;
|
use Illuminate\Bus\Queueable;
|
||||||
@ -39,6 +42,14 @@ class RestoreAssignmentsJob implements ShouldQueue
|
|||||||
|
|
||||||
public int $backoff = 0;
|
public int $backoff = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, object>
|
||||||
|
*/
|
||||||
|
public function middleware(): array
|
||||||
|
{
|
||||||
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new job instance.
|
* Create a new job instance.
|
||||||
*/
|
*/
|
||||||
@ -403,6 +414,8 @@ private static function operationRunContext(
|
|||||||
'policy_type' => trim($policyType),
|
'policy_type' => trim($policyType),
|
||||||
'policy_id' => trim($policyId),
|
'policy_id' => trim($policyId),
|
||||||
'assignment_item_count' => count($assignments),
|
'assignment_item_count' => count($assignments),
|
||||||
|
'execution_authority_mode' => ExecutionAuthorityMode::ActorBound->value,
|
||||||
|
'required_capability' => 'tenant.manage',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\BackupSchedule;
|
use App\Models\BackupSchedule;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
@ -39,6 +40,10 @@ class RunBackupScheduleJob implements ShouldQueue
|
|||||||
|
|
||||||
public int $tries = 3;
|
public int $tries = 3;
|
||||||
|
|
||||||
|
public int $timeout = 300;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compatibility-only legacy field.
|
* Compatibility-only legacy field.
|
||||||
*
|
*
|
||||||
@ -58,7 +63,7 @@ public function __construct(
|
|||||||
|
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(
|
public function handle(
|
||||||
|
|||||||
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\Tenant;
|
use App\Models\Tenant;
|
||||||
@ -23,7 +25,15 @@
|
|||||||
|
|
||||||
class RunInventorySyncJob implements ShouldQueue
|
class RunInventorySyncJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
@ -45,7 +55,7 @@ public function __construct(
|
|||||||
*/
|
*/
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -89,11 +99,6 @@ public function handle(InventorySyncService $inventorySyncService, AuditLogger $
|
|||||||
$successCount = 0;
|
$successCount = 0;
|
||||||
$failedCount = 0;
|
$failedCount = 0;
|
||||||
|
|
||||||
// Note: The TrackOperationRun middleware will automatically set status to 'running' at start.
|
|
||||||
// It will also handle success completion if no exceptions thrown.
|
|
||||||
// However, InventorySyncService execution logic might be complex with partial failures.
|
|
||||||
// We might want to explicitly update the OperationRun if partial failures occur.
|
|
||||||
|
|
||||||
$result = $inventorySyncService->executeSelection(
|
$result = $inventorySyncService->executeSelection(
|
||||||
$this->operationRun,
|
$this->operationRun,
|
||||||
$tenant,
|
$tenant,
|
||||||
|
|||||||
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Jobs;
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Jobs\Concerns\BridgesFailedOperationRun;
|
||||||
|
use App\Jobs\Middleware\EnsureQueuedExecutionLegitimate;
|
||||||
use App\Jobs\Middleware\TrackOperationRun;
|
use App\Jobs\Middleware\TrackOperationRun;
|
||||||
use App\Models\OperationRun;
|
use App\Models\OperationRun;
|
||||||
use App\Models\Policy;
|
use App\Models\Policy;
|
||||||
@ -20,7 +22,15 @@
|
|||||||
|
|
||||||
class SyncPoliciesJob implements ShouldQueue
|
class SyncPoliciesJob implements ShouldQueue
|
||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use BridgesFailedOperationRun;
|
||||||
|
use Dispatchable;
|
||||||
|
use InteractsWithQueue;
|
||||||
|
use Queueable;
|
||||||
|
use SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 180;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
@ -39,7 +49,7 @@ public function __construct(
|
|||||||
|
|
||||||
public function middleware(): array
|
public function middleware(): array
|
||||||
{
|
{
|
||||||
return [new TrackOperationRun];
|
return [new EnsureQueuedExecutionLegitimate, new TrackOperationRun];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function handle(PolicySyncService $service, OperationRunService $operationRunService): void
|
public function handle(PolicySyncService $service, OperationRunService $operationRunService): void
|
||||||
|
|||||||
@ -20,6 +20,10 @@ class SyncRoleDefinitionsJob implements ShouldQueue
|
|||||||
{
|
{
|
||||||
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
||||||
|
|
||||||
|
public int $timeout = 240;
|
||||||
|
|
||||||
|
public bool $failOnTimeout = true;
|
||||||
|
|
||||||
public ?OperationRun $operationRun = null;
|
public ?OperationRun $operationRun = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -18,6 +18,14 @@ class AuditLog extends Model
|
|||||||
{
|
{
|
||||||
use HasFactory;
|
use HasFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<int, string>
|
||||||
|
*/
|
||||||
|
private const INTERNAL_METADATA_KEYS = [
|
||||||
|
'_actor_type',
|
||||||
|
'_dedupe_key',
|
||||||
|
];
|
||||||
|
|
||||||
protected $guarded = [];
|
protected $guarded = [];
|
||||||
|
|
||||||
protected $casts = [
|
protected $casts = [
|
||||||
@ -202,7 +210,12 @@ public function contextItems(): array
|
|||||||
}
|
}
|
||||||
|
|
||||||
foreach ($metadata as $key => $value) {
|
foreach ($metadata as $key => $value) {
|
||||||
if (in_array($key, $seen, true) || in_array($key, ['before', 'after'], true)) {
|
if (
|
||||||
|
in_array($key, $seen, true)
|
||||||
|
|| in_array($key, ['before', 'after'], true)
|
||||||
|
|| str_starts_with((string) $key, '_')
|
||||||
|
|| in_array((string) $key, self::INTERNAL_METADATA_KEYS, true)
|
||||||
|
) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user