logging
$
npx mdskill add EpicenterHQ/epicenter/loggingProvides structured logging for library diagnostics with 5 levels and typed errors
- Solves the problem of tracking internal state and errors in library code
- Uses wellcrafted/logger with consoleSink, memorySink, and composeSinks
- Applies level-based filtering and structured data to log messages
- Delivers logs to stdout/stderr for operator-defined external storage
SKILL.md
.github/skills/loggingView on GitHub ↗
---
name: logging
description: 'wellcrafted/logger for library diagnostics: 5 levels, typed errors, DI sink. Use for attach primitives, background errors, file-backed logs. No console.* in library code.'
metadata:
author: epicenter
version: '2.0'
---
# Workspace Logger
Structured, level-keyed, field-oriented logging for library code. Modeled on Rust's `tracing`. Completes the `defineErrors` story: errors are structured data; level lives at the call site.
## Where it lives
All of it ships from **`wellcrafted/logger`** : `createLogger`, `consoleSink`, `memorySink`, `composeSinks`, `tapErr`, and the types. Runtime-agnostic, browser-safe. No file sink in-process: durability is a *host* concern (shell redirect, systemd journal, Cloudflare tail). The library emits to `consoleSink`; the operator decides where stdout/stderr go.
## Quickstart
```ts
import { createLogger } from 'wellcrafted/logger';
const log = createLogger('markdown-materializer'); // defaults to consoleSink
log.info('materializer ready');
log.warn(MarkdownError.TableWrite({ path, cause }));
```
## The 5 levels
`trace | debug | info | warn | error`. No `fatal` : process termination is the app's call, not the library's.
| Level | Signature | Use for |
|---|---|---|
| `trace` | `(message, data?)` | Per-token / per-message noise; off in prod |
| `debug` | `(message, data?)` | Internal state transitions (handshakes, cache loads) |
| `info` | `(message, data?)` | Lifecycle events (connected, loaded, flushed) |
| `warn` | `(err)` | Recoverable failure : retry, fallback, partial result |
| `error` | `(err)` | Unrecoverable at this layer; the operation has given up |
**Shape split is intentional.** `warn` / `error` take a typed error unary : the variant carries `message`, `name`, and captured fields. `trace` / `debug` / `info` are free-form because free-running diagnostic events don't need enumeration.
## Level is a call-site decision, not a variant property
```ts
// Right : same error, different levels in different contexts
log.warn(SyncError.ConnectionFailed({ cause })); // inside retry loop
log.error(SyncError.ConnectionFailed({ cause })); // last attempt, giving up
```
Do NOT attach a `severity` to `defineErrors` variants. That's `miette`'s pattern; `tracing`, `log`, and every production Rust logger put level on the call. Context-dependent data belongs at the context.
## Sinks
A sink is `((event) => void) & Partial<AsyncDisposable>` : a callable with optional resource cleanup.
```ts
import {
createLogger,
consoleSink, // default; routes to console[level]
memorySink, // for tests; returns { sink, events }
composeSinks, // fan out to multiple sinks
tapErr, // Result-flow combinator
} from 'wellcrafted/logger';
```
### Durability is the host's job
For a long-running daemon or CLI that needs durable logs, the library still emits to `consoleSink`; the operator decides where the stream goes:
```bash
bun run start # dev: console
bun run start 2>> ~/.app/app.jsonl # ad-hoc file capture
systemd-run --user bun run start # journal (structured queries via journalctl)
```
This used to be `jsonlFileSink`; that primitive was removed because owning a file writer in-process bought complexity (backpressure, dispose semantics, error fallbacks) that shell redirection already solves.
### `composeSinks(...)` : fan out
```ts
const sink = composeSinks(consoleSink, myCustomSink);
const log = createLogger('source', sink);
```
`composeSinks` forwards disposal to members that implement it (via `sink[Symbol.asyncDispose]?.()`). `consoleSink` is a no-op on dispose; stateful sinks flush and close.
### `memorySink()` : for tests
```ts
const { sink, events } = memorySink();
const log = createLogger('test', sink);
log.warn(MyError.Thing({ cause: new Error('boom') }));
expect(events).toHaveLength(1);
expect(events[0]).toMatchObject({ level: 'warn', source: 'test' });
```
Do NOT assert on `console.*` output. Inject a `memorySink` and inspect the event array.
## `tapErr` : the Result-flow combinator
`tapErr(logFn)` logs on the Err branch and returns the Result unchanged. Takes a log *method*, not a whole logger, so the caller picks the level at the pipeline site.
```ts
const result = await tryAsync({
try: () => writeTable(path),
catch: (cause) => MarkdownError.TableWrite({ path, cause }),
}).then(tapErr(log.warn));
```
Mirrors Rust's `.inspect_err` and Effect's `tapErrorCause`. No message argument : the typed error owns its message.
## DI, not globals
No module-level logger registry. No `setDefaultLogger()`. Each attach primitive takes an optional `log?: Logger` option and defaults to `createLogger(<source>)` (console sink). Caller wires sinks explicitly.
```ts
const markdown = attachMarkdownMaterializer(ydoc, { dir, log });
const sqlite = attachSqliteMaterializer(ydoc, { db, log });
const collaboration = openCollaboration(ydoc, { url, log, openWebSocket, replicaId });
```
Share one sink across loggers when you build a custom one:
```ts
const sink = composeSinks(consoleSink, myCustomSink);
const markdown = attachMarkdownMaterializer(ydoc, { dir, log: createLogger('workspace/markdown', sink) });
const sqlite = attachSqliteMaterializer(ydoc, { db, log: createLogger('workspace/sqlite', sink) });
```
## Browser
The whole surface is pure JS and browser-safe.
## Event shape
Every sink receives:
```ts
type LogEvent = {
ts: number; // epoch millis
level: LogLevel; // 'trace' | 'debug' | 'info' | 'warn' | 'error'
source: string; // from createLogger()
message: string; // human text : for warn/error, inherited from the typed error
data?: unknown; // the typed error for warn/error; free-form for info/debug/trace
};
```
Custom sinks that serialize for the wire should convert `ts` to ISO-8601 and flatten native `Error` instances (otherwise they JSON.stringify to `{}`).
## See also
- `error-handling` skill : the `tryAsync.catch:` → `tapErr(log.warn)` pipeline
- `define-errors` skill : how to mint the typed error variants the logger consumes
- `rust-errors` skill : full `tracing` ↔ `Logger` mapping