PSA/shared/workflow/expression-authoring/adapters/webhookPayloadContextAdapter.ts
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

83 lines
2.4 KiB
TypeScript

import type {
SharedExpressionContextRoot,
SharedExpressionPathOption,
SharedExpressionSchemaNode,
} from '../context';
import type { ExpressionMode } from '../modes';
import { buildPathOptionsFromContextRoots } from '../pathDiscovery';
function inferSchema(value: unknown): SharedExpressionSchemaNode {
if (value === null) {
return { type: 'null' };
}
if (Array.isArray(value)) {
return {
type: 'array',
items: inferArrayItemSchema(value),
};
}
switch (typeof value) {
case 'string':
return { type: 'string' };
case 'number':
return { type: Number.isInteger(value) ? 'integer' : 'number' };
case 'boolean':
return { type: 'boolean' };
case 'object':
return {
type: 'object',
properties: Object.fromEntries(
Object.entries(value as Record<string, unknown>)
.sort(([left], [right]) => left.localeCompare(right))
.map(([key, nested]) => [key, inferSchema(nested)]),
),
};
default:
return { type: 'unknown' };
}
}
function inferArrayItemSchema(values: unknown[]): SharedExpressionSchemaNode {
const firstNonNull = values.find((value) => value !== null && value !== undefined);
return firstNonNull === undefined ? { type: 'unknown' } : inferSchema(firstNonNull);
}
export function buildWebhookPayloadExpressionContextRoots(samplePayload: unknown): SharedExpressionContextRoot[] {
if (!samplePayload || typeof samplePayload !== 'object' || Array.isArray(samplePayload)) {
return [
{
key: 'value',
label: 'Payload',
description: 'Webhook request body',
schema: inferSchema(samplePayload),
allowInModes: ['path-only', 'template'],
},
];
}
return Object.entries(samplePayload as Record<string, unknown>)
.sort(([left], [right]) => left.localeCompare(right))
.map(([key, value]) => ({
key,
label: key,
description: 'Webhook request body field',
schema: inferSchema(value),
allowInModes: ['path-only', 'template'],
}));
}
export function buildWebhookPayloadExpressionPathOptions(
samplePayload: unknown,
params: {
mode?: ExpressionMode;
includeRootPaths?: boolean;
} = {},
): SharedExpressionPathOption[] {
return buildPathOptionsFromContextRoots(
buildWebhookPayloadExpressionContextRoots(samplePayload),
params,
);
}