express, koa, hapi, and elysia are all server-side frameworks for building APIs and web services in JavaScript and TypeScript. express is the industry standard with a massive ecosystem, using a request-response middleware model. koa is created by the same team but focuses on async/await and a cleaner context object. hapi emphasizes configuration over code with strong validation built-in. elysia is a modern, TypeScript-first framework optimized for the Bun runtime but compatible with Node, focusing on speed and type safety.
Both express and koa are established Node.js frameworks, while hapi offers a configuration-heavy alternative, and elysia brings modern TypeScript features to the table. All four help you build servers, but they handle requests, errors, and types in very different ways. Let's compare how they tackle common engineering problems.
express uses the classic req and res objects passed through middleware.
req to add data and res to send responses.next() to pass control.// express: Classic req/res
app.use((req, res, next) => {
req.user = { id: 1 };
next();
});
app.get('/', (req, res) => {
res.send(`Hello ${req.user.id}`);
});
koa uses a single ctx (context) object instead of separate req/res.
ctx.request and ctx.body live on one object.next() callback needed.// koa: Context object
app.use(async (ctx, next) => {
ctx.state.user = { id: 1 };
await next();
});
app.use(async (ctx) => {
ctx.body = `Hello ${ctx.state.user.id}`;
});
hapi separates the request object and the h response toolkit.
// hapi: Request and toolkit
server.route({
method: 'GET',
path: '/',
handler: (request, h) => {
return `Hello ${request.auth.credentials.id}`;
}
});
elysia uses a single context object similar to Koa but with types.
// elysia: Typed context
app.get('/', ({ user }) => {
return `Hello ${user.id}`;
});
express requires external packages for validation (like joi or zod).
// express: Manual validation
app.post('/user', (req, res, next) => {
if (!req.body.name) return res.status(400).send('Missing name');
res.send('OK');
});
koa also needs external routers and validation libraries.
@koa/router for path handling.// koa: External router
router.post('/user', async (ctx) => {
if (!ctx.request.body.name) {
ctx.status = 400;
return;
}
ctx.body = 'OK';
});
hapi has validation built into the route configuration.
// hapi: Built-in validation
server.route({
method: 'POST',
path: '/user',
options: {
validate: {
payload: Joi.object({ name: Joi.string().required() })
}
},
handler: (request) => 'OK'
});
elysia uses TypeScript types and plugins for validation.
// elysia: Type-safe validation
app.post('/user', ({ body }) => 'OK', {
body: t.Object({
name: t.String()
})
});
express uses a special middleware function with four arguments.
next(err) to trigger it.// express: Error middleware
app.use((err, req, res, next) => {
res.status(500).send(err.message);
});
koa relies on standard try/catch blocks in async functions.
// koa: Try/catch
app.use(async (ctx, next) => {
try {
await next();
} catch (err) {
ctx.status = 500;
ctx.body = err.message;
}
});
hapi uses the boom library for HTTP errors.
// hapi: Boom errors
handler: (request, h) => {
throw Boom.badRequest('Invalid input');
}
elysia has dedicated error handling hooks.
// elysia: Error hook
app.onError(({ code, error }) => {
return { message: error.message };
});
express runs on Node.js and is mature but slower than newer options.
// express: Node.js runtime
import express from 'express';
const app = express();
app.listen(3000);
koa runs on Node.js and is slightly lighter than Express.
// koa: Node.js runtime
import Koa from 'koa';
const app = new Koa();
app.listen(3000);
hapi runs on Node.js with a focus on stability over speed.
// hapi: Node.js runtime
import { Server } from '@hapi/hapi';
const server = new Server({ port: 3000 });
await server.start();
elysia is optimized for Bun but supports Node.js.
// elysia: Bun or Node runtime
import { Elysia } from 'elysia';
new Elysia().listen(3000);
While the differences are clear, all four frameworks share core concepts for building web servers.
http module (or Bun equivalent).// All frameworks ultimately listen on a port
app.listen(3000); // express, koa, elysia
server.start(); // hapi
// express
app.use(logger);
// koa
app.use(logger);
// hapi
server.ext('onRequest', logger);
// elysia
app.use(logger());
// express
app.use(cors());
// koa
app.use(cors());
// hapi
server.register(require('@hapi/cors'));
// elysia
app.use(cors());
// express
app.use(express.json());
// koa
app.use(bodyparser.json());
// hapi
// Built-in payload parsing
// elysia
// Built-in JSON parsing
| Feature | express | koa | hapi | elysia |
|---|---|---|---|---|
| Runtime | 🟢 Node.js | 🟢 Node.js | 🟢 Node.js | 🟣 Bun (Primary) / Node |
| Style | 📝 Callback/Promise | ⏳ Async/Await | ⚙️ Configuration | 🛡️ TypeScript First |
| Validation | 🔌 External | 🔌 External | ✅ Built-in | ✅ Built-in (Types) |
| Context | 📦 req + res | 📦 ctx | 📦 request + h | 📦 Typed Context |
| Learning Curve | 📉 Low | 📉 Low | 📈 High | 📉 Low (for TS users) |
| Feature | Shared by All Four |
|---|---|
| Core Goal | 🌐 Build HTTP APIs |
| Middleware | 🔌 Request interception |
| Ecosystem | 📦 NPM Plugins |
| Language | 💻 JavaScript/TypeScript |
| Deployment | ☁️ Container/Serverless |
express is the reliable workhorse 🐴 — it runs everything, has a plugin for anything, and every host supports it. Ideal for legacy projects, quick prototypes, or teams that need maximum hiring pool compatibility.
koa is the modernized Express 🧹 — same team, cleaner code, no callback hell. Best for teams who want Express flexibility but prefer modern async/await syntax without the baggage.
hapi is the enterprise vault 🏦 — strict, configurable, and secure by default. Perfect for large organizations where configuration rules and validation matter more than setup speed.
elysia is the speed racer 🏎️ — built for today's TypeScript and Bun ecosystem. Choose this for new greenfield projects where performance and type safety are top priorities.
Final Thought: All four frameworks can build robust APIs. Your choice depends on whether you value ecosystem size (express), code cleanliness (koa), configuration strictness (hapi), or modern performance (elysia).
Choose elysia if you are starting a new project and want the best TypeScript experience with high performance. It is perfect for teams adopting the Bun runtime or those who want end-to-end type safety without extra code generation tools.
Choose express if you need maximum compatibility with existing tutorials, plugins, and hosting providers. It is the safest bet for teams that want a vast library of middleware and don't mind managing callback-style middleware patterns.
Choose hapi if you need strict input validation and a configuration-driven architecture out of the box. It suits enterprise environments where security policies and detailed request schemas are more important than raw speed or minimal setup.
Choose koa if you want a lighter foundation than Express with modern async/await support without the baggage of older patterns. It is ideal for teams that prefer building their own middleware stack from scratch rather than relying on a massive ecosystem.
Ergonomic Framework for Humans
Documentation | Discord | Sponsors
TypeScript with End-to-End Type Safety, type integrity, and exceptional developer experience. Supercharged by Bun.