|
| 1 | +# CLAUDE.md |
| 2 | + |
| 3 | +This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository. |
| 4 | + |
| 5 | +## Commands |
| 6 | + |
| 7 | +All commands use **Yarn** (not npm). |
| 8 | + |
| 9 | +```bash |
| 10 | +yarn dev # Start dev server (processes env YAML first via scripts/unyamlify-env-local.ts) |
| 11 | +yarn build # Type-check (tsc) then build for production |
| 12 | +yarn test # Run all tests once (Vitest, non-watch) |
| 13 | +yarn format # Format all .ts/.tsx/.json/.md files with Prettier |
| 14 | +yarn format:check # Check formatting without writing |
| 15 | +yarn storybook # Launch Storybook on port 6006 |
| 16 | +``` |
| 17 | + |
| 18 | +**Run a single test file:** |
| 19 | + |
| 20 | +```bash |
| 21 | +yarn vitest run src/core/usecases/launcher/decoupledLogic/computeHelmValues.test.ts |
| 22 | +``` |
| 23 | + |
| 24 | +**Run tests matching a name pattern:** |
| 25 | + |
| 26 | +```bash |
| 27 | +yarn vitest run --reporter=verbose -t "pattern" |
| 28 | +``` |
| 29 | + |
| 30 | +Pre-commit hooks run `eslint --fix` and `prettier --write` via lint-staged. |
| 31 | + |
| 32 | +## Architecture |
| 33 | + |
| 34 | +Onyxia Web is a React SPA — a data science platform portal for launching Kubernetes services (Helm charts), browsing catalogs, managing S3 files, managing Vault secrets, and querying data via DuckDB. It is deployed as static files served by nginx. |
| 35 | + |
| 36 | +### Core principles |
| 37 | + |
| 38 | +- **React is only for rendering.** Business logic is React-agnostic and lives in `src/core/`. The `src/ui/` layer is strictly for React components and hooks. |
| 39 | +- **Unidirectional dependencies.** `src/core/` never imports from `src/ui/`, not even for types. |
| 40 | +- **Reactive over promise-based.** Thunks update observable state; the UI reacts to state changes. Prefer dispatching actions and reading state over returning values from thunks. |
| 41 | +- **Constants outside Redux state.** Values that don't change are not stored in state — they are retrieved from thunks when needed, to avoid unnecessary re-renders. |
| 42 | + |
| 43 | +### `src/core/` — Business logic |
| 44 | + |
| 45 | +Follows a clean-architecture / ports-and-adapters pattern using the `clean-architecture` npm package (a Redux-like store without Redux). |
| 46 | + |
| 47 | +- **`ports/`** — TypeScript interfaces defining contracts for external dependencies (`OnyxiaApi`, `Oidc`, `S3Client`, `SecretsManager`, `SqlOlap`). |
| 48 | +- **`adapters/`** — Concrete implementations: `onyxiaApi/` (axios-based HTTP), `oidc/` (oidc-spa), `s3Client/` (AWS SDK v3), `secretManager/` (Vault), `sqlOlap/` (DuckDB WASM). Each adapter has a mock counterpart for dev/testing. |
| 49 | +- **`usecases/`** — One folder per feature (20+ total: `catalog`, `launcher`, `serviceManagement`, `fileExplorer`, `secretExplorer`, `dataExplorer`, etc.). Each usecase follows the pattern: |
| 50 | + - `state.ts` — state shape + `createUsecaseActions` (slice-like) |
| 51 | + - `thunks.ts` — async side effects, accesses adapters via `createUsecaseContextApi` |
| 52 | + - `selectors.ts` — memoized state derivations |
| 53 | + - `index.ts` — re-exports all three |
| 54 | +- **`bootstrap.ts`** — Wires adapters together and creates the core store. |
| 55 | +- **`index.ts`** — Exports `useCoreState`, `getCore`, `createReactApi` bindings consumed by `src/ui/`. |
| 56 | + |
| 57 | +**Complex use-cases** (especially `launcher/`) have a `decoupledLogic/` subfolder with pure functions and no framework dependencies — this is where most unit tests live. |
| 58 | + |
| 59 | +### `src/ui/` — React layer |
| 60 | + |
| 61 | +- **`App/`** — Root layout: Header, LeftBar, Main, Footer. `App.tsx` triggers core bootstrap; `Main.tsx` is the route-based page switcher. |
| 62 | +- **`pages/`** — One folder per route/page. Each page exports `routeDefs` (via `type-route`'s `defineRoute`) and `routeGroup`. All are merged in `pages/index.ts`. |
| 63 | +- **`routes.tsx`** — Router instantiation. Navigation uses `routes.catalog(...).push()` or `session.push()`. |
| 64 | +- **`i18n/`** — i18nifty setup. Translation keys are declared at the component level via `declareComponentKeys`, collected into a `ComponentKey` union in `i18n/types.ts`. Nine languages: en, fr, zh-CN, no, fi, nl, it, es, de. |
| 65 | +- **`theme/`** — onyxia-ui theme setup (palette, fonts, favicon). |
| 66 | +- **`shared/`** — Reusable components (CommandBar, CodeBlock, SettingField, etc.). |
| 67 | + |
| 68 | +### Key patterns |
| 69 | + |
| 70 | +**Consuming core state in React:** |
| 71 | + |
| 72 | +```ts |
| 73 | +import { useCoreState, getCore } from "core"; |
| 74 | +const helmReleases = useCoreState(state => state.serviceManagement.helmReleases); |
| 75 | +await getCore().dispatch(usecases.serviceManagement.thunks.initialize()); |
| 76 | +``` |
| 77 | + |
| 78 | +**Styling — tss-react** (not plain CSS modules): |
| 79 | + |
| 80 | +```ts |
| 81 | +import { tss } from "tss"; |
| 82 | +const useStyles = tss.withName({ MyComponent }).create(({ theme }) => ({ ... })); |
| 83 | +const { classes, cx } = useStyles(); |
| 84 | +``` |
| 85 | + |
| 86 | +**Absolute imports** — `tsconfig.json` sets `baseUrl: "src"`, so use `import { foo } from "core/usecases/catalog"` (not relative paths). |
| 87 | + |
| 88 | +**Environment variables** — All env vars are centrally parsed and validated in `src/env.ts`. The `index.html` is an EJS template processed by `vite-envs` at build time. |
| 89 | + |
| 90 | +**Authentication** — OIDC init (`oidc-spa`) happens before React renders, in `main.tsx`. Use the `Oidc` port interface, not the adapter directly. |
| 91 | + |
| 92 | +**Plugin system** — `src/pluginSystem.ts` exposes `window.onyxia` after boot and fires an `"onyxiaready"` `CustomEvent`, allowing external JS to interact with core state, routes, theme, and i18n. |
| 93 | + |
| 94 | +**Keycloak theme** — `src/keycloak-theme/` is a Keycloakify login theme that shares env and i18n infrastructure with the main app. Build with `yarn build-keycloak-theme`. |
| 95 | + |
| 96 | +## Key libraries |
| 97 | + |
| 98 | +| Library | Role | |
| 99 | +| -------------------- | ------------------------------------------------------------ | |
| 100 | +| `onyxia-ui` | In-house design system on top of MUI v6 | |
| 101 | +| `type-route` | Strongly-typed client-side router | |
| 102 | +| `i18nifty` | Component-level i18n | |
| 103 | +| `clean-architecture` | Redux-like store (ports/usecases pattern) | |
| 104 | +| `oidc-spa` | OIDC/OAuth2 authentication | |
| 105 | +| `keycloakify` | Keycloak login theme from React components | |
| 106 | +| `tss-react` | CSS-in-JS bound to onyxia-ui theme | |
| 107 | +| `vite-envs` | Env var injection into EJS `index.html` at build time | |
| 108 | +| DuckDB WASM | In-browser SQL OLAP queries (`dataExplorer`, `sqlOlapShell`) | |
0 commit comments