[{"data":1,"prerenderedAt":1912},["ShallowReactive",2],{"navigation_docs":3,"-logging-audit-recipes":547,"-logging-audit-recipes-surround":1907},[4,35,159,201,289,446,531],{"title":5,"path":6,"stem":7,"children":8,"page":34},"Getting Started","\u002Fgetting-started","1.getting-started",[9,14,19,24,29],{"title":10,"path":11,"stem":12,"icon":13},"Introduction","\u002Fgetting-started\u002Fintroduction","1.getting-started\u002F1.introduction","i-lucide-info",{"title":15,"path":16,"stem":17,"icon":18},"Installation","\u002Fgetting-started\u002Finstallation","1.getting-started\u002F2.installation","i-lucide-download",{"title":20,"path":21,"stem":22,"icon":23},"Quick Start","\u002Fgetting-started\u002Fquick-start","1.getting-started\u002F3.quick-start","i-lucide-zap",{"title":25,"path":26,"stem":27,"icon":28},"Agent Skills","\u002Fgetting-started\u002Fagent-skills","1.getting-started\u002F4.agent-skills","i-lucide-sparkles",{"title":30,"path":31,"stem":32,"icon":33},"vs Other Loggers","\u002Fgetting-started\u002Fvs-other-loggers","1.getting-started\u002F5.vs-other-loggers","i-lucide-scale",false,{"title":36,"path":37,"stem":38,"children":39,"page":34},"Logging","\u002Flogging","2.logging",[40,45,50,55,60,65,70,99,127],{"title":41,"path":42,"stem":43,"icon":44},"Overview","\u002Flogging\u002Foverview","2.logging\u002F0.overview","i-lucide-list",{"title":46,"path":47,"stem":48,"icon":49},"Simple Logging","\u002Flogging\u002Fsimple-logging","2.logging\u002F1.simple-logging","i-lucide-terminal",{"title":51,"path":52,"stem":53,"icon":54},"Wide Events","\u002Flogging\u002Fwide-events","2.logging\u002F2.wide-events","i-lucide-layers",{"title":56,"path":57,"stem":58,"icon":59},"Structured Errors","\u002Flogging\u002Fstructured-errors","2.logging\u002F3.structured-errors","i-lucide-shield-alert",{"title":61,"path":62,"stem":63,"icon":64},"Catalogs","\u002Flogging\u002Fcatalogs","2.logging\u002F4.catalogs","i-lucide-book-open",{"title":66,"path":67,"stem":68,"icon":69},"Client Logging","\u002Flogging\u002Fclient-logging","2.logging\u002F5.client-logging","i-lucide-monitor",{"title":71,"icon":72,"path":73,"stem":74,"children":75,"page":34},"AI SDK","i-simple-icons-vercel","\u002Flogging\u002Fai-sdk","2.logging\u002F6.ai-sdk",[76,79,84,89,94],{"title":41,"path":77,"stem":78,"icon":44},"\u002Flogging\u002Fai-sdk\u002Foverview","2.logging\u002F6.ai-sdk\u002F01.overview",{"title":80,"path":81,"stem":82,"icon":83},"Usage","\u002Flogging\u002Fai-sdk\u002Fusage","2.logging\u002F6.ai-sdk\u002F02.usage","i-lucide-code",{"title":85,"path":86,"stem":87,"icon":88},"Options","\u002Flogging\u002Fai-sdk\u002Foptions","2.logging\u002F6.ai-sdk\u002F03.options","i-lucide-sliders",{"title":90,"path":91,"stem":92,"icon":93},"Metadata","\u002Flogging\u002Fai-sdk\u002Fmetadata","2.logging\u002F6.ai-sdk\u002F04.metadata","i-lucide-database",{"title":95,"path":96,"stem":97,"icon":98},"Telemetry","\u002Flogging\u002Fai-sdk\u002Ftelemetry","2.logging\u002F6.ai-sdk\u002F05.telemetry","i-lucide-activity",{"title":100,"icon":101,"path":102,"stem":103,"children":104,"page":34},"Better Auth","i-simple-icons-betterauth","\u002Flogging\u002Fbetter-auth","2.logging\u002F7.better-auth",[105,108,113,118,122],{"title":41,"path":106,"stem":107,"icon":44},"\u002Flogging\u002Fbetter-auth\u002Foverview","2.logging\u002F7.better-auth\u002F01.overview",{"title":109,"path":110,"stem":111,"icon":112},"Identify User","\u002Flogging\u002Fbetter-auth\u002Fidentify-user","2.logging\u002F7.better-auth\u002F02.identify-user","i-lucide-user-check",{"title":114,"path":115,"stem":116,"icon":117},"Middleware","\u002Flogging\u002Fbetter-auth\u002Fmiddleware","2.logging\u002F7.better-auth\u002F03.middleware","i-lucide-shield",{"title":119,"path":120,"stem":121,"icon":69},"Client Sync","\u002Flogging\u002Fbetter-auth\u002Fclient-sync","2.logging\u002F7.better-auth\u002F04.client-sync",{"title":123,"path":124,"stem":125,"icon":126},"Performance","\u002Flogging\u002Fbetter-auth\u002Fperformance","2.logging\u002F7.better-auth\u002F05.performance","i-lucide-gauge",{"title":128,"icon":129,"path":130,"stem":131,"children":132,"page":34},"Audit Logs","i-lucide-shield-check","\u002Flogging\u002Faudit","2.logging\u002F8.audit",[133,136,141,146,151,155],{"title":41,"path":134,"stem":135,"icon":44},"\u002Flogging\u002Faudit\u002Foverview","2.logging\u002F8.audit\u002F01.overview",{"title":137,"path":138,"stem":139,"icon":140},"Schema","\u002Flogging\u002Faudit\u002Fschema","2.logging\u002F8.audit\u002F02.schema","i-lucide-file-text",{"title":142,"path":143,"stem":144,"icon":145},"Recording","\u002Flogging\u002Faudit\u002Frecording","2.logging\u002F8.audit\u002F03.recording","i-lucide-pen-line",{"title":147,"path":148,"stem":149,"icon":150},"Drains","\u002Flogging\u002Faudit\u002Fpipeline","2.logging\u002F8.audit\u002F04.pipeline","i-lucide-link",{"title":152,"path":153,"stem":154,"icon":129},"Compliance","\u002Flogging\u002Faudit\u002Fcompliance","2.logging\u002F8.audit\u002F05.compliance",{"title":156,"path":157,"stem":158,"icon":64},"Recipes","\u002Flogging\u002Faudit\u002Frecipes","2.logging\u002F8.audit\u002F06.recipes",{"title":160,"path":161,"stem":162,"children":163,"page":34},"Core Concepts","\u002Fcore-concepts","3.core-concepts",[164,169,174,179,184,188,191,196],{"title":165,"path":166,"stem":167,"icon":168},"Lifecycle","\u002Fcore-concepts\u002Flifecycle","3.core-concepts\u002F0.lifecycle","i-lucide-arrow-right-left",{"title":170,"path":171,"stem":172,"icon":173},"Configuration","\u002Fcore-concepts\u002Fconfiguration","3.core-concepts\u002F1.configuration","i-lucide-settings",{"title":175,"path":176,"stem":177,"icon":178},"Sampling","\u002Fcore-concepts\u002Fsampling","3.core-concepts\u002F2.sampling","i-lucide-filter",{"title":180,"path":181,"stem":182,"icon":183},"Typed Fields","\u002Fcore-concepts\u002Ftyped-fields","3.core-concepts\u002F3.typed-fields","i-simple-icons-typescript",{"title":185,"path":186,"stem":187,"icon":129},"Best Practices","\u002Fcore-concepts\u002Fbest-practices","3.core-concepts\u002F4.best-practices",{"title":123,"path":189,"stem":190,"icon":126},"\u002Fcore-concepts\u002Fperformance","3.core-concepts\u002F5.performance",{"title":192,"path":193,"stem":194,"icon":195},"Vite Plugin","\u002Fcore-concepts\u002Fvite-plugin","3.core-concepts\u002F6.vite-plugin","i-custom-vite",{"title":197,"path":198,"stem":199,"icon":200},"Auto-Redaction","\u002Fcore-concepts\u002Fredaction","3.core-concepts\u002F7.redaction","i-lucide-eye-off",{"title":202,"path":203,"stem":204,"children":205,"page":34},"Frameworks","\u002Fframeworks","4.frameworks",[206,210,215,220,225,230,235,240,245,250,255,260,265,270,274,279,284],{"title":41,"path":207,"stem":208,"icon":209},"\u002Fframeworks\u002Foverview","4.frameworks\u002F00.overview","i-lucide-layout-grid",{"title":211,"path":212,"stem":213,"icon":214},"Nuxt","\u002Fframeworks\u002Fnuxt","4.frameworks\u002F01.nuxt","i-simple-icons-nuxtdotjs",{"title":216,"path":217,"stem":218,"icon":219},"Next.js","\u002Fframeworks\u002Fnextjs","4.frameworks\u002F02.nextjs","i-simple-icons-nextdotjs",{"title":221,"path":222,"stem":223,"icon":224},"SvelteKit","\u002Fframeworks\u002Fsveltekit","4.frameworks\u002F03.sveltekit","i-simple-icons-svelte",{"title":226,"path":227,"stem":228,"icon":229},"Nitro","\u002Fframeworks\u002Fnitro","4.frameworks\u002F04.nitro","i-custom-nitro",{"title":231,"path":232,"stem":233,"icon":234},"TanStack Start","\u002Fframeworks\u002Ftanstack-start","4.frameworks\u002F05.tanstack-start","i-custom-tanstack",{"title":236,"path":237,"stem":238,"icon":239},"NestJS","\u002Fframeworks\u002Fnestjs","4.frameworks\u002F06.nestjs","i-simple-icons-nestjs",{"title":241,"path":242,"stem":243,"icon":244},"Express","\u002Fframeworks\u002Fexpress","4.frameworks\u002F07.express","i-simple-icons-express",{"title":246,"path":247,"stem":248,"icon":249},"Hono","\u002Fframeworks\u002Fhono","4.frameworks\u002F08.hono","i-simple-icons-hono",{"title":251,"path":252,"stem":253,"icon":254},"Fastify","\u002Fframeworks\u002Ffastify","4.frameworks\u002F09.fastify","i-simple-icons-fastify",{"title":256,"path":257,"stem":258,"icon":259},"Elysia","\u002Fframeworks\u002Felysia","4.frameworks\u002F10.elysia","i-custom-elysia",{"title":261,"path":262,"stem":263,"icon":264},"React Router","\u002Fframeworks\u002Freact-router","4.frameworks\u002F11.react-router","i-custom-reactrouter",{"title":266,"path":267,"stem":268,"icon":269},"Cloudflare Workers","\u002Fframeworks\u002Fcloudflare-workers","4.frameworks\u002F12.cloudflare-workers","i-simple-icons-cloudflare",{"title":271,"path":272,"stem":273,"icon":183},"Standalone","\u002Fframeworks\u002Fstandalone","4.frameworks\u002F13.standalone",{"title":275,"path":276,"stem":277,"icon":278},"Astro","\u002Fframeworks\u002Fastro","4.frameworks\u002F14.astro","i-simple-icons-astro",{"title":280,"path":281,"stem":282,"icon":283},"AWS Lambda","\u002Fframeworks\u002Faws-lambda","4.frameworks\u002F16.aws-lambda","i-custom-lambda",{"title":285,"path":286,"stem":287,"icon":288},"Custom Integration","\u002Fframeworks\u002Fcustom-integration","4.frameworks\u002F17.custom-integration","i-lucide-puzzle",{"title":290,"path":291,"stem":292,"children":293,"page":34},"Build on top","\u002Fbuild-on-top","5.build-on-top",[294,297,325,346,373,400,428],{"title":41,"path":295,"stem":296,"icon":54},"\u002Fbuild-on-top\u002Foverview","5.build-on-top\u002F0.overview",{"title":298,"path":299,"stem":300,"children":301,"page":34},"Observers","\u002Fbuild-on-top\u002Fobservers","5.build-on-top\u002Fobservers",[302,306,311,316,321],{"title":41,"path":303,"stem":304,"icon":305},"\u002Fbuild-on-top\u002Fobservers\u002Foverview","5.build-on-top\u002Fobservers\u002F1.overview","i-lucide-eye",{"title":307,"path":308,"stem":309,"icon":310},"Stream API","\u002Fbuild-on-top\u002Fobservers\u002Fin-process-stream","5.build-on-top\u002Fobservers\u002F2.in-process-stream","i-lucide-radio-tower",{"title":312,"path":313,"stem":314,"icon":315},"Stream server","\u002Fbuild-on-top\u002Fobservers\u002Fstream-server","5.build-on-top\u002Fobservers\u002F3.stream-server","i-lucide-radio",{"title":317,"path":318,"stem":319,"icon":320},"FS reader","\u002Fbuild-on-top\u002Fobservers\u002Ffs-reader","5.build-on-top\u002Fobservers\u002F4.fs-reader","i-lucide-folder-search",{"title":156,"path":322,"stem":323,"icon":324},"\u002Fbuild-on-top\u002Fobservers\u002Fconsumer-recipes","5.build-on-top\u002Fobservers\u002F5.consumer-recipes","i-lucide-chef-hat",{"title":326,"path":327,"stem":328,"children":329,"page":34},"Pipeline","\u002Fbuild-on-top\u002Fpipeline","5.build-on-top\u002Fpipeline",[330,334,338,342],{"title":41,"path":331,"stem":332,"icon":333},"\u002Fbuild-on-top\u002Fpipeline\u002Foverview","5.build-on-top\u002Fpipeline\u002F1.overview","i-lucide-workflow",{"title":335,"path":336,"stem":337,"icon":288},"Plugins","\u002Fbuild-on-top\u002Fpipeline\u002Fplugins","5.build-on-top\u002Fpipeline\u002F2.plugins",{"title":339,"path":340,"stem":341,"icon":28},"Custom enrichers","\u002Fbuild-on-top\u002Fpipeline\u002Fcustom-enrichers","5.build-on-top\u002Fpipeline\u002F3.custom-enrichers",{"title":343,"path":344,"stem":345,"icon":178},"Tail sampling","\u002Fbuild-on-top\u002Fpipeline\u002Ftail-sampling","5.build-on-top\u002Fpipeline\u002F4.tail-sampling",{"title":347,"path":348,"stem":349,"children":350,"page":34},"Scenarios","\u002Fbuild-on-top\u002Fscenarios","5.build-on-top\u002Fscenarios",[351,355,359,364,368],{"title":41,"path":352,"stem":353,"icon":354},"\u002Fbuild-on-top\u002Fscenarios\u002Foverview","5.build-on-top\u002Fscenarios\u002F0.overview","i-lucide-clipboard-list",{"title":356,"path":357,"stem":358,"icon":49},"Local debugging toolkit","\u002Fbuild-on-top\u002Fscenarios\u002Flocal-debugging-toolkit","5.build-on-top\u002Fscenarios\u002F1.local-debugging-toolkit",{"title":360,"path":361,"stem":362,"icon":363},"Tenant-aware logging","\u002Fbuild-on-top\u002Fscenarios\u002Ftenant-aware-logging","5.build-on-top\u002Fscenarios\u002F2.tenant-aware-logging","i-lucide-users",{"title":365,"path":366,"stem":367,"icon":129},"Compliance audit","\u002Fbuild-on-top\u002Fscenarios\u002Fcompliance-audit","5.build-on-top\u002Fscenarios\u002F3.compliance-audit",{"title":369,"path":370,"stem":371,"icon":372},"Cross-app error vocab","\u002Fbuild-on-top\u002Fscenarios\u002Fcross-app-error-vocab","5.build-on-top\u002Fscenarios\u002F4.cross-app-error-vocab","i-lucide-book",{"title":374,"path":375,"stem":376,"children":377,"page":34},"Shared packages","\u002Fbuild-on-top\u002Fshared","5.build-on-top\u002Fshared",[378,382,386,391,395],{"title":41,"path":379,"stem":380,"icon":381},"\u002Fbuild-on-top\u002Fshared\u002Foverview","5.build-on-top\u002Fshared\u002F1.overview","i-lucide-package",{"title":383,"path":384,"stem":385,"icon":381},"Catalogs as packages","\u002Fbuild-on-top\u002Fshared\u002Fcatalogs-as-packages","5.build-on-top\u002Fshared\u002F2.catalogs-as-packages",{"title":387,"path":388,"stem":389,"icon":390},"Drains as packages","\u002Fbuild-on-top\u002Fshared\u002Fdrains-as-packages","5.build-on-top\u002Fshared\u002F3.drains-as-packages","i-lucide-package-2",{"title":392,"path":393,"stem":394,"icon":28},"Enrichers as packages","\u002Fbuild-on-top\u002Fshared\u002Fenrichers-as-packages","5.build-on-top\u002Fshared\u002F4.enrichers-as-packages",{"title":396,"path":397,"stem":398,"icon":399},"Integration as package","\u002Fbuild-on-top\u002Fshared\u002Fintegration-as-package","5.build-on-top\u002Fshared\u002F5.integration-as-package","i-lucide-package-check",{"title":401,"path":402,"stem":403,"children":404,"page":34},"Sinks","\u002Fbuild-on-top\u002Fsinks","5.build-on-top\u002Fsinks",[405,409,414,418,423],{"title":41,"path":406,"stem":407,"icon":408},"\u002Fbuild-on-top\u002Fsinks\u002Foverview","5.build-on-top\u002Fsinks\u002F1.overview","i-lucide-arrow-up-from-line",{"title":410,"path":411,"stem":412,"icon":413},"Custom drains","\u002Fbuild-on-top\u002Fsinks\u002Fcustom-drains","5.build-on-top\u002Fsinks\u002F2.custom-drains","i-lucide-code-2",{"title":415,"path":416,"stem":417,"icon":333},"Drain pipeline","\u002Fbuild-on-top\u002Fsinks\u002Fdrain-pipeline","5.build-on-top\u002Fsinks\u002F3.drain-pipeline",{"title":419,"path":420,"stem":421,"icon":422},"Identity headers","\u002Fbuild-on-top\u002Fsinks\u002Fidentity-headers","5.build-on-top\u002Fsinks\u002F4.identity-headers","i-lucide-fingerprint",{"title":424,"path":425,"stem":426,"icon":427},"Fanout","\u002Fbuild-on-top\u002Fsinks\u002Ffanout-and-multi-drain","5.build-on-top\u002Fsinks\u002F5.fanout-and-multi-drain","i-lucide-share-2",{"title":429,"path":430,"stem":431,"children":432,"page":34},"Sources","\u002Fbuild-on-top\u002Fsources","5.build-on-top\u002Fsources",[433,437,441],{"title":41,"path":434,"stem":435,"icon":436},"\u002Fbuild-on-top\u002Fsources\u002Foverview","5.build-on-top\u002Fsources\u002F1.overview","i-lucide-arrow-down-to-line",{"title":438,"path":439,"stem":440,"icon":288},"Custom framework","\u002Fbuild-on-top\u002Fsources\u002Fcustom-framework","5.build-on-top\u002Fsources\u002F2.custom-framework",{"title":442,"path":443,"stem":444,"icon":445},"Middleware toolkit","\u002Fbuild-on-top\u002Fsources\u002Fmiddleware-toolkit","5.build-on-top\u002Fsources\u002F3.middleware-toolkit","i-lucide-wrench",{"title":447,"path":448,"stem":449,"children":450,"page":34},"Adapters","\u002Fadapters","6.adapters",[451,454,494,509],{"title":41,"path":452,"stem":453,"icon":44},"\u002Fadapters\u002Foverview","6.adapters\u002F01.overview",{"title":455,"path":456,"stem":457,"children":458,"page":34},"Cloud destinations","\u002Fadapters\u002Fcloud","6.adapters\u002F02.cloud",[459,464,469,474,479,484,489],{"title":460,"path":461,"stem":462,"icon":463},"Axiom","\u002Fadapters\u002Fcloud\u002Faxiom","6.adapters\u002F02.cloud\u002F01.axiom","i-custom-axiom",{"title":465,"path":466,"stem":467,"icon":468},"OTLP","\u002Fadapters\u002Fcloud\u002Fotlp","6.adapters\u002F02.cloud\u002F02.otlp","i-simple-icons-opentelemetry",{"title":470,"path":471,"stem":472,"icon":473},"PostHog","\u002Fadapters\u002Fcloud\u002Fposthog","6.adapters\u002F02.cloud\u002F03.posthog","i-simple-icons-posthog",{"title":475,"path":476,"stem":477,"icon":478},"Sentry","\u002Fadapters\u002Fcloud\u002Fsentry","6.adapters\u002F02.cloud\u002F04.sentry","i-simple-icons-sentry",{"title":480,"path":481,"stem":482,"icon":483},"Better Stack","\u002Fadapters\u002Fcloud\u002Fbetter-stack","6.adapters\u002F02.cloud\u002F05.better-stack","i-simple-icons-betterstack",{"title":485,"path":486,"stem":487,"icon":488},"Datadog","\u002Fadapters\u002Fcloud\u002Fdatadog","6.adapters\u002F02.cloud\u002F06.datadog","i-simple-icons-datadog",{"title":490,"path":491,"stem":492,"icon":493},"HyperDX","\u002Fadapters\u002Fcloud\u002Fhyperdx","6.adapters\u002F02.cloud\u002F07.hyperdx","i-custom-hyperdx",{"title":495,"path":496,"stem":497,"children":498,"page":34},"Self-hosted","\u002Fadapters\u002Fself-hosted","6.adapters\u002F03.self-hosted",[499,504],{"title":500,"path":501,"stem":502,"icon":503},"File System","\u002Fadapters\u002Fself-hosted\u002Ffs","6.adapters\u002F03.self-hosted\u002F01.fs","i-lucide-hard-drive",{"title":505,"path":506,"stem":507,"icon":508},"NuxtHub","\u002Fadapters\u002Fself-hosted\u002Fnuxthub","6.adapters\u002F03.self-hosted\u002F02.nuxthub","i-simple-icons-nuxt",{"title":510,"path":511,"stem":512,"children":513,"page":34},"Building blocks","\u002Fadapters\u002Fbuilding-blocks","6.adapters\u002F04.building-blocks",[514,517,522,526],{"title":326,"path":515,"stem":516,"icon":333},"\u002Fadapters\u002Fbuilding-blocks\u002Fpipeline","6.adapters\u002F04.building-blocks\u002F01.pipeline",{"title":518,"path":519,"stem":520,"icon":521},"HTTP","\u002Fadapters\u002Fbuilding-blocks\u002Fhttp","6.adapters\u002F04.building-blocks\u002F02.http","i-lucide-globe",{"title":523,"path":524,"stem":525,"icon":83},"Custom Adapters","\u002Fadapters\u002Fbuilding-blocks\u002Fcustom","6.adapters\u002F04.building-blocks\u002F03.custom",{"title":527,"path":528,"stem":529,"icon":530},"Toolkit","\u002Fadapters\u002Fbuilding-blocks\u002Ftoolkit","6.adapters\u002F04.building-blocks\u002F04.toolkit","i-lucide-blocks",{"title":532,"path":533,"stem":534,"children":535,"page":34},"Enrichers","\u002Fenrichers","7.enrichers",[536,539,543],{"title":41,"path":537,"stem":538,"icon":28},"\u002Fenrichers\u002Foverview","7.enrichers\u002F1.overview",{"title":540,"path":541,"stem":542,"icon":288},"Built-in","\u002Fenrichers\u002Fbuilt-in","7.enrichers\u002F2.built-in",{"title":544,"path":545,"stem":546,"icon":83},"Custom","\u002Fenrichers\u002Fcustom","7.enrichers\u002F3.custom",{"id":548,"title":549,"body":550,"description":1895,"extension":1896,"links":1897,"meta":1903,"navigation":1904,"path":157,"seo":1905,"stem":158,"__hash__":1906},"docs\u002F2.logging\u002F8.audit\u002F06.recipes.md","Recipes & Reference",{"type":551,"value":552,"toc":1888},"minimark",[553,570,575,802,813,817,1035,1038,1042,1347,1357,1361,1367,1666,1677,1681,1878,1884],[554,555,556,557,561,562,565,566,569],"p",{},"Pick the recipe that matches your sink, drop it in, and you have a tamper-evident audit log. Each recipe composes the same primitives (",[558,559,560],"code",{},"auditOnly",", ",[558,563,564],{},"signed",", optional ",[558,567,568],{},"await: true",") over different drains.",[571,572,574],"h2",{"id":573},"audit-logs-on-disk","Audit logs on disk",[576,577,578,784],"code-group",{},[579,580,586],"pre",{"className":581,"code":582,"filename":583,"language":584,"meta":585,"style":585},"language-typescript shiki shiki-themes material-theme-lighter material-theme material-theme-palenight","import { auditOnly, signed } from 'evlog'\nimport { createFsDrain } from 'evlog\u002Ffs'\n\nnitro.hooks.hook('evlog:drain', auditOnly(\n  signed(createFsDrain({ dir: '.audit', maxFiles: 30 }), { strategy: 'hash-chain' }),\n  { await: true },\n))\n","Input — server\u002Fplugins\u002Fevlog.ts","typescript","",[558,587,588,627,648,655,691,760,778],{"__ignoreMap":585},[589,590,593,597,601,605,608,611,614,617,620,624],"span",{"class":591,"line":592},"line",1,[589,594,596],{"class":595},"s7zQu","import",[589,598,600],{"class":599},"sMK4o"," {",[589,602,604],{"class":603},"sTEyZ"," auditOnly",[589,606,607],{"class":599},",",[589,609,610],{"class":603}," signed",[589,612,613],{"class":599}," }",[589,615,616],{"class":595}," from",[589,618,619],{"class":599}," '",[589,621,623],{"class":622},"sfazB","evlog",[589,625,626],{"class":599},"'\n",[589,628,630,632,634,637,639,641,643,646],{"class":591,"line":629},2,[589,631,596],{"class":595},[589,633,600],{"class":599},[589,635,636],{"class":603}," createFsDrain",[589,638,613],{"class":599},[589,640,616],{"class":595},[589,642,619],{"class":599},[589,644,645],{"class":622},"evlog\u002Ffs",[589,647,626],{"class":599},[589,649,651],{"class":591,"line":650},3,[589,652,654],{"emptyLinePlaceholder":653},true,"\n",[589,656,658,661,664,667,669,673,676,679,682,684,686,688],{"class":591,"line":657},4,[589,659,660],{"class":603},"nitro",[589,662,663],{"class":599},".",[589,665,666],{"class":603},"hooks",[589,668,663],{"class":599},[589,670,672],{"class":671},"s2Zo4","hook",[589,674,675],{"class":603},"(",[589,677,678],{"class":599},"'",[589,680,681],{"class":622},"evlog:drain",[589,683,678],{"class":599},[589,685,607],{"class":599},[589,687,604],{"class":671},[589,689,690],{"class":603},"(\n",[589,692,694,697,699,702,704,707,711,714,716,719,721,723,726,728,732,734,737,739,741,744,746,748,751,753,755,757],{"class":591,"line":693},5,[589,695,696],{"class":671},"  signed",[589,698,675],{"class":603},[589,700,701],{"class":671},"createFsDrain",[589,703,675],{"class":603},[589,705,706],{"class":599},"{",[589,708,710],{"class":709},"swJcz"," dir",[589,712,713],{"class":599},":",[589,715,619],{"class":599},[589,717,718],{"class":622},".audit",[589,720,678],{"class":599},[589,722,607],{"class":599},[589,724,725],{"class":709}," maxFiles",[589,727,713],{"class":599},[589,729,731],{"class":730},"sbssI"," 30",[589,733,613],{"class":599},[589,735,736],{"class":603},")",[589,738,607],{"class":599},[589,740,600],{"class":599},[589,742,743],{"class":709}," strategy",[589,745,713],{"class":599},[589,747,619],{"class":599},[589,749,750],{"class":622},"hash-chain",[589,752,678],{"class":599},[589,754,613],{"class":599},[589,756,736],{"class":603},[589,758,759],{"class":599},",\n",[589,761,763,766,769,771,775],{"class":591,"line":762},6,[589,764,765],{"class":599},"  {",[589,767,768],{"class":709}," await",[589,770,713],{"class":599},[589,772,774],{"class":773},"sfNiH"," true",[589,776,777],{"class":599}," },\n",[589,779,781],{"class":591,"line":780},7,[589,782,783],{"class":603},"))\n",[579,785,790],{"className":786,"code":787,"filename":788,"language":789,"meta":585,"style":585},"language-ndjson shiki shiki-themes material-theme-lighter material-theme material-theme-palenight","{\"audit\":{\"action\":\"invoice.refund\",\"actor\":{\"type\":\"user\",\"id\":\"usr_42\"},\"target\":{\"type\":\"invoice\",\"id\":\"inv_889\"},\"outcome\":\"success\",\"version\":1,\"idempotencyKey\":\"ak_8f3c4b2a1e5d6f7c\",\"prevHash\":null,\"hash\":\"3f2c8e1a...\"}}\n{\"audit\":{\"action\":\"user.update\",\"actor\":{\"type\":\"user\",\"id\":\"usr_42\"},\"target\":{\"type\":\"user\",\"id\":\"usr_99\"},\"outcome\":\"success\",\"version\":1,\"idempotencyKey\":\"ak_5e7d8f9a0b1c2d3e\",\"prevHash\":\"3f2c8e1a...\",\"hash\":\"9a1b4d7c...\"}}\n","Output — .audit\u002F2026-04-24.ndjson","ndjson",[558,791,792,797],{"__ignoreMap":585},[589,793,794],{"class":591,"line":592},[589,795,796],{},"{\"audit\":{\"action\":\"invoice.refund\",\"actor\":{\"type\":\"user\",\"id\":\"usr_42\"},\"target\":{\"type\":\"invoice\",\"id\":\"inv_889\"},\"outcome\":\"success\",\"version\":1,\"idempotencyKey\":\"ak_8f3c4b2a1e5d6f7c\",\"prevHash\":null,\"hash\":\"3f2c8e1a...\"}}\n",[589,798,799],{"class":591,"line":629},[589,800,801],{},"{\"audit\":{\"action\":\"user.update\",\"actor\":{\"type\":\"user\",\"id\":\"usr_42\"},\"target\":{\"type\":\"user\",\"id\":\"usr_99\"},\"outcome\":\"success\",\"version\":1,\"idempotencyKey\":\"ak_5e7d8f9a0b1c2d3e\",\"prevHash\":\"3f2c8e1a...\",\"hash\":\"9a1b4d7c...\"}}\n",[554,803,804,805,808,809,812],{},"Each line's ",[558,806,807],{},"prevHash"," matches the previous line's ",[558,810,811],{},"hash",". Tampering with any row breaks the chain forward of that point — a verifier replays the hashes and reports the first mismatch.",[571,814,816],{"id":815},"audit-logs-to-a-dedicated-axiom-dataset","Audit logs to a dedicated Axiom dataset",[576,818,819,987,1010],{},[579,820,822],{"className":581,"code":821,"filename":583,"language":584,"meta":585,"style":585},"import { auditOnly } from 'evlog'\nimport { createAxiomDrain } from 'evlog\u002Faxiom'\n\nnitro.hooks.hook('evlog:drain', createAxiomDrain({ dataset: 'logs' }))\nnitro.hooks.hook('evlog:drain', auditOnly(\n  createAxiomDrain({ dataset: 'audit', token: process.env.AXIOM_AUDIT_TOKEN }),\n))\n",[558,823,824,842,862,866,910,936,983],{"__ignoreMap":585},[589,825,826,828,830,832,834,836,838,840],{"class":591,"line":592},[589,827,596],{"class":595},[589,829,600],{"class":599},[589,831,604],{"class":603},[589,833,613],{"class":599},[589,835,616],{"class":595},[589,837,619],{"class":599},[589,839,623],{"class":622},[589,841,626],{"class":599},[589,843,844,846,848,851,853,855,857,860],{"class":591,"line":629},[589,845,596],{"class":595},[589,847,600],{"class":599},[589,849,850],{"class":603}," createAxiomDrain",[589,852,613],{"class":599},[589,854,616],{"class":595},[589,856,619],{"class":599},[589,858,859],{"class":622},"evlog\u002Faxiom",[589,861,626],{"class":599},[589,863,864],{"class":591,"line":650},[589,865,654],{"emptyLinePlaceholder":653},[589,867,868,870,872,874,876,878,880,882,884,886,888,890,892,894,897,899,901,904,906,908],{"class":591,"line":657},[589,869,660],{"class":603},[589,871,663],{"class":599},[589,873,666],{"class":603},[589,875,663],{"class":599},[589,877,672],{"class":671},[589,879,675],{"class":603},[589,881,678],{"class":599},[589,883,681],{"class":622},[589,885,678],{"class":599},[589,887,607],{"class":599},[589,889,850],{"class":671},[589,891,675],{"class":603},[589,893,706],{"class":599},[589,895,896],{"class":709}," dataset",[589,898,713],{"class":599},[589,900,619],{"class":599},[589,902,903],{"class":622},"logs",[589,905,678],{"class":599},[589,907,613],{"class":599},[589,909,783],{"class":603},[589,911,912,914,916,918,920,922,924,926,928,930,932,934],{"class":591,"line":693},[589,913,660],{"class":603},[589,915,663],{"class":599},[589,917,666],{"class":603},[589,919,663],{"class":599},[589,921,672],{"class":671},[589,923,675],{"class":603},[589,925,678],{"class":599},[589,927,681],{"class":622},[589,929,678],{"class":599},[589,931,607],{"class":599},[589,933,604],{"class":671},[589,935,690],{"class":603},[589,937,938,941,943,945,947,949,951,954,956,958,961,963,966,968,971,973,976,979,981],{"class":591,"line":762},[589,939,940],{"class":671},"  createAxiomDrain",[589,942,675],{"class":603},[589,944,706],{"class":599},[589,946,896],{"class":709},[589,948,713],{"class":599},[589,950,619],{"class":599},[589,952,953],{"class":622},"audit",[589,955,678],{"class":599},[589,957,607],{"class":599},[589,959,960],{"class":709}," token",[589,962,713],{"class":599},[589,964,965],{"class":603}," process",[589,967,663],{"class":599},[589,969,970],{"class":603},"env",[589,972,663],{"class":599},[589,974,975],{"class":603},"AXIOM_AUDIT_TOKEN ",[589,977,978],{"class":599},"}",[589,980,736],{"class":603},[589,982,759],{"class":599},[589,984,985],{"class":591,"line":780},[589,986,783],{"class":603},[579,988,993],{"className":989,"code":990,"filename":991,"language":992,"meta":585,"style":585},"language-kusto shiki shiki-themes material-theme-lighter material-theme material-theme-palenight","['audit']\n| where audit.action == \"invoice.refund\"\n| summarize count() by audit.outcome, bin(_time, 1h)\n","Output — Axiom query","kusto",[558,994,995,1000,1005],{"__ignoreMap":585},[589,996,997],{"class":591,"line":592},[589,998,999],{},"['audit']\n",[589,1001,1002],{"class":591,"line":629},[589,1003,1004],{},"| where audit.action == \"invoice.refund\"\n",[589,1006,1007],{"class":591,"line":650},[589,1008,1009],{},"| summarize count() by audit.outcome, bin(_time, 1h)\n",[579,1011,1014],{"className":989,"code":1012,"filename":1013,"language":992,"meta":585,"style":585},"['audit']\n| where audit.outcome == \"denied\"\n| summarize count() by audit.actor.id, audit.action\n| order by count_ desc\n","Output — denials by actor",[558,1015,1016,1020,1025,1030],{"__ignoreMap":585},[589,1017,1018],{"class":591,"line":592},[589,1019,999],{},[589,1021,1022],{"class":591,"line":629},[589,1023,1024],{},"| where audit.outcome == \"denied\"\n",[589,1026,1027],{"class":591,"line":650},[589,1028,1029],{},"| summarize count() by audit.actor.id, audit.action\n",[589,1031,1032],{"class":591,"line":657},[589,1033,1034],{},"| order by count_ desc\n",[554,1036,1037],{},"Splitting datasets means the audit dataset can have a longer retention (7y), tighter access controls, and a separate billing line — without touching the rest of your pipeline.",[571,1039,1041],{"id":1040},"audit-logs-in-postgres","Audit logs in Postgres",[576,1043,1044,1300],{},[579,1045,1047],{"className":581,"code":1046,"filename":583,"language":584,"meta":585,"style":585},"import { auditOnly } from 'evlog'\nimport type { DrainContext } from 'evlog'\n\nconst postgresAudit = async (ctx: DrainContext) => {\n  await db.insert(auditEvents).values({\n    id: ctx.event.audit!.idempotencyKey,\n    timestamp: new Date(ctx.event.timestamp),\n    payload: ctx.event,\n  }).onConflictDoNothing()\n}\n\nnitro.hooks.hook('evlog:drain', auditOnly(postgresAudit, { await: true }))\n",[558,1048,1049,1067,1089,1093,1128,1158,1185,1215,1231,1247,1253,1258],{"__ignoreMap":585},[589,1050,1051,1053,1055,1057,1059,1061,1063,1065],{"class":591,"line":592},[589,1052,596],{"class":595},[589,1054,600],{"class":599},[589,1056,604],{"class":603},[589,1058,613],{"class":599},[589,1060,616],{"class":595},[589,1062,619],{"class":599},[589,1064,623],{"class":622},[589,1066,626],{"class":599},[589,1068,1069,1071,1074,1076,1079,1081,1083,1085,1087],{"class":591,"line":629},[589,1070,596],{"class":595},[589,1072,1073],{"class":595}," type",[589,1075,600],{"class":599},[589,1077,1078],{"class":603}," DrainContext",[589,1080,613],{"class":599},[589,1082,616],{"class":595},[589,1084,619],{"class":599},[589,1086,623],{"class":622},[589,1088,626],{"class":599},[589,1090,1091],{"class":591,"line":650},[589,1092,654],{"emptyLinePlaceholder":653},[589,1094,1095,1099,1102,1105,1108,1111,1115,1117,1120,1122,1125],{"class":591,"line":657},[589,1096,1098],{"class":1097},"spNyl","const",[589,1100,1101],{"class":603}," postgresAudit ",[589,1103,1104],{"class":599},"=",[589,1106,1107],{"class":1097}," async",[589,1109,1110],{"class":599}," (",[589,1112,1114],{"class":1113},"sHdIc","ctx",[589,1116,713],{"class":599},[589,1118,1078],{"class":1119},"sBMFI",[589,1121,736],{"class":599},[589,1123,1124],{"class":1097}," =>",[589,1126,1127],{"class":599}," {\n",[589,1129,1130,1133,1136,1138,1141,1143,1146,1148,1150,1153,1155],{"class":591,"line":693},[589,1131,1132],{"class":595},"  await",[589,1134,1135],{"class":603}," db",[589,1137,663],{"class":599},[589,1139,1140],{"class":671},"insert",[589,1142,675],{"class":709},[589,1144,1145],{"class":603},"auditEvents",[589,1147,736],{"class":709},[589,1149,663],{"class":599},[589,1151,1152],{"class":671},"values",[589,1154,675],{"class":709},[589,1156,1157],{"class":599},"{\n",[589,1159,1160,1163,1165,1168,1170,1173,1175,1177,1180,1183],{"class":591,"line":762},[589,1161,1162],{"class":709},"    id",[589,1164,713],{"class":599},[589,1166,1167],{"class":603}," ctx",[589,1169,663],{"class":599},[589,1171,1172],{"class":603},"event",[589,1174,663],{"class":599},[589,1176,953],{"class":603},[589,1178,1179],{"class":599},"!.",[589,1181,1182],{"class":603},"idempotencyKey",[589,1184,759],{"class":599},[589,1186,1187,1190,1192,1195,1198,1200,1202,1204,1206,1208,1211,1213],{"class":591,"line":780},[589,1188,1189],{"class":709},"    timestamp",[589,1191,713],{"class":599},[589,1193,1194],{"class":599}," new",[589,1196,1197],{"class":671}," Date",[589,1199,675],{"class":709},[589,1201,1114],{"class":603},[589,1203,663],{"class":599},[589,1205,1172],{"class":603},[589,1207,663],{"class":599},[589,1209,1210],{"class":603},"timestamp",[589,1212,736],{"class":709},[589,1214,759],{"class":599},[589,1216,1218,1221,1223,1225,1227,1229],{"class":591,"line":1217},8,[589,1219,1220],{"class":709},"    payload",[589,1222,713],{"class":599},[589,1224,1167],{"class":603},[589,1226,663],{"class":599},[589,1228,1172],{"class":603},[589,1230,759],{"class":599},[589,1232,1234,1237,1239,1241,1244],{"class":591,"line":1233},9,[589,1235,1236],{"class":599},"  }",[589,1238,736],{"class":709},[589,1240,663],{"class":599},[589,1242,1243],{"class":671},"onConflictDoNothing",[589,1245,1246],{"class":709},"()\n",[589,1248,1250],{"class":591,"line":1249},10,[589,1251,1252],{"class":599},"}\n",[589,1254,1256],{"class":591,"line":1255},11,[589,1257,654],{"emptyLinePlaceholder":653},[589,1259,1261,1263,1265,1267,1269,1271,1273,1275,1277,1279,1281,1283,1286,1288,1290,1292,1294,1296,1298],{"class":591,"line":1260},12,[589,1262,660],{"class":603},[589,1264,663],{"class":599},[589,1266,666],{"class":603},[589,1268,663],{"class":599},[589,1270,672],{"class":671},[589,1272,675],{"class":603},[589,1274,678],{"class":599},[589,1276,681],{"class":622},[589,1278,678],{"class":599},[589,1280,607],{"class":599},[589,1282,604],{"class":671},[589,1284,1285],{"class":603},"(postgresAudit",[589,1287,607],{"class":599},[589,1289,600],{"class":599},[589,1291,768],{"class":709},[589,1293,713],{"class":599},[589,1295,774],{"class":773},[589,1297,613],{"class":599},[589,1299,783],{"class":603},[579,1301,1306],{"className":1302,"code":1303,"filename":1304,"language":1305,"meta":585,"style":585},"language-sql shiki shiki-themes material-theme-lighter material-theme material-theme-palenight","SELECT id, timestamp, payload->'audit'->>'action' AS action,\n       payload->'audit'->>'outcome' AS outcome\nFROM audit_events\nWHERE id = 'ak_8f3c4b2a1e5d6f7c';\n\n--          id          |       timestamp       |     action      | outcome\n-- ---------------------+-----------------------+-----------------+---------\n--  ak_8f3c4b2a1e5d6f7c | 2026-04-24 10:23:45.6 | invoice.refund  | success\n","Output — audit_events row","sql",[558,1307,1308,1313,1318,1323,1328,1332,1337,1342],{"__ignoreMap":585},[589,1309,1310],{"class":591,"line":592},[589,1311,1312],{},"SELECT id, timestamp, payload->'audit'->>'action' AS action,\n",[589,1314,1315],{"class":591,"line":629},[589,1316,1317],{},"       payload->'audit'->>'outcome' AS outcome\n",[589,1319,1320],{"class":591,"line":650},[589,1321,1322],{},"FROM audit_events\n",[589,1324,1325],{"class":591,"line":657},[589,1326,1327],{},"WHERE id = 'ak_8f3c4b2a1e5d6f7c';\n",[589,1329,1330],{"class":591,"line":693},[589,1331,654],{"emptyLinePlaceholder":653},[589,1333,1334],{"class":591,"line":762},[589,1335,1336],{},"--          id          |       timestamp       |     action      | outcome\n",[589,1338,1339],{"class":591,"line":780},[589,1340,1341],{},"-- ---------------------+-----------------------+-----------------+---------\n",[589,1343,1344],{"class":591,"line":1217},[589,1345,1346],{},"--  ak_8f3c4b2a1e5d6f7c | 2026-04-24 10:23:45.6 | invoice.refund  | success\n",[554,1348,1349,1350,1352,1353,1356],{},"The deterministic ",[558,1351,1182],{}," makes retries safe — duplicate inserts collapse via ",[558,1354,1355],{},"ON CONFLICT DO NOTHING",". Without it, a transient network blip during a retry would create a duplicate audit row, which is exactly what you don't want.",[571,1358,1360],{"id":1359},"testing-audits","Testing audits",[554,1362,1363,1366],{},[558,1364,1365],{},"mockAudit()"," captures every audit event emitted during a test:",[579,1368,1370],{"className":581,"code":1369,"language":584,"meta":585,"style":585},"import { mockAudit } from 'evlog'\n\nit('refunds the invoice and records an audit', async () => {\n  const captured = mockAudit()\n\n  await refundInvoice({ id: 'inv_889' }, { actor: { type: 'user', id: 'u1' } })\n\n  expect(captured.events).toHaveLength(1)\n  expect(captured.toIncludeAuditOf({\n    action: 'invoice.refund',\n    target: { type: 'invoice', id: 'inv_889' },\n    outcome: 'success',\n  })).toBe(true)\n\n  captured.restore()\n})\n",[558,1371,1372,1391,1395,1420,1435,1439,1505,1509,1538,1555,1571,1605,1621,1641,1646,1659],{"__ignoreMap":585},[589,1373,1374,1376,1378,1381,1383,1385,1387,1389],{"class":591,"line":592},[589,1375,596],{"class":595},[589,1377,600],{"class":599},[589,1379,1380],{"class":603}," mockAudit",[589,1382,613],{"class":599},[589,1384,616],{"class":595},[589,1386,619],{"class":599},[589,1388,623],{"class":622},[589,1390,626],{"class":599},[589,1392,1393],{"class":591,"line":629},[589,1394,654],{"emptyLinePlaceholder":653},[589,1396,1397,1400,1402,1404,1407,1409,1411,1413,1416,1418],{"class":591,"line":650},[589,1398,1399],{"class":671},"it",[589,1401,675],{"class":603},[589,1403,678],{"class":599},[589,1405,1406],{"class":622},"refunds the invoice and records an audit",[589,1408,678],{"class":599},[589,1410,607],{"class":599},[589,1412,1107],{"class":1097},[589,1414,1415],{"class":599}," ()",[589,1417,1124],{"class":1097},[589,1419,1127],{"class":599},[589,1421,1422,1425,1428,1431,1433],{"class":591,"line":657},[589,1423,1424],{"class":1097},"  const",[589,1426,1427],{"class":603}," captured",[589,1429,1430],{"class":599}," =",[589,1432,1380],{"class":671},[589,1434,1246],{"class":709},[589,1436,1437],{"class":591,"line":693},[589,1438,654],{"emptyLinePlaceholder":653},[589,1440,1441,1443,1446,1448,1450,1453,1455,1457,1460,1462,1465,1467,1470,1472,1474,1476,1478,1480,1483,1485,1487,1489,1491,1493,1496,1498,1500,1502],{"class":591,"line":762},[589,1442,1132],{"class":595},[589,1444,1445],{"class":671}," refundInvoice",[589,1447,675],{"class":709},[589,1449,706],{"class":599},[589,1451,1452],{"class":709}," id",[589,1454,713],{"class":599},[589,1456,619],{"class":599},[589,1458,1459],{"class":622},"inv_889",[589,1461,678],{"class":599},[589,1463,1464],{"class":599}," },",[589,1466,600],{"class":599},[589,1468,1469],{"class":709}," actor",[589,1471,713],{"class":599},[589,1473,600],{"class":599},[589,1475,1073],{"class":709},[589,1477,713],{"class":599},[589,1479,619],{"class":599},[589,1481,1482],{"class":622},"user",[589,1484,678],{"class":599},[589,1486,607],{"class":599},[589,1488,1452],{"class":709},[589,1490,713],{"class":599},[589,1492,619],{"class":599},[589,1494,1495],{"class":622},"u1",[589,1497,678],{"class":599},[589,1499,613],{"class":599},[589,1501,613],{"class":599},[589,1503,1504],{"class":709},")\n",[589,1506,1507],{"class":591,"line":780},[589,1508,654],{"emptyLinePlaceholder":653},[589,1510,1511,1514,1516,1519,1521,1524,1526,1528,1531,1533,1536],{"class":591,"line":1217},[589,1512,1513],{"class":671},"  expect",[589,1515,675],{"class":709},[589,1517,1518],{"class":603},"captured",[589,1520,663],{"class":599},[589,1522,1523],{"class":603},"events",[589,1525,736],{"class":709},[589,1527,663],{"class":599},[589,1529,1530],{"class":671},"toHaveLength",[589,1532,675],{"class":709},[589,1534,1535],{"class":730},"1",[589,1537,1504],{"class":709},[589,1539,1540,1542,1544,1546,1548,1551,1553],{"class":591,"line":1233},[589,1541,1513],{"class":671},[589,1543,675],{"class":709},[589,1545,1518],{"class":603},[589,1547,663],{"class":599},[589,1549,1550],{"class":671},"toIncludeAuditOf",[589,1552,675],{"class":709},[589,1554,1157],{"class":599},[589,1556,1557,1560,1562,1564,1567,1569],{"class":591,"line":1249},[589,1558,1559],{"class":709},"    action",[589,1561,713],{"class":599},[589,1563,619],{"class":599},[589,1565,1566],{"class":622},"invoice.refund",[589,1568,678],{"class":599},[589,1570,759],{"class":599},[589,1572,1573,1576,1578,1580,1582,1584,1586,1589,1591,1593,1595,1597,1599,1601,1603],{"class":591,"line":1255},[589,1574,1575],{"class":709},"    target",[589,1577,713],{"class":599},[589,1579,600],{"class":599},[589,1581,1073],{"class":709},[589,1583,713],{"class":599},[589,1585,619],{"class":599},[589,1587,1588],{"class":622},"invoice",[589,1590,678],{"class":599},[589,1592,607],{"class":599},[589,1594,1452],{"class":709},[589,1596,713],{"class":599},[589,1598,619],{"class":599},[589,1600,1459],{"class":622},[589,1602,678],{"class":599},[589,1604,777],{"class":599},[589,1606,1607,1610,1612,1614,1617,1619],{"class":591,"line":1260},[589,1608,1609],{"class":709},"    outcome",[589,1611,713],{"class":599},[589,1613,619],{"class":599},[589,1615,1616],{"class":622},"success",[589,1618,678],{"class":599},[589,1620,759],{"class":599},[589,1622,1624,1626,1629,1631,1634,1636,1639],{"class":591,"line":1623},13,[589,1625,1236],{"class":599},[589,1627,1628],{"class":709},"))",[589,1630,663],{"class":599},[589,1632,1633],{"class":671},"toBe",[589,1635,675],{"class":709},[589,1637,1638],{"class":773},"true",[589,1640,1504],{"class":709},[589,1642,1644],{"class":591,"line":1643},14,[589,1645,654],{"emptyLinePlaceholder":653},[589,1647,1649,1652,1654,1657],{"class":591,"line":1648},15,[589,1650,1651],{"class":603},"  captured",[589,1653,663],{"class":599},[589,1655,1656],{"class":671},"restore",[589,1658,1246],{"class":709},[589,1660,1662,1664],{"class":591,"line":1661},16,[589,1663,978],{"class":599},[589,1665,1504],{"class":603},[554,1667,1668,1669,1672,1673,1676],{},"Always call ",[558,1670,1671],{},"captured.restore()"," in an ",[558,1674,1675],{},"afterEach"," (or wrap with a fixture) so a failing assertion never leaks into the next test.",[571,1678,1680],{"id":1679},"api-reference","API Reference",[1682,1683,1684,1700],"table",{},[1685,1686,1687],"thead",{},[1688,1689,1690,1694,1697],"tr",{},[1691,1692,1693],"th",{},"Symbol",[1691,1695,1696],{},"Kind",[1691,1698,1699],{},"Notes",[1701,1702,1703,1717,1730,1742,1759,1771,1784,1797,1813,1825,1838,1853,1865],"tbody",{},[1688,1704,1705,1711,1714],{},[1706,1707,1708],"td",{},[558,1709,1710],{},"AuditFields",[1706,1712,1713],{},"type",[1706,1715,1716],{},"Reserved field on the wide event",[1688,1718,1719,1724,1727],{},[1706,1720,1721],{},[558,1722,1723],{},"defineAuditAction(name, opts?)",[1706,1725,1726],{},"factory",[1706,1728,1729],{},"Typed action registry, infers target shape",[1688,1731,1732,1737,1739],{},[1706,1733,1734],{},[558,1735,1736],{},"defineAuditCatalog(prefix, map)",[1706,1738,1726],{},[1706,1740,1741],{},"Bundle of typed audit actions sharing a prefix",[1688,1743,1744,1749,1752],{},[1706,1745,1746],{},[558,1747,1748],{},"log.audit(fields)",[1706,1750,1751],{},"method",[1706,1753,1754,1755,1758],{},"Sugar over ",[558,1756,1757],{},"log.set({ audit })"," + force-keep",[1688,1760,1761,1766,1768],{},[1706,1762,1763],{},[558,1764,1765],{},"log.audit.deny(reason, fields)",[1706,1767,1751],{},[1706,1769,1770],{},"Records a denied action",[1688,1772,1773,1778,1781],{},[1706,1774,1775],{},[558,1776,1777],{},"audit(fields)",[1706,1779,1780],{},"function",[1706,1782,1783],{},"Standalone for scripts \u002F jobs",[1688,1785,1786,1791,1794],{},[1706,1787,1788],{},[558,1789,1790],{},"withAudit({ action, target })(fn)",[1706,1792,1793],{},"wrapper",[1706,1795,1796],{},"Auto-emit success \u002F failure \u002F denied",[1688,1798,1799,1804,1807],{},[1706,1800,1801],{},[558,1802,1803],{},"auditDiff(before, after)",[1706,1805,1806],{},"helper",[1706,1808,1809,1810],{},"Redact-aware JSON Patch for ",[558,1811,1812],{},"changes",[1688,1814,1815,1819,1822],{},[1706,1816,1817],{},[558,1818,1365],{},[1706,1820,1821],{},"test util",[1706,1823,1824],{},"Capture + assert audits in tests",[1688,1826,1827,1832,1835],{},[1706,1828,1829],{},[558,1830,1831],{},"auditEnricher(opts?)",[1706,1833,1834],{},"enricher",[1706,1836,1837],{},"Auto-fill request \u002F runtime \u002F tenant context",[1688,1839,1840,1845,1847],{},[1706,1841,1842],{},[558,1843,1844],{},"auditOnly(drain, { await? })",[1706,1846,1793],{},[1706,1848,1849,1850,1852],{},"Routes only events with an ",[558,1851,953],{}," field",[1688,1854,1855,1860,1862],{},[1706,1856,1857],{},[558,1858,1859],{},"signed(drain, opts)",[1706,1861,1793],{},[1706,1863,1864],{},"Generic integrity wrapper (hmac \u002F hash-chain)",[1688,1866,1867,1872,1875],{},[1706,1868,1869],{},[558,1870,1871],{},"auditRedactPreset",[1706,1873,1874],{},"config",[1706,1876,1877],{},"Strict PII for audit events",[554,1879,1880,1881,1883],{},"Everything ships from the main ",[558,1882,623],{}," entrypoint.",[1885,1886,1887],"style",{},"html pre.shiki code .s7zQu, html code.shiki .s7zQu{--shiki-light:#39ADB5;--shiki-light-font-style:italic;--shiki-default:#89DDFF;--shiki-default-font-style:italic;--shiki-dark:#89DDFF;--shiki-dark-font-style:italic}html pre.shiki code .sMK4o, html code.shiki .sMK4o{--shiki-light:#39ADB5;--shiki-default:#89DDFF;--shiki-dark:#89DDFF}html pre.shiki code .sTEyZ, html code.shiki .sTEyZ{--shiki-light:#90A4AE;--shiki-default:#EEFFFF;--shiki-dark:#BABED8}html pre.shiki code .sfazB, html code.shiki .sfazB{--shiki-light:#91B859;--shiki-default:#C3E88D;--shiki-dark:#C3E88D}html pre.shiki code .s2Zo4, html code.shiki .s2Zo4{--shiki-light:#6182B8;--shiki-default:#82AAFF;--shiki-dark:#82AAFF}html pre.shiki code .swJcz, html code.shiki .swJcz{--shiki-light:#E53935;--shiki-default:#F07178;--shiki-dark:#F07178}html pre.shiki code .sbssI, html code.shiki .sbssI{--shiki-light:#F76D47;--shiki-default:#F78C6C;--shiki-dark:#F78C6C}html pre.shiki code .sfNiH, html code.shiki .sfNiH{--shiki-light:#FF5370;--shiki-default:#FF9CAC;--shiki-dark:#FF9CAC}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .spNyl, html code.shiki .spNyl{--shiki-light:#9C3EDA;--shiki-default:#C792EA;--shiki-dark:#C792EA}html pre.shiki code .sHdIc, html code.shiki .sHdIc{--shiki-light:#90A4AE;--shiki-light-font-style:italic;--shiki-default:#EEFFFF;--shiki-default-font-style:italic;--shiki-dark:#BABED8;--shiki-dark-font-style:italic}html pre.shiki code .sBMFI, html code.shiki .sBMFI{--shiki-light:#E2931D;--shiki-default:#FFCB6B;--shiki-dark:#FFCB6B}",{"title":585,"searchDepth":629,"depth":629,"links":1889},[1890,1891,1892,1893,1894],{"id":573,"depth":629,"text":574},{"id":815,"depth":629,"text":816},{"id":1040,"depth":629,"text":1041},{"id":1359,"depth":629,"text":1360},{"id":1679,"depth":629,"text":1680},"File system, Axiom, and Postgres recipes for audit logs, plus mockAudit for tests and the full API reference.","md",[1898,1901],{"label":152,"icon":129,"to":153,"color":1899,"variant":1900},"neutral","subtle",{"label":447,"icon":1902,"to":452,"color":1899,"variant":1900},"i-lucide-plug",{},{"title":156,"icon":64},{"title":549,"description":1895},"H3Qr_9083_axnLlOraNZDhtYetJLfzBabxjPxx7f9jA",[1908,1910],{"title":152,"path":153,"stem":154,"description":1909,"icon":129,"children":-1},"Integrity, redact presets, GDPR vs append-only, retention windows, and the most common pitfalls when shipping audit logs to production.",{"title":165,"path":166,"stem":167,"description":1911,"icon":168,"children":-1},"Understand the full lifecycle of an evlog event, from creation to drain. Covers all three modes (simple logging, wide events, request logging), sampling, enrichment, and delivery.",1778338247654]