mirror of
https://github.com/shankar0123/certctl.git
synced 2026-06-07 21:01:31 +00:00
78485f7429
The MED-11 closure shipped users.deactivated_at + DELETE /api/v1/auth/users/{id}
+ cascade-revoke, but the federated-user soft-delete was reversible: the next
OIDC login under the same (provider, subject) tuple re-minted a session and
re-elevated the user.
Three legs of the chain were severed (each independently CRIT-shaped):
Leg A — postgres/user.go::userColumns omitted `deactivated_at`, so scanUser
never populated User.DeactivatedAt. Every Get / GetByOIDCSubject /
ListAll returned DeactivatedAt = nil regardless of the column value.
Leg B — postgres/user.go::Update SQL omitted `deactivated_at = $X`, so the
handler's `u.DeactivatedAt = now()` mutation was a no-op write at
the SQL level. Even with leg A closed, no row ever flipped.
Leg C — oidc/service.go::upsertUser did not inspect DeactivatedAt on the
existing-user path. Even with legs A + B closed, the OIDC login
would still proceed normally.
The cascade-session-revoke half of the original closure remained correct, but
only for the duration of the user's current cookie. SOC 2 CC6.3 + ISO 27001
A.9.2.6 "user access removal" controls require both immediate revoke AND
persistent block — this fix restores the persistent-block leg.
Closure across layers:
internal/repository/postgres/user.go
- userColumns adds `deactivated_at`
- scanUser reads via sql.NullTime intermediate (column is nullable)
- Create writes deactivated_at explicitly (NULL for new active users;
forward-compat for future seed-data flows that pre-populate the column)
- Update writes deactivated_at on every call; nil DeactivatedAt → NULL
(supports reactivation)
internal/auth/oidc/service.go
- New sentinel ErrUserDeactivated
- upsertUser checks existing.DeactivatedAt != nil BEFORE mutating email /
display_name / last_login_at — preserves last_login_at forensics on
rejected login attempts (defense-in-depth pin against future
"performance optimization" that reorders the gate)
internal/api/handler/auth_session_oidc.go
- classifyOIDCFailure adds typed errors.Is dispatch for ErrUserDeactivated
→ audit category "user_deactivated" (SOC/SIEM observability surface)
internal/api/handler/auth_users.go
- Self-deactivate guard on Deactivate: HTTP 409 + audit row
auth.user_deactivate_self_rejected when caller targets own User row.
Prevents an admin from one-way-door locking themselves out via the
standard handler; break-glass remains the recovery path.
- New Reactivate handler: inverse of Deactivate. Clears DeactivatedAt
via Update; emits auth.user_reactivated audit row. Idempotent on
already-active rows. Sessions revoked at deactivation stay revoked
(cascade irreversible by design — user must complete fresh OIDC
login).
internal/api/router/router.go
- POST /api/v1/auth/users/{id}/reactivate wired with auth.user.deactivate
gate (reactivation is the inverse op, not a separate privilege)
web/src/api/client.ts + web/src/pages/auth/UsersPage.tsx
- authReactivateUser() client function
- Reactivate button on deactivated rows in UsersPage
Regression coverage:
Postgres (testcontainers, skipped under -short):
TestUserRepository_DeactivatedAt_RoundTrip — Create → set DeactivatedAt
→ Update → Get / GetByOIDCSubject / ListAll round-trip the value
TestUserRepository_DeactivatedAt_CreateWritesNullForActive — new active
user reads back DeactivatedAt = nil
TestUserRepository_DeactivatedAt_CreatePersistsPreDeactivated — Create
with non-nil DeactivatedAt round-trips (forward-compat path)
OIDC service:
TestService_HandleCallback_RejectsDeactivatedUser — errors.Is
ErrUserDeactivated; CallbackResult nil; persisted email / last_login_at
/ deactivated_at NOT mutated by the rejected attempt
TestService_HandleCallback_AllowsReactivatedUser — DeactivatedAt = nil
→ happy path resumes
TestService_HandleCallback_DeactivatedUserPreservesForensics —
defense-in-depth pin against future regressions that reorder the
gate-vs-mutation sequence
Classifier:
TestClassifyOIDCFailure extended — typed dispatch + wrapped variant
round-trip through errors.Is
Handler:
TestAuthUsers_Deactivate_RejectsSelfDeactivate — HTTP 409 + audit
row + cascade-revoke NOT fired + row stays active
TestAuthUsers_Deactivate_OtherUser_HappyPath — HTTP 204 + cascade
fires + row soft-deleted
TestAuthUsers_Reactivate_HappyPath / _IdempotentOnActiveUser /
_UnknownID / _MissingID / _UpdateError
Phase 6 verify gate green on the targeted packages: gofmt clean, go vet
clean, go test -short pass across internal/auth/oidc, internal/api/handler,
internal/api/router, internal/repository/postgres, internal/auth/...,
internal/service/..., internal/tlsprobe/..., internal/trustanchor/...,
internal/validation/...
Spec at cowork/auth-bundles-fixes-2026-05-11/02-crit-deactivated-at-enforcement.md
Closure annotation at cowork/auth-bundles-audit-2026-05-10.md MED-11 row.
Operator advisory in CHANGELOG.md v2.1.0 release notes.
1380 lines
54 KiB
TypeScript
1380 lines
54 KiB
TypeScript
import type { Certificate, CertificateVersion, Agent, Job, Notification, AuditEvent, PolicyRule, PolicyViolation, RenewalPolicy, Issuer, Target, CertificateProfile, Owner, Team, AgentGroup, PaginatedResponse, DashboardSummary, CertificateStatusCount, ExpirationBucket, JobTrendDataPoint, IssuanceRateDataPoint, MetricsResponse, DiscoveredCertificate, DiscoveryScan, DiscoverySummary, NetworkScanTarget, EndpointHealthCheck, HealthHistoryEntry, HealthCheckSummary, AgentDependencyCounts, RetireAgentResponse, BlockedByDependenciesResponse, CRLCacheResponse, IntuneStatsResponse, IntuneReloadTrustResponse, SCEPProfilesResponse, SCEPProbeResult, SCEPProbesResponse, ESTProfilesResponse, ESTReloadTrustResponse } from './types';
|
|
|
|
const BASE = '/api/v1';
|
|
|
|
// P-1 closure (diff-04x03-d24864996ad4 P2 + cat-b-dc46aadab98e P3):
|
|
// the audit flagged 26+16 orphan client functions. Recon at HEAD
|
|
// found 17 actual orphans (the 26+16 audit numbers conflated; many
|
|
// were eliminated by the B-1 / S-1 / I-2 / D-2 closures since the
|
|
// audit was written). The remaining 17 are all detail-page
|
|
// candidates — singleton-getter `getX(id)` fns that detail pages
|
|
// will need when the corresponding `XPage` grows a `XDetailPage`
|
|
// route. Preserved here (rather than deleted) so the future
|
|
// detail-page work doesn't have to relitigate the client.ts surface.
|
|
//
|
|
// Intentionally-orphan client functions:
|
|
// getAgentGroup, getAgentGroupMembers, getAuditEvent,
|
|
// getCertificateDeployments, getDiscoveredCertificate,
|
|
// getHealthCheck, getHealthCheckHistory, getNetworkScanTarget,
|
|
// getNotification, getOwner, getPolicy,
|
|
// getPolicyViolations, getRenewalPolicy, getTeam, registerAgent
|
|
// (by-design pull-only; see C-1 closure docblock above its export),
|
|
// updateHealthCheck.
|
|
//
|
|
// CRL/OCSP-Responder Phase 5 closed the getOCSPStatus orphan: the
|
|
// CertificateDetailPage Revocation Endpoints panel now exercises it
|
|
// via the "Check OCSP status" button, so it's removed from the list
|
|
// above (and from the CI guardrail's DOCUMENTED list).
|
|
//
|
|
// CI guardrail at .github/workflows/ci.yml::"Documented orphan
|
|
// client fns sync guard (P-1)" enforces the docblock list ↔
|
|
// export list relationship: every name above must still be
|
|
// declared somewhere in this file, and conversely if a name is
|
|
// removed from the list its export must also be removed (orphans
|
|
// must never silently accumulate).
|
|
//
|
|
// See coverage-gap-audit-2026-04-24-v5/unified-audit.md
|
|
// diff-04x03-d24864996ad4 + cat-b-dc46aadab98e for closure rationale.
|
|
|
|
// API key stored in memory (not localStorage for security)
|
|
let apiKey: string | null = null;
|
|
|
|
export function setApiKey(key: string | null) {
|
|
apiKey = key;
|
|
}
|
|
|
|
export function getApiKey(): string | null {
|
|
return apiKey;
|
|
}
|
|
|
|
function authHeaders(): Record<string, string> {
|
|
const headers: Record<string, string> = { 'Content-Type': 'application/json' };
|
|
if (apiKey) {
|
|
headers['Authorization'] = `Bearer ${apiKey}`;
|
|
}
|
|
return headers;
|
|
}
|
|
|
|
// Bundle 2 Phase 8 — read the __Host-certctl_csrf cookie value (set
|
|
// by the OIDC-callback / break-glass-login flows; JS-readable by
|
|
// design so the GUI can echo it into the X-CSRF-Token header on every
|
|
// state-changing request). Returns empty string when the cookie isn't
|
|
// set (Bearer-mode deployments don't need CSRF; the server's middleware
|
|
// short-circuits CSRF for Bearer-authenticated requests).
|
|
//
|
|
// Audit 2026-05-10 MED-14 — cookie name carries the `__Host-` prefix
|
|
// (subdomain-takeover defense). The browser includes the prefix in
|
|
// document.cookie verbatim; the comparison below matches that.
|
|
function readCSRFCookie(): string {
|
|
if (typeof document === 'undefined' || !document.cookie) return '';
|
|
for (const part of document.cookie.split(';')) {
|
|
const [k, ...rest] = part.trim().split('=');
|
|
if (k === '__Host-certctl_csrf') {
|
|
return decodeURIComponent(rest.join('='));
|
|
}
|
|
}
|
|
return '';
|
|
}
|
|
|
|
// Audit 2026-05-10 HIGH-8 — extract the session-failure cause from the
|
|
// WWW-Authenticate header the server emits on 401. The server format
|
|
// (RFC 6750 §3) is: `Bearer realm="certctl", error="invalid_token",
|
|
// error_description="<cause>"` where <cause> is one of the stable
|
|
// categories `idle_timeout` / `absolute_timeout` /
|
|
// `back_channel_revoked` / `invalid_token`. Returns "" when the
|
|
// header is missing, malformed, or carries an unrecognised cause —
|
|
// the AuthProvider falls back to the generic "Session expired" UX
|
|
// in that case (forward-compat with future categories).
|
|
function parseWWWAuthenticateCause(header: string | null): string {
|
|
if (!header) return '';
|
|
const m = header.match(/error_description="([^"]+)"/i);
|
|
if (!m) return '';
|
|
const cause = m[1];
|
|
switch (cause) {
|
|
case 'idle_timeout':
|
|
case 'absolute_timeout':
|
|
case 'back_channel_revoked':
|
|
case 'invalid_token':
|
|
return cause;
|
|
default:
|
|
return '';
|
|
}
|
|
}
|
|
|
|
// isStateChangingMethod mirrors the server-side
|
|
// internal/auth/session/middleware.go::isStateChangingMethod predicate.
|
|
// State-changing requests get the X-CSRF-Token header auto-attached
|
|
// when in session-cookie mode; safe methods don't need it.
|
|
function isStateChangingMethod(method?: string): boolean {
|
|
switch ((method || 'GET').toUpperCase()) {
|
|
case 'POST':
|
|
case 'PUT':
|
|
case 'DELETE':
|
|
case 'PATCH':
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
async function fetchJSON<T>(url: string, init?: RequestInit): Promise<T> {
|
|
// Bundle 2 Phase 8 — credentials:'include' lets the certctl_session
|
|
// cookie ride along on every request. Bearer-mode deployments work
|
|
// unchanged (the cookie just isn't there). Auto-attach X-CSRF-Token
|
|
// header on state-changing methods when the cookie is present.
|
|
const headers: Record<string, string> = { ...authHeaders(), ...(init?.headers as Record<string, string> | undefined) };
|
|
if (isStateChangingMethod(init?.method)) {
|
|
const csrf = readCSRFCookie();
|
|
if (csrf && !headers['X-CSRF-Token']) {
|
|
headers['X-CSRF-Token'] = csrf;
|
|
}
|
|
}
|
|
const res = await fetch(url, {
|
|
...init,
|
|
credentials: 'include',
|
|
headers, // intentional: spread init first, then override headers with the merged map (init.headers already merged into `headers` above)
|
|
});
|
|
if (res.status === 401) {
|
|
// Audit 2026-05-10 HIGH-8 — propagate the WWW-Authenticate
|
|
// error_description so the AuthProvider can route the user into
|
|
// OIDC-aware re-login UX instead of generic "session expired."
|
|
// Stable cause categories: idle_timeout, absolute_timeout,
|
|
// back_channel_revoked, invalid_token. Anything else is treated
|
|
// as invalid_token by the server-side classifier.
|
|
const cause = parseWWWAuthenticateCause(res.headers.get('WWW-Authenticate'));
|
|
const event = new CustomEvent('certctl:auth-required', { detail: { cause } });
|
|
window.dispatchEvent(event);
|
|
throw new Error('Authentication required');
|
|
}
|
|
if (!res.ok) {
|
|
let errorMsg = res.statusText;
|
|
try {
|
|
const body = await res.json();
|
|
errorMsg = body.message || body.error || errorMsg;
|
|
} catch {
|
|
// Response body is not JSON, use status text
|
|
}
|
|
throw new Error(errorMsg || `HTTP ${res.status}`);
|
|
}
|
|
if (res.status === 204) return {} as T;
|
|
return res.json();
|
|
}
|
|
|
|
// Auth
|
|
//
|
|
// Bundle 2 Phase 6 / Category E — /auth/info now optionally returns
|
|
// the list of configured OIDC providers (id + display_name + login_url)
|
|
// when the server has any configured. The Login page renders the
|
|
// "Sign in with X" buttons from this list; older servers (pre-Phase-6)
|
|
// just return {auth_type, required} and the GUI falls back to the
|
|
// API-key form. Both shapes are valid; oidc_providers is an
|
|
// optional field on the wire.
|
|
export interface AuthInfoOIDCProvider {
|
|
id: string;
|
|
display_name: string;
|
|
login_url: string;
|
|
}
|
|
export interface AuthInfoResponse {
|
|
auth_type: string;
|
|
required: boolean;
|
|
oidc_providers?: AuthInfoOIDCProvider[];
|
|
}
|
|
export const getAuthInfo = () =>
|
|
fetch(`${BASE}/auth/info`, { headers: { 'Content-Type': 'application/json' } })
|
|
.then(r => r.json() as Promise<AuthInfoResponse>);
|
|
|
|
// AuthCheckResponse mirrors the /auth/check handler payload. Post-M-003 it
|
|
// surfaces `user` (named-key identity) and `admin` (named-key admin flag) so
|
|
// the GUI can gate admin-only affordances. When CERTCTL_AUTH_TYPE=none the
|
|
// backend returns {user: "", admin: false}.
|
|
export interface AuthCheckResponse {
|
|
status: string;
|
|
user: string;
|
|
admin: boolean;
|
|
}
|
|
|
|
export const checkAuth = (key: string) =>
|
|
fetch(`${BASE}/auth/check`, {
|
|
headers: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${key}` },
|
|
}).then(r => {
|
|
if (!r.ok) throw new Error('Invalid API key');
|
|
return r.json() as Promise<AuthCheckResponse>;
|
|
});
|
|
|
|
// =============================================================================
|
|
// Bundle 1 Phase 10 — RBAC management API surface.
|
|
//
|
|
// Backs the Roles / Keys / Auth Settings GUI pages (web/src/pages/auth/*).
|
|
// Every function maps 1:1 to a Phase-4 / Phase-7 server endpoint;
|
|
// permission gates fire server-side, the GUI's permission-aware
|
|
// renders are a UX layer on top.
|
|
// =============================================================================
|
|
|
|
export interface AuthRole {
|
|
id: string;
|
|
tenant_id: string;
|
|
name: string;
|
|
description?: string;
|
|
created_at?: string;
|
|
updated_at?: string;
|
|
}
|
|
|
|
export interface AuthRolePermission {
|
|
role_id: string;
|
|
permission_id: string;
|
|
scope_type: 'global' | 'profile' | 'issuer';
|
|
scope_id?: string;
|
|
}
|
|
|
|
export interface AuthPermission {
|
|
id: string;
|
|
name: string;
|
|
namespace: string;
|
|
}
|
|
|
|
export interface AuthEffectivePermission {
|
|
permission: string;
|
|
scope_type: 'global' | 'profile' | 'issuer';
|
|
scope_id?: string;
|
|
}
|
|
|
|
export interface AuthMeResponse {
|
|
actor_id: string;
|
|
actor_type: string;
|
|
tenant_id: string;
|
|
admin: boolean;
|
|
roles: string[];
|
|
effective_permissions: AuthEffectivePermission[];
|
|
}
|
|
|
|
export interface AuthKeyEntry {
|
|
actor_id: string;
|
|
actor_type: string;
|
|
tenant_id: string;
|
|
role_ids: string[];
|
|
}
|
|
|
|
export const authMe = () => fetchJSON<AuthMeResponse>(`${BASE}/auth/me`);
|
|
|
|
export const authListRoles = () =>
|
|
fetchJSON<{ roles: AuthRole[] }>(`${BASE}/auth/roles`).then(r => r.roles);
|
|
|
|
export const authGetRole = (id: string) =>
|
|
fetchJSON<{ role: AuthRole; permissions: AuthRolePermission[] }>(
|
|
`${BASE}/auth/roles/${id}`,
|
|
);
|
|
|
|
export const authCreateRole = (body: { name: string; description?: string }) =>
|
|
fetchJSON<AuthRole>(`${BASE}/auth/roles`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
export const authUpdateRole = (id: string, body: { name: string; description?: string }) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/roles/${id}`, {
|
|
method: 'PUT',
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
export const authDeleteRole = (id: string) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/roles/${id}`, { method: 'DELETE' });
|
|
|
|
export const authListPermissions = () =>
|
|
fetchJSON<{ permissions: AuthPermission[] }>(`${BASE}/auth/permissions`).then(
|
|
r => r.permissions,
|
|
);
|
|
|
|
export const authAddRolePermission = (
|
|
roleId: string,
|
|
body: { permission: string; scope_type?: string; scope_id?: string },
|
|
) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/roles/${roleId}/permissions`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
export const authRemoveRolePermission = (roleId: string, perm: string) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/roles/${roleId}/permissions/${perm}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
export const authListKeys = () =>
|
|
fetchJSON<{ keys: AuthKeyEntry[] }>(`${BASE}/auth/keys`).then(r => r.keys);
|
|
|
|
// Audit 2026-05-10 HIGH-10 — extended grant body. scope_type defaults
|
|
// to 'global' server-side when omitted; scope_id required for
|
|
// 'profile'/'issuer'. expires_at is RFC3339; omitted = no expiry.
|
|
export interface AssignKeyRoleOptions {
|
|
scope_type?: 'global' | 'profile' | 'issuer';
|
|
scope_id?: string;
|
|
expires_at?: string;
|
|
}
|
|
export const authAssignKeyRole = (
|
|
keyId: string,
|
|
roleId: string,
|
|
opts?: AssignKeyRoleOptions,
|
|
) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/keys/${keyId}/roles`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ role_id: roleId, ...(opts ?? {}) }),
|
|
});
|
|
|
|
// =============================================================================
|
|
// Audit 2026-05-10 — GUI batch additions.
|
|
// =============================================================================
|
|
|
|
// MED-11 — federated users.
|
|
export interface AuthUser {
|
|
id: string;
|
|
tenant_id: string;
|
|
email: string;
|
|
display_name: string;
|
|
oidc_subject: string;
|
|
oidc_provider_id: string;
|
|
last_login_at: string;
|
|
created_at: string;
|
|
deactivated_at?: string;
|
|
}
|
|
export const authListUsers = (providerID?: string) => {
|
|
const q = providerID ? `?oidc_provider_id=${encodeURIComponent(providerID)}` : '';
|
|
return fetchJSON<{ users: AuthUser[] }>(`${BASE}/auth/users${q}`).then(r => r.users);
|
|
};
|
|
export const authDeactivateUser = (id: string) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/users/${id}`, { method: 'DELETE' });
|
|
// Audit 2026-05-11 A-2 — inverse of authDeactivateUser. Clears
|
|
// users.deactivated_at; next OIDC login proceeds normally.
|
|
export const authReactivateUser = (id: string) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/users/${id}/reactivate`, { method: 'POST' });
|
|
|
|
// MED-12 — runtime config.
|
|
export const authRuntimeConfig = () =>
|
|
fetchJSON<{ runtime_config: Record<string, string> }>(`${BASE}/auth/runtime-config`)
|
|
.then(r => r.runtime_config);
|
|
|
|
// MED-7 — JWKS status.
|
|
export interface JWKSStatusSnapshot {
|
|
last_refresh_at?: string;
|
|
current_kids: string[];
|
|
refresh_count: number;
|
|
last_error?: string;
|
|
rejected_jws_count: number;
|
|
iss_param_supported: boolean;
|
|
}
|
|
export const authOIDCJWKSStatus = (providerID: string) =>
|
|
fetchJSON<JWKSStatusSnapshot>(`${BASE}/auth/oidc/providers/${providerID}/jwks-status`);
|
|
|
|
// MED-5 — OIDC provider test (dry-run).
|
|
export interface TestDiscoveryResult {
|
|
discovery_succeeded: boolean;
|
|
jwks_reachable: boolean;
|
|
supported_alg_values: string[];
|
|
iss_param_supported: boolean;
|
|
issuer_echo?: string;
|
|
authorization_url?: string;
|
|
token_url?: string;
|
|
jwks_uri?: string;
|
|
userinfo_endpoint?: string;
|
|
errors?: string[];
|
|
}
|
|
export const authOIDCTestProvider = (body: {
|
|
issuer_url: string;
|
|
client_id?: string;
|
|
client_secret?: string;
|
|
scopes?: string[];
|
|
}) =>
|
|
fetchJSON<TestDiscoveryResult>(`${BASE}/auth/oidc/test`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
export const authRevokeKeyRole = (keyId: string, roleId: string) =>
|
|
fetchJSON<unknown>(`${BASE}/auth/keys/${keyId}/roles/${roleId}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
export interface BootstrapAvailability {
|
|
available: boolean;
|
|
}
|
|
|
|
export const authBootstrapAvailable = () =>
|
|
fetch(`${BASE}/auth/bootstrap`, {
|
|
headers: { 'Content-Type': 'application/json' },
|
|
}).then(r => r.json() as Promise<BootstrapAvailability>);
|
|
|
|
// =============================================================================
|
|
// Bundle 2 Phase 8 — OIDC providers + group mappings + sessions +
|
|
// break-glass admin API surface. Backs:
|
|
// - LoginPage (OIDC provider buttons + breakglass form)
|
|
// - OIDCProvidersPage + OIDCProviderDetailPage
|
|
// - GroupMappingsPage
|
|
// - SessionsPage (own + admin)
|
|
// - ProfilePage session-list panel
|
|
//
|
|
// Every function maps 1:1 to a Phase 5 / Phase 7.5 server endpoint;
|
|
// permission gates fire server-side, the GUI's permission-aware
|
|
// renders are a UX layer on top.
|
|
// =============================================================================
|
|
|
|
export interface OIDCProvider {
|
|
id: string;
|
|
tenant_id: string;
|
|
name: string;
|
|
issuer_url: string;
|
|
client_id: string;
|
|
redirect_uri: string;
|
|
groups_claim_path: string;
|
|
groups_claim_format: string;
|
|
fetch_userinfo: boolean;
|
|
scopes: string[];
|
|
allowed_email_domains?: string[];
|
|
iat_window_seconds: number;
|
|
jwks_cache_ttl_seconds: number;
|
|
created_at: string;
|
|
updated_at: string;
|
|
}
|
|
|
|
export interface OIDCProviderRequest {
|
|
name: string;
|
|
issuer_url: string;
|
|
client_id: string;
|
|
client_secret?: string; // sent on create + rotate; omitted on edit-without-rotate
|
|
redirect_uri: string;
|
|
groups_claim_path?: string;
|
|
groups_claim_format?: string;
|
|
fetch_userinfo?: boolean;
|
|
scopes?: string[];
|
|
allowed_email_domains?: string[];
|
|
iat_window_seconds?: number;
|
|
jwks_cache_ttl_seconds?: number;
|
|
}
|
|
|
|
export interface GroupRoleMapping {
|
|
id: string;
|
|
provider_id: string;
|
|
group_name: string;
|
|
role_id: string;
|
|
tenant_id: string;
|
|
created_at: string;
|
|
}
|
|
|
|
export interface SessionInfo {
|
|
id: string;
|
|
actor_id: string;
|
|
actor_type: string;
|
|
ip_address?: string;
|
|
user_agent?: string;
|
|
created_at: string;
|
|
last_seen_at: string;
|
|
idle_expires_at: string;
|
|
absolute_expires_at: string;
|
|
revoked: boolean;
|
|
}
|
|
|
|
// OIDC provider CRUD (auth.oidc.list / .create / .edit / .delete).
|
|
export const listOIDCProviders = () =>
|
|
fetchJSON<{ providers: OIDCProvider[] }>(`${BASE}/auth/oidc/providers`);
|
|
|
|
export const createOIDCProvider = (req: OIDCProviderRequest) =>
|
|
fetchJSON<OIDCProvider>(`${BASE}/auth/oidc/providers`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(req),
|
|
});
|
|
|
|
export const updateOIDCProvider = (id: string, req: OIDCProviderRequest) =>
|
|
fetchJSON<OIDCProvider>(`${BASE}/auth/oidc/providers/${encodeURIComponent(id)}`, {
|
|
method: 'PUT',
|
|
body: JSON.stringify(req),
|
|
});
|
|
|
|
export const deleteOIDCProvider = (id: string) =>
|
|
fetchJSON<void>(`${BASE}/auth/oidc/providers/${encodeURIComponent(id)}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
export const refreshOIDCProvider = (id: string) =>
|
|
fetchJSON<{ refreshed: boolean }>(`${BASE}/auth/oidc/providers/${encodeURIComponent(id)}/refresh`, {
|
|
method: 'POST',
|
|
});
|
|
|
|
// Group→role mapping CRUD (auth.oidc.list / .edit).
|
|
export const listGroupMappings = (providerID: string) =>
|
|
fetchJSON<{ mappings: GroupRoleMapping[] }>(
|
|
`${BASE}/auth/oidc/group-mappings?provider_id=${encodeURIComponent(providerID)}`,
|
|
);
|
|
|
|
export const addGroupMapping = (providerID: string, groupName: string, roleID: string) =>
|
|
fetchJSON<GroupRoleMapping>(`${BASE}/auth/oidc/group-mappings`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ provider_id: providerID, group_name: groupName, role_id: roleID }),
|
|
});
|
|
|
|
export const removeGroupMapping = (id: string) =>
|
|
fetchJSON<void>(`${BASE}/auth/oidc/group-mappings/${encodeURIComponent(id)}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
// Session list + revoke. The GET also accepts ?actor_id=<other>
|
|
// for the admin all-actors view (auth.session.list.all gated server-
|
|
// side; see internal/api/router::router.go).
|
|
export const listSessions = (actorID?: string, actorType?: string) => {
|
|
const q = actorID ? `?actor_id=${encodeURIComponent(actorID)}${actorType ? '&actor_type=' + encodeURIComponent(actorType) : ''}` : '';
|
|
return fetchJSON<{ sessions: SessionInfo[] }>(`${BASE}/auth/sessions${q}`);
|
|
};
|
|
|
|
export const revokeSession = (sessionID: string) =>
|
|
fetchJSON<void>(`${BASE}/auth/sessions/${encodeURIComponent(sessionID)}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
// Logout — POST /auth/logout. Auth-exempt (the handler accepts the
|
|
// caller's session cookie OR a missing cookie; both 204).
|
|
export const logout = () =>
|
|
fetch(`/auth/logout`, { method: 'POST', credentials: 'include' }).then(r => {
|
|
if (!r.ok && r.status !== 204) throw new Error(`logout failed: ${r.status}`);
|
|
});
|
|
|
|
// =============================================================================
|
|
// Bundle 2 Phase 7.5 — break-glass admin surface. The login endpoint
|
|
// is auth-exempt; the admin endpoints require auth.breakglass.admin.
|
|
// All four endpoints return 404 when CERTCTL_BREAKGLASS_ENABLED=false
|
|
// (surface invisibility).
|
|
// =============================================================================
|
|
|
|
export const breakglassLogin = (actorID: string, password: string) =>
|
|
fetch(`/auth/breakglass/login`, {
|
|
method: 'POST',
|
|
credentials: 'include',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ actor_id: actorID, password }),
|
|
}).then(async r => {
|
|
if (r.status === 204) return;
|
|
if (r.status === 404) throw new Error('break-glass admin not enabled on this server');
|
|
if (!r.ok) throw new Error('invalid credentials');
|
|
});
|
|
|
|
export const breakglassSetPassword = (targetActorID: string, password: string) =>
|
|
fetchJSON<{ actor_id: string; created_at: string }>(`${BASE}/auth/breakglass/credentials`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ actor_id: targetActorID, password }),
|
|
});
|
|
|
|
export const breakglassUnlock = (targetActorID: string) =>
|
|
fetchJSON<void>(`${BASE}/auth/breakglass/credentials/${encodeURIComponent(targetActorID)}/unlock`, {
|
|
method: 'POST',
|
|
});
|
|
|
|
export const breakglassRemove = (targetActorID: string) =>
|
|
fetchJSON<void>(`${BASE}/auth/breakglass/credentials/${encodeURIComponent(targetActorID)}`, {
|
|
method: 'DELETE',
|
|
});
|
|
|
|
export type BreakglassCredentialRow = {
|
|
actor_id: string;
|
|
created_at: string;
|
|
last_password_change_at: string;
|
|
failure_count: number;
|
|
locked_until?: string;
|
|
last_failure_at?: string;
|
|
};
|
|
|
|
// Audit 2026-05-10 CRIT-4 closure — admin GUI Break-glass page. The
|
|
// password hash is never returned by the server; this lists only the
|
|
// metadata operators need to render the credentialed-actor table.
|
|
// Returns 404 when CERTCTL_BREAKGLASS_ENABLED=false (surface invisibility).
|
|
export const breakglassListCredentials = () =>
|
|
fetchJSON<{ credentials: BreakglassCredentialRow[] }>(`${BASE}/auth/breakglass/credentials`)
|
|
.then(r => r.credentials);
|
|
|
|
// =============================================================================
|
|
// Bundle 1 Phase 10 — approvals queue.
|
|
//
|
|
// Backs ApprovalsPage. Bundle-1's ApprovalKind enum includes
|
|
// `cert_issuance` (existing) and `profile_edit` (Phase 9). The list
|
|
// surface returns both kinds; the page renders them with a kind
|
|
// pill so an approver can tell them apart at a glance.
|
|
// =============================================================================
|
|
|
|
export type ApprovalKind = 'cert_issuance' | 'profile_edit';
|
|
export type ApprovalState = 'pending' | 'approved' | 'rejected' | 'expired';
|
|
|
|
export interface ApprovalRequest {
|
|
id: string;
|
|
kind: ApprovalKind;
|
|
certificate_id?: string;
|
|
job_id?: string;
|
|
profile_id: string;
|
|
requested_by: string;
|
|
state: ApprovalState;
|
|
decided_by?: string;
|
|
decided_at?: string;
|
|
decision_note?: string;
|
|
metadata?: Record<string, string>;
|
|
payload?: string; // base64 / raw JSON pass-through
|
|
created_at: string;
|
|
updated_at: string;
|
|
}
|
|
|
|
export const listApprovals = (state: ApprovalState = 'pending') =>
|
|
fetchJSON<PaginatedResponse<ApprovalRequest>>(`${BASE}/approvals?state=${state}`);
|
|
|
|
export const approveApproval = (id: string, note: string) =>
|
|
fetchJSON<unknown>(`${BASE}/approvals/${id}/approve`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ note }),
|
|
});
|
|
|
|
export const rejectApproval = (id: string, note: string) =>
|
|
fetchJSON<unknown>(`${BASE}/approvals/${id}/reject`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ note }),
|
|
});
|
|
|
|
// Certificates
|
|
export const getCertificates = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Certificate>>(`${BASE}/certificates?${qs}`);
|
|
};
|
|
|
|
export const getCertificate = (id: string) =>
|
|
fetchJSON<Certificate>(`${BASE}/certificates/${id}`);
|
|
|
|
export const getCertificateVersions = (id: string) =>
|
|
fetchJSON<PaginatedResponse<CertificateVersion>>(`${BASE}/certificates/${id}/versions`);
|
|
|
|
export const createCertificate = (data: Partial<Certificate>) =>
|
|
fetchJSON<Certificate>(`${BASE}/certificates`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const triggerRenewal = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/certificates/${id}/renew`, { method: 'POST' });
|
|
|
|
export const updateCertificate = (id: string, data: Partial<Certificate>) =>
|
|
fetchJSON<Certificate>(`${BASE}/certificates/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const archiveCertificate = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/certificates/${id}`, { method: 'DELETE' });
|
|
|
|
export const triggerDeployment = (id: string, targetId: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/certificates/${id}/deploy`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ target_id: targetId }),
|
|
});
|
|
|
|
export const revokeCertificate = (id: string, reason: string) =>
|
|
fetchJSON<{ status: string }>(`${BASE}/certificates/${id}/revoke`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ reason }),
|
|
});
|
|
|
|
export interface BulkRevokeCriteria {
|
|
reason: string;
|
|
profile_id?: string;
|
|
owner_id?: string;
|
|
agent_id?: string;
|
|
issuer_id?: string;
|
|
team_id?: string;
|
|
certificate_ids?: string[];
|
|
}
|
|
|
|
export interface BulkRevokeResult {
|
|
total_matched: number;
|
|
total_revoked: number;
|
|
total_skipped: number;
|
|
total_failed: number;
|
|
errors?: { certificate_id: string; error: string }[];
|
|
}
|
|
|
|
export const bulkRevokeCertificates = (criteria: BulkRevokeCriteria) =>
|
|
fetchJSON<BulkRevokeResult>(`${BASE}/certificates/bulk-revoke`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(criteria),
|
|
});
|
|
|
|
// L-1 master closure (cat-l-fa0c1ac07ab5): bulk renew. Mirrors
|
|
// BulkRevokeCriteria field-for-field so operators who already know the
|
|
// bulk-revoke contract have zero new surface to learn. Pre-L-1 the GUI
|
|
// looped `await triggerRenewal(id)` over the selection; 100 certs = 100
|
|
// HTTP round-trips. Post-L-1 it's a single POST returning per-cert
|
|
// {certificate_id, job_id} pairs in enqueued_jobs and per-cert errors
|
|
// in errors. The "renew all certs of profile X" use case is the
|
|
// canonical reason to support criteria-mode in addition to explicit IDs.
|
|
export interface BulkRenewalCriteria {
|
|
profile_id?: string;
|
|
owner_id?: string;
|
|
agent_id?: string;
|
|
issuer_id?: string;
|
|
team_id?: string;
|
|
certificate_ids?: string[];
|
|
}
|
|
|
|
export interface BulkRenewalResult {
|
|
total_matched: number;
|
|
total_enqueued: number;
|
|
total_skipped: number;
|
|
total_failed: number;
|
|
enqueued_jobs?: { certificate_id: string; job_id: string }[];
|
|
errors?: { certificate_id: string; error: string }[];
|
|
}
|
|
|
|
export const bulkRenewCertificates = (criteria: BulkRenewalCriteria) =>
|
|
fetchJSON<BulkRenewalResult>(`${BASE}/certificates/bulk-renew`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(criteria),
|
|
});
|
|
|
|
// L-2 closure (cat-l-8a1fb258a38a): bulk reassign owner (and optionally
|
|
// team) for a set of certificates. Narrower than bulk-renew — explicit
|
|
// IDs only, no criteria-mode (operators query first, then reassign by
|
|
// ID). Pre-L-2 the GUI looped `await updateCertificate(id, { owner_id })`.
|
|
// owner_id is required; team_id is optional and updates only when
|
|
// non-empty (matches the existing per-cert PUT contract).
|
|
export interface BulkReassignmentRequest {
|
|
certificate_ids: string[];
|
|
owner_id: string;
|
|
team_id?: string;
|
|
}
|
|
|
|
export interface BulkReassignmentResult {
|
|
total_matched: number;
|
|
total_reassigned: number;
|
|
total_skipped: number;
|
|
total_failed: number;
|
|
errors?: { certificate_id: string; error: string }[];
|
|
}
|
|
|
|
export const bulkReassignCertificates = (request: BulkReassignmentRequest) =>
|
|
fetchJSON<BulkReassignmentResult>(`${BASE}/certificates/bulk-reassign`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(request),
|
|
});
|
|
|
|
// Certificate Export
|
|
//
|
|
// B-1 master closure (cat-b-9b97ffb35ef7): the previous `exportCertificatePEM`
|
|
// helper that returned `{cert_pem, chain_pem, full_pem}` JSON was removed —
|
|
// it had zero consumers across web/, MCP, CLI, and tests, and was a dead
|
|
// duplicate of `downloadCertificatePEM` which is the only call site that
|
|
// actually exists in `CertificateDetailPage` (browser file-download path).
|
|
// If a JSON variant is ever needed again, re-add an explicit fetcher with a
|
|
// page consumer in the same commit; do not resurrect the orphan.
|
|
export const downloadCertificatePEM = (id: string) => {
|
|
const headers: Record<string, string> = {};
|
|
if (apiKey) headers['Authorization'] = `Bearer ${apiKey}`;
|
|
return fetch(`${BASE}/certificates/${id}/export/pem?download=true`, { headers })
|
|
.then(r => {
|
|
if (!r.ok) throw new Error('Export failed');
|
|
return r.blob();
|
|
});
|
|
};
|
|
|
|
export const exportCertificatePKCS12 = (id: string, password: string = '') => {
|
|
const headers: Record<string, string> = { 'Content-Type': 'application/json' };
|
|
if (apiKey) headers['Authorization'] = `Bearer ${apiKey}`;
|
|
return fetch(`${BASE}/certificates/${id}/export/pkcs12`, {
|
|
method: 'POST',
|
|
headers,
|
|
body: JSON.stringify({ password }),
|
|
}).then(r => {
|
|
if (!r.ok) throw new Error('Export failed');
|
|
return r.blob();
|
|
});
|
|
};
|
|
|
|
// Certificate Deployments
|
|
export const getCertificateDeployments = (id: string, params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Job>>(`${BASE}/certificates/${id}/deployments?${qs}`);
|
|
};
|
|
|
|
// OCSP (RFC 6960) — served unauthenticated under /.well-known/pki/ per RFC 8615
|
|
// (M-006 relocation). The legacy JSON CRL endpoint (`GET /api/v1/crl`) was
|
|
// removed entirely; relying parties fetch the DER-encoded CRL directly from
|
|
// `/.well-known/pki/crl/{issuer_id}` (no GUI wrapper — binary download only).
|
|
export const getOCSPStatus = (issuerId: string, serial: string) => {
|
|
// No Authorization header — the OCSP responder is intentionally unauthenticated
|
|
// so relying parties without certctl API keys can check revocation status.
|
|
return fetch(`/.well-known/pki/ocsp/${issuerId}/${serial}`)
|
|
.then(r => {
|
|
if (!r.ok) throw new Error(`OCSP request failed: ${r.status}`);
|
|
return r.arrayBuffer();
|
|
});
|
|
};
|
|
|
|
// CRL/OCSP-Responder Phase 5: GUI-side helper for the "Test CRL fetch" button
|
|
// on CertificateDetailPage. Fetches the DER-encoded CRL from the well-known
|
|
// endpoint and returns the byte length so the panel can show "OK — N bytes".
|
|
// The Authorization header is intentionally omitted: /.well-known/pki/crl/ is
|
|
// the standards-compliant relying-party surface and runs unauthenticated.
|
|
export const fetchCRL = (issuerId: string) => {
|
|
return fetch(`/.well-known/pki/crl/${issuerId}`)
|
|
.then(async r => {
|
|
if (!r.ok) throw new Error(`CRL fetch failed: ${r.status}`);
|
|
const buf = await r.arrayBuffer();
|
|
return { byteLength: buf.byteLength, contentType: r.headers.get('content-type') ?? '' };
|
|
});
|
|
};
|
|
|
|
// CRL/OCSP-Responder Phase 5 admin endpoint mirror.
|
|
//
|
|
// Backend handler: internal/api/handler/admin_crl_cache.go::ListCache.
|
|
// M-008 admin-gated; non-admin Bearer callers get HTTP 403 — the GUI hides
|
|
// the badge entirely (rather than letting it 403 noisily) by gating the
|
|
// React-Query enabled flag on useAuth().admin at the call site.
|
|
export const getAdminCRLCache = () =>
|
|
fetchJSON<CRLCacheResponse>(`${BASE}/admin/crl/cache`);
|
|
|
|
// SCEP RFC 8894 + Intune master bundle Phase 9.2 admin endpoint mirror.
|
|
//
|
|
// Backend handler: internal/api/handler/admin_scep_intune.go.
|
|
// Both endpoints are M-008 admin-gated; the SCEPAdminPage component
|
|
// gates the React-Query `enabled` flag on useAuth().admin so non-admin
|
|
// callers never see the page (the route itself is also conditional on
|
|
// the admin flag in main.tsx).
|
|
export const getAdminSCEPIntuneStats = () =>
|
|
fetchJSON<IntuneStatsResponse>(`${BASE}/admin/scep/intune/stats`);
|
|
|
|
export const reloadAdminSCEPIntuneTrust = (pathID: string) =>
|
|
fetchJSON<IntuneReloadTrustResponse>(`${BASE}/admin/scep/intune/reload-trust`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ path_id: pathID }),
|
|
});
|
|
|
|
// SCEP RFC 8894 + Intune master bundle Phase 9 follow-up
|
|
// (the project's SCEP GUI restructure spec): per-profile SCEP admin
|
|
// surface backing the Profiles tab on the SCEP Administration page.
|
|
// M-008 admin-gated; same gating semantics as the existing
|
|
// getAdminSCEPIntuneStats helper.
|
|
export const getAdminSCEPProfiles = () =>
|
|
fetchJSON<SCEPProfilesResponse>(`${BASE}/admin/scep/profiles`);
|
|
|
|
// EST RFC 7030 hardening master bundle Phase 7.2 admin endpoints.
|
|
//
|
|
// Backend handler: internal/api/handler/admin_est.go.
|
|
// Both endpoints are M-008 admin-gated; the ESTAdminPage component
|
|
// gates the React-Query `enabled` flag on useAuth().admin so non-admin
|
|
// callers never see the page (the route itself is also conditional on
|
|
// the admin flag in main.tsx).
|
|
export const getAdminESTProfiles = () =>
|
|
fetchJSON<ESTProfilesResponse>(`${BASE}/admin/est/profiles`);
|
|
|
|
export const reloadAdminESTTrust = (pathID: string) =>
|
|
fetchJSON<ESTReloadTrustResponse>(`${BASE}/admin/est/reload-trust`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ path_id: pathID }),
|
|
});
|
|
|
|
// SCEP RFC 8894 + Intune master bundle Phase 11.5: SCEP probe
|
|
// (capability + posture). Synchronous — the caller blocks until the
|
|
// probe completes (cap: 30s server-side). Persists to the history
|
|
// table that listSCEPProbes reads from.
|
|
export const probeSCEPServer = (url: string) =>
|
|
fetchJSON<SCEPProbeResult>(`${BASE}/network-scan/scep-probe`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ url }),
|
|
});
|
|
|
|
export const listSCEPProbes = () =>
|
|
fetchJSON<SCEPProbesResponse>(`${BASE}/network-scan/scep-probes`);
|
|
|
|
// Agents
|
|
export const getAgents = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Agent>>(`${BASE}/agents?${qs}`);
|
|
};
|
|
|
|
export const getAgent = (id: string) =>
|
|
fetchJSON<Agent>(`${BASE}/agents/${id}`);
|
|
|
|
// C-1 closure (cat-b-6177f36636fb): registerAgent is intentionally
|
|
// orphan in the GUI per certctl's pull-only deployment model. Agents
|
|
// enroll via install-agent.sh + cmd/agent/main.go and register
|
|
// themselves at first heartbeat — operators don't (and shouldn't)
|
|
// drive registration from the dashboard. The client fn is preserved
|
|
// here (rather than deleted) so future features that want to drive
|
|
// registration from the GUI (e.g. a one-click "register proxy agent"
|
|
// panel for network-appliance topologies) can reach the endpoint
|
|
// without a client.ts edit. See docs/architecture.md::Agents for
|
|
// the architectural rationale and unified-audit.md cat-b-6177f36636fb
|
|
// for closure rationale.
|
|
export const registerAgent = (data: Partial<Agent>) =>
|
|
fetchJSON<Agent>(`${BASE}/agents`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
// I-004: typed error thrown by retireAgent when the server returns HTTP 409 with
|
|
// {error: "blocked_by_dependencies", ...}. Callers that want to show the
|
|
// dependency-counts dialog should `catch (e)` and check `e instanceof
|
|
// BlockedByDependenciesError` — the counts field is the same shape the
|
|
// backend handler returns from its inline struct in
|
|
// internal/api/handler/agents.go. Generic network / 5xx failures still throw
|
|
// plain Error so existing error-boundary code is unaffected.
|
|
export class BlockedByDependenciesError extends Error {
|
|
readonly counts: AgentDependencyCounts;
|
|
constructor(message: string, counts: AgentDependencyCounts) {
|
|
super(message);
|
|
this.name = 'BlockedByDependenciesError';
|
|
this.counts = counts;
|
|
}
|
|
}
|
|
|
|
// I-004: retire an agent via DELETE /api/v1/agents/{id}. Three distinct
|
|
// success paths the UI needs to distinguish:
|
|
// * 200 — fresh retire; body has retired_at, already_retired=false, cascade
|
|
// flag, counts of what was cascaded.
|
|
// * 204 — idempotent re-retire; the row was already retired. No body. We
|
|
// synthesize a RetireAgentResponse with already_retired=true and zero
|
|
// counts so the caller can keep a single return type.
|
|
// * 409 — blocked_by_dependencies; thrown as BlockedByDependenciesError so
|
|
// the caller can surface the active_targets/active_certificates/pending_jobs
|
|
// counts in a confirmation dialog and offer force=true.
|
|
// Anything else bubbles up via the standard fetchJSON error path.
|
|
export const retireAgent = async (
|
|
id: string,
|
|
opts: { force?: boolean; reason?: string } = {},
|
|
): Promise<RetireAgentResponse> => {
|
|
const qs = new URLSearchParams();
|
|
if (opts.force) qs.set('force', 'true');
|
|
if (opts.reason) qs.set('reason', opts.reason);
|
|
const url = qs.toString()
|
|
? `${BASE}/agents/${id}?${qs.toString()}`
|
|
: `${BASE}/agents/${id}`;
|
|
|
|
const res = await fetch(url, {
|
|
method: 'DELETE',
|
|
headers: authHeaders(),
|
|
});
|
|
|
|
if (res.status === 401) {
|
|
// Audit 2026-05-10 HIGH-8 — see fetchAPI() for the cause-extraction rationale.
|
|
const cause = parseWWWAuthenticateCause(res.headers.get('WWW-Authenticate'));
|
|
window.dispatchEvent(new CustomEvent('certctl:auth-required', { detail: { cause } }));
|
|
throw new Error('Authentication required');
|
|
}
|
|
|
|
// 204 No Content — idempotent re-retire. Synthesize a response so callers
|
|
// get a uniform shape; already_retired=true tells them the agent was
|
|
// already in the retired state before this call.
|
|
if (res.status === 204) {
|
|
return {
|
|
retired_at: '',
|
|
already_retired: true,
|
|
cascade: false,
|
|
counts: { active_targets: 0, active_certificates: 0, pending_jobs: 0 },
|
|
};
|
|
}
|
|
|
|
if (res.status === 409) {
|
|
// Body is always JSON for 409 per the handler contract.
|
|
const body = (await res.json()) as BlockedByDependenciesResponse;
|
|
throw new BlockedByDependenciesError(
|
|
body.message || 'agent has active dependencies',
|
|
body.counts,
|
|
);
|
|
}
|
|
|
|
if (!res.ok) {
|
|
let errorMsg = res.statusText;
|
|
try {
|
|
const body = await res.json();
|
|
errorMsg = body.message || body.error || errorMsg;
|
|
} catch {
|
|
// not JSON
|
|
}
|
|
throw new Error(errorMsg || `HTTP ${res.status}`);
|
|
}
|
|
|
|
return (await res.json()) as RetireAgentResponse;
|
|
};
|
|
|
|
// I-004: list retired agents via GET /api/v1/agents/retired. Kept separate
|
|
// from getAgents (which hits the default active-only listing) so the retired
|
|
// tab on AgentsPage can page independently. per_page is capped server-side at
|
|
// 500 (see handler ListRetiredAgents).
|
|
export const listRetiredAgents = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Agent>>(`${BASE}/agents/retired?${qs}`);
|
|
};
|
|
|
|
// Jobs
|
|
export const getJobs = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Job>>(`${BASE}/jobs?${qs}`);
|
|
};
|
|
|
|
export const cancelJob = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/jobs/${id}/cancel`, { method: 'POST' });
|
|
|
|
// Notifications
|
|
export const getNotifications = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Notification>>(`${BASE}/notifications?${qs}`);
|
|
};
|
|
|
|
export const getNotification = (id: string) =>
|
|
fetchJSON<Notification>(`${BASE}/notifications/${id}`);
|
|
|
|
export const markNotificationRead = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/notifications/${id}/read`, { method: 'POST' });
|
|
|
|
/**
|
|
* I-005: requeue a dead notification back to the retry queue. Flips status
|
|
* 'dead' → 'pending' and clears next_retry_at so the retry sweep picks it up
|
|
* on its next tick (default 2 minutes, CERTCTL_NOTIFICATION_RETRY_INTERVAL).
|
|
* Used by the Dead letter tab's "Requeue" button after an operator fixes the
|
|
* underlying delivery failure (SMTP config, webhook endpoint, etc.). The
|
|
* handler returns a StatusResponse ({ status: "requeued" }) — the frontend
|
|
* only needs to know the call succeeded so the mutation can invalidate the
|
|
* notifications query.
|
|
*/
|
|
export const requeueNotification = (id: string) =>
|
|
fetchJSON<{ status: string }>(`${BASE}/notifications/${id}/requeue`, { method: 'POST' });
|
|
|
|
// Audit
|
|
export const getAuditEvents = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '200', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<AuditEvent>>(`${BASE}/audit?${qs}`);
|
|
};
|
|
|
|
export const getAuditEvent = (id: string) =>
|
|
fetchJSON<AuditEvent>(`${BASE}/audit/${id}`);
|
|
|
|
// Policies
|
|
export const getPolicies = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<PolicyRule>>(`${BASE}/policies?${qs}`);
|
|
};
|
|
|
|
export const createPolicy = (data: Partial<PolicyRule>) =>
|
|
fetchJSON<PolicyRule>(`${BASE}/policies`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updatePolicy = (id: string, data: Partial<PolicyRule>) =>
|
|
fetchJSON<PolicyRule>(`${BASE}/policies/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const getPolicy = (id: string) =>
|
|
fetchJSON<PolicyRule>(`${BASE}/policies/${id}`);
|
|
|
|
export const deletePolicy = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/policies/${id}`, { method: 'DELETE' });
|
|
|
|
export const getPolicyViolations = (id: string) =>
|
|
fetchJSON<PaginatedResponse<PolicyViolation>>(`${BASE}/policies/${id}/violations`);
|
|
|
|
// G-1: Renewal Policies (/api/v1/renewal-policies) — lifecycle policies with
|
|
// rp-* IDs in the renewal_policies table. Distinct from getPolicies() above
|
|
// which hits /api/v1/policies and returns PolicyRule (compliance, pol-* IDs).
|
|
// OnboardingWizard, CertificatesPage, and CertificateDetailPage populate the
|
|
// `renewal_policy_id` dropdown from this endpoint; populating it from
|
|
// getPolicies() produced FK violations on certificate insert/update.
|
|
export const getRenewalPolicies = (page = 1, perPage = 50) => {
|
|
const qs = new URLSearchParams({ page: String(page), per_page: String(perPage) }).toString();
|
|
return fetchJSON<PaginatedResponse<RenewalPolicy>>(`${BASE}/renewal-policies?${qs}`);
|
|
};
|
|
|
|
export const getRenewalPolicy = (id: string) =>
|
|
fetchJSON<RenewalPolicy>(`${BASE}/renewal-policies/${id}`);
|
|
|
|
export const createRenewalPolicy = (data: Partial<RenewalPolicy>) =>
|
|
fetchJSON<RenewalPolicy>(`${BASE}/renewal-policies`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateRenewalPolicy = (id: string, data: Partial<RenewalPolicy>) =>
|
|
fetchJSON<RenewalPolicy>(`${BASE}/renewal-policies/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteRenewalPolicy = (id: string) =>
|
|
fetchJSON<void>(`${BASE}/renewal-policies/${id}`, { method: 'DELETE' });
|
|
|
|
// Issuers
|
|
export const getIssuers = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Issuer>>(`${BASE}/issuers?${qs}`);
|
|
};
|
|
|
|
export const createIssuer = (data: Partial<Issuer>) =>
|
|
fetchJSON<Issuer>(`${BASE}/issuers`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const testIssuerConnection = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}/test`, { method: 'POST' });
|
|
|
|
export const updateIssuer = (id: string, data: Partial<Issuer>) =>
|
|
fetchJSON<Issuer>(`${BASE}/issuers/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteIssuer = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}`, { method: 'DELETE' });
|
|
|
|
// Targets
|
|
export const getTargets = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Target>>(`${BASE}/targets?${qs}`);
|
|
};
|
|
|
|
export const createTarget = (data: Partial<Target>) =>
|
|
fetchJSON<Target>(`${BASE}/targets`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateTarget = (id: string, data: Partial<Target>) =>
|
|
fetchJSON<Target>(`${BASE}/targets/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteTarget = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/targets/${id}`, { method: 'DELETE' });
|
|
|
|
export const testTargetConnection = (id: string) =>
|
|
fetchJSON<{ status: string; message: string }>(`${BASE}/targets/${id}/test`, { method: 'POST' });
|
|
|
|
// Profiles
|
|
export const getProfiles = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<CertificateProfile>>(`${BASE}/profiles?${qs}`);
|
|
};
|
|
|
|
export const getProfile = (id: string) =>
|
|
fetchJSON<CertificateProfile>(`${BASE}/profiles/${id}`);
|
|
|
|
export const createProfile = (data: Partial<CertificateProfile>) =>
|
|
fetchJSON<CertificateProfile>(`${BASE}/profiles`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateProfile = (id: string, data: Partial<CertificateProfile>) =>
|
|
fetchJSON<CertificateProfile>(`${BASE}/profiles/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteProfile = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/profiles/${id}`, { method: 'DELETE' });
|
|
|
|
// Owners
|
|
export const getOwners = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Owner>>(`${BASE}/owners?${qs}`);
|
|
};
|
|
|
|
export const getOwner = (id: string) =>
|
|
fetchJSON<Owner>(`${BASE}/owners/${id}`);
|
|
|
|
export const createOwner = (data: Partial<Owner>) =>
|
|
fetchJSON<Owner>(`${BASE}/owners`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateOwner = (id: string, data: Partial<Owner>) =>
|
|
fetchJSON<Owner>(`${BASE}/owners/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteOwner = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/owners/${id}`, { method: 'DELETE' });
|
|
|
|
// Teams
|
|
export const getTeams = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<Team>>(`${BASE}/teams?${qs}`);
|
|
};
|
|
|
|
export const getTeam = (id: string) =>
|
|
fetchJSON<Team>(`${BASE}/teams/${id}`);
|
|
|
|
export const createTeam = (data: Partial<Team>) =>
|
|
fetchJSON<Team>(`${BASE}/teams`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateTeam = (id: string, data: Partial<Team>) =>
|
|
fetchJSON<Team>(`${BASE}/teams/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteTeam = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/teams/${id}`, { method: 'DELETE' });
|
|
|
|
// Agent Groups
|
|
export const getAgentGroups = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<AgentGroup>>(`${BASE}/agent-groups?${qs}`);
|
|
};
|
|
|
|
export const getAgentGroup = (id: string) =>
|
|
fetchJSON<AgentGroup>(`${BASE}/agent-groups/${id}`);
|
|
|
|
export const createAgentGroup = (data: Partial<AgentGroup>) =>
|
|
fetchJSON<AgentGroup>(`${BASE}/agent-groups`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateAgentGroup = (id: string, data: Partial<AgentGroup>) =>
|
|
fetchJSON<AgentGroup>(`${BASE}/agent-groups/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteAgentGroup = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/agent-groups/${id}`, { method: 'DELETE' });
|
|
|
|
export const getAgentGroupMembers = (id: string) =>
|
|
fetchJSON<PaginatedResponse<Agent>>(`${BASE}/agent-groups/${id}/members`);
|
|
|
|
// Renewal Approvals
|
|
export const approveRenewal = (jobId: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/jobs/${jobId}/approve`, { method: 'POST' });
|
|
|
|
export const rejectRenewal = (jobId: string, reason: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/jobs/${jobId}/reject`, { method: 'POST', body: JSON.stringify({ reason }) });
|
|
|
|
// Discovery
|
|
export const getDiscoveredCertificates = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<DiscoveredCertificate>>(`${BASE}/discovered-certificates?${qs}`);
|
|
};
|
|
|
|
export const getDiscoveredCertificate = (id: string) =>
|
|
fetchJSON<DiscoveredCertificate>(`${BASE}/discovered-certificates/${id}`);
|
|
|
|
export const claimDiscoveredCertificate = (id: string, managedCertificateId: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/discovered-certificates/${id}/claim`, {
|
|
method: 'POST',
|
|
body: JSON.stringify({ managed_certificate_id: managedCertificateId }),
|
|
});
|
|
|
|
export const dismissDiscoveredCertificate = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/discovered-certificates/${id}/dismiss`, { method: 'POST' });
|
|
|
|
export const getDiscoveryScans = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<DiscoveryScan>>(`${BASE}/discovery-scans?${qs}`);
|
|
};
|
|
|
|
export const getDiscoverySummary = () =>
|
|
fetchJSON<DiscoverySummary>(`${BASE}/discovery-summary`);
|
|
|
|
// Network Scan Targets
|
|
export const getNetworkScanTargets = (params: Record<string, string> = {}) => {
|
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
|
return fetchJSON<PaginatedResponse<NetworkScanTarget>>(`${BASE}/network-scan-targets?${qs}`);
|
|
};
|
|
|
|
export const getNetworkScanTarget = (id: string) =>
|
|
fetchJSON<NetworkScanTarget>(`${BASE}/network-scan-targets/${id}`);
|
|
|
|
export const createNetworkScanTarget = (data: Partial<NetworkScanTarget>) =>
|
|
fetchJSON<NetworkScanTarget>(`${BASE}/network-scan-targets`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateNetworkScanTarget = (id: string, data: Partial<NetworkScanTarget>) =>
|
|
fetchJSON<NetworkScanTarget>(`${BASE}/network-scan-targets/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteNetworkScanTarget = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/network-scan-targets/${id}`, { method: 'DELETE' });
|
|
|
|
export const triggerNetworkScan = (id: string) =>
|
|
fetchJSON<{ message: string }>(`${BASE}/network-scan-targets/${id}/scan`, { method: 'POST' });
|
|
|
|
// Stats
|
|
export const getDashboardSummary = () =>
|
|
fetchJSON<DashboardSummary>(`${BASE}/stats/summary`);
|
|
|
|
export const getCertificatesByStatus = () =>
|
|
fetchJSON<CertificateStatusCount[]>(`${BASE}/stats/certificates-by-status`);
|
|
|
|
export const getExpirationTimeline = (days = 30) =>
|
|
fetchJSON<ExpirationBucket[]>(`${BASE}/stats/expiration-timeline?days=${days}`);
|
|
|
|
export const getJobTrends = (days = 30) =>
|
|
fetchJSON<JobTrendDataPoint[]>(`${BASE}/stats/job-trends?days=${days}`);
|
|
|
|
export const getIssuanceRate = (days = 30) =>
|
|
fetchJSON<IssuanceRateDataPoint[]>(`${BASE}/stats/issuance-rate?days=${days}`);
|
|
|
|
export const getMetrics = () =>
|
|
fetchJSON<MetricsResponse>(`${BASE}/metrics`);
|
|
|
|
// Digest
|
|
export const previewDigest = () => {
|
|
const headers: Record<string, string> = {};
|
|
if (apiKey) headers['Authorization'] = `Bearer ${apiKey}`;
|
|
return fetch(`${BASE}/digest/preview`, { headers })
|
|
.then(r => {
|
|
if (!r.ok) throw new Error(`Digest preview failed: ${r.status}`);
|
|
return r.text();
|
|
});
|
|
};
|
|
|
|
export const sendDigest = () =>
|
|
fetchJSON<{ message: string }>(`${BASE}/digest/send`, { method: 'POST' });
|
|
|
|
// Jobs (single)
|
|
export const getJob = (id: string) =>
|
|
fetchJSON<Job>(`${BASE}/jobs/${id}`);
|
|
|
|
// Job Verification
|
|
export const getJobVerification = (id: string) =>
|
|
fetchJSON<{ job_id: string; target_id: string; verified: boolean; actual_fingerprint: string; expected_fingerprint: string; verified_at: string; error?: string }>(`${BASE}/jobs/${id}/verification`);
|
|
|
|
// Issuers (single)
|
|
export const getIssuer = (id: string) =>
|
|
fetchJSON<Issuer>(`${BASE}/issuers/${id}`);
|
|
|
|
// Targets (single)
|
|
export const getTarget = (id: string) =>
|
|
fetchJSON<Target>(`${BASE}/targets/${id}`);
|
|
|
|
// Prometheus metrics (text format)
|
|
export const getPrometheusMetrics = () => {
|
|
const headers: Record<string, string> = {};
|
|
if (apiKey) headers['Authorization'] = `Bearer ${apiKey}`;
|
|
return fetch(`${BASE}/metrics/prometheus`, { headers })
|
|
.then(r => {
|
|
if (!r.ok) throw new Error(`Prometheus metrics failed: ${r.status}`);
|
|
return r.text();
|
|
});
|
|
};
|
|
|
|
// Health
|
|
export const getHealth = () => fetchJSON<{ status: string }>('/health');
|
|
|
|
// Health checks (M48)
|
|
export const listHealthChecks = (params?: { status?: string; certificate_id?: string; enabled?: string; page?: number; per_page?: number }): Promise<PaginatedResponse<EndpointHealthCheck>> => {
|
|
const query = new URLSearchParams();
|
|
if (params?.status) query.set('status', params.status);
|
|
if (params?.certificate_id) query.set('certificate_id', params.certificate_id);
|
|
if (params?.enabled) query.set('enabled', params.enabled);
|
|
if (params?.page) query.set('page', String(params.page));
|
|
if (params?.per_page) query.set('per_page', String(params.per_page));
|
|
const qs = query.toString();
|
|
return fetchJSON<PaginatedResponse<EndpointHealthCheck>>(`${BASE}/health-checks${qs ? '?' + qs : ''}`);
|
|
};
|
|
|
|
export const getHealthCheck = (id: string) =>
|
|
fetchJSON<EndpointHealthCheck>(`${BASE}/health-checks/${id}`);
|
|
|
|
export const createHealthCheck = (data: Partial<EndpointHealthCheck>) =>
|
|
fetchJSON<EndpointHealthCheck>(`${BASE}/health-checks`, { method: 'POST', body: JSON.stringify(data) });
|
|
|
|
export const updateHealthCheck = (id: string, data: Partial<EndpointHealthCheck>) =>
|
|
fetchJSON<EndpointHealthCheck>(`${BASE}/health-checks/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
|
|
|
export const deleteHealthCheck = (id: string) =>
|
|
fetchJSON<void>(`${BASE}/health-checks/${id}`, { method: 'DELETE' });
|
|
|
|
export const getHealthCheckHistory = (id: string, limit?: number) => {
|
|
const query = limit ? `?limit=${limit}` : '';
|
|
return fetchJSON<HealthHistoryEntry[]>(`${BASE}/health-checks/${id}/history${query}`);
|
|
};
|
|
|
|
export const acknowledgeHealthCheck = (id: string) =>
|
|
fetchJSON<void>(`${BASE}/health-checks/${id}/acknowledge`, { method: 'POST', body: JSON.stringify({}) });
|
|
|
|
export const getHealthCheckSummary = () =>
|
|
fetchJSON<HealthCheckSummary>(`${BASE}/health-checks/summary`);
|
|
|
|
// IntermediateCA hierarchy (Rank 8 of the 2026-05-03 deep-research
|
|
// deliverable). Admin-gated at the handler layer; non-admin Bearer
|
|
// callers get 403. Operators drive the hierarchy from
|
|
// IssuerHierarchyPage; the recursive tree render is built from the
|
|
// flat list returned here by walking each row's parent_ca_id.
|
|
export interface IntermediateCA {
|
|
id: string;
|
|
owning_issuer_id: string;
|
|
parent_ca_id?: string | null;
|
|
name: string;
|
|
subject: string;
|
|
state: 'active' | 'retiring' | 'retired';
|
|
cert_pem: string;
|
|
key_driver_id: string;
|
|
not_before: string;
|
|
not_after: string;
|
|
path_len_constraint?: number | null;
|
|
name_constraints?: { permitted?: string[]; excluded?: string[] }[];
|
|
ocsp_responder_url?: string;
|
|
metadata?: Record<string, string>;
|
|
created_at: string;
|
|
updated_at: string;
|
|
}
|
|
|
|
export const listIntermediateCAs = (issuerID: string) =>
|
|
fetchJSON<{ data: IntermediateCA[] }>(`${BASE}/issuers/${issuerID}/intermediates`);
|
|
|
|
export const getIntermediateCA = (id: string) =>
|
|
fetchJSON<IntermediateCA>(`${BASE}/intermediates/${id}`);
|
|
|
|
export const retireIntermediateCA = (id: string, note: string, confirm: boolean) =>
|
|
fetchJSON<{ id: string; decided_by: string; confirmed: boolean }>(
|
|
`${BASE}/intermediates/${id}/retire`,
|
|
{ method: 'POST', body: JSON.stringify({ note, confirm }) },
|
|
);
|