elysiajs
ElysiaJS Development Skill
Always consult elysiajs.com/llms.txt for code examples and latest API.
Overview
ElysiaJS is a TypeScript framework for building Bun-first (but not limited to Bun) type-safe, high-performance backend servers. This skill provides comprehensive guidance for developing with Elysia, including routing, validation, authentication, plugins, integrations, and deployment.
When to Use This Skill
Trigger this skill when the user asks to:
- Create or modify ElysiaJS routes, handlers, or servers
- Setup validation with any Standard Schema library (Zod, Valibot, ArkType) or TypeBox
- Implement authentication (JWT, session-based, macros, guards)
- Add plugins (CORS, OpenAPI, Static files, JWT)
- Integrate with external services (Drizzle ORM, Better Auth, Next.js, Eden Treaty)
- Setup WebSocket endpoints for real-time features
- Create unit tests for Elysia instances
- Deploy Elysia servers to production
Quick Start
Quick scaffold:
bun create elysia app
Basic Server
import { Elysia, t, status } from 'elysia'
const app = new Elysia()
.get('/', () => 'Hello World')
.post('/user', ({ body }) => body, {
body: t.Object({
name: t.String(),
age: t.Number()
})
})
.get('/id/:id', ({ params: { id } }) => {
if(id > 1_000_000) return status(404, 'Not Found')
return id
}, {
params: t.Object({
id: t.Number({ minimum: 1 })
}),
response: {
200: t.Number(),
404: t.Literal('Not Found')
}
})
.listen(3000)
Note: This example uses TypeBox (
t) for brevity. If your project uses Zod, Valibot, or another Standard Schema library, use that instead. See the "Validation" section below.
Basic Usage
HTTP Methods
import { Elysia } from 'elysia'
new Elysia()
.get('/', 'GET')
.post('/', 'POST')
.put('/', 'PUT')
.patch('/', 'PATCH')
.delete('/', 'DELETE')
.options('/', 'OPTIONS')
.head('/', 'HEAD')
Path Parameters
.get('/user/:id', ({ params: { id } }) => id)
.get('/post/:id/:slug', ({ params }) => params)
Query Parameters
.get('/search', ({ query }) => query.q)
// GET /search?q=elysia → "elysia"
Request Body
.post('/user', ({ body }) => body)
Headers
.get('/', ({ headers }) => headers.authorization)
Validation (Standard Schema)
Elysia natively supports any schema library that implements the Standard Schema spec — Zod, Valibot, ArkType, Effect Schema, Yup, and others. You can also use the built-in TypeBox (t from 'elysia'). Mix and match validators freely within the same handler.
IMPORTANT: When the user's project already uses a Standard Schema library (Zod, Valibot, etc.), prefer that library over TypeBox. Only use TypeBox (t) if the project has no existing schema library or explicitly uses TypeBox.
With Zod
import { Elysia } from 'elysia'
import { z } from 'zod'
new Elysia()
.post('/user', ({ body }) => body, {
body: z.object({
name: z.string(),
age: z.number().min(0),
email: z.string().email(),
website: z.string().url().optional()
})
})
With Valibot
import { Elysia } from 'elysia'
import * as v from 'valibot'
new Elysia()
.post('/user', ({ body }) => body, {
body: v.object({
name: v.string(),
age: v.pipe(v.number(), v.minValue(0)),
email: v.pipe(v.string(), v.email())
})
})
Mixing Validators
import { z } from 'zod'
import * as v from 'valibot'
new Elysia()
.get('/id/:id', ({ params, query }) => params.id, {
params: z.object({ id: z.coerce.number() }),
query: v.object({ name: v.literal('Lilith') })
})
With TypeBox (Built-in)
import { Elysia, t } from 'elysia'
new Elysia()
.post('/user', ({ body }) => body, {
body: t.Object({
name: t.String(),
age: t.Number(),
email: t.String({ format: 'email' }),
website: t.Optional(t.String({ format: 'uri' }))
})
})
File Upload
// With TypeBox
.post('/upload', ({ body }) => body.file, {
body: t.Object({
file: t.File({ type: 'image', maxSize: '5m' }),
files: t.Files({ type: ['image/png', 'image/jpeg'] })
})
})
// With Standard Schema — use fileType() for secure content-type validation
import { fileType } from 'elysia'
import { z } from 'zod'
.post('/upload', ({ body }) => body.file, {
body: z.object({
file: z.file().refine((file) => fileType(file, 'image/jpeg'))
})
})
Response Validation
// Works with any schema library
.get('/user/:id', ({ params: { id } }) => ({
id,
name: 'John',
email: 'john@example.com'
}), {
params: z.object({ id: z.coerce.number() }),
response: {
200: z.object({
id: z.number(),
name: z.string(),
email: z.string()
}),
404: z.string()
}
})
TypeBox vs Standard Schema
| Feature | TypeBox (t) |
Standard Schema (Zod, etc.) |
|---|---|---|
| Runtime validation | Yes | Yes |
| Type inference | Yes | Yes |
| OpenAPI schema generation | Automatic | Not supported |
Reference Models ('modelName') |
Yes | Not supported |
File validation (t.File) |
Built-in | Use fileType() utility |
| Auto-coercion (params/query) | Automatic | Library-dependent (e.g., z.coerce) |
Error Handling
.get('/user/:id', ({ params: { id }, status }) => {
const user = findUser(id)
if (!user) {
return status(404, 'User not found')
}
return user
})
Guards (Apply to Multiple Routes)
// With TypeBox
.guard({
params: t.Object({ id: t.Number() })
}, app => app
.get('/user/:id', ({ params: { id } }) => id)
.delete('/user/:id', ({ params: { id } }) => id)
)
// With Zod
.guard({
params: z.object({ id: z.coerce.number() })
}, app => app
.get('/user/:id', ({ params: { id } }) => id)
.delete('/user/:id', ({ params: { id } }) => id)
)
Macro
.macro({
hi: (word: string) => ({
beforeHandle() { console.log(word) }
})
})
.get('/', () => 'hi', { hi: 'Elysia' })
Project Structure (Recommended)
Elysia takes an unopinionated approach but based on user request. But without any specific preference, we recommend a feature-based and domain driven folder structure where each feature has its own folder containing controllers, services, and models.
src/
├── index.ts # Main server entry
├── modules/
│ ├── auth/
│ │ ├── index.ts # Auth routes (Elysia instance)
│ │ ├── service.ts # Business logic
│ │ └── model.ts # TypeBox schemas/DTOs
│ └── user/
│ ├── index.ts
│ ├── service.ts
│ └── model.ts
└── plugins/
└── custom.ts
public/ # Static files (if using static plugin)
test/ # Unit tests
Each file has its own responsibility as follows:
- Controller (index.ts): Handle HTTP routing, request validation, and cookie.
- Service (service.ts): Handle business logic, decoupled from Elysia controller if possible.
- Model (model.ts): Define the data structure and validation for the request and response.
Best Practice
Elysia is unopinionated on design pattern, but if not provided, we can relies on MVC pattern pair with feature based folder structure.
- Controller:
- Prefers Elysia as a controller for HTTP dependant controller
- For non HTTP dependent, prefers service instead unless explicitly asked
- Use
onErrorto handle local custom errors - Register Model to Elysia instance via
Elysia.models({ ...models })and prefix model by namespace `Elysia.prefix('model', 'Namespace.') - Prefers Reference Model by name provided by Elysia instead of using an actual
Model.name
- Service:
- Prefers class (or abstract class if possible)
- Prefers interface/type derive from
Model - Return
status(import { status } from 'elysia') for error - Prefers
return Errorinstead ofthrow Error
- Models:
- Always export validation model and type of validation model
- Custom Error should be in contains in Model
Elysia Key Concept
Elysia has a every important concepts/rules to understand before use.
Encapsulation - Isolates by Default
Lifecycles (hooks, middleware) don't leak between instances unless scoped.
Scope levels:
local(default) - current instance + descendantsscoped- parent + current + descendantsglobal- all instances
.onBeforeHandle(() => {}) // only local instance
.onBeforeHandle({ as: 'global' }, () => {}) // exports to all
Method Chaining - Required for Types
Must chain. Each method returns new type reference.
❌ Don't:
const app = new Elysia()
app.state('build', 1) // loses type
app.get('/', ({ store }) => store.build) // build doesn't exists
✅ Do:
new Elysia()
.state('build', 1)
.get('/', ({ store }) => store.build)
Explicit Dependencies
Each instance independent. Declare what you use.
const auth = new Elysia()
.decorate('Auth', Auth)
.model(Auth.models)
new Elysia()
.get('/', ({ Auth }) => Auth.getProfile()) // Auth doesn't exists
new Elysia()
.use(auth) // must declare
.get('/', ({ Auth }) => Auth.getProfile())
Global scope when:
- No types added (cors, helmet)
- Global lifecycle (logging, tracing)
Explicit when:
- Adds types (state, models)
- Business logic (auth, db)
Deduplication
Plugins re-execute unless named:
new Elysia() // rerun on `.use`
new Elysia({ name: 'ip' }) // runs once across all instances
Order Matters
Events apply to routes registered after them.
.onBeforeHandle(() => console.log('1'))
.get('/', () => 'hi') // has hook
.onBeforeHandle(() => console.log('2')) // doesn't affect '/'
Type Inference
Inline functions only for accurate types.
For controllers, destructure in inline wrapper:
.post('/', ({ body }) => Controller.greet(body), {
body: t.Object({ name: t.String() })
})
Get type from schema:
type MyType = typeof MyType.static
Reference Model
Model can be reference by name, especially great for documenting an API
new Elysia()
.model({
book: t.Object({
name: t.String()
})
})
.post('/', ({ body }) => body.name, {
body: 'book'
})
Model can be renamed by using .prefix / .suffix
new Elysia()
.model({
book: t.Object({
name: t.String()
})
})
.prefix('model', 'Namespace')
.post('/', ({ body }) => body.name, {
body: 'Namespace.Book'
})
Once prefix, model name will be capitalized by default.
Technical Terms
The following are technical terms that is use for Elysia:
OpenAPI Type Gen- function namefromTypesfrom@elysiajs/openapifor generating OpenAPI from types, seeplugins/openapi.mdEden,Eden Treaty- e2e type safe RPC client for share type from backend to frontend
Resources
Use the following references as needed.
It's recommended to checkout route.md for as it contains the most important foundation building blocks with examples.
plugin.md and validation.md is important as well but can be check as needed.
references/
Detailed documentation split by topic:
bun-fullstack-dev-server.md- Bun Fullstack Dev Server with HMR. React without bundler.cookie.md- Detailed documentation on cookiedeployment.md- Production deployment guide / Dockereden.md- e2e type safe RPC client for share type from backend to frontendguard.md- Setting validation/lifecycle all at oncemacro.md- Compose multiple schema/lifecycle as a reusable Elysia via key-value (recommended for complex setup, eg. authentication, authorization, Role-based Access Check)plugin.md- Decouple part of Elysia into a standalone componentroute.md- Elysia foundation building block: Routing, Handler and Contexttesting.md- Unit tests with examplesvalidation.md- Validation with Standard Schema (Zod, Valibot, ArkType, etc.) and TypeBox, including all custom validation ruleswebsocket.md- Real-time features
plugins/
Detailed documentation, usage and configuration reference for official Elysia plugin:
bearer.md- Add bearer capability to Elysia (@elysiajs/bearer)cors.md- Out of box configuration for CORS (@elysiajs/cors)cron.md- Run cron job with access to Elysia context (@elysiajs/cron)graphql-apollo.md- Integration GraphQL Apollo (@elysiajs/graphql-apollo)graphql-yoga.md- Integration with GraphQL Yoga (@elysiajs/graphql-yoga)html.md- HTML and JSX plugin setup and usage (@elysiajs/html)jwt.md- JWT / JWK plugin (@elysiajs/jwt)openapi.md- OpenAPI documentation and OpenAPI Type Gen / OpenAPI from types (@elysiajs/openapi)opentelemetry.md- OpenTelemetry, instrumentation, and record span utilities (@elysiajs/opentelemetry)server-timing.md- Server Timing metric for debug (@elysiajs/server-timing)static.md- Serve static files/folders for Elysia Server (@elysiajs/static)
integrations/
Guide to integrate Elysia with external library/runtime:
ai-sdk.md- Using Vercel AI SDK with Elysiaastro.md- Elysia in Astro API routebetter-auth.md- Integrate Elysia with better-authcloudflare-worker.md- Elysia on Cloudflare Worker adapterdeno.md- Elysia on Denodrizzle.md- Integrate Elysia with Drizzle ORMexpo.md- Elysia in Expo API routenextjs.md- Elysia in Nextjs API routenodejs.md- Run Elysia on Node.jsnuxt.md- Elysia on API routeprisma.md- Integrate Elysia with Prismareact-email.d- Create and Send Email with React and Elysiasveltekit.md- Run Elysia on Svelte Kit API routetanstack-start.md- Run Elysia on Tanstack Start / React Queryvercel.md- Deploy Elysia to Vercel
examples/ (optional)
basic.ts- Basic Elysia examplebody-parser.ts- Custom body parser example via.onParsecomplex.ts- Comprehensive usage of Elysia servercookie.ts- Setting cookieerror.ts- Error handlingfile.ts- Returning local file from serverguard.ts- Setting mulitple validation schema and lifecyclemap-response.ts- Custom response mapperredirect.ts- Redirect responserename.ts- Rename context's propertyschema.ts- Setup validationstate.ts- Setup global stateupload-file.ts- File upload with validationwebsocket.ts- Web Socket for realtime communication
patterns/ (optional)
patterns/mvc.md- Detail guideline for using Elysia with MVC patterns