practical-haskell
Practical Haskell (GHC)
Use this skill when the task is Haskell code quality, performance, or reasoning about evaluation. Assume GHC with optimizations (-O / -O2) unless the user says otherwise.
Core ideas
- Purity lets the compiler rewrite code safely; prefer explicit effects in
IOor appropriate abstraction. - Lazy by default: values are evaluated when needed. That enables composition but can hide space leaks.
- Types catch many bugs early; use them to encode intent (including
newtypefor domain distinctions). - Know what GHC emits: when performance matters, treat Core (
-ddump-simpl) as ground truth after optimization.
Always
- Be explicit about strict vs lazy data and bindings when modeling accumulators, parsers, or long-lived state.
- Prefer
foldl'fromData.List(or strict folds from the right library) for numeric accumulation over plainfoldlon strict values. - Profile (
profiling, eventlog,ghc-debug, etc.) before micro-optimizing. - Write small composable functions; rely on inlining and specialization rather than giant monoliths.
- Use fusion-friendly pipelines (
map,filter,foldr-based idioms) where appropriate; validate hot paths in Core if allocation matters.
Never
- Accidentally build large chains of thunks (classic
foldl (+) 0on large strict sums). - Ignore space leaks from unevaluated structure holding onto memory.
- Micro-optimize without evidence from profiling or Core.
- Treat laziness as universally good or bad; decide per use case.
Prefer
- Strict fields (
!) on accumulator-like constructor fields;UNPACKfor small unboxed numeric fields when profiling supports it. - Newtypes for zero-runtime-cost distinctions vs
datawith a single field. INLINE/INLINABLE/SPECIALIZEon hot polymorphic glue when dictionaries or lack of specialization shows up in Core.- Worker/wrapper style: a strict internal worker and a small external API.
- Monomorphic hot loops when polymorphism still costs after specialization attempts.
Laziness and strictness
import Data.List (foldl')
-- Infinite lists are fine when consumption is bounded.
naturals :: [Integer]
naturals = [1..]
firstTen :: [Integer]
firstTen = take 10 naturals
-- foldl on strict arithmetic often leaks thunks; foldl' forces as it goes.
badSum :: [Int] -> Int
badSum = foldl (+) 0
goodSum :: [Int] -> Int
goodSum = foldl' (+) 0
Bang patterns ({-# LANGUAGE BangPatterns #-}) force evaluation of a binding; use at strategic places (accumulators, fields that must not retain thunks).
Strict fields on data constructors evaluate to WHNF when the constructor is entered; combine with profiling to avoid over-forcing.
Fusion and lists
List pipelines like sum . map f . filter p often fuse under -O2 into a single loop. If allocation persists, inspect Core. Avoid forcing materialization unnecessarily (e.g. redundant length or indexing on huge intermediates in hot code).
GHC applies rewrite rules internally; custom {-# RULES #-} is advanced and must be validated (correctness and phase interactions).
Newtypes
newtype UserId = UserId Int
deriving (Eq, Ord, Show)
newtype Email = Email String
deriving (Eq, Show)
Use newtype for distinct types with identical representation. GeneralizedNewtypeDeriving can derive classes when appropriate and policy allows.
Specialization and inlining
Polymorphic hot code may pass type-class dictionaries. Mitigations:
- Give a monomorphic variant for the hot path.
- Use
{-# SPECIALIZE #-}for concrete instantiations. - Use
{-# INLINABLE #-}on small polymorphic helpers so call sites can specialize.
Verify with Core, not assumptions.
Reading Core (quick guide)
Compile with something like ghc -O2 -ddump-simpl -dsuppress-all -dno-suppress-type-signatures YourModule.hs (flags vary by need).
caseusually forces evaluation; extraletbindings can mean allocation.- Look for fusion: one tight recursive loop vs multiple passes.
- Check whether dictionary calls remain in hot loops.
Mental checklist
- When does each subexpression get forced?
- Where might thunks retain memory (closures, lazy fields,
foldl-style accumulation)? - Will this pipeline fuse or allocate intermediates?
- What does simplified Core show for the hot path?
- Is the hot code monomorphic and specialized?
Signature moves (when profiling says so)
- Strict accumulators:
foldl', bang patterns, strict fields. UNPACKsmall strict numeric fields to reduce indirection.INLINE/INLINABLE/SPECIALIZEto recover specialization.- Fusion-friendly combinators; avoid accidental intermediate lists in inner loops.
- Worker/wrapper refactor for clearer strict internals.
- Re-check Core after each change.
Additional resources
For extended examples and GHC flag recipes, see reference.md.
More from kaynetik/skills
ultimate-nixos
Comprehensive Nix ecosystem guidance covering NixOS system configuration, nixpkgs packaging and maintainer workflow, nix-darwin macOS management, Home Manager, flakes, derivations, NixOS module design, security hardening, and community processes. Use when writing Nix expressions, packaging software for nixpkgs, configuring NixOS or nix-darwin systems, designing NixOS modules, reviewing or merging nixpkgs PRs, managing secrets, hardening systems, working with flakes and overlays, or when the user mentions Nix, NixOS, nixpkgs, nix-darwin, Home Manager, flake, derivation, overlay, OfBorg, nixpkgs-review, or darwin-rebuild.
14lua-projects
>-
11c-cpp-compilers
C and C++ compiler toolchain skill covering GCC, Clang/LLVM, build modes, warnings, sanitizers, static analysis, LTO, PGO, C++20/23/26 features, and debugging. Use when writing or reviewing C/C++ code, choosing compiler flags, interpreting errors or warnings, enabling sanitizers, running clang-tidy or cppcheck, optimizing builds, working with C++20 modules or C23 features, or troubleshooting linker issues.
9tdd-red-green-refactor
>-
9argocd
ArgoCD GitOps operations -- Application and AppProject authoring, multi-source apps, ApplicationSet generators (Go templates, progressive sync), sync strategies, RBAC, SSO, health checks, CLI commands, and troubleshooting. Use when writing ArgoCD manifests, managing deployments, configuring sync policies, debugging sync/health status, or when the user mentions ArgoCD, GitOps, ApplicationSet, AppProject, argocd CLI, sync wave, or self-heal.
8meta-cognition-parallel
EXPERIMENTAL: Three-layer parallel meta-cognition analysis. Triggers on: /meta-parallel, parallel analysis
7