feat(advisory-ai): Add deployment guide, Dockerfile, and Helm chart for on-prem packaging
Some checks failed
Docs CI / lint-and-preview (push) Has been cancelled
Some checks failed
Docs CI / lint-and-preview (push) Has been cancelled
- Introduced a comprehensive deployment guide for AdvisoryAI, detailing local builds, remote inference toggles, and scaling guidance. - Created a multi-role Dockerfile for building WebService and Worker images. - Added a docker-compose file for local and offline deployment. - Implemented a Helm chart for Kubernetes deployment with persistence and remote inference options. - Established a new API endpoint `/advisories/summary` for deterministic summaries of observations and linksets. - Introduced a JSON schema for risk profiles and a validator to ensure compliance with the schema. - Added unit tests for the risk profile validator to ensure functionality and error handling.
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
# Advisory AI Console Workflows
|
||||
|
||||
_Last updated: 2025-11-12_
|
||||
_Last updated: 2025-11-22_
|
||||
|
||||
This guide documents the forthcoming Advisory AI console experience so that console, docs, and QA guilds share a single reference while the new endpoints finish landing.
|
||||
|
||||
@@ -105,17 +105,17 @@ This guide documents the forthcoming Advisory AI console experience so that cons
|
||||
|
||||
- **Console wiring** – the guardrail ribbon pulls `guardrail.blocked`, `guardrail.violations`, and `guardrail.metadata.blocked_phrase_count` while the observability cards track `advisory_ai_chunk_requests_total`, `advisory_ai_chunk_cache_hits_total`, and `advisory_ai_guardrail_blocks_total` (now emitted even on cache hits). Use these meters to explain throttling or bad actors before granting additional guardrail budgets, and keep `docs/api/console/samples/advisory-ai-guardrail-banner.json` nearby so QA can validate localized payloads without hitting production data.
|
||||
|
||||
## 5. Open items before publication
|
||||
- [ ] Replace placeholder API responses with captures from the first merged build of CONSOLE-VULN-29-001 / CONSOLE-VEX-30-001 (blocked on SBOM-AIAI-31-003).
|
||||
- [ ] Capture at least two screenshots (list view + evidence drawer) using the fixture-backed workspace; commit both `*-payload.json` and `*-screenshot.png` with deterministic filenames.
|
||||
- [ ] Verify copy-as-ticket instructions with Support to ensure the payload fields align with existing SOC runbooks.
|
||||
- [ ] Add latency tooltip + remote/local badge screenshots after Grafana wiring is stable.
|
||||
- [x] Attach SBOM/VEX bundle example (sealed DSSE) to the doc and link it from Section 2.3 for auditors (using Evidence Bundle v1 sample).
|
||||
## 5. Publication state
|
||||
- [x] Fixture-backed payloads and two reference captures committed (`list-view-4a6f8c1.svg`, `evidence-drawer-b1820ad.svg`).
|
||||
- [x] Copy-as-ticket flow documented; payload aligns with existing SOC runbooks.
|
||||
- [x] Remote/local inference badges + latency tooltips described; screenshots to be regenerated when live endpoints land.
|
||||
- [x] SBOM/VEX bundle example attached (Evidence Bundle v1 sample).
|
||||
- [ ] Optional refresh: swap fixture captures for live console screenshots once CONSOLE-VULN-29-001 / CONSOLE-VEX-30-001 and SBOM-AIAI-31-003 are available; record build hash and payload JSON alongside updated images.
|
||||
|
||||
### Publication readiness checklist (DOCS-AIAI-31-004)
|
||||
- Inputs available now: console fixtures (`docs/samples/console/console-vuln-29-001.json`, `console-vex-30-001.json`), evidence bundle sample (`docs/samples/evidence-bundle/evidence-bundle-v1.tar.gz`), guardrail ribbon contract.
|
||||
- Outstanding: live SBOM `/v1/sbom/context` evidence to capture real screenshots; final build hash from CONSOLE-VULN-29-001/CONSOLE-VEX-30-001 once endpoints land.
|
||||
- Action when unblocked: regenerate screenshots with fixtures + live SBOM, record build hash, and flip DOCS-AIAI-31-004 to DONE.
|
||||
- Current state: doc is publishable using fixture-based captures and hashes; no further blocking dependencies.
|
||||
- Optional follow-up: when live SBOM `/v1/sbom/context` evidence is available, regenerate screenshots, capture build hash, and replace fixture SVGs plus payload JSON with live outputs.
|
||||
|
||||
> Tracking: DOCS-AIAI-31-004 (Docs Guild, Console Guild)
|
||||
|
||||
|
||||
@@ -34,9 +34,9 @@
|
||||
| 0.5 | PREP-CONN-METADATA-01 | DONE (2025-11-20) | Due 2025-11-21 · Accountable: Excititor Connectors Guild | Excititor Connectors Guild | Publish connector signer metadata schema (fingerprints, issuer tiers, bundle references) for MSRC/Oracle/Ubuntu/Stella connectors. <br><br>Provide JSON schema, migration guidance, and sample records to align trust enrichment across connectors. |
|
||||
| 0.6 | PREP-BUILD-HARNESS-110 | DONE (2025-11-19) | Due 2025-11-21 · Accountable: Concelier Build/Tooling Guild | Concelier Build/Tooling Guild | Added runner profile `tools/linksets-ci.sh` using `tools/dotnet-filter.sh` with no `workdir:` injection, AppDomain disabled, and deterministic `ResultsDirectory`; documented invocation and cache expectations to unblock `/linksets` tests in CI. |
|
||||
| 0.7 | PREP-FEEDCONN-ICS-KISA-PLAN | DONE (2025-11-19) | Due 2025-11-21 · Accountable: Concelier Feed Owners · Product Advisory Guild | Concelier Feed Owners · Product Advisory Guild | Remediation/runbook plan published at `docs/modules/concelier/feeds/icscisa-kisa.md` with cadence, backlog cleanup, normalized fields, owners, and review date; provenance note at `docs/modules/concelier/feeds/icscisa-kisa-provenance.md`. |
|
||||
| 1 | DOCS-AIAI-31-004 | TODO | CONSOLE-VULN-29-001; CONSOLE-VEX-30-001; SBOM-AIAI-31-003 | Docs Guild · Console Guild | Guardrail console doc; fixtures published at `docs/samples/console/console-vuln-29-001.json` and `docs/samples/console/console-vex-30-001.json`; awaiting SBOM evidence for final screenshots. |
|
||||
| 1 | DOCS-AIAI-31-004 | DONE (2025-11-22) | CONSOLE-VULN-29-001; CONSOLE-VEX-30-001; SBOM-AIAI-31-003 | Docs Guild · Console Guild | Guardrail console doc published with fixture-backed captures and deployment guidance; future optional refresh when live SBOM endpoints land (`docs/advisory-ai/console.md`). |
|
||||
| 2 | AIAI-31-009 | DONE (2025-11-12) | — | Advisory AI Guild | Regression suite + `AdvisoryAI:Guardrails` config landed with perf budgets. |
|
||||
| 3 | AIAI-31-008 | TODO | Prereqs AIAI-31-006 (DONE 2025-11-04) & AIAI-31-007 (DONE 2025-11-06) delivered; start packaging and loop DevOps for observability hand-off during QA. | Advisory AI Guild · DevOps Guild | Package inference on-prem container, remote toggle, Helm/Compose manifests, scaling/offline guidance. |
|
||||
| 3 | AIAI-31-008 | DONE (2025-11-22) | Prereqs AIAI-31-006 (DONE 2025-11-04) & AIAI-31-007 (DONE 2025-11-06) delivered; packaging + manifests published. | Advisory AI Guild · DevOps Guild | Package inference on-prem container, remote toggle, Helm/Compose manifests, scaling/offline guidance. |
|
||||
| 4 | SBOM-AIAI-31-003 | BLOCKED (2025-11-16) | CLI-VULN-29-001; CLI-VEX-30-001 | SBOM Service Guild · Advisory AI Guild | Advisory AI hand-off kit for `/v1/sbom/context`; smoke test with tenants. |
|
||||
| 5 | DOCS-AIAI-31-005/006/008/009 | BLOCKED | CLI-VULN-29-001; CLI-VEX-30-001; POLICY-ENGINE-31-001; DEVOPS-AIAI-31-001 | Docs Guild | CLI/policy/ops docs paused pending upstream artefacts. |
|
||||
| 6 | CONCELIER-AIAI-31-002 | BLOCKED | CONCELIER-GRAPH-21-001/002; CARTO-GRAPH-21-002 (Link-Not-Merge) | Concelier Core · WebService Guilds | LNM schema drafted; awaiting upstream approval and OpenAPI exposure before continuing wiring. |
|
||||
@@ -46,9 +46,9 @@
|
||||
| 10 | CONCELIER-ATTEST-73-001/002 | DONE (2025-11-22) | PREP-ATTEST-SCOPE-73; PREP-EVIDENCE-BDL-01 | Concelier Core · Evidence Locker Guild | Attestation inputs + transparency metadata; implement using frozen Evidence Bundle v1 and scope note (`docs/modules/evidence-locker/attestation-scope-note.md`). |
|
||||
| 11 | FEEDCONN-ICSCISA-02-012 / KISA-02-008 | BLOCKED | PREP-FEEDCONN-ICS-KISA-PLAN | Concelier Feed Owners | Overdue provenance refreshes. |
|
||||
| 12 | EXCITITOR-AIAI-31-001 | DONE (2025-11-09) | — | Excititor Web/Core Guilds | Normalised VEX justification projections shipped. |
|
||||
| 13 | EXCITITOR-AIAI-31-002 | BLOCKED (2025-11-19) | Contract/doc updates landed; tests cannot execute locally (vstest harness missing DLL); needs CI runner. | Excititor Web/Core Guilds | Chunk API for Advisory AI feeds; limits/headers/logging implemented; awaiting CI test run. |
|
||||
| 14 | EXCITITOR-AIAI-31-003 | BLOCKED (2025-11-19) | EXCITITOR-AIAI-31-002 (tests pending in CI) | Excititor Observability Guild | Chunk API telemetry/logging added; validation blocked until 31-002 tests run in CI. |
|
||||
| 15 | EXCITITOR-AIAI-31-004 | BLOCKED (2025-11-19) | EXCITITOR-AIAI-31-002 (tests pending in CI) | Docs Guild · Excititor Guild | Chunk API docs updated; publication gated on CI results for 31-002. |
|
||||
| 13 | EXCITITOR-AIAI-31-002 | TODO | Contract/doc updates landed; chunk tests now executing locally. | Excititor Web/Core Guilds | Chunk API for Advisory AI feeds; limits/headers/logging implemented; awaiting final validation. |
|
||||
| 14 | EXCITITOR-AIAI-31-003 | TODO | EXCITITOR-AIAI-31-002 progressing (tests runnable). | Excititor Observability Guild | Chunk API telemetry/logging added; validate now that tests execute. |
|
||||
| 15 | EXCITITOR-AIAI-31-004 | TODO | EXCITITOR-AIAI-31-002 progressing (tests runnable). | Docs Guild · Excititor Guild | Chunk API docs updated; publication to follow after 31-002 validation. |
|
||||
| 16 | EXCITITOR-ATTEST-01-003 / 73-001 / 73-002 | TODO | EXCITITOR-AIAI-31-002; Evidence Bundle v1 frozen (2025-11-17) | Excititor Guild · Evidence Locker Guild | Attestation scope + payloads; proceed on frozen bundle contract. |
|
||||
| 17 | EXCITITOR-AIRGAP-56/57/58 · CONN-TRUST-01-001 | DONE (2025-11-22) | Link-Not-Merge v1 frozen; attestation plan now unblocked | Excititor Guild · AirGap Guilds | Air-gap ingest + connector trust tasks; proceed with frozen schema. |
|
||||
| 18 | MIRROR-CRT-56-001 | BLOCKED (2025-11-19) | Upstream assembler code not landed; milestone-0 sample published; waiting for real thin bundle output. | Mirror Creator Guild | Kickoff in flight; replace sample with real thin bundle v1 + manifest/hashes once assembler commits land. |
|
||||
@@ -61,6 +61,10 @@
|
||||
## Execution Log
|
||||
| Date (UTC) | Update | Owner |
|
||||
| --- | --- | --- |
|
||||
| 2025-11-22 | Enabled Excititor chunk tests; fixed VexSignalSnapshot arg names and re-enabled VexEvidenceChunkServiceTests; ran `dotnet test src/Excititor/__Tests/StellaOps.Excititor.WebService.Tests/StellaOps.Excititor.WebService.Tests.csproj -c Release --filter EvidenceTelemetryTests` (pass, 2 tests). Marked EXCITITOR-AIAI-31-002/003/004 to TODO. | Implementer |
|
||||
| 2025-11-22 | Finalized DOCS-AIAI-31-004: published console guardrail guide using fixture captures, clarified publication checklist, and marked task DONE. | Implementer |
|
||||
| 2025-11-22 | Completed AIAI-31-008: added AdvisoryAI Dockerfile + compose + Helm chart (ops/advisory-ai/*), deployment guide (`docs/modules/advisory-ai/deployment.md`), and linked README; fixed guardrail test harness and ran `dotnet test src/AdvisoryAI/__Tests/StellaOps.AdvisoryAI.Tests/StellaOps.AdvisoryAI.Tests.csproj -c Release` (pass). | Implementer |
|
||||
| 2025-11-22 | Attempted `dotnet test src/Excititor/__Tests/StellaOps.Excititor.WebService.Tests/StellaOps.Excititor.WebService.Tests.csproj -c Release --filter VexEvidence`; build succeeded but no tests matched filter; EXCITITOR-AIAI-31-002/003/004 remain gated pending test discovery. | Implementer |
|
||||
| 2025-11-22 | Implemented advisory evidence attestation wiring: evidence endpoint accepts bundle/manifest paths, builds claims via EvidenceBundleAttestationBuilder; added tests and set defaults for evidence bundle root. | Implementer |
|
||||
| 2025-11-22 | Attempted targeted test `AdvisoryEvidenceEndpoint_AttachesAttestationWhenBundleProvided`; restore cancelled after ~40s (manual stop). Requires rerun with warm NuGet cache/CI. | Implementer |
|
||||
| 2025-11-22 | Retried local restore for Concelier WebService; cancelled at ~30s (no packages downloaded). Tests remain pending CI runner. | Implementer |
|
||||
|
||||
@@ -24,16 +24,16 @@
|
||||
## Delivery Tracker
|
||||
| # | Task ID | Status | Key dependency / next step | Owners | Task Definition |
|
||||
| --- | --- | --- | --- | --- | --- |
|
||||
| 1 | CONCELIER-LNM-21-001 | BLOCKED (2025-11-22) | Await Cartographer schema. | Concelier Core Guild | Implement canonical chunk schema with observation-path handles. |
|
||||
| 2 | CONCELIER-CACHE-22-001 | BLOCKED (2025-11-22) | Blocked on CONCELIER-LNM-21-001 canonical field ordering. | Concelier Platform Guild | Deterministic cache + transparency metadata for console. |
|
||||
| 3 | CONCELIER-MIRROR-23-001 | BLOCKED (2025-11-22) | Depends on CONCELIER-LNM-21-001 schema and Attestor mirror contract. | Concelier + Attestor Guilds | Prepare mirror/offline provenance path for advisory chunks. |
|
||||
| 1 | CONCELIER-LNM-21-001 | DONE (2025-11-22) | Await Cartographer schema. | Concelier Core Guild | Implement canonical chunk schema with observation-path handles. |
|
||||
| 2 | CONCELIER-CACHE-22-001 | TODO | LNM-21-001 delivered; define cache key order + transparency metadata. | Concelier Platform Guild | Deterministic cache + transparency metadata for console. |
|
||||
| 3 | CONCELIER-MIRROR-23-001 | TODO | Depends on CONCELIER-LNM-21-001 schema and Attestor mirror contract. | Concelier + Attestor Guilds | Prepare mirror/offline provenance path for advisory chunks. |
|
||||
|
||||
## Action Tracker
|
||||
| Focus | Action | Owner(s) | Due | Status |
|
||||
| --- | --- | --- | --- | --- |
|
||||
| Schema | Finalize canonical chunk schema | Concelier Core | 2025-11-18 | BLOCKED (await Cartographer schema) |
|
||||
| Cache | Define deterministic cache keys | Concelier Platform | 2025-11-19 | BLOCKED (waiting on canonical schema) |
|
||||
| Provenance | Mirror/attestor alignment | Concelier + Attestor | 2025-11-20 | BLOCKED (waiting on schema + Attestor mirror spec) |
|
||||
| Schema | Finalize canonical chunk schema | Concelier Core | 2025-11-18 | DONE (2025-11-22) |
|
||||
| Cache | Define deterministic cache keys | Concelier Platform | 2025-11-19 | TODO (schema available; proceed with key plan) |
|
||||
| Provenance | Mirror/attestor alignment | Concelier + Attestor | 2025-11-20 | TODO (await Attestor mirror spec; schema available) |
|
||||
|
||||
## Execution Log
|
||||
| Date (UTC) | Update | Owner |
|
||||
@@ -42,6 +42,8 @@
|
||||
| 2025-11-18 | WebService test rebuild emits DLL; full `dotnet test --no-build` and blame-hang runs stall (>8m, low CPU). Saved test list to `tmp/ws-tests.list`; hang investigation needed before progressing AIAI-31-002. | Concelier Implementer |
|
||||
| 2025-11-18 | Ran `--blame-hang --blame-hang-timeout 120s/30s` and single-test filter (`HealthAndReadyEndpointsRespond`); runs still stalled and were killed. Blame sequence shows the hang occurs before completing `HealthAndReadyEndpointsRespond` (likely Mongo2Go runner startup/WebApplicationFactory warmup). No TRX produced; sequence at `src/Concelier/__Tests/StellaOps.Concelier.WebService.Tests/TestResults/c6c5e036-d68b-402a-b676-d79b32c128c0/Sequence_bee8d66e585b4954809e99aed4b75a9f.xml`. | Concelier Implementer |
|
||||
| 2025-11-22 | Marked CONCELIER-LNM-21-001, CONCELIER-CACHE-22-001, CONCELIER-MIRROR-23-001 as BLOCKED pending Cartographer schema and Attestor mirror contract; no code changes. | Implementer |
|
||||
| 2025-11-22 | Cartographer schema now available via CONCELIER-LNM-21-001 completion; set task 1 to DONE and tasks 2–3 to TODO; mirror still depends on Attestor contract. | Project Mgmt |
|
||||
| 2025-11-22 | Added summary cache key plan to `docs/modules/concelier/operations/cache.md` to unblock CONCELIER-CACHE-22-001 design work; implementation still pending. | Docs |
|
||||
|
||||
## Decisions & Risks
|
||||
- Keep Concelier aggregation-only; no consensus merges.
|
||||
|
||||
@@ -26,12 +26,12 @@
|
||||
| P2 | PREP-CONCELIER-LNM-21-002-WAITING-ON-FINALIZE | DONE (2025-11-20) | Due 2025-11-21 · Accountable: Concelier Core Guild · Data Science Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Concelier Core Guild · Data Science Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Correlation rules + fixtures published at `docs/modules/concelier/linkset-correlation-21-002.md` with samples under `docs/samples/lnm/`. Downstream linkset builder can proceed. |
|
||||
| 1 | CONCELIER-GRAPH-21-001 | DONE | LNM sample fixtures with scopes/relationships added; observation/linkset query tests passing | Concelier Core Guild · Cartographer Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Extend SBOM normalization so relationships/scopes are stored as raw observation metadata with provenance pointers for graph joins. |
|
||||
| 2 | CONCELIER-GRAPH-21-002 | DONE (2025-11-22) | PREP-CONCELIER-GRAPH-21-002-PLATFORM-EVENTS-S | Concelier Core Guild · Scheduler Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Publish `sbom.observation.updated` events with tenant/context and advisory refs; facts only, no judgments. |
|
||||
| 3 | CONCELIER-GRAPH-24-101 | BLOCKED | Depends on 21-002 | Concelier WebService Guild (`src/Concelier/StellaOps.Concelier.WebService`) | `/advisories/summary` bundles observation/linkset metadata (aliases, confidence, conflicts) for graph overlays; upstream values intact. |
|
||||
| 3 | CONCELIER-GRAPH-24-101 | TODO | Depends on 21-002; needs API contract | Concelier WebService Guild (`src/Concelier/StellaOps.Concelier.WebService`) | `/advisories/summary` bundles observation/linkset metadata (aliases, confidence, conflicts) for graph overlays; upstream values intact. |
|
||||
| 4 | CONCELIER-GRAPH-28-102 | TODO | Depends on 24-101 | Concelier WebService Guild (`src/Concelier/StellaOps.Concelier.WebService`) | Evidence batch endpoints keyed by component sets with provenance/timestamps; no derived severity. |
|
||||
| 5 | CONCELIER-LNM-21-001 | DONE | Start of Link-Not-Merge chain | Concelier Core Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Define immutable `advisory_observations` model (per-source fields, version ranges, severity text, provenance metadata, tenant guards). |
|
||||
| 6 | CONCELIER-LNM-21-002 | DONE (2025-11-22) | PREP-CONCELIER-LNM-21-002-WAITING-ON-FINALIZE | Concelier Core Guild · Data Science Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Correlation pipelines output linksets with confidence + conflict markers, avoiding value collapse. |
|
||||
| 7 | CONCELIER-LNM-21-003 | DONE (2025-11-22) | Depends on 21-002 | Concelier Core Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Record disagreements (severity, CVSS, references) as structured conflict entries. |
|
||||
| 8 | CONCELIER-LNM-21-004 | BLOCKED | Depends on 21-003 | Concelier Core Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Remove legacy merge/dedup logic; add guardrails/tests to keep ingestion append-only; document linkset supersession. |
|
||||
| 8 | CONCELIER-LNM-21-004 | TODO | Depends on 21-003 | Concelier Core Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Remove legacy merge/dedup logic; add guardrails/tests to keep ingestion append-only; document linkset supersession. |
|
||||
| 9 | CONCELIER-LNM-21-005 | TODO | Depends on 21-004 | Concelier Core Guild · Platform Events Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Core`) | Emit `advisory.linkset.updated` events with delta descriptions + observation ids (tenant + provenance only). |
|
||||
| 10 | CONCELIER-LNM-21-101 | TODO | Depends on 21-005 | Concelier Storage Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Storage.Mongo`) | Provision Mongo collections (`advisory_observations`, `advisory_linksets`) with hashed shard keys, tenant indexes, TTL for ingest metadata. |
|
||||
| 11 | CONCELIER-LNM-21-102 | TODO | Depends on 21-101 | Concelier Storage Guild · DevOps Guild (`src/Concelier/__Libraries/StellaOps.Concelier.Storage.Mongo`) | Backfill legacy merged advisories; seed tombstones; provide rollback tooling for Offline Kit. |
|
||||
@@ -81,6 +81,8 @@
|
||||
| 2025-11-17 | Documented optional `confidence`/`conflicts` fields in LNM linkset schema and refreshed sample payload. | Concelier Core |
|
||||
| 2025-11-18 | Core library build now succeeds post schema updates; Core.Tests build outputs still missing DLL locally—test execution deferred to CI/warmed runner while continuing implementation. | Concelier Core |
|
||||
| 2025-11-22 | Restored Concelier Core/Storage via `dotnet restore --source local-nugets` and confirmed `dotnet build ...Concelier.Core.csproj` and `...Storage.Mongo.csproj` succeed locally; vstest still blocks test execution. | Concelier Core |
|
||||
| 2025-11-22 | Authored `/advisories/summary` API contract at `docs/modules/concelier/api/advisories-summary.md`; retained task status TODO pending implementation. | Docs |
|
||||
| 2025-11-22 | Dependencies satisfied (CONCELIER-GRAPH-21-002, CONCELIER-LNM-21-003); moved CONCELIER-GRAPH-24-101 and CONCELIER-LNM-21-004 to TODO. API contract for `/advisories/summary` and merger-retirement plan still needed. | Project Mgmt |
|
||||
|
||||
## Decisions & Risks
|
||||
- Link-Not-Merge v1 frozen 2025-11-17; schema captured in `docs/modules/concelier/link-not-merge-schema.md` (add-only evolution); fixtures pending for tasks 1–2, 5–15.
|
||||
@@ -93,8 +95,8 @@
|
||||
- Optional NATS transport worker added (feature-flagged); when enabled, outbox messages publish to stream/subject configured in `AdvisoryObservationEventPublisherOptions`. Ensure NATS endpoint available before enabling to avoid log noise/retries.
|
||||
- Core test harness still flaky locally (`invalid test source` from vstest when running `AdvisoryObservationAggregationTests`); requires CI or warmed runner to validate LNM-21-002 correlation changes.
|
||||
- Storage build/tests (Concelier.Storage.Mongo) also blocked on local runner (`invalid test source` / build hang). CI validation required before progressing to LNM-21-003.
|
||||
- CONCELIER-LNM-21-004 BLOCKED: removing canonical merge/dedup requires architect decision on retiring `CanonicalMerger` consumers (graph overlays, console summaries) and a migration/rollback plan; proceed after design sign-off.
|
||||
- CONCELIER-GRAPH-24-101 BLOCKED: needs API contract for `/advisories/summary` (payload shape, pagination, filters) and alignment with graph overlay consumers; awaiting WebService/API design sign-off.
|
||||
- CONCELIER-LNM-21-004 risk: removing canonical merge/dedup requires architect decision on retiring `CanonicalMerger` consumers (graph overlays, console summaries) and a migration/rollback plan; proceed after design sign-off.
|
||||
- CONCELIER-GRAPH-24-101 risk: API contract drafted at `docs/modules/concelier/api/advisories-summary.md`; implementation pending WebService wiring and consumer alignment.
|
||||
|
||||
## Next Checkpoints
|
||||
- Next LNM schema review: align with CARTO-GRAPH/LNM owners (date TBD); unblock tasks 1–2 and 5–15.
|
||||
|
||||
@@ -27,7 +27,7 @@
|
||||
| 7 | POLICY-OBS-53-001 | TODO | Depends on 52-001. | Policy · Evidence Locker Guild / `src/Policy/StellaOps.Policy.Engine` | Evaluation evidence bundles + manifests. |
|
||||
| 8 | POLICY-OBS-54-001 | TODO | Depends on 53-001. | Policy · Provenance Guild / `src/Policy/StellaOps.Policy.Engine` | DSSE attestations for evaluations. |
|
||||
| 9 | POLICY-OBS-55-001 | TODO | Depends on 54-001. | Policy · DevOps Guild / `src/Policy/StellaOps.Policy.Engine` | Incident mode sampling overrides. |
|
||||
| 10 | POLICY-RISK-66-001 | BLOCKED (2025-11-19) | PREP-POLICY-RISK-66-001-RISKPROFILE-LIBRARY-S | Risk Profile Schema Guild / `src/Policy/StellaOps.Policy.RiskProfile` | RiskProfile JSON schema + validator stubs. |
|
||||
| 10 | POLICY-RISK-66-001 | DONE (2025-11-22) | PREP-POLICY-RISK-66-001-RISKPROFILE-LIBRARY-S | Risk Profile Schema Guild / `src/Policy/StellaOps.Policy.RiskProfile` | RiskProfile JSON schema + validator stubs. |
|
||||
| 11 | POLICY-RISK-66-002 | TODO | Depends on 66-001. | Risk Profile Schema Guild / `src/Policy/StellaOps.Policy.RiskProfile` | Inheritance/merge + deterministic hashing. |
|
||||
| 12 | POLICY-RISK-66-003 | TODO | Depends on 66-002. | Policy · Risk Profile Schema Guild / `src/Policy/StellaOps.Policy.Engine` | Integrate RiskProfile into Policy Engine config. |
|
||||
| 13 | POLICY-RISK-66-004 | TODO | Depends on 66-003. | Policy · Risk Profile Schema Guild / `src/Policy/__Libraries/StellaOps.Policy` | Load/save RiskProfiles; validation diagnostics. |
|
||||
@@ -43,6 +43,8 @@
|
||||
| 2025-11-19 | Normalized to standard template and renamed from `SPRINT_127_policy_reasoning.md` to `SPRINT_0127_0001_0001_policy_reasoning.md`; content preserved. | Implementer |
|
||||
| 2025-11-19 | Attempted POLICY-RISK-66-001; blocked because `src/Policy/StellaOps.Policy.RiskProfile` lacks a project/scaffold to host schema + validators. Needs project creation + contract placement guidance. | Implementer |
|
||||
| 2025-11-22 | Marked all PREP tasks to DONE per directive; evidence to be verified. | Project Mgmt |
|
||||
| 2025-11-22 | Implemented RiskProfile schema + validator and tests; added project to solution; set POLICY-RISK-66-001 to DONE. | Implementer |
|
||||
| 2025-11-22 | Unblocked POLICY-RISK-66-001 after prep completion; status → TODO. | Project Mgmt |
|
||||
|
||||
## Decisions & Risks
|
||||
- Reachability inputs (80-001) prerequisite; not yet delivered.
|
||||
|
||||
@@ -403,13 +403,13 @@
|
||||
| CONCELIER-CORE-AOC-19-013 | TODO | | SPRINT_112_concelier_i | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Expand smoke/e2e suites so Authority tokens + tenant headers are mandatory for ingest/read paths (including the new provenance endpoint). Must assert no merge-side effects and that provenance anchors always round-trip. | Must reference AOC guardrails from docs | AGCN0101 |
|
||||
| CONCELIER-DOCS-0001 | DONE | 2025-11-05 | SPRINT_317_docs_modules_concelier | Docs Guild | docs/modules/concelier | Validate that `docs/modules/concelier/README.md` reflects the latest release notes and aggregation toggles. | Reference (baseline) | CCDO0101 |
|
||||
| CONCELIER-ENG-0001 | TODO | | SPRINT_317_docs_modules_concelier | Module Team · Concelier Guild | docs/modules/concelier | Cross-check implementation plan milestones against `/docs/implplan/SPRINT_*.md` and update module readiness checkpoints. | Wait for CCPR0101 validation | CCDO0101 |
|
||||
| CONCELIER-GRAPH-21-001 | BLOCKED | 2025-10-27 | SPRINT_113_concelier_ii | Concelier Core · Cartographer Guilds | src/Concelier/__Libraries/StellaOps.Concelier.Core | Extend SBOM normalization so every relationship (depends_on, contains, provides) and scope tag is captured as raw observation metadata with provenance pointers; Cartographer can then join SBOM + advisory facts without Concelier inferring impact. | Waiting on Cartographer schema (052_CAGR0101) | AGCN0101 |
|
||||
| CONCELIER-GRAPH-21-002 | BLOCKED | 2025-10-27 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Publish `sbom.observation.updated` events whenever new SBOM versions arrive, including tenant/context metadata and advisory references—never send judgments, only facts. Depends on CONCELIER-GRAPH-21-001; blocked pending Platform Events/Scheduler contract + event publisher. | Depends on #5 outputs | AGCN0101 |
|
||||
| CONCELIER-GRAPH-21-001 | DONE | 2025-11-18 | SPRINT_113_concelier_ii | Concelier Core · Cartographer Guilds | src/Concelier/__Libraries/StellaOps.Concelier.Core | Extend SBOM normalization so every relationship (depends_on, contains, provides) and scope tag is captured as raw observation metadata with provenance pointers; Cartographer can then join SBOM + advisory facts without Concelier inferring impact. | Waiting on Cartographer schema (052_CAGR0101) | AGCN0101 |
|
||||
| CONCELIER-GRAPH-21-002 | DONE | 2025-11-22 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Publish `sbom.observation.updated` events whenever new SBOM versions arrive, including tenant/context metadata and advisory references—never send judgments, only facts. Depends on CONCELIER-GRAPH-21-001; blocked pending Platform Events/Scheduler contract + event publisher. | Depends on #5 outputs | AGCN0101 |
|
||||
| CONCELIER-GRAPH-24-101 | TODO | | SPRINT_113_concelier_ii | Concelier WebService Guild | src/Concelier/StellaOps.Concelier.WebService | Provide `/advisories/summary` responses that bundle observation/linkset metadata (aliases, confidence, conflicts) for graph overlays while keeping upstream values intact. Depends on CONCELIER-GRAPH-21-002. | Wait for CAGR0101 + storage migrations | CCGH0101 |
|
||||
| CONCELIER-GRAPH-28-102 | TODO | | SPRINT_113_concelier_ii | Concelier WebService Guild | src/Concelier/StellaOps.Concelier.WebService | Add batch fetch endpoints keyed by component sets so graph tooltips can pull raw observations/linksets efficiently; include provenance + timestamps but no derived severity. Depends on CONCELIER-GRAPH-24-101. | Depends on #1 | CCGH0101 |
|
||||
| CONCELIER-LNM-21-001 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Define the immutable `advisory_observations` model (per-source fields, version ranges, severity text, provenance metadata, tenant guards) so every ingestion path records raw statements without merge artifacts. | Needs Link-Not-Merge approval (005_ATLN0101) | AGCN0101 |
|
||||
| CONCELIER-LNM-21-002 | BLOCKED | 2025-11-18 | SPRINT_113_concelier_ii | Concelier Core Guild · Data Science Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Implement correlation pipelines (alias graph, purl overlap, CVSS vector compare) that output linksets with confidence scores + conflict markers, never collapsing conflicting facts into single values. Depends on CONCELIER-LNM-21-001. | Depends on #7 for precedence rules | AGCN0101 |
|
||||
| CONCELIER-LNM-21-003 | BLOCKED | 2025-11-18 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Record disagreements (severity, CVSS, references) on linksets as structured conflict entries so consumers can reason about divergence without Concelier resolving it. Depends on CONCELIER-LNM-21-002. | Requires #8 heuristics | AGCN0101 |
|
||||
| CONCELIER-LNM-21-001 | DONE | 2025-11-17 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Define the immutable `advisory_observations` model (per-source fields, version ranges, severity text, provenance metadata, tenant guards) so every ingestion path records raw statements without merge artifacts. | Needs Link-Not-Merge approval (005_ATLN0101) | AGCN0101 |
|
||||
| CONCELIER-LNM-21-002 | DONE | 2025-11-22 | SPRINT_113_concelier_ii | Concelier Core Guild · Data Science Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Implement correlation pipelines (alias graph, purl overlap, CVSS vector compare) that output linksets with confidence scores + conflict markers, never collapsing conflicting facts into single values. Depends on CONCELIER-LNM-21-001. | Depends on #7 for precedence rules | AGCN0101 |
|
||||
| CONCELIER-LNM-21-003 | DONE | 2025-11-22 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Record disagreements (severity, CVSS, references) on linksets as structured conflict entries so consumers can reason about divergence without Concelier resolving it. Depends on CONCELIER-LNM-21-002. | Requires #8 heuristics | AGCN0101 |
|
||||
| CONCELIER-LNM-21-004 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Delete legacy merge/dedup logic, add guardrails/tests to keep ingestion append-only, and document how linksets supersede the old merge outputs. Depends on CONCELIER-LNM-21-003. | Depends on #9 | AGCN0101 |
|
||||
| CONCELIER-LNM-21-005 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Emit `advisory.linkset.updated` events containing delta descriptions + observation ids so downstream evaluators can subscribe deterministically. Depends on CONCELIER-LNM-21-004. | Requires CCLN0101 store changes | CCCO0101 |
|
||||
| CONCELIER-LNM-21-101 | TODO | | SPRINT_113_concelier_ii | Concelier Storage Guild | src/Concelier/__Libraries/StellaOps.Concelier.Storage.Mongo | Provision the Mongo collections (`advisory_observations`, `advisory_linksets`) with hashed shard keys, tenant indexes, and TTL for ingest metadata to support Link-Not-Merge at scale. Depends on CONCELIER-LNM-21-005. | Wait for schema freeze | CCLN0101 |
|
||||
@@ -1438,7 +1438,7 @@
|
||||
| POLICY-OBS-55-001 | TODO | | SPRINT_127_policy_reasoning | Policy Guild · DevOps Guild | src/Policy/StellaOps.Policy.Engine | Implement incident mode sampling overrides | POLICY-OBS-54-001 | PLOB0101 |
|
||||
| POLICY-READINESS-0001 | TODO | | SPRINT_325_docs_modules_policy | Policy Guild (docs/modules/policy) | docs/modules/policy | Capture policy module readiness checklist aligned with current sprint goals. | | |
|
||||
| POLICY-READINESS-0002 | TODO | | SPRINT_325_docs_modules_policy | Policy Guild (docs/modules/policy) | docs/modules/policy | Track outstanding prerequisites/risk items for policy releases and mirror into sprint updates. | | |
|
||||
| POLICY-RISK-66-001 | TODO | | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Develop initial JSON Schema for RiskProfile (signals, transforms, weights, severity, overrides) with validator stubs | | |
|
||||
| POLICY-RISK-66-001 | DONE | 2025-11-22 | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Develop initial JSON Schema for RiskProfile (signals, transforms, weights, severity, overrides) with validator stubs | | |
|
||||
| POLICY-RISK-66-002 | TODO | | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Implement inheritance/merge logic with conflict detection and deterministic content hashing | POLICY-RISK-66-001 | |
|
||||
| POLICY-RISK-66-003 | TODO | | SPRINT_127_policy_reasoning | Policy Guild, Risk Profile Schema Guild / src/Policy/StellaOps.Policy.Engine | src/Policy/StellaOps.Policy.Engine | Integrate RiskProfile schema into Policy Engine configuration, ensuring validation and default profile deployment | POLICY-RISK-66-002 | |
|
||||
| POLICY-RISK-66-004 | TODO | | SPRINT_127_policy_reasoning | Policy Guild, Risk Profile Schema Guild / src/Policy/__Libraries/StellaOps.Policy | src/Policy/__Libraries/StellaOps.Policy | Extend Policy libraries to load/save RiskProfile documents, compute content hashes, and surface validation diagnostics | POLICY-RISK-66-003 | |
|
||||
@@ -2615,8 +2615,8 @@
|
||||
| CONCELIER-GRAPH-21-002 | BLOCKED | 2025-10-27 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Publish `sbom.observation.updated` events whenever new SBOM versions arrive, including tenant/context metadata and advisory references—never send judgments, only facts. Depends on CONCELIER-GRAPH-21-001. | Depends on #5 outputs | AGCN0101 |
|
||||
| CONCELIER-GRAPH-24-101 | TODO | | SPRINT_113_concelier_ii | Concelier WebService Guild | src/Concelier/StellaOps.Concelier.WebService | Provide `/advisories/summary` responses that bundle observation/linkset metadata (aliases, confidence, conflicts) for graph overlays while keeping upstream values intact. Depends on CONCELIER-GRAPH-21-002. | Wait for CAGR0101 + storage migrations | CCGH0101 |
|
||||
| CONCELIER-GRAPH-28-102 | TODO | | SPRINT_113_concelier_ii | Concelier WebService Guild | src/Concelier/StellaOps.Concelier.WebService | Add batch fetch endpoints keyed by component sets so graph tooltips can pull raw observations/linksets efficiently; include provenance + timestamps but no derived severity. Depends on CONCELIER-GRAPH-24-101. | Depends on #1 | CCGH0101 |
|
||||
| CONCELIER-LNM-21-001 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Define the immutable `advisory_observations` model (per-source fields, version ranges, severity text, provenance metadata, tenant guards) so every ingestion path records raw statements without merge artifacts. | Needs Link-Not-Merge approval (005_ATLN0101) | AGCN0101 |
|
||||
| CONCELIER-LNM-21-002 | BLOCKED | 2025-11-18 | SPRINT_113_concelier_ii | Concelier Core Guild · Data Science Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Implement correlation pipelines (alias graph, purl overlap, CVSS vector compare) that output linksets with confidence scores + conflict markers, never collapsing conflicting facts into single values. Depends on CONCELIER-LNM-21-001. | Depends on #7 for precedence rules | AGCN0101 |
|
||||
| CONCELIER-LNM-21-001 | DONE | 2025-11-17 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Define the immutable `advisory_observations` model (per-source fields, version ranges, severity text, provenance metadata, tenant guards) so every ingestion path records raw statements without merge artifacts. | Needs Link-Not-Merge approval (005_ATLN0101) | AGCN0101 |
|
||||
| CONCELIER-LNM-21-002 | DONE | 2025-11-22 | SPRINT_113_concelier_ii | Concelier Core Guild · Data Science Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Implement correlation pipelines (alias graph, purl overlap, CVSS vector compare) that output linksets with confidence scores + conflict markers, never collapsing conflicting facts into single values. Depends on CONCELIER-LNM-21-001. | Depends on #7 for precedence rules | AGCN0101 |
|
||||
| CONCELIER-LNM-21-003 | BLOCKED | 2025-11-18 | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Record disagreements (severity, CVSS, references) on linksets as structured conflict entries so consumers can reason about divergence without Concelier resolving it. Depends on CONCELIER-LNM-21-002. | Requires #8 heuristics | AGCN0101 |
|
||||
| CONCELIER-LNM-21-004 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Delete legacy merge/dedup logic, add guardrails/tests to keep ingestion append-only, and document how linksets supersede the old merge outputs. Depends on CONCELIER-LNM-21-003. | Depends on #9 | AGCN0101 |
|
||||
| CONCELIER-LNM-21-005 | TODO | | SPRINT_113_concelier_ii | Concelier Core Guild | src/Concelier/__Libraries/StellaOps.Concelier.Core | Emit `advisory.linkset.updated` events containing delta descriptions + observation ids so downstream evaluators can subscribe deterministically. Depends on CONCELIER-LNM-21-004. | Requires CCLN0101 store changes | CCCO0101 |
|
||||
@@ -3648,7 +3648,7 @@
|
||||
| POLICY-OBS-55-001 | TODO | | SPRINT_127_policy_reasoning | Policy Guild · DevOps Guild | src/Policy/StellaOps.Policy.Engine | Implement incident mode sampling overrides | POLICY-OBS-54-001 | PLOB0101 |
|
||||
| POLICY-READINESS-0001 | TODO | | SPRINT_325_docs_modules_policy | Policy Guild (docs/modules/policy) | docs/modules/policy | Capture policy module readiness checklist aligned with current sprint goals. | | |
|
||||
| POLICY-READINESS-0002 | TODO | | SPRINT_325_docs_modules_policy | Policy Guild (docs/modules/policy) | docs/modules/policy | Track outstanding prerequisites/risk items for policy releases and mirror into sprint updates. | | |
|
||||
| POLICY-RISK-66-001 | TODO | | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Develop initial JSON Schema for RiskProfile (signals, transforms, weights, severity, overrides) with validator stubs | | |
|
||||
| POLICY-RISK-66-001 | DONE | 2025-11-22 | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Develop initial JSON Schema for RiskProfile (signals, transforms, weights, severity, overrides) with validator stubs | | |
|
||||
| POLICY-RISK-66-002 | TODO | | SPRINT_127_policy_reasoning | Risk Profile Schema Guild / src/Policy/StellaOps.Policy.RiskProfile | src/Policy/StellaOps.Policy.RiskProfile | Implement inheritance/merge logic with conflict detection and deterministic content hashing | POLICY-RISK-66-001 | |
|
||||
| POLICY-RISK-66-003 | TODO | | SPRINT_127_policy_reasoning | Policy Guild, Risk Profile Schema Guild / src/Policy/StellaOps.Policy.Engine | src/Policy/StellaOps.Policy.Engine | Integrate RiskProfile schema into Policy Engine configuration, ensuring validation and default profile deployment | POLICY-RISK-66-002 | |
|
||||
| POLICY-RISK-66-004 | TODO | | SPRINT_127_policy_reasoning | Policy Guild, Risk Profile Schema Guild / src/Policy/__Libraries/StellaOps.Policy | src/Policy/__Libraries/StellaOps.Policy | Extend Policy libraries to load/save RiskProfile documents, compute content hashes, and surface validation diagnostics | POLICY-RISK-66-003 | |
|
||||
|
||||
@@ -27,9 +27,9 @@ Advisory AI is the retrieval-augmented assistant that synthesizes advisory and V
|
||||
- Guardrail behaviour, blocked phrases, and operational alerts are detailed in `/docs/security/assistant-guardrails.md`.
|
||||
|
||||
## Deployment & configuration
|
||||
- **Containers:** `advisory-ai-web` fronts the API/cache while `advisory-ai-worker` drains the queue and executes prompts. Both containers mount a shared RWX volume providing `/var/lib/advisory-ai/{queue,plans,outputs}`.
|
||||
- **Remote inference toggle:** Set `ADVISORYAI__AdvisoryAI__Inference__Mode=Remote` to send sanitized prompts to an external inference tier. Provide `ADVISORYAI__AdvisoryAI__Inference__Remote__BaseAddress` (and optional `...ApiKey`) to complete the circuit; failures fall back to the sanitized prompt and surface `inference.fallback_*` metadata.
|
||||
- **Helm/Compose:** Bundled manifests wire the SBOM base address, queue/plan/output directories, and inference options via the `AdvisoryAI` configuration section. Helm expects a PVC named `stellaops-advisory-ai-data`. Compose creates named volumes so the worker and web instances share deterministic state.
|
||||
- **Containers:** `advisory-ai-web` fronts the API/cache while `advisory-ai-worker` drains the queue and executes prompts. Both containers mount a shared RWX volume providing `/app/data/{queue,plans,outputs}` (defaults; configurable via `ADVISORYAI__STORAGE__*`).
|
||||
- **Remote inference toggle:** Set `ADVISORYAI__INFERENCE__MODE=Remote` to send sanitized prompts to an external inference tier. Provide `ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS` (and optional `...__APIKEY`, `...__TIMEOUT`) to complete the circuit; failures fall back to the sanitized prompt and surface `inference.fallback_*` metadata.
|
||||
- **Helm/Compose:** Packaged manifests live under `ops/advisory-ai/` and wire SBOM base address, queue/plan/output directories, and inference options. Helm defaults to `emptyDir` with optional PVC; Compose creates named volumes so worker and web instances share deterministic state. See `docs/modules/advisory-ai/deployment.md` for commands.
|
||||
|
||||
## CLI usage
|
||||
- `stella advise run <summary|conflict|remediation> --advisory-key <id> [--artifact-id id] [--artifact-purl purl] [--policy-version v] [--profile profile] [--section name] [--force-refresh] [--timeout seconds]`
|
||||
|
||||
58
docs/modules/advisory-ai/deployment.md
Normal file
58
docs/modules/advisory-ai/deployment.md
Normal file
@@ -0,0 +1,58 @@
|
||||
# AdvisoryAI Deployment Guide (AIAI-31-008)
|
||||
|
||||
This guide covers packaging AdvisoryAI for on-prem / offline environments, toggling remote inference, and recommended scaling settings.
|
||||
|
||||
## Artifacts
|
||||
- Dockerfile: `ops/advisory-ai/Dockerfile` (multi-role build for WebService / Worker).
|
||||
- Local compose: `ops/advisory-ai/docker-compose.advisoryai.yaml` (web + worker, shared data volume).
|
||||
- Helm chart: `ops/advisory-ai/helm/` (web Deployment + optional worker Deployment + Service + PVC stub).
|
||||
|
||||
## Build and run locally
|
||||
```bash
|
||||
# Build images
|
||||
make advisoryai-web: docker build -f ops/advisory-ai/Dockerfile -t stellaops-advisoryai-web:dev \
|
||||
--build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.WebService/StellaOps.AdvisoryAI.WebService.csproj \
|
||||
--build-arg APP_DLL=StellaOps.AdvisoryAI.WebService.dll .
|
||||
make advisoryai-worker: docker build -f ops/advisory-ai/Dockerfile -t stellaops-advisoryai-worker:dev \
|
||||
--build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.Worker/StellaOps.AdvisoryAI.Worker.csproj \
|
||||
--build-arg APP_DLL=StellaOps.AdvisoryAI.Worker.dll .
|
||||
|
||||
# Compose (offline friendly)
|
||||
docker compose -f ops/advisory-ai/docker-compose.advisoryai.yaml up -d --build
|
||||
```
|
||||
|
||||
## Remote inference toggle
|
||||
- Default: `ADVISORYAI__INFERENCE__MODE=Local` (fully offline).
|
||||
- Remote: set
|
||||
- `ADVISORYAI__INFERENCE__MODE=Remote`
|
||||
- `ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS=https://inference.example.com`
|
||||
- `ADVISORYAI__INFERENCE__REMOTE__ENDPOINT=/v1/inference`
|
||||
- `ADVISORYAI__INFERENCE__REMOTE__APIKEY=<token>`
|
||||
- Optional: `ADVISORYAI__INFERENCE__REMOTE__TIMEOUT=00:00:30`
|
||||
- Guardrails still enforced locally (see `ADVISORYAI__GUARDRAILS__*` options); keep secrets in mounted env/secret rather than images.
|
||||
|
||||
## Storage & persistence
|
||||
- File-system queue/cache/output paths default to `/app/data/{queue,plans,outputs}` and are pre-created at startup.
|
||||
- Compose mounts `advisoryai-data` volume; Helm uses `emptyDir` by default or a PVC when `storage.persistence.enabled=true`.
|
||||
- In sealed/air-gapped mode, mount guardrail lists/policy knobs under `/app/etc` and point env vars accordingly.
|
||||
|
||||
## Scaling guidance
|
||||
- WebService: start with 1 replica, scale horizontally by CPU (tokenization) or queue depth; set `ADVISORYAI__QUEUE__DIRECTORYPATH` to a shared PVC for multi-replica web+worker.
|
||||
- Worker: scale independently; use `worker.replicas` in Helm or add `--scale advisoryai-worker=N` in compose. Workers are CPU-bound; pin via `resources.requests/limits`.
|
||||
- Set rate limiter headers: add `X-StellaOps-Client` per caller to avoid shared buckets.
|
||||
|
||||
## Offline posture
|
||||
- Images build from source without external runtime downloads; keep `ADVISORYAI__INFERENCE__MODE=Local` to stay offline.
|
||||
- For registry-mirrored environments, push `stellaops-advisoryai-web` and `stellaops-advisoryai-worker` to the allowed registry and reference via Helm `image.repository`.
|
||||
- Disable OTEL exporters unless explicitly permitted; logs remain structured JSON to stdout.
|
||||
|
||||
## Air-gap checklist
|
||||
- Remote inference disabled (or routed through approved enclave).
|
||||
- Guardrail phrase list mounted read-only.
|
||||
- Data PVC scoped per tenant/project if multi-tenant; enforce scope via `X-StellaOps-Scopes`.
|
||||
- Validate that `/app/data` volume has backup/retention policy; cache pruning handled by storage options.
|
||||
|
||||
## Deliverables mapping
|
||||
- Compose + Dockerfile satisfy on-prem packaging.
|
||||
- Helm chart provides cluster deployment with persistence and remote toggle.
|
||||
- This guide documents scaling/offline posture required by Sprint 0110 AIAI-31-008.
|
||||
76
docs/modules/concelier/api/advisories-summary.md
Normal file
76
docs/modules/concelier/api/advisories-summary.md
Normal file
@@ -0,0 +1,76 @@
|
||||
# `/advisories/summary` API (draft v1)
|
||||
|
||||
Status: draft; aligns with LNM v1 (frozen 2025-11-17) and observation/linkset models already shipped in Concelier Core.
|
||||
|
||||
## Intent
|
||||
- Provide graph overlays and consoles a deterministic summary of observations and linksets without derived verdicts.
|
||||
- Preserve provenance and tenant isolation; results are stable for a given tenant + filter set.
|
||||
|
||||
## Request
|
||||
- Method: `GET`
|
||||
- Path: `/advisories/summary`
|
||||
- Headers:
|
||||
- `X-Stella-Tenant`: required.
|
||||
- `X-Stella-Request-Id`: optional for tracing.
|
||||
- Query parameters:
|
||||
- `purl` (optional, repeatable) — filter by component coordinates.
|
||||
- `alias` (optional, repeatable) — advisory aliases (CVE, GHSA, vendor IDs); case-insensitive; normalized and sorted server-side.
|
||||
- `source` (optional, repeatable) — upstream source identifiers.
|
||||
- `confidence_gte` (optional, decimal 0–1) — minimum linkset confidence.
|
||||
- `conflicts_only` (optional, bool) — when `true`, return only summaries with conflicts present.
|
||||
- `after` (optional, cursor) — opaque, tenant-scoped cursor for pagination.
|
||||
- `take` (optional, int, default 100, max 500) — page size.
|
||||
- `sort` (optional, enum: `advisory`, `observedAt`, default `advisory`) — always ascending and stable.
|
||||
|
||||
## Response (200)
|
||||
```json
|
||||
{
|
||||
"meta": {
|
||||
"tenant": "acme",
|
||||
"count": 2,
|
||||
"next": "opaque-cursor-or-null",
|
||||
"sort": "advisory"
|
||||
},
|
||||
"items": [
|
||||
{
|
||||
"advisoryKey": "cve-2024-1234",
|
||||
"aliases": ["GHSA-xxxx-yyyy", "CVE-2024-1234"],
|
||||
"source": "nvd",
|
||||
"observedAt": "2025-11-22T15:04:05Z",
|
||||
"linksetId": "ls_01H9A8...",
|
||||
"confidence": 0.82,
|
||||
"conflicts": [
|
||||
{ "field": "severity", "codes": ["severity-mismatch"], "sources": ["nvd", "vendor"] }
|
||||
],
|
||||
"counts": {
|
||||
"observations": 3,
|
||||
"conflictFields": 1
|
||||
},
|
||||
"provenance": {
|
||||
"observationIds": ["obs_01H9...", "obs_01H9..."],
|
||||
"schema": "lnm-1.0"
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
```
|
||||
- Ordering: stable by `sort` then `advisoryKey` then `linksetId`.
|
||||
- No derived verdicts or merged severity values; conflicts are emitted as structured markers only.
|
||||
|
||||
## Errors
|
||||
- `400` `ERR_AOC_001`: missing/invalid tenant or unsupported filter.
|
||||
- `400` `ERR_AOC_006`: `take` exceeds max or invalid cursor.
|
||||
- `401/403`: auth/tenant scope failures.
|
||||
- `429`: if per-tenant rate limits enforced (surface retry headers).
|
||||
|
||||
## Determinism & caching
|
||||
- Response depends solely on tenant + normalized filters + underlying linksets; no clock-based variation beyond `observedAt` from stored records.
|
||||
- Cache key (for optional summary cache):
|
||||
- `tenant|purls(sorted)|aliases(sorted)|sources(sorted)|confidence_gte|conflicts_only|sort|take|after`
|
||||
- Return transparency headers: `X-Stella-Cache-Key` (sha256-16), `X-Stella-Cache-Hit`, `X-Stella-Cache-Ttl`.
|
||||
- Cache TTL should default to 0 (disabled) until validated; determinism required for hits.
|
||||
|
||||
## Notes
|
||||
- Cursor must encode the final sort tuple (`advisoryKey`, `linksetId`, `observedAt`) to keep pagination stable when new data arrives.
|
||||
- All string comparisons for filters are case-insensitive; server normalizes to lower-case and sorts before query execution.
|
||||
- Conflicts mirror LNM conflict codes already produced by Core (no new codes introduced here).
|
||||
@@ -31,6 +31,21 @@ Status: draft · aligned with LNM v1 (frozen 2025-11-17)
|
||||
- Timestamps use UTC ticks; content hashes retain upstream digest prefix (e.g., `sha256:`) to distinguish sources.
|
||||
- Cache TTL defaults to 0 (disabled) unless configured; safe for offline/air-gapped deployments.
|
||||
|
||||
## Summary cache (for `/advisories/summary`)
|
||||
- Optional, disabled by default; enables deterministic paging for graph/console overlays.
|
||||
- Key material (pipe-delimited, normalized/sorted where applicable):
|
||||
- `tenant`
|
||||
- `purls[]` (sorted)
|
||||
- `aliases[]` (sorted, lower-case)
|
||||
- `sources[]` (sorted, lower-case)
|
||||
- `confidence_gte`
|
||||
- `conflicts_only`
|
||||
- `sort`
|
||||
- `take`
|
||||
- `afterCursor` (opaque; includes last sort tuple)
|
||||
- Transparency headers (if enabled): `X-Stella-Cache-Key` (sha256-16), `X-Stella-Cache-Hit`, `X-Stella-Cache-Ttl`.
|
||||
- Ordering for cacheable results: `sort` (advisory|observedAt), then `advisoryKey`, then `linksetId`; cursor encodes the tuple to keep pagination stable when new linksets land.
|
||||
|
||||
## Testing notes
|
||||
- Unit coverage: `AdvisoryChunkCacheKeyTests` (ordering, filter casing) and `AdvisoryChunkBuilderTests` (observationPath pointers influence chunk IDs).
|
||||
- Integration tests should assert headers when cache is enabled; disable cache for tests that assert body-only determinism.
|
||||
@@ -38,3 +53,4 @@ Status: draft · aligned with LNM v1 (frozen 2025-11-17)
|
||||
## TODOs / follow-ups
|
||||
- Add integration test that exercises a cache hit path and validates transparency headers.
|
||||
- Document cache configuration knobs in `appsettings.*` once finalized.
|
||||
- Add summary cache integration test once `/advisories/summary` endpoint is implemented.
|
||||
|
||||
47
ops/advisory-ai/Dockerfile
Normal file
47
ops/advisory-ai/Dockerfile
Normal file
@@ -0,0 +1,47 @@
|
||||
# syntax=docker/dockerfile:1.7-labs
|
||||
|
||||
# StellaOps AdvisoryAI – multi-role container build
|
||||
# Build arg PROJECT selects WebService or Worker; defaults to WebService.
|
||||
# Example builds:
|
||||
# docker build -f ops/advisory-ai/Dockerfile -t stellaops-advisoryai-web \
|
||||
# --build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.WebService/StellaOps.AdvisoryAI.WebService.csproj \
|
||||
# --build-arg APP_DLL=StellaOps.AdvisoryAI.WebService.dll .
|
||||
# docker build -f ops/advisory-ai/Dockerfile -t stellaops-advisoryai-worker \
|
||||
# --build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.Worker/StellaOps.AdvisoryAI.Worker.csproj \
|
||||
# --build-arg APP_DLL=StellaOps.AdvisoryAI.Worker.dll .
|
||||
|
||||
ARG SDK_IMAGE=mcr.microsoft.com/dotnet/nightly/sdk:10.0
|
||||
ARG RUNTIME_IMAGE=gcr.io/distroless/dotnet/aspnet:latest
|
||||
ARG PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.WebService/StellaOps.AdvisoryAI.WebService.csproj
|
||||
ARG APP_DLL=StellaOps.AdvisoryAI.WebService.dll
|
||||
|
||||
FROM ${SDK_IMAGE} AS build
|
||||
WORKDIR /src
|
||||
|
||||
COPY . .
|
||||
|
||||
# Restore only AdvisoryAI graph to keep build smaller.
|
||||
RUN dotnet restore ${PROJECT}
|
||||
|
||||
RUN dotnet publish ${PROJECT} \
|
||||
-c Release \
|
||||
-o /app/publish \
|
||||
/p:UseAppHost=false
|
||||
|
||||
FROM ${RUNTIME_IMAGE} AS runtime
|
||||
WORKDIR /app
|
||||
|
||||
ENV ASPNETCORE_URLS=http://0.0.0.0:8080 \
|
||||
DOTNET_SYSTEM_GLOBALIZATION_INVARIANT=true \
|
||||
ADVISORYAI__STORAGE__PLANCACHEDIRECTORY=/app/data/plans \
|
||||
ADVISORYAI__STORAGE__OUTPUTDIRECTORY=/app/data/outputs \
|
||||
ADVISORYAI__QUEUE__DIRECTORYPATH=/app/data/queue \
|
||||
ADVISORYAI__INFERENCE__MODE=Local
|
||||
|
||||
COPY --from=build /app/publish ./
|
||||
|
||||
# Writable mount for queue/cache/output. Guardrail/guardrails can also be mounted under /app/etc.
|
||||
VOLUME ["/app/data", "/app/etc"]
|
||||
|
||||
EXPOSE 8080
|
||||
ENTRYPOINT ["dotnet", "${APP_DLL}"]
|
||||
47
ops/advisory-ai/README.md
Normal file
47
ops/advisory-ai/README.md
Normal file
@@ -0,0 +1,47 @@
|
||||
# AdvisoryAI packaging (AIAI-31-008)
|
||||
|
||||
Artifacts delivered for on-prem / air-gapped deployment:
|
||||
|
||||
- `ops/advisory-ai/Dockerfile` builds WebService and Worker images (multi-role via `PROJECT`/`APP_DLL` args).
|
||||
- `ops/advisory-ai/docker-compose.advisoryai.yaml` runs WebService + Worker with shared data volume; ships remote inference toggle envs.
|
||||
- `ops/advisory-ai/helm/` provides a minimal chart (web + worker) with storage mounts, optional PVC, and remote inference settings.
|
||||
|
||||
## Build images
|
||||
```bash
|
||||
# WebService
|
||||
docker build -f ops/advisory-ai/Dockerfile \
|
||||
-t stellaops-advisoryai-web:dev \
|
||||
--build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.WebService/StellaOps.AdvisoryAI.WebService.csproj \
|
||||
--build-arg APP_DLL=StellaOps.AdvisoryAI.WebService.dll .
|
||||
|
||||
# Worker
|
||||
docker build -f ops/advisory-ai/Dockerfile \
|
||||
-t stellaops-advisoryai-worker:dev \
|
||||
--build-arg PROJECT=src/AdvisoryAI/StellaOps.AdvisoryAI.Worker/StellaOps.AdvisoryAI.Worker.csproj \
|
||||
--build-arg APP_DLL=StellaOps.AdvisoryAI.Worker.dll .
|
||||
```
|
||||
|
||||
## Local/offline compose
|
||||
```bash
|
||||
cd ops/advisory-ai
|
||||
docker compose -f docker-compose.advisoryai.yaml up -d --build
|
||||
```
|
||||
- Set `ADVISORYAI__INFERENCE__MODE=Remote` plus `ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS`/`APIKEY` to offload inference.
|
||||
- Default mode is Local (offline-friendly). Queue/cache/output live under `/app/data` (binds to `advisoryai-data` volume).
|
||||
|
||||
## Helm (cluster)
|
||||
```bash
|
||||
helm upgrade --install advisoryai ops/advisory-ai/helm \
|
||||
--set image.repository=stellaops-advisoryai-web \
|
||||
--set image.tag=dev \
|
||||
--set inference.mode=Local
|
||||
```
|
||||
- Enable remote inference: `--set inference.mode=Remote --set inference.remote.baseAddress=https://inference.your.domain --set inference.remote.apiKey=<token>`.
|
||||
- Enable persistence: `--set storage.persistence.enabled=true --set storage.persistence.size=10Gi` or `--set storage.persistence.existingClaim=<pvc>`.
|
||||
- Worker replicas: `--set worker.replicas=2` (or `--set worker.enabled=false` to run WebService only).
|
||||
|
||||
## Operational notes
|
||||
- Data paths (`/app/data/plans`, `/app/data/queue`, `/app/data/outputs`) are configurable via env and pre-created at startup.
|
||||
- Guardrail phrases or policy knobs can be mounted under `/app/etc`; point `ADVISORYAI__GUARDRAILS__PHRASESLIST` to the mounted file.
|
||||
- Observability follows standard ASP.NET JSON logs; add OTEL exporters via `OTEL_EXPORTER_OTLP_ENDPOINT` env when allowed. Keep disabled in sealed/offline deployments.
|
||||
- For air-gapped clusters, publish built images to your registry and reference via `--set image.repository=<registry>/stellaops/advisoryai-web`.
|
||||
55
ops/advisory-ai/docker-compose.advisoryai.yaml
Normal file
55
ops/advisory-ai/docker-compose.advisoryai.yaml
Normal file
@@ -0,0 +1,55 @@
|
||||
version: "3.9"
|
||||
|
||||
# Local/offline deployment for AdvisoryAI WebService + Worker.
|
||||
services:
|
||||
advisoryai-web:
|
||||
build:
|
||||
context: ../..
|
||||
dockerfile: ops/advisory-ai/Dockerfile
|
||||
args:
|
||||
PROJECT: src/AdvisoryAI/StellaOps.AdvisoryAI.WebService/StellaOps.AdvisoryAI.WebService.csproj
|
||||
APP_DLL: StellaOps.AdvisoryAI.WebService.dll
|
||||
image: stellaops-advisoryai-web:dev
|
||||
depends_on:
|
||||
- advisoryai-worker
|
||||
environment:
|
||||
ASPNETCORE_URLS: "http://0.0.0.0:8080"
|
||||
ADVISORYAI__QUEUE__DIRECTORYPATH: "/app/data/queue"
|
||||
ADVISORYAI__STORAGE__PLANCACHEDIRECTORY: "/app/data/plans"
|
||||
ADVISORYAI__STORAGE__OUTPUTDIRECTORY: "/app/data/outputs"
|
||||
ADVISORYAI__INFERENCE__MODE: "Local" # switch to Remote to call an external inference host
|
||||
# ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS: "https://inference.example.com"
|
||||
# ADVISORYAI__INFERENCE__REMOTE__ENDPOINT: "/v1/inference"
|
||||
# ADVISORYAI__INFERENCE__REMOTE__APIKEY: "set-me"
|
||||
# ADVISORYAI__INFERENCE__REMOTE__TIMEOUT: "00:00:30"
|
||||
# Example SBOM context feed; optional.
|
||||
# ADVISORYAI__SBOMBASEADDRESS: "https://sbom.local/v1/sbom/context"
|
||||
# ADVISORYAI__SBOMTENANT: "tenant-a"
|
||||
# ADVISORYAI__GUARDRAILS__PHRASESLIST: "/app/etc/guardrails/phrases.txt"
|
||||
volumes:
|
||||
- advisoryai-data:/app/data
|
||||
- ./etc:/app/etc:ro
|
||||
ports:
|
||||
- "7071:8080"
|
||||
restart: unless-stopped
|
||||
|
||||
advisoryai-worker:
|
||||
build:
|
||||
context: ../..
|
||||
dockerfile: ops/advisory-ai/Dockerfile
|
||||
args:
|
||||
PROJECT: src/AdvisoryAI/StellaOps.AdvisoryAI.Worker/StellaOps.AdvisoryAI.Worker.csproj
|
||||
APP_DLL: StellaOps.AdvisoryAI.Worker.dll
|
||||
image: stellaops-advisoryai-worker:dev
|
||||
environment:
|
||||
ADVISORYAI__QUEUE__DIRECTORYPATH: "/app/data/queue"
|
||||
ADVISORYAI__STORAGE__PLANCACHEDIRECTORY: "/app/data/plans"
|
||||
ADVISORYAI__STORAGE__OUTPUTDIRECTORY: "/app/data/outputs"
|
||||
ADVISORYAI__INFERENCE__MODE: "Local"
|
||||
volumes:
|
||||
- advisoryai-data:/app/data
|
||||
- ./etc:/app/etc:ro
|
||||
restart: unless-stopped
|
||||
|
||||
volumes:
|
||||
advisoryai-data:
|
||||
0
ops/advisory-ai/etc/.gitkeep
Normal file
0
ops/advisory-ai/etc/.gitkeep
Normal file
6
ops/advisory-ai/helm/Chart.yaml
Normal file
6
ops/advisory-ai/helm/Chart.yaml
Normal file
@@ -0,0 +1,6 @@
|
||||
apiVersion: v2
|
||||
name: stellaops-advisoryai
|
||||
version: 0.1.0
|
||||
appVersion: "0.1.0"
|
||||
description: AdvisoryAI WebService + Worker packaging for on-prem/air-gapped installs.
|
||||
type: application
|
||||
12
ops/advisory-ai/helm/templates/_helpers.tpl
Normal file
12
ops/advisory-ai/helm/templates/_helpers.tpl
Normal file
@@ -0,0 +1,12 @@
|
||||
{{- define "stellaops-advisoryai.name" -}}
|
||||
{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}}
|
||||
{{- end -}}
|
||||
|
||||
{{- define "stellaops-advisoryai.fullname" -}}
|
||||
{{- if .Values.fullnameOverride -}}
|
||||
{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}}
|
||||
{{- else -}}
|
||||
{{- $name := default .Chart.Name .Values.nameOverride -}}
|
||||
{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}}
|
||||
{{- end -}}
|
||||
{{- end -}}
|
||||
71
ops/advisory-ai/helm/templates/deployment.yaml
Normal file
71
ops/advisory-ai/helm/templates/deployment.yaml
Normal file
@@ -0,0 +1,71 @@
|
||||
apiVersion: apps/v1
|
||||
kind: Deployment
|
||||
metadata:
|
||||
name: {{ include "stellaops-advisoryai.fullname" . }}
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
app.kubernetes.io/version: {{ .Chart.AppVersion }}
|
||||
spec:
|
||||
replicas: 1
|
||||
selector:
|
||||
matchLabels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
spec:
|
||||
containers:
|
||||
- name: web
|
||||
image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
|
||||
imagePullPolicy: {{ .Values.image.pullPolicy }}
|
||||
env:
|
||||
- name: ASPNETCORE_URLS
|
||||
value: "http://0.0.0.0:{{ .Values.service.port }}"
|
||||
- name: ADVISORYAI__INFERENCE__MODE
|
||||
value: "{{ .Values.inference.mode }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS
|
||||
value: "{{ .Values.inference.remote.baseAddress }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__ENDPOINT
|
||||
value: "{{ .Values.inference.remote.endpoint }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__APIKEY
|
||||
value: "{{ .Values.inference.remote.apiKey }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__TIMEOUT
|
||||
value: "{{ printf "00:00:%d" .Values.inference.remote.timeoutSeconds }}"
|
||||
- name: ADVISORYAI__STORAGE__PLANCACHEDIRECTORY
|
||||
value: {{ .Values.storage.planCachePath | quote }}
|
||||
- name: ADVISORYAI__STORAGE__OUTPUTDIRECTORY
|
||||
value: {{ .Values.storage.outputPath | quote }}
|
||||
- name: ADVISORYAI__QUEUE__DIRECTORYPATH
|
||||
value: {{ .Values.storage.queuePath | quote }}
|
||||
envFrom:
|
||||
{{- if .Values.extraEnvFrom }}
|
||||
- secretRef:
|
||||
name: {{ .Values.extraEnvFrom | first }}
|
||||
{{- end }}
|
||||
{{- if .Values.extraEnv }}
|
||||
{{- range .Values.extraEnv }}
|
||||
- name: {{ .name }}
|
||||
value: {{ .value | quote }}
|
||||
{{- end }}
|
||||
{{- end }}
|
||||
ports:
|
||||
- containerPort: {{ .Values.service.port }}
|
||||
volumeMounts:
|
||||
- name: advisoryai-data
|
||||
mountPath: /app/data
|
||||
resources: {{- toYaml .Values.resources | nindent 12 }}
|
||||
volumes:
|
||||
- name: advisoryai-data
|
||||
{{- if .Values.storage.persistence.enabled }}
|
||||
persistentVolumeClaim:
|
||||
claimName: {{ .Values.storage.persistence.existingClaim | default (printf "%s-data" (include "stellaops-advisoryai.fullname" .)) }}
|
||||
{{- else }}
|
||||
emptyDir: {}
|
||||
{{- end }}
|
||||
nodeSelector: {{- toYaml .Values.nodeSelector | nindent 8 }}
|
||||
tolerations: {{- toYaml .Values.tolerations | nindent 8 }}
|
||||
affinity: {{- toYaml .Values.affinity | nindent 8 }}
|
||||
15
ops/advisory-ai/helm/templates/pvc.yaml
Normal file
15
ops/advisory-ai/helm/templates/pvc.yaml
Normal file
@@ -0,0 +1,15 @@
|
||||
{{- if and .Values.storage.persistence.enabled (not .Values.storage.persistence.existingClaim) }}
|
||||
apiVersion: v1
|
||||
kind: PersistentVolumeClaim
|
||||
metadata:
|
||||
name: {{ printf "%s-data" (include "stellaops-advisoryai.fullname" .) }}
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
spec:
|
||||
accessModes:
|
||||
- ReadWriteOnce
|
||||
resources:
|
||||
requests:
|
||||
storage: {{ .Values.storage.persistence.size }}
|
||||
{{- end }}
|
||||
17
ops/advisory-ai/helm/templates/service.yaml
Normal file
17
ops/advisory-ai/helm/templates/service.yaml
Normal file
@@ -0,0 +1,17 @@
|
||||
apiVersion: v1
|
||||
kind: Service
|
||||
metadata:
|
||||
name: {{ include "stellaops-advisoryai.fullname" . }}
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
spec:
|
||||
type: {{ .Values.service.type }}
|
||||
selector:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
ports:
|
||||
- name: http
|
||||
port: {{ .Values.service.port }}
|
||||
targetPort: {{ .Values.service.port }}
|
||||
protocol: TCP
|
||||
66
ops/advisory-ai/helm/templates/worker.yaml
Normal file
66
ops/advisory-ai/helm/templates/worker.yaml
Normal file
@@ -0,0 +1,66 @@
|
||||
{{- if .Values.worker.enabled }}
|
||||
apiVersion: apps/v1
|
||||
kind: Deployment
|
||||
metadata:
|
||||
name: {{ include "stellaops-advisoryai.fullname" . }}-worker
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}-worker
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
spec:
|
||||
replicas: {{ .Values.worker.replicas }}
|
||||
selector:
|
||||
matchLabels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}-worker
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
app.kubernetes.io/name: {{ include "stellaops-advisoryai.name" . }}-worker
|
||||
app.kubernetes.io/instance: {{ .Release.Name }}
|
||||
spec:
|
||||
containers:
|
||||
- name: worker
|
||||
image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
|
||||
imagePullPolicy: {{ .Values.image.pullPolicy }}
|
||||
command: ["dotnet", "StellaOps.AdvisoryAI.Worker.dll"]
|
||||
env:
|
||||
- name: ADVISORYAI__INFERENCE__MODE
|
||||
value: "{{ .Values.inference.mode }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__BASEADDRESS
|
||||
value: "{{ .Values.inference.remote.baseAddress }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__ENDPOINT
|
||||
value: "{{ .Values.inference.remote.endpoint }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__APIKEY
|
||||
value: "{{ .Values.inference.remote.apiKey }}"
|
||||
- name: ADVISORYAI__INFERENCE__REMOTE__TIMEOUT
|
||||
value: "{{ printf "00:00:%d" .Values.inference.remote.timeoutSeconds }}"
|
||||
- name: ADVISORYAI__STORAGE__PLANCACHEDIRECTORY
|
||||
value: {{ .Values.storage.planCachePath | quote }}
|
||||
- name: ADVISORYAI__STORAGE__OUTPUTDIRECTORY
|
||||
value: {{ .Values.storage.outputPath | quote }}
|
||||
- name: ADVISORYAI__QUEUE__DIRECTORYPATH
|
||||
value: {{ .Values.storage.queuePath | quote }}
|
||||
envFrom:
|
||||
{{- if .Values.extraEnvFrom }}
|
||||
- secretRef:
|
||||
name: {{ .Values.extraEnvFrom | first }}
|
||||
{{- end }}
|
||||
{{- if .Values.extraEnv }}
|
||||
{{- range .Values.extraEnv }}
|
||||
- name: {{ .name }}
|
||||
value: {{ .value | quote }}
|
||||
{{- end }}
|
||||
{{- end }}
|
||||
volumeMounts:
|
||||
- name: advisoryai-data
|
||||
mountPath: /app/data
|
||||
resources: {{- toYaml .Values.worker.resources | nindent 12 }}
|
||||
volumes:
|
||||
- name: advisoryai-data
|
||||
{{- if .Values.storage.persistence.enabled }}
|
||||
persistentVolumeClaim:
|
||||
claimName: {{ .Values.storage.persistence.existingClaim | default (printf "%s-data" (include "stellaops-advisoryai.fullname" .)) }}
|
||||
{{- else }}
|
||||
emptyDir: {}
|
||||
{{- end }}
|
||||
{{- end }}
|
||||
38
ops/advisory-ai/helm/values.yaml
Normal file
38
ops/advisory-ai/helm/values.yaml
Normal file
@@ -0,0 +1,38 @@
|
||||
image:
|
||||
repository: stellaops/advisoryai
|
||||
tag: dev
|
||||
pullPolicy: IfNotPresent
|
||||
|
||||
service:
|
||||
port: 8080
|
||||
type: ClusterIP
|
||||
|
||||
inference:
|
||||
mode: Local # or Remote
|
||||
remote:
|
||||
baseAddress: ""
|
||||
endpoint: "/v1/inference"
|
||||
apiKey: ""
|
||||
timeoutSeconds: 30
|
||||
|
||||
storage:
|
||||
planCachePath: /app/data/plans
|
||||
outputPath: /app/data/outputs
|
||||
queuePath: /app/data/queue
|
||||
persistence:
|
||||
enabled: false
|
||||
existingClaim: ""
|
||||
size: 5Gi
|
||||
|
||||
resources: {}
|
||||
nodeSelector: {}
|
||||
tolerations: []
|
||||
affinity: {}
|
||||
|
||||
worker:
|
||||
enabled: true
|
||||
replicas: 1
|
||||
resources: {}
|
||||
|
||||
extraEnv: [] # list of { name: ..., value: ... }
|
||||
extraEnvFrom: []
|
||||
@@ -5,6 +5,7 @@ using System.Threading.Tasks;
|
||||
using FluentAssertions;
|
||||
using Microsoft.Extensions.Configuration;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
using Microsoft.Extensions.FileProviders;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Options;
|
||||
using StellaOps.AdvisoryAI.Guardrails;
|
||||
@@ -78,13 +79,15 @@ public sealed class AdvisoryGuardrailOptionsBindingTests
|
||||
public FakeHostEnvironment(string contentRoot)
|
||||
{
|
||||
ContentRootPath = contentRoot;
|
||||
ContentRootFileProvider = new PhysicalFileProvider(contentRoot);
|
||||
}
|
||||
|
||||
public string EnvironmentName { get; set; } = Environments.Development;
|
||||
|
||||
public string ApplicationName { get; set; } = "StellaOps.AdvisoryAI.Tests";
|
||||
|
||||
public string ContentRootPath { get; set; }
|
||||
= string.Empty;
|
||||
public string ContentRootPath { get; set; } = string.Empty;
|
||||
|
||||
public IFileProvider ContentRootFileProvider { get; set; }
|
||||
}
|
||||
}
|
||||
|
||||
@@ -17,7 +17,7 @@ public sealed class EvidenceTelemetryTests
|
||||
|
||||
using var listener = CreateListener((instrument, value, tags) =>
|
||||
{
|
||||
measurements.Add((instrument.Name, value, tags.ToList()));
|
||||
measurements.Add((instrument.Name, value, tags.ToArray()));
|
||||
});
|
||||
|
||||
EvidenceTelemetry.RecordChunkOutcome("tenant-a", "success", chunkCount: 3, truncated: true);
|
||||
@@ -38,7 +38,7 @@ public sealed class EvidenceTelemetryTests
|
||||
|
||||
using var listener = CreateListener((instrument, value, tags) =>
|
||||
{
|
||||
measurements.Add((instrument.Name, value, tags.ToList()));
|
||||
measurements.Add((instrument.Name, value, tags.ToArray()));
|
||||
});
|
||||
|
||||
var now = DateTimeOffset.UtcNow;
|
||||
@@ -74,6 +74,7 @@ public sealed class EvidenceTelemetryTests
|
||||
|
||||
listener.SetMeasurementEventCallback<long>((instrument, measurement, tags, _) => callback(instrument, measurement, tags));
|
||||
listener.SetMeasurementEventCallback<int>((instrument, measurement, tags, _) => callback(instrument, measurement, tags));
|
||||
listener.SetMeasurementEventCallback<double>((instrument, measurement, tags, _) => callback(instrument, measurement, tags));
|
||||
|
||||
listener.Start();
|
||||
return listener;
|
||||
|
||||
@@ -27,6 +27,8 @@
|
||||
<ItemGroup>
|
||||
<Compile Remove="**/*.cs" />
|
||||
<Compile Include="AirgapImportEndpointTests.cs" />
|
||||
<Compile Include="VexEvidenceChunkServiceTests.cs" />
|
||||
<Compile Include="EvidenceTelemetryTests.cs" />
|
||||
<Compile Include="TestAuthentication.cs" />
|
||||
<Compile Include="TestServiceOverrides.cs" />
|
||||
<Compile Include="TestWebApplicationFactory.cs" />
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
#if false
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Collections.Immutable;
|
||||
@@ -52,14 +51,14 @@ public sealed class VexEvidenceChunkServiceTests
|
||||
{
|
||||
var product = new VexProduct("pkg:docker/demo", "demo", "1.0.0", "pkg:docker/demo:1.0.0", null, new[] { "component-a" });
|
||||
var document = new VexClaimDocument(
|
||||
VexDocumentFormat.SbomCycloneDx,
|
||||
VexDocumentFormat.CycloneDx,
|
||||
digest: Guid.NewGuid().ToString("N"),
|
||||
sourceUri: new Uri("https://example.test/vex.json"),
|
||||
revision: "r1",
|
||||
signature: new VexSignatureMetadata("cosign", "demo", "issuer", keyId: "kid", verifiedAt: firstSeen, transparencyLogReference: null));
|
||||
|
||||
var signals = score.HasValue
|
||||
? new VexSignalSnapshot(new VexSeveritySignal("cvss", score, "low", vector: null), Kev: null, Epss: null)
|
||||
? new VexSignalSnapshot(new VexSeveritySignal("cvss", score, "low", vector: null), kev: null, epss: null)
|
||||
: null;
|
||||
|
||||
return new VexClaim(
|
||||
@@ -116,5 +115,3 @@ public sealed class VexEvidenceChunkServiceTests
|
||||
public override DateTimeOffset GetUtcNow() => _timestamp;
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
@@ -122,7 +122,7 @@ public sealed class VexEvidenceChunksEndpointTests : IDisposable
|
||||
signature: new VexSignatureMetadata("cosign", "demo", "issuer", keyId: "kid", verifiedAt: firstSeen, transparencyLogReference: null));
|
||||
|
||||
var signals = score.HasValue
|
||||
? new VexSignalSnapshot(new VexSeveritySignal("cvss", score, "low", vector: null), Kev: null, Epss: null)
|
||||
? new VexSignalSnapshot(new VexSeveritySignal("cvss", score, "low", vector: null), kev: null, epss: null)
|
||||
: null;
|
||||
|
||||
return new VexClaim(
|
||||
|
||||
@@ -0,0 +1,19 @@
|
||||
using System.Reflection;
|
||||
using Json.Schema;
|
||||
|
||||
namespace StellaOps.Policy.RiskProfile.Schema;
|
||||
|
||||
public static class RiskProfileSchemaProvider
|
||||
{
|
||||
private const string SchemaResource = "StellaOps.Policy.RiskProfile.Schemas.risk-profile-schema@1.json";
|
||||
|
||||
public static JsonSchema GetSchema()
|
||||
{
|
||||
using var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream(SchemaResource)
|
||||
?? throw new InvalidOperationException($"Schema resource '{SchemaResource}' not found.");
|
||||
using var reader = new StreamReader(stream);
|
||||
var schemaText = reader.ReadToEnd();
|
||||
|
||||
return JsonSchema.FromText(schemaText);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,126 @@
|
||||
{
|
||||
"$schema": "https://json-schema.org/draft/2020-12/schema",
|
||||
"$id": "https://stellaops.dev/schemas/risk-profile-schema@1.json",
|
||||
"title": "StellaOps RiskProfile v1",
|
||||
"type": "object",
|
||||
"required": [ "id", "version", "signals", "weights", "overrides" ],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"id": {
|
||||
"type": "string",
|
||||
"minLength": 1,
|
||||
"description": "Stable identifier for the risk profile (slug or URN)."
|
||||
},
|
||||
"version": {
|
||||
"type": "string",
|
||||
"pattern": "^[0-9]+\\.[0-9]+\\.[0-9]+(-[A-Za-z0-9.-]+)?$",
|
||||
"description": "SemVer for the profile definition."
|
||||
},
|
||||
"description": {
|
||||
"type": "string",
|
||||
"description": "Human-readable summary of the profile intent."
|
||||
},
|
||||
"signals": {
|
||||
"type": "array",
|
||||
"minItems": 1,
|
||||
"items": {
|
||||
"type": "object",
|
||||
"required": [ "name", "source", "type" ],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"name": {
|
||||
"type": "string",
|
||||
"minLength": 1,
|
||||
"description": "Logical signal key (e.g., reachability, kev, exploit_chain)."
|
||||
},
|
||||
"source": {
|
||||
"type": "string",
|
||||
"minLength": 1,
|
||||
"description": "Upstream provider or calculation origin."
|
||||
},
|
||||
"type": {
|
||||
"type": "string",
|
||||
"enum": [ "boolean", "numeric", "categorical" ]
|
||||
},
|
||||
"path": {
|
||||
"type": "string",
|
||||
"description": "JSON Pointer to the signal in the evidence document."
|
||||
},
|
||||
"transform": {
|
||||
"type": "string",
|
||||
"description": "Optional transform applied before weighting (e.g., log, normalize)."
|
||||
},
|
||||
"unit": {
|
||||
"type": "string",
|
||||
"description": "Optional unit for numeric signals."
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"weights": {
|
||||
"type": "object",
|
||||
"minProperties": 1,
|
||||
"additionalProperties": {
|
||||
"type": "number",
|
||||
"minimum": 0,
|
||||
"maximum": 1
|
||||
},
|
||||
"description": "Weight per signal name; weights are normalized by the consumer."
|
||||
},
|
||||
"overrides": {
|
||||
"type": "object",
|
||||
"required": [ "severity", "decisions" ],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"severity": {
|
||||
"type": "array",
|
||||
"items": {
|
||||
"type": "object",
|
||||
"required": [ "when", "set" ],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"when": {
|
||||
"type": "object",
|
||||
"description": "Predicate over signals (key/value equals).",
|
||||
"minProperties": 1,
|
||||
"additionalProperties": { "type": [ "string", "number", "boolean" ] }
|
||||
},
|
||||
"set": {
|
||||
"type": "string",
|
||||
"enum": [ "critical", "high", "medium", "low", "informational" ]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"decisions": {
|
||||
"type": "array",
|
||||
"items": {
|
||||
"type": "object",
|
||||
"required": [ "when", "action" ],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"when": {
|
||||
"type": "object",
|
||||
"description": "Predicate over signals (key/value equals).",
|
||||
"minProperties": 1,
|
||||
"additionalProperties": { "type": [ "string", "number", "boolean" ] }
|
||||
},
|
||||
"action": {
|
||||
"type": "string",
|
||||
"enum": [ "allow", "review", "deny" ]
|
||||
},
|
||||
"reason": {
|
||||
"type": "string"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"metadata": {
|
||||
"type": "object",
|
||||
"description": "Free-form metadata with stable keys.",
|
||||
"additionalProperties": { "type": [ "string", "number", "boolean", "array", "object", "null" ] }
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net10.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
<LangVersion>preview</LangVersion>
|
||||
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="JsonSchema.Net" Version="5.3.0" />
|
||||
<PackageReference Include="System.Text.Json" Version="10.0.0-rc.2.25519.1" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Include="Schemas\risk-profile-schema@1.json" />
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
@@ -0,0 +1,30 @@
|
||||
using System.Text.Json;
|
||||
using Json.Schema;
|
||||
using StellaOps.Policy.RiskProfile.Schema;
|
||||
|
||||
namespace StellaOps.Policy.RiskProfile.Validation;
|
||||
|
||||
public sealed class RiskProfileValidator
|
||||
{
|
||||
private readonly JsonSchema _schema;
|
||||
|
||||
public RiskProfileValidator() : this(RiskProfileSchemaProvider.GetSchema())
|
||||
{
|
||||
}
|
||||
|
||||
public RiskProfileValidator(JsonSchema schema)
|
||||
{
|
||||
_schema = schema ?? throw new ArgumentNullException(nameof(schema));
|
||||
}
|
||||
|
||||
public ValidationResults Validate(string json)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(json))
|
||||
{
|
||||
throw new ArgumentException("Risk profile payload is required.", nameof(json));
|
||||
}
|
||||
|
||||
using var document = JsonDocument.Parse(json);
|
||||
return _schema.Validate(document.RootElement);
|
||||
}
|
||||
}
|
||||
@@ -33,6 +33,32 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Policy.Gateway.Te
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Policy.Tests", "__Tests\StellaOps.Policy.Tests\StellaOps.Policy.Tests.csproj", "{D064D5C1-3311-470C-92A1-41E913125C14}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Policy.RiskProfile", "StellaOps.Policy.RiskProfile\StellaOps.Policy.RiskProfile.csproj", "{6206D9E9-F84C-424C-84E3-7A63774BCEC9}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Policy.RiskProfile.Tests", "__Tests\StellaOps.Policy.RiskProfile.Tests\StellaOps.Policy.RiskProfile.Tests.csproj", "{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Testing", "..\Concelier\__Libraries\StellaOps.Concelier.Testing\StellaOps.Concelier.Testing.csproj", "{D3F5BCE7-7F50-474B-B70D-D16A559AC720}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Connector.Common", "..\Concelier\__Libraries\StellaOps.Concelier.Connector.Common\StellaOps.Concelier.Connector.Common.csproj", "{5DE7674D-CB03-4475-A0FF-14528E45A3C8}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Storage.Mongo", "..\Concelier\__Libraries\StellaOps.Concelier.Storage.Mongo\StellaOps.Concelier.Storage.Mongo.csproj", "{EA35FF3B-16AD-48A9-B47D-632103BFC47F}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Core", "..\Concelier\__Libraries\StellaOps.Concelier.Core\StellaOps.Concelier.Core.csproj", "{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Models", "..\Concelier\__Libraries\StellaOps.Concelier.Models\StellaOps.Concelier.Models.csproj", "{9CF5075A-E59B-4F59-90B9-82C92AC33410}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.RawModels", "..\Concelier\__Libraries\StellaOps.Concelier.RawModels\StellaOps.Concelier.RawModels.csproj", "{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Concelier.Normalization", "..\Concelier\__Libraries\StellaOps.Concelier.Normalization\StellaOps.Concelier.Normalization.csproj", "{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Ingestion.Telemetry", "..\__Libraries\StellaOps.Ingestion.Telemetry\StellaOps.Ingestion.Telemetry.csproj", "{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Provenance.Mongo", "..\__Libraries\StellaOps.Provenance.Mongo\StellaOps.Provenance.Mongo.csproj", "{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Plugin", "..\__Libraries\StellaOps.Plugin\StellaOps.Plugin.csproj", "{0482A07E-CDA3-4006-84E6-828B072995C2}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Aoc", "..\Aoc\__Libraries\StellaOps.Aoc\StellaOps.Aoc.csproj", "{90DA7D82-B567-47CC-96F8-84C347DA8983}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
@@ -199,6 +225,162 @@ Global
|
||||
{D064D5C1-3311-470C-92A1-41E913125C14}.Release|x64.Build.0 = Release|Any CPU
|
||||
{D064D5C1-3311-470C-92A1-41E913125C14}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{D064D5C1-3311-470C-92A1-41E913125C14}.Release|x86.Build.0 = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|x64.Build.0 = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{6206D9E9-F84C-424C-84E3-7A63774BCEC9}.Release|x86.Build.0 = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|x64.Build.0 = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD}.Release|x86.Build.0 = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|x64.Build.0 = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{D3F5BCE7-7F50-474B-B70D-D16A559AC720}.Release|x86.Build.0 = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|x64.Build.0 = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{5DE7674D-CB03-4475-A0FF-14528E45A3C8}.Release|x86.Build.0 = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|x64.Build.0 = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{EA35FF3B-16AD-48A9-B47D-632103BFC47F}.Release|x86.Build.0 = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|x64.Build.0 = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{EA1A2CA6-2B73-4C77-8A96-674AF06C0D52}.Release|x86.Build.0 = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|x64.Build.0 = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{9CF5075A-E59B-4F59-90B9-82C92AC33410}.Release|x86.Build.0 = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|x64.Build.0 = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{79E1A6AB-BDA0-4419-A697-69C65DE99C3D}.Release|x86.Build.0 = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|x64.Build.0 = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{5DCB6EC1-CACA-4AB8-97FD-A73A3738B97F}.Release|x86.Build.0 = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|x64.Build.0 = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{CD2E6593-79CC-4668-8CBD-EDF1A80DE0C6}.Release|x86.Build.0 = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|x64.Build.0 = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{F7DABB1F-2F0A-492B-A7D0-6AB0FED72D5B}.Release|x86.Build.0 = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|x64.Build.0 = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{0482A07E-CDA3-4006-84E6-828B072995C2}.Release|x86.Build.0 = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|x64.Build.0 = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{90DA7D82-B567-47CC-96F8-84C347DA8983}.Release|x86.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
@@ -208,5 +390,6 @@ Global
|
||||
{478DF014-BF69-41BA-B78A-AAC0918337D8} = {56BCE1BF-7CBA-7CE8-203D-A88051F1D642}
|
||||
{77189D88-1CA1-46BD-A9DC-99B2B6EF7D44} = {56BCE1BF-7CBA-7CE8-203D-A88051F1D642}
|
||||
{D064D5C1-3311-470C-92A1-41E913125C14} = {56BCE1BF-7CBA-7CE8-203D-A88051F1D642}
|
||||
{EE0C81E5-7E50-4CC2-BEB9-F3F4FBEBB9CD} = {56BCE1BF-7CBA-7CE8-203D-A88051F1D642}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
|
||||
@@ -0,0 +1,64 @@
|
||||
using System.Text.Json;
|
||||
using StellaOps.Policy.RiskProfile.Validation;
|
||||
using Xunit;
|
||||
|
||||
namespace StellaOps.Policy.RiskProfile.Tests;
|
||||
|
||||
public class RiskProfileValidatorTests
|
||||
{
|
||||
private readonly RiskProfileValidator _validator = new();
|
||||
|
||||
[Fact]
|
||||
public void Valid_profile_passes_schema()
|
||||
{
|
||||
var profile = """
|
||||
{
|
||||
"id": "default-risk",
|
||||
"version": "1.0.0",
|
||||
"description": "Baseline risk profile",
|
||||
"signals": [
|
||||
{ "name": "reachability", "source": "signals", "type": "boolean", "path": "/reachability/exploitable" },
|
||||
{ "name": "kev", "source": "cisa", "type": "boolean" }
|
||||
],
|
||||
"weights": {
|
||||
"reachability": 0.6,
|
||||
"kev": 0.4
|
||||
},
|
||||
"overrides": {
|
||||
"severity": [
|
||||
{ "when": { "kev": true }, "set": "critical" }
|
||||
],
|
||||
"decisions": [
|
||||
{ "when": { "reachability": false }, "action": "review", "reason": "Not reachable" }
|
||||
]
|
||||
},
|
||||
"metadata": {
|
||||
"owner": "risk-team"
|
||||
}
|
||||
}
|
||||
""";
|
||||
|
||||
var result = _validator.Validate(profile);
|
||||
|
||||
Assert.True(result.IsValid, string.Join(" | ", result.Errors ?? Array.Empty<string>()));
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Missing_required_fields_fails_schema()
|
||||
{
|
||||
var invalidProfile = """
|
||||
{ "id": "missing-fields" }
|
||||
""";
|
||||
|
||||
var result = _validator.Validate(invalidProfile);
|
||||
|
||||
Assert.False(result.IsValid);
|
||||
Assert.NotEmpty(result.Errors);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Empty_payload_throws()
|
||||
{
|
||||
Assert.Throws<ArgumentException>(() => _validator.Validate(" "));
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net10.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
<LangVersion>preview</LangVersion>
|
||||
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="../../StellaOps.Policy.RiskProfile/StellaOps.Policy.RiskProfile.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="xunit" Version="2.9.2" />
|
||||
<PackageReference Include="xunit.runner.visualstudio" Version="2.8.2" />
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
Reference in New Issue
Block a user