generated from coulomb/repo-seed
A2 — Compilation fixes: - Remove inline FK constraints from Schema.sql; IHP schema compiler cannot parse them. Add 1744329600-restore-fk-constraints.sql migration to restore referential integrity at the DB level. - Rename `#label` → `#label_` throughout to avoid clash with Haskell built-in. - Fix `hub.id == hid` UUID comparisons to use `toUUID hub.id`. - Replace non-existent `setStatus`/`respondJson` calls with `renderJsonWithStatusCode` throughout Api controllers. - Fix qualified package import for `cryptohash-sha256` in Auth.hs. - Add `CanSelect (Text, Text)` instance in Helper.View. - Refactor HSX inline lambdas to named helper functions in 100+ views (GHC cannot infer types for anonymous functions inside quasi-quoted HSX). - Fix missing imports (IHP.QueryBuilder, IHP.Fetch, Web.Routes, Only, etc.) across helpers and controllers. - Remove duplicate `diffUTCTime` definition in BottleneckDetector. - Change `createEventForHub` return type from `IO ResponseReceived` to `IO ()`. - Seed type-registry vocabulary via 1744502400-seed-type-registries.sql (moved from Schema.sql where IHP does not execute INSERT statements). A3 — Tailwind build pipeline: - Add `tailwindcss` to flake.nix native packages. - Uncomment `tailwind.exec` process in devenv shell config. - Add tailwind/tailwind.config.js (scans Web/View/**/*.hs). - Add tailwind/app.css with @tailwind directives. A4 — Admin user seed: - Add 1744416000-seed-admin-user.sql: inserts admin@inter-hub.local with bcrypt-hashed password admin1234! (cost 10). - Add .env.example documenting all required environment variables and default admin credentials. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
93 lines
3.6 KiB
Haskell
93 lines
3.6 KiB
Haskell
module Web.View.CrossHubPropagations.Index where
|
||
|
||
import Web.Types
|
||
import Generated.Types
|
||
import IHP.Prelude
|
||
import IHP.ViewPrelude
|
||
import Web.Routes ()
|
||
|
||
data IndexView = IndexView
|
||
{ propagations :: ![CrossHubPropagation]
|
||
, hubs :: ![Hub]
|
||
}
|
||
|
||
instance View IndexView where
|
||
html IndexView { .. } = [hsx|
|
||
<div class="flex items-center justify-between mb-6">
|
||
<h1 class="text-2xl font-semibold">Cross-Hub Propagations</h1>
|
||
<a href={DetectPropagationsAction}
|
||
class="text-sm bg-indigo-600 text-white px-3 py-1.5 rounded hover:bg-indigo-700">
|
||
Detect
|
||
</a>
|
||
</div>
|
||
|
||
{renderPropagationsList propagations hubs}
|
||
|]
|
||
renderPropagationsList :: [CrossHubPropagation] -> [Hub] -> Html
|
||
renderPropagationsList [] _ = [hsx|<p class="text-sm text-gray-400">No propagation events detected yet.</p>|]
|
||
renderPropagationsList propagations hubs = [hsx|
|
||
<div class="bg-white rounded-lg border border-gray-200 overflow-hidden">
|
||
<table class="w-full text-sm">
|
||
<thead class="bg-gray-50 border-b border-gray-200">
|
||
<tr>
|
||
<th class="text-left px-4 py-3 font-medium text-gray-700">Pattern</th>
|
||
<th class="text-left px-4 py-3 font-medium text-gray-700">Summary</th>
|
||
<th class="text-left px-4 py-3 font-medium text-gray-700">Source Hub</th>
|
||
<th class="text-left px-4 py-3 font-medium text-gray-700">Status</th>
|
||
<th class="text-left px-4 py-3 font-medium text-gray-700">Detected</th>
|
||
<th class="px-4 py-3"></th>
|
||
</tr>
|
||
</thead>
|
||
<tbody class="divide-y divide-gray-100">
|
||
{forEach propagations (renderPropRow hubs)}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
|]
|
||
|
||
renderPropRow :: [Hub] -> CrossHubPropagation -> Html
|
||
renderPropRow hubs p =
|
||
let hubName hid = maybe "–" (.name) (find (\h -> h.id == hid) hubs)
|
||
in [hsx|
|
||
<tr class="hover:bg-gray-50">
|
||
<td class="px-4 py-3">
|
||
<span class="bg-purple-100 text-purple-700 text-xs px-1.5 py-0.5 rounded">
|
||
{p.patternType}
|
||
</span>
|
||
</td>
|
||
<td class="px-4 py-3 text-gray-700">{p.summary}</td>
|
||
<td class="px-4 py-3 text-gray-500 text-xs">
|
||
{maybe "–" hubName p.sourceHubId}
|
||
</td>
|
||
<td class="px-4 py-3">
|
||
<span class={statusBadge p.status <> " text-xs px-2 py-0.5 rounded font-medium"}>
|
||
{p.status}
|
||
</span>
|
||
</td>
|
||
<td class="px-4 py-3 text-xs text-gray-400">{show p.detectedAt}</td>
|
||
<td class="px-4 py-3 text-right">
|
||
{renderAcknowledgeLink p}
|
||
{renderResolveLink p}
|
||
</td>
|
||
</tr>
|
||
|]
|
||
|
||
renderAcknowledgeLink :: CrossHubPropagation -> Html
|
||
renderAcknowledgeLink p
|
||
| p.status == "open" = [hsx|<a href={AcknowledgePropagationAction (p.id)}
|
||
class="text-xs text-yellow-600 hover:underline mr-2">Acknowledge</a>|]
|
||
| otherwise = mempty
|
||
|
||
renderResolveLink :: CrossHubPropagation -> Html
|
||
renderResolveLink p
|
||
| p.status /= "resolved" = [hsx|<a href={ResolvePropagationAction (p.id)}
|
||
class="text-xs text-green-600 hover:underline">Resolve</a>|]
|
||
| otherwise = mempty
|
||
|
||
statusBadge :: Text -> Text
|
||
statusBadge s = case s of
|
||
"open" -> "bg-yellow-100 text-yellow-800"
|
||
"acknowledged" -> "bg-blue-100 text-blue-800"
|
||
"resolved" -> "bg-green-100 text-green-800"
|
||
_ -> "bg-gray-100 text-gray-600"
|