Files
inter-hub/Web/Controller/GovernanceTemplates.hs
Bernd Worsch 3737845e02 fix(WP-0017/E4): Layer 3 error fixes — round 2 (18 files)
Fixes 46 compile errors across 18 controllers and views:
- BridgeResponse missing from explicit import lists (Widgets, RequirementCandidates,
  DecisionRecords, AgentDelegations) — dot-notation HasField resolution fails without
  the type in scope under DuplicateRecordFields
- unId not in IHP v1.5 — replaced all fmap (Id . unId) with fmap coerce
- respondWith not in IHP — replaced with plain redirectTo in 5 controllers
- [hubId] list param to sqlQuery — replaced with (Only hubId) tuple
- deleteWhere not in IHP — replaced with query/filterWhere/fetch/deleteRecords
- fill @'["label"] mismatch — field is label_ in generated types, not label
- PersistUUID/toUUID (persistent-style) — replaced with (Only id)
- intercalate + jsonArrayTexts ambiguity in GovernanceTemplates — hid Index import,
  removed local duplicates, added Data.Text (intercalate)
- Int16 not in scope in AntifragilityDashboard — changed to Int (score :: Int)
- typeArraySection type mismatch in HubCapabilityManifests/Edit — unified to [Text]
- renderForm arity mismatch — added action param to DecisionRecords/New.renderForm
- Missing qualified Data.Aeson import in AdaptiveThresholds
- Missing ?request::Request constraint in Api/V2/WidgetPatterns.renderJsonWithStatus

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-12 12:17:45 +00:00

146 lines
7.1 KiB
Haskell

module Web.Controller.GovernanceTemplates where
import Web.Types
import Web.View.GovernanceTemplates.Index hiding (jsonArrayTexts)
import Web.View.GovernanceTemplates.Show
import Web.View.GovernanceTemplates.New
import Generated.Types
import IHP.Prelude
import IHP.ControllerPrelude
import Data.Aeson (Value(..), decode, encode, toJSON)
import qualified Data.ByteString.Lazy as LBS
import Data.Text (intercalate)
instance Controller GovernanceTemplatesController where
beforeAction = ensureIsUser
-- List published templates with clone count
action GovernanceTemplatesAction = autoRefresh do
templates <- sqlQuery
"SELECT gt.*, COUNT(gtc.id) AS clone_count \
\ FROM governance_templates gt \
\ LEFT JOIN governance_template_clones gtc ON gtc.governance_template_id = gt.id \
\ WHERE gt.is_published = TRUE \
\ GROUP BY gt.id \
\ ORDER BY clone_count DESC, gt.name ASC"
()
render IndexView { templates }
-- Template detail with clone count
action ShowGovernanceTemplateAction { governanceTemplateId } = do
template <- fetch governanceTemplateId
hub <- fetch template.hubId
cloneCount <- sqlQueryScalar
"SELECT COUNT(*) FROM governance_template_clones WHERE governance_template_id = ?"
(Only governanceTemplateId)
render ShowView { template, hub, cloneCount = fromMaybe 0 cloneCount }
action NewGovernanceTemplateAction = do
hubs <- query @Hub |> orderByAsc #name |> fetch
categories <- sqlQuery
"SELECT name, label FROM annotation_category_registry WHERE status = 'active' ORDER BY label"
()
let template = newRecord @GovernanceTemplate
render NewView { template, hubs, categories }
action CreateGovernanceTemplateAction = do
hubs <- query @Hub |> orderByAsc #name |> fetch
categories <- sqlQuery
"SELECT name, label FROM annotation_category_registry WHERE status = 'active' ORDER BY label"
()
let template = newRecord @GovernanceTemplate
let selectedCats = paramList @Text "categories"
let templateBodyRaw = param @Text "templateBody"
let mBody = decode (LBS.fromStrict (cs templateBodyRaw)) :: Maybe Value
case mBody of
Nothing -> do
setErrorMessage "Template body must be valid JSON."
render NewView { template, hubs, categories }
Just bodyVal -> do
-- Validate each selected category is in the registry
mErrors <- validateCategories selectedCats
case mErrors of
Left unknown -> do
setErrorMessage ("Unknown categories: " <> intercalate ", " unknown)
render NewView { template, hubs, categories }
Right () -> do
template
|> fill @'["hubId", "name", "description"]
|> set #categories (toJSON selectedCats)
|> set #templateBody bodyVal
|> set #isPublished False
|> validateField #name nonEmpty
|> validateField #hubId nonEmpty
|> ifValid \case
Left template -> render NewView { template, hubs, categories }
Right template -> do
t <- createRecord template
setSuccessMessage "Governance template created"
redirectTo ShowGovernanceTemplateAction { governanceTemplateId = t.id }
-- Clone template + manifest amendment if needed
action CloneGovernanceTemplateAction { governanceTemplateId } = do
template <- fetch governanceTemplateId
hubId <- getUserHubId
existing <- query @GovernanceTemplateClone
|> filterWhere (#governanceTemplateId, governanceTemplateId)
|> filterWhere (#cloningHubId, hubId)
|> fetchOneOrNothing
case existing of
Just _ -> do
setSuccessMessage "Your hub has already cloned this template."
redirectTo ShowGovernanceTemplateAction { governanceTemplateId }
Nothing -> do
newRecord @GovernanceTemplateClone
|> set #governanceTemplateId governanceTemplateId
|> set #cloningHubId hubId
|> createRecord
-- Check if template categories are in hub's manifest
mManifest <- query @HubCapabilityManifest
|> filterWhere (#hubId, hubId)
|> filterWhere (#status, "active")
|> fetchOneOrNothing
let templateCats = jsonArrayTexts template.categories
let existingCats = maybe [] (jsonArrayTexts . (.declaredAnnotationCategories)) mManifest
let missingCats = filter (`notElem` existingCats) templateCats
if not (null missingCats)
then do
let newCats = existingCats ++ missingCats
draft <- newRecord @HubCapabilityManifest
|> set #hubId hubId
|> set #status "draft"
|> set #declaredWidgetTypes
(maybe (toJSON ([] :: [Text])) (.declaredWidgetTypes) mManifest)
|> set #declaredEventTypes
(maybe (toJSON ([] :: [Text])) (.declaredEventTypes) mManifest)
|> set #declaredAnnotationCategories (toJSON newCats)
|> set #declaredPolicyScopes
(maybe (toJSON ([] :: [Text])) (.declaredPolicyScopes) mManifest)
|> createRecord
setSuccessMessage "Template cloned. A manifest amendment draft has been created for the new categories."
redirectTo ShowHubCapabilityManifestAction { hubCapabilityManifestId = draft.id }
else do
setSuccessMessage "Template cloned."
redirectTo ShowGovernanceTemplateAction { governanceTemplateId }
-- | Validate that all category names exist in the active annotation_category_registry.
validateCategories ::
(?modelContext :: ModelContext) =>
[Text] -> IO (Either [Text] ())
validateCategories cats = do
registered <- sqlQuery
"SELECT name FROM annotation_category_registry WHERE status = 'active'"
()
let known = map (\(Only n) -> n) (registered :: [Only Text])
let unknown = filter (`notElem` known) cats
pure $ if null unknown then Right () else Left unknown
-- | Resolve the hub for the current session (first hub fallback).
getUserHubId :: (?modelContext :: ModelContext, ?context :: ControllerContext) => IO (Id Hub)
getUserHubId = do
hubs <- query @Hub |> limit 1 |> fetch
case hubs of
(h:_) -> pure h.id
[] -> error "No hubs found"