feat(WP-0009): IHF GAAF Compliance Foundation — type registries, extension manifests, architectural contracts
Some checks failed
Test / test (push) Has been cancelled

Implements IHUB-WP-0009: closes four GAAF-2026 gaps before domain hub work begins.
- TypeRegistry helper + controllers/views (hub_kind, hub_capability_manifest)
- HubCapabilityManifest entity with validation and registry linkage
- ARCHITECTURE-LAYERS.md + CI-enforced boundary contracts
- Alembic migration 1743724800, fitness tests (Test/Architecture/)
- GAAF spec, Operational Architecture spec, domain hub extension guide
- Updates to CLAUDE.md, SCOPE.md, Schema.sql, Routes, FrontController, Types

state_hub_sync: pending (tunnel was STALE at completion time; run fix-consistency)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-31 21:17:39 +00:00
parent 1a7732d7da
commit b5d73aa18b
47 changed files with 4855 additions and 104 deletions

View File

@@ -545,3 +545,159 @@ ALTER TABLE widgets
CREATE INDEX widgets_is_archived_idx ON widgets (is_archived)
WHERE is_archived = TRUE;
-- ============================================================
-- GAAF Compliance Foundation (IHUB-WP-0009)
-- T02: hub_kind | T03: type registries + seed | T04: maturity columns | T05: manifests
-- ============================================================
-- T02 — Hub kind classification
ALTER TABLE hubs
ADD COLUMN hub_kind TEXT NOT NULL DEFAULT 'domain';
CREATE INDEX hubs_hub_kind_idx ON hubs (hub_kind);
CREATE UNIQUE INDEX hubs_one_framework_idx ON hubs (hub_kind)
WHERE hub_kind = 'framework';
-- T03 — Type registries
CREATE TABLE widget_type_registry (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
name TEXT NOT NULL UNIQUE,
label TEXT NOT NULL,
description TEXT,
owner_hub_id UUID REFERENCES hubs(id),
status TEXT NOT NULL DEFAULT 'active',
deprecated_in_favour_of TEXT,
created_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL
);
CREATE INDEX widget_type_registry_status_idx ON widget_type_registry (status);
CREATE INDEX widget_type_registry_owner_hub_idx ON widget_type_registry (owner_hub_id);
CREATE TABLE event_type_registry (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
name TEXT NOT NULL UNIQUE,
label TEXT NOT NULL,
description TEXT,
owner_hub_id UUID REFERENCES hubs(id),
status TEXT NOT NULL DEFAULT 'active',
deprecated_in_favour_of TEXT,
created_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL
);
CREATE INDEX event_type_registry_status_idx ON event_type_registry (status);
CREATE INDEX event_type_registry_owner_hub_idx ON event_type_registry (owner_hub_id);
CREATE TABLE annotation_category_registry (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
name TEXT NOT NULL UNIQUE,
label TEXT NOT NULL,
description TEXT,
owner_hub_id UUID REFERENCES hubs(id),
status TEXT NOT NULL DEFAULT 'active',
deprecated_in_favour_of TEXT,
created_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL
);
CREATE INDEX annotation_category_registry_status_idx ON annotation_category_registry (status);
CREATE INDEX annotation_category_registry_owner_hub_idx ON annotation_category_registry (owner_hub_id);
CREATE TABLE policy_scope_registry (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
name TEXT NOT NULL UNIQUE,
label TEXT NOT NULL,
description TEXT,
owner_hub_id UUID REFERENCES hubs(id),
status TEXT NOT NULL DEFAULT 'active',
deprecated_in_favour_of TEXT,
created_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL
);
CREATE INDEX policy_scope_registry_status_idx ON policy_scope_registry (status);
CREATE INDEX policy_scope_registry_owner_hub_idx ON policy_scope_registry (owner_hub_id);
-- T03 — Seed framework-level vocabulary (owner_hub_id = NULL)
INSERT INTO widget_type_registry (name, label, description) VALUES
('chart', 'Chart', 'Data visualisation chart widget'),
('form', 'Form', 'Data entry form widget'),
('table', 'Table', 'Tabular data display widget'),
('action', 'Action Control', 'Button, link, or trigger widget'),
('panel', 'Status Panel', 'Summary or status information panel'),
('workflow-step', 'Workflow Step', 'Single step in a multi-step workflow'),
('recommendation','Recommendation', 'AI or system recommendation block'),
('chat', 'Chat Region', 'Conversational interaction region'),
('diff', 'Diff / Review', 'Code diff or change review element');
INSERT INTO event_type_registry (name, label, description) VALUES
('viewed', 'Viewed', 'Widget was rendered and visible to the user'),
('focused', 'Focused', 'Widget received input focus'),
('clicked', 'Clicked', 'Widget was clicked or tapped'),
('submitted', 'Submitted', 'Form or action was submitted'),
('abandoned', 'Abandoned', 'User navigated away without completing'),
('retried', 'Retried', 'Action was retried after failure'),
('failed', 'Failed', 'Action or submission resulted in an error'),
('commented', 'Commented', 'User added a comment or annotation'),
('flagged_confusing', 'Flagged Confusing', 'User flagged the widget as confusing'),
('flagged_helpful', 'Flagged Helpful', 'User flagged the widget as helpful'),
('blocked_by_policy', 'Blocked by Policy', 'Action was blocked by a policy rule'),
('escalated', 'Escalated', 'Issue was escalated for review'),
('accepted_recommendation', 'Accepted Recommendation', 'User accepted an AI recommendation'),
('rejected_recommendation', 'Rejected Recommendation', 'User rejected an AI recommendation'),
('retracted', 'Retracted', 'Correction marker referencing original event in metadata');
INSERT INTO annotation_category_registry (name, label, description) VALUES
('friction', 'Friction', 'Interaction caused user effort or difficulty'),
('missing_capability', 'Missing Capability', 'Required feature or function is absent'),
('policy_conflict', 'Policy Conflict', 'Widget behaviour conflicts with a policy'),
('trust_deficit', 'Trust Deficit', 'User lacks confidence in the widget output'),
('accessibility', 'Accessibility', 'Accessibility or inclusive design concern'),
('workflow_bottleneck', 'Workflow Bottleneck', 'Widget creates a slowdown in the workflow'),
('documentation_gap', 'Documentation Gap', 'Missing or insufficient documentation'),
('product_opportunity', 'Product Opportunity', 'Observation suggesting a product improvement'),
('governance_concern', 'Governance Concern', 'Concern about governance, audit, or compliance');
INSERT INTO policy_scope_registry (name, label, description) VALUES
('internal', 'Internal', 'Applies to internal operators only'),
('org-wide', 'Organisation-Wide', 'Applies across the entire organisation'),
('external', 'External-Facing', 'Applies to externally visible surfaces'),
('regulatory', 'Regulatory', 'Driven by regulatory or compliance requirements'),
('security', 'Security', 'Security policy scope');
-- T04 — Maturity columns on existing contract tables
ALTER TABLE envelope_emission_contracts
ADD COLUMN maturity TEXT NOT NULL DEFAULT 'stable';
ALTER TABLE interaction_reporting_contracts
ADD COLUMN maturity TEXT NOT NULL DEFAULT 'stable';
ALTER TABLE widget_adapter_specs
ADD COLUMN maturity TEXT NOT NULL DEFAULT 'beta';
-- T05 — Hub Capability Manifest
CREATE TABLE hub_capability_manifests (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
hub_id UUID NOT NULL UNIQUE REFERENCES hubs(id),
manifest_version TEXT NOT NULL DEFAULT '1.0',
declared_widget_types JSONB NOT NULL DEFAULT '[]',
declared_event_types JSONB NOT NULL DEFAULT '[]',
declared_annotation_categories JSONB NOT NULL DEFAULT '[]',
declared_policy_scopes JSONB NOT NULL DEFAULT '[]',
capability_description TEXT,
contact TEXT,
status TEXT NOT NULL DEFAULT 'draft',
activated_at TIMESTAMP WITH TIME ZONE,
created_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL,
updated_at TIMESTAMP WITH TIME ZONE DEFAULT now() NOT NULL
);
CREATE INDEX hub_capability_manifests_hub_id_idx ON hub_capability_manifests (hub_id);
CREATE INDEX hub_capability_manifests_status_idx ON hub_capability_manifests (status);
-- GAAF: type registries enforced from here (IHUB-WP-0009)
-- All new type discriminator columns (widget_type, event_type, category,
-- policy_scope) must reference a registry table or carry a CHECK constraint.