fix(WP-0017/E5): Layer 3 error fixes — round 3 (24 files)

Int16→Int in score/stars functions; uuid-based readMay→UUID.fromText;
autoRefresh do-notation fix; id→\x->x ambiguity in HubRoutingRules;
MarketplaceDashboard replaced raw SQL with IHP query builder; optional
hub selector in TypeRegistry views via CanSelect (Text, Maybe Id) instance
added to Web.View.Prelude; import consolidations to Web.View.Prelude.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-12 13:11:32 +00:00
parent 3737845e02
commit 2c22766cd6
24 changed files with 81 additions and 110 deletions

View File

@@ -5,8 +5,6 @@ import Web.View.MarketplaceDashboard.Show
import Generated.Types
import IHP.Prelude
import IHP.ControllerPrelude
import Database.PostgreSQL.Simple (Query)
instance Controller MarketplaceDashboardController where
beforeAction = ensureIsUser
@@ -15,15 +13,32 @@ instance Controller MarketplaceDashboardController where
let mWType = paramOrNothing @Text "widgetType"
let sortBy = paramOrDefault @Text "adopted" "sort"
-- Widget patterns: full-text search + filter
patterns <- sqlQuery (patternQuery mSearch mWType sortBy) ()
-- Widget patterns: fetch then count adoptions
basePatterns <- query @WidgetPattern
|> filterWhere (#isPublished, True)
|> orderByAsc #name
|> limit 50
|> fetch
patterns <- mapM (\p -> do
cnt <- sqlQueryScalar
"SELECT COUNT(*) FROM pattern_adoptions WHERE widget_pattern_id = ?"
(Only p.id)
pure (p, fromMaybe 0 cnt)) basePatterns
-- Governance templates: full-text search
templates <- sqlQuery (templateQuery mSearch) ()
-- Governance templates: fetch then count clones
baseTemplates <- query @GovernanceTemplate
|> filterWhere (#isPublished, True)
|> limit 50
|> fetch
templates <- mapM (\t -> do
cnt <- sqlQueryScalar
"SELECT COUNT(*) FROM governance_template_clones WHERE governance_template_id = ?"
(Only t.id)
pure (t, fromMaybe 0 cnt)) baseTemplates
-- Trending patterns (most adoptions in last 30 days)
trending <- sqlQuery
"SELECT wp.id, wp.name, wp.widget_type, COUNT(pa.id) AS recent_adoptions \
trendingRaw :: [(Id WidgetPattern, Text, Text, Int)] <- sqlQuery
"SELECT wp.id, wp.name, wp.widget_type, CAST(COUNT(pa.id) AS integer) AS recent_adoptions \
\ FROM widget_patterns wp \
\ JOIN pattern_adoptions pa ON pa.widget_pattern_id = wp.id \
\ WHERE wp.is_published = TRUE \
@@ -32,6 +47,7 @@ instance Controller MarketplaceDashboardController where
\ ORDER BY recent_adoptions DESC \
\ LIMIT 5"
()
let trending = trendingRaw
widgetTypeOptions <- sqlQuery
"SELECT name, label FROM widget_type_registry WHERE status = 'active' ORDER BY label"
@@ -45,40 +61,3 @@ instance Controller MarketplaceDashboardController where
, sortOrder = sortBy
}
-- | Widget pattern list query with optional search and type filter.
patternQuery :: Maybe Text -> Maybe Text -> Text -> Query
patternQuery mSearch mWType sortBy =
let baseWhere = "wp.is_published = TRUE"
searchClause = case mSearch of
Nothing -> ""
Just _ -> " AND to_tsvector('english', wp.name || ' ' || COALESCE(wp.description,'')) \
\ @@ plainto_tsquery(?)"
typeClause = case mWType of
Nothing -> ""
Just _ -> " AND wp.widget_type = ?"
orderClause = case sortBy of
"recent" -> "wp.created_at DESC"
"alpha" -> "wp.name ASC"
_ -> "adopter_count DESC"
in "SELECT wp.*, COUNT(pa.id) AS adopter_count \
\ FROM widget_patterns wp \
\ LEFT JOIN pattern_adoptions pa ON pa.widget_pattern_id = wp.id \
\ WHERE " <> baseWhere <> searchClause <> typeClause <>
" GROUP BY wp.id \
\ ORDER BY " <> orderClause <>
" LIMIT 50"
-- | Governance template list query with optional search.
templateQuery :: Maybe Text -> Query
templateQuery mSearch =
let searchClause = case mSearch of
Nothing -> ""
Just _ -> " AND to_tsvector('english', gt.name || ' ' || COALESCE(gt.description,'')) \
\ @@ plainto_tsquery(?)"
in "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" <> searchClause <>
" GROUP BY gt.id \
\ ORDER BY clone_count DESC \
\ LIMIT 50"