Hermes 284313f908
Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

11 KiB

PRD — Tanium Criticality Asset Facts

  • Slug: tanium-criticality-asset-facts
  • Date: 2026-04-29
  • Status: Draft

Summary

Add a small, generic Asset Facts data model in CE and use it as the day-one persistence/display surface for Tanium endpoint criticality in EE.

Tanium exposes endpoint criticality through Gateway endpoint sensor readings, not as a first-class Endpoint GraphQL field. Alga should fetch the Tanium Endpoint Criticality with Level virtual sensor as a best-effort enrichment during Tanium inventory sync, persist the parsed result as a provider-sourced asset fact, and display it on Tanium-synced asset detail pages.

The first release is display-focused. The data model must remain queryable and neutral enough to support later asset filtering, workflow predicates, AI context, and possible future user-defined fields without requiring a rewrite.

Problem

Tanium has a device criticality concept that can change how endpoints are handled operationally. Alga currently has no universal asset criticality concept and no generic custom-field/fact model for integration-sourced asset metadata.

A purely Tanium-specific UI-only implementation would satisfy immediate display needs but would work against later goals:

  • asset filtering by criticality,
  • workflow conditions such as “Tanium criticality is High or Critical,”
  • AI-assisted queries over endpoint posture,
  • a broader custom/facts model for asset metadata.

At the same time, building a full custom-fields platform now is too much scope for the first Tanium criticality release.

Goals

  1. Create a CE/base asset_facts data structure for provider/system/manual facts about assets.
  2. Keep Tanium criticality provider-sourced and read-only in day one.
  3. Fetch Tanium criticality through Gateway Endpoint Criticality with Level sensor readings.
  4. Keep the criticality fetch best-effort so missing Tanium Criticality permissions do not break inventory sync.
  5. Persist the parsed display value and raw Tanium sensor response in a queryable fact row.
  6. Display Tanium criticality on asset detail pages when an available fact exists.
  7. Preserve a clean path to future filtering, workflow, AI, and custom-field-like usage.

Non-goals

  • Build generic user-editable custom fields.
  • Add workflow predicates over asset facts in this phase.
  • Add AI tools or agent context retrieval for asset facts in this phase.
  • Add asset list filtering by facts in this phase.
  • Add Tanium write-back based on Alga state.
  • Add manual editing of provider-sourced facts.
  • Treat Tanium criticality as an Alga-native first-class asset column.

Users and Primary Flows

MSP technician viewing an asset

  1. Technician opens a Tanium-synced asset detail page.
  2. Alga displays the usual RMM/Tanium-enriched asset data.
  3. If Tanium criticality is available, Alga shows a compact Tanium Criticality indicator, such as Tanium Criticality: High.
  4. If no criticality fact is available, the page remains clean and does not show an empty placeholder.

Admin running Tanium sync

  1. Admin runs Tanium full inventory sync from RMM integration settings.
  2. Alga fetches endpoint inventory as it does today.
  3. Alga separately attempts to fetch Tanium Endpoint Criticality with Level readings for the same mapped scopes.
  4. Inventory sync succeeds even if criticality readings fail due to content-set/RBAC availability.
  5. Where readings are available, Alga upserts asset facts after assets are mapped/created.

Future consumer: filters/workflows/AI

Not in day-one UI scope, but later consumers should be able to query asset facts directly instead of scraping provider JSON from system_info.

UX / UI Notes

Day-one UI should be intentionally modest:

  • Render a provider-sourced fact area or small provider badge/card only when available facts exist.
  • Known Tanium renderer:
    • Label: Tanium Criticality
    • Value: Low, Medium, High, or Critical
  • Use theme-aware badge/status styling.
  • If only a numeric multiplier is available, display the multiplier in a neutral form.
  • Hide unavailable facts from the primary asset detail surface.
  • Do not introduce a generic editable facts/custom-fields UI in this phase.

Requirements

Functional Requirements

  1. Add a CE/base migration for an asset_facts table.
  2. Add a reusable server-side helper/service to upsert provider-sourced asset facts.
  3. Add a Tanium Gateway client method to discover/verify the Endpoint Criticality with Level sensor metadata when needed.
  4. Add a Tanium Gateway client method to query endpoint criticality readings by optional computer group filter.
  5. Parse Tanium criticality values into a normalized fact shape:
    • display text where available,
    • numeric multiplier where available,
    • raw sensor columns/values in JSON.
  6. Run the criticality query separately from the main endpoint inventory query.
  7. Continue Tanium inventory sync when the criticality query fails globally.
  8. Upsert an available criticality fact after shared RMM ingestion returns an asset id.
  9. Mark an endpoint-specific criticality fact unavailable when Tanium explicitly returns no result for that endpoint.
  10. Leave existing facts untouched when the entire criticality query fails for a scope.
  11. Add asset detail loading for available asset facts.
  12. Render Tanium criticality on asset detail pages when the fact exists and is available.

Non-functional Requirements

  1. The asset_facts table must follow tenant/Citus conventions, including tenant in primary/unique indexes and query predicates.
  2. Provider-sourced facts must be read-only from UI surfaces in this phase.
  3. Criticality enrichment must not materially increase Tanium sync fragility.
  4. Raw provider data must be preserved for traceability without forcing UI consumers to parse it.
  5. The implementation must not require EE-only schema for the generic facts table.

Data / API / Integrations

CE data model

Create a CE/base table, tentatively named asset_facts:

asset_fact_id uuid
tenant uuid
asset_id uuid
source_type text          -- integration | manual | system; v1 uses integration
provider text             -- nullable; tanium for this use case
integration_id uuid       -- nullable; RMM integration id where applicable
namespace text            -- tanium, alga, ninjaone, etc.
fact_key text             -- criticality
label text                -- Criticality
value_text text           -- High / Critical / Medium / Low
value_number numeric      -- 1.67 / 2 / 1.33 / 1
value_bool boolean        -- future use
value_json jsonb          -- raw/structured source value
source text               -- tanium.gateway.sensor.Endpoint Criticality with Level
source_updated_at timestamptz nullable
last_synced_at timestamptz nullable
is_available boolean
created_at timestamptz
updated_at timestamptz

Practical uniqueness for v1:

unique (tenant, asset_id, source_type, namespace, fact_key)

For Tanium criticality:

source_type: integration
provider: tanium
namespace: tanium
fact_key: criticality
label: Criticality
value_text: High
value_number: 1.67
value_json: {
  sensorName: "Endpoint Criticality with Level",
  columns: [...],
  rawValues: [...]
}
source: "tanium.gateway.sensor.Endpoint Criticality with Level"
is_available: true

Tanium Gateway

Docs findings:

  • Endpoint Criticality with Level is a Criticality virtual sensor.
  • It returns endpoint criticality value and text.
  • Gateway supports arbitrary endpoint sensor reads through:
    • Endpoint.sensorReading(sensor: EndpointSensorRef!): EndpointSensorReading
    • Endpoint.sensorReadings(sensors: [EndpointSensorRef!]!): EndpointSensorReadings!
  • If a sensor is unavailable from the data source, Gateway returns an error.

Recommended criticality query shape:

query TaniumEndpointCriticality(
  $first: Int!
  $after: Cursor
  $filter: EndpointFieldFilter
) {
  endpoints(first: $first, after: $after, filter: $filter) {
    edges {
      node {
        id
        criticality: sensorReadings(
          sensors: [
            { name: "Endpoint Criticality with Level" }
          ]
        ) {
          columns {
            name
            values
          }
        }
      }
    }
    pageInfo {
      hasNextPage
      endCursor
    }
  }
}

Optional metadata verification query:

query TaniumCriticalitySensorMetadata {
  sensors(
    first: 10
    filter: {
      path: "name"
      op: EQ
      value: "Endpoint Criticality with Level"
    }
  ) {
    edges {
      node {
        name
        description
        valueType
        virtual
        harvested
        contentSetName
        columns {
          name
          valueType
          hidden
        }
      }
    }
  }
}

Security / Permissions

  • Tanium settings and sync actions continue to use existing withAuth, RBAC, and ADVANCED_ASSETS tier gating.
  • The asset_facts table exists in CE, but Tanium fact production remains gated by the existing EE Tanium integration feature/tier controls.
  • Asset fact reads should respect existing asset access patterns and tenant filtering.
  • Provider-sourced facts are not user-editable in this phase.

Observability

  • Log a warning when Tanium criticality enrichment fails globally for a scope.
  • Do not mark the whole Tanium inventory sync as failed solely due to criticality enrichment failure.
  • Preserve raw sensor response in value_json for later debugging.

No new metrics/monitoring work is required for this phase.

Rollout / Migration

  1. Add CE/base migration for asset_facts.
  2. Existing tenants receive an empty table; no backfill is required.
  3. Tanium facts are populated on the next full Tanium inventory sync.
  4. If the UI deploys before facts exist, no empty criticality section is shown.
  5. Existing system_info storage remains unchanged and can continue carrying raw provider metadata.

Open Questions

  1. What exact column names does each Tanium tenant return for Endpoint Criticality with Level? Implementation should discover/preserve raw columns and parse defensively.
  2. Should unavailable facts be shown in any diagnostic/admin-only UI later? Day one hides them from primary asset detail.
  3. Should future fact uniqueness distinguish multiple integrations of the same namespace for one asset? V1 assumes one current fact per asset/source_type/namespace/key.

Acceptance Criteria (Definition of Done)

  1. CE/base migrations create and rollback asset_facts successfully.
  2. Tanium full sync continues to ingest assets if criticality sensor access fails.
  3. Tanium full sync upserts asset_facts rows for endpoints with criticality readings.
  4. Tanium criticality facts preserve both normalized display data and raw Tanium sensor values.
  5. Explicit per-endpoint no-result values mark facts unavailable without deleting raw traceability.
  6. Asset detail displays Tanium criticality when an available Tanium criticality fact exists.
  7. Asset detail hides unavailable/missing criticality facts.
  8. Unit/integration tests cover Gateway parsing, fact upsert behavior, sync failure isolation, and UI rendering.