Development
The repo is a Go module plus a pnpm workspace. The Go binary embeds the built SPA, so a full local build runs both toolchains.
#Prerequisites
- Go (matching
go.mod). - pnpm 11 (auto-managed via
corepack). - TypeScript runs via
tsgofrom@typescript/native-preview— installed - Lint/format use
oxlintandoxfmt— installed through pnpm.
through pnpm.
#First run
pnpm install
pnpm build # builds SPA + SDK and copies dist into apps/api
go run ./apps/api/cmd/clickclack serve
open http://localhost:8080
The dev fallback creates Local Captain as the first user, a ClickClack workspace, and a general channel, so the SPA loads into a working state on first hit.
#Two-process dev loop
# terminal 1
pnpm dev:api # go run ... serve --dev-bootstrap=true
# terminal 2
pnpm dev:web # vite dev server with API proxy
The Vite dev server proxies /api and /api/realtime/ws to localhost:8080.
#Scripts
| Command | What it does |
|---|---|
pnpm build | Builds the Svelte app and the SDK, then embeds apps/web/dist into apps/api/internal/webassets/dist. |
pnpm build:web | Builds and normalizes the Svelte app without touching embedded Go assets. |
pnpm build:sdk | Builds the TypeScript SDK. |
pnpm check | Full local gate: pnpm test, root/workspace tsgo, oxlint, and format checks. |
pnpm coverage | Go tests with coverage; fails under 85% line coverage. |
pnpm dev:api | go run ./apps/api/cmd/clickclack serve --dev-bootstrap=true. |
pnpm dev:web | vite dev for the SPA. |
pnpm fmt | gofmt + oxfmt over Go and TS/Svelte. |
pnpm fmt:check | CI-compatible formatting check with gofmt -l and oxfmt --check. |
pnpm lint | oxlint over web, SDK, examples, and tests. |
goreleaser release --snapshot --clean | Local release smoke test for all configured OS/arch targets. |
pnpm typecheck | tsgo --noEmit -p tsconfig.json for root Playwright config/tests. |
pnpm test | Builds the web app and SDK, then runs Go tests against those fresh web assets in a temp copy without rewriting tracked embedded assets. |
pnpm test:e2e | Playwright suite in tests/e2e. |
pnpm build uses CLICKCLACK_WEB_VERSION=dev by default. That keeps repeated local builds deterministic while still allowing real source changes to update content-hashed assets. Release and Docker builds should set CLICKCLACK_WEB_VERSION to the commit or tag being shipped.
#Layout
apps/
api/ # Go backend, single-binary entrypoint
cmd/clickclack/ # CLI main
internal/
auth/ # placeholder
config/ # flag/env/file resolution
httpapi/ # chi router, handlers, auth resolution
realtime/ # in-process pub/sub hub
store/ # store interface + types
sqlite/ # SQLite implementation, migrations, backup, export
postgres/ # Postgres implementation, migrations, export
webassets/ # go:embed for the built SPA
web/ # Svelte 5 SPA
packages/
protocol/ # OpenAPI spec, source of truth for the wire shape
sdk-ts/ # TypeScript SDK (generated types + friendly wrapper)
examples/
bot-ts/ # SDK usage example
infra/
migrations/sqlite # mirror of embedded SQLite migrations for tooling
migrations/postgres # mirror of embedded Postgres migrations for tooling
tests/
e2e/ # Playwright tests
docs/ # this directory
#Adding a feature
- Update
packages/protocol/openapi.yamlfirst when the wire shape - Add the store method on
apps/api/internal/store/types.goand implement - Wire the handler in
apps/api/internal/httpapi. - Update the SDK in
packages/sdk-ts/src/index.tsso TS clients have a - Update or add a
docs/features/<thing>.md. - Run
pnpm checkandpnpm coverage.
changes. It is the contract.
it in both apps/api/internal/store/sqlite and apps/api/internal/store/postgres when the feature touches durable state.
typed surface.
#Testing
apps/api/internal/...is the bulk of the test suite. Coverage gate istests/e2e/chat.spec.tsexercises the SPA end-to-end via Playwright.- The SDK has no test target yet — the bot example is the smoke test.
85%.
#Coding rules
- IDs are sortable ULID-style with semantic prefixes (
usr_,wsp_,chn_, - Keep transactions short. Outbox events are inserted in the same tx as the
- Keep SQL behind the store interface. Dialect-specific SQL belongs in the
- Use
sqlcfor typed SQL. Edit schema/query files, then run - TypeScript: no Svelte imports in
packages/sdk-ts. The SDK must stay
msg_, evt_, upl_, idn_).
durable write that produced them.
SQLite/Postgres store packages, not in HTTP handlers.
pnpm generate:sqlc; do not hand-maintain generated storedb code.
framework-neutral.