Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

@curveball/core

curveball4.4kMIT1.0.2TypeScript support: included

Curveball is a framework writting in Typescript for Node.js

http, framework, nodejs, typescript, push

readme

Curveball

Curveball is a framework for building web services in Node.js. It fullfills a similar role to Express and it's heavily inspired by Koa.

This web framework has the following goals:

  • A minimal foundation.
  • Completely written in and for TypeScript.
  • Modern Ecmascript features.
  • Async/await-based middleware.
  • Support for Node, Bun, AWS Lambda, Azure functions.
  • Native support for HTTP/2, including easy access to HTTP/2 Push.
  • Native, deeply integrated Websocket.
  • Native support for modern HTTP features, such as 103 Early Hints.
  • The ability to easily do internal sub-requests without having to do a real HTTP request.

If you used Koa in the past, this is going to look pretty familiar. I'm a big fan of Koa myself and would recommend it over this project if you don't need any of the things this project offers.

Installation

npm install @curveball/core

Getting started

Curveball only provides a basic framework. Using it means implementing or using curveball middleware. For example, if you want a router, use or build a Router middleware.

All of the following examples are written in typescript, but it is also possible to use the framework with plain javascript.

import { Application, Context } from '@curveball/core';

const app = new Application();
app.use((ctx: Context) => {

  ctx.status = 200;
  ctx.response.body = 'Hello world!'

});

app.listen(4000);

Middlewares you might want

Authentication

You might like a12n-server, a full OAuth2 authorization server, written in Curveball and works well with the OAuth2 middleware.

AWS Lambda support / 'Serverless'

Bun support

To use Curveball with Bun, use the kernel package:

import { Application } from '@curveball/kernel';

const app = new Application();

// Add all your middlewares here!
app.use( ctx => {
  ctx.response.body = {msg: 'hello world!'};
});

export default {
  port: 3000,
  fetch: app.fetch.bind(app)
};

Some more details can be found in this article.

Doing internal subrequests

Many Node.js HTTP frameworks don't easily allow doing internal sub-requests. Instead, they recommend doing a real HTTP request. These requests are more expensive though, as it has to go through the network stack.

Curveball allows you to do an internal request with 'mock' request and response objects.

Suggested use-cases:

  • Running cheaper integration tests.
  • Embedding resources in REST apis.

Example:

import { Application } from '@curveball/core';

const app = new Application();
const response = await app.subRequest('POST', '/foo/bar', { 'Content-Type': 'text/html' }, '<h1>Hi</h1>');

Only the first 2 arguments are required. It's also possible to pass a Request object instead.

import { Application, MemoryRequest } from '@curveball/core';

const app = new Application();
const request = new MemoryRequest('POST', '/foo/bar', { 'Content-Type': 'text/html' }, '<h1>Hi</h1>');
const response = await app.subRequest(request);

HTTP/2 push

HTTP/2 push can be used to anticipate GET requests client might want to do in the near future.

Example use-cases are:

  • Sending scripts and stylesheets earlier for HTML-based sites.
  • REST api's sending resources based on relationships clients might want to follow.
import { Application } from '@curveball/core';
import http2 from 'http2';

const app = new Application();
const server = http2.createSecureServer({
  key: fs.readFileSync('server-key.pem'),
  cert: fs.readFileSync('server-cert.pem')
}, app.callback());
server.listen(4443);

app.use( ctx => {

  ctx.response.status = 200;
  ctx.response.headers.set('Content-Type', 'text/html');
  ctx.response.body = '';

  await ctx.response.push( pushCtx => {

    pushCtx.path = '/script.js';
    return app.handle(pushCtx);

  });

});

HTTP/2 push works by sending HTTP responses to the client, but it also includes HTTP requests. This is because HTTP clients need to know which request the response belongs to.

The push function simply takes a middleware, similar to use on Application. The callback will only be triggered if the clients supports push and wants to receive pushes.

In the preceding example, we are using app.handle() to do a full HTTP request through all the regular middlewares.

It's not required to do this. You can also generate responses right in the callback or call an alternative middleware.

Lastly, pushCtx.request.method will be set to GET by default. GET is also the only supported method for pushes.

Sending 1xx Informational responses

Curveball has native support for sending informational responses. Examples are:

  • 100 Continue to let a client know even before the request completed that it makes sense to continue, or that it should break off the request.
  • 102 Processing to periodically indicate that the server is still working on the response. This might not be very useful anymore.
  • 103 Early Hints a new standard to let a client or proxy know early in the process that some headers might be coming, allowing clients or proxies to for example pre-fetch certain resources even before the initial request completes.

Here's an example of a middleware using 103 Early Hints:

import { Application, Context, Middleware } from '@curveball/core';

const app = new Curveball();
app.use(async (ctx: Context, next: Middleware) => {

  await ctx.response.sendInformational(103, {
    'Link' : [
      '</style.css> rel="prefetch" as="style"',
      '</script.js> rel="prefetch" as="script"',
    ]
  });
  await next();

});

Websocket

To get Websocket up and running, just run:

app.listenWs(port);

This will start a websocket server on the specified port. Any incoming Websocket connections will now just work.

If a Websocket connection was started, the Context object will now have a webSocket property. This property is simply an instance of Websocket from the ws NPM package.

Example usage:

import { UpgradeRequired } from '@curveball/http-errors';

app.use( ctx => {
  if (!ctx.webSocket) {
    throw new UpgradeRequired('This endpoint only supports WebSocket');
  }

  ctx.webSocket.send('Hello');
  ctx.webSocket.on('message', (msg) => {
    console.log('Received %s', msg);
  });

});

If you use typescript, install the @types/ws package to get all the correct typings:

npm i -D @types/ws

The Controller package also has built-in features to make this even easier.

API

The Application class

The application is main class for your project. It's mainly responsible for calling middlewares and hooking into the HTTP server.

It has the following methods

  • use(m: Middleware) - Add a middleware to your application.
  • handle(c: Context) - Take a Context object, and run all middlewares in order on it.
  • listen(port: number) - Run a HTTP server on the specified port.
  • listenWs(port: number) - Start a websocket server on the specified port.
  • callback() - The result of this function can be used as a requestListener for node.js http, https and http2 packages.
  • subRequest(method: string, path:string, headers: object, body: any) - Run an internal HTTP request and return the result.
  • subRequest(request: Request) - Run an internal HTTP request and return the result.
  • origin - Sets the 'origin' for the application. This is used to determine absolute URIs. You can set the origin directly on the application, but you can also set a CURVEBALL_ORIGIN environment variable. If nothing is set this value will default to http://localhost.

The Context class

The Context object has the following properties:

  • request - An instance of Request.
  • response - An instance of Response.
  • state - An object you can use to store request-specific state information. this object can be used to pass information between middlewares. A common example is that an authentication middlware might set 'currently logged in user' information here.
  • ip() - Get the ip address of the HTTP client that's trying to connect.
  • path - The path of the request, for example /foo.html.
  • method - For example, POST.
  • query - An object containing the query parametes.
  • status - The HTTP status code, for example 200 or 404.
  • sendInformational(status, headers?) - Sends a 100 Continue, 102 Processing or 103 Early Hints - response with optional headers.
  • push(callback: Middleware) - Do a HTTP/2 push.
  • redirect(status, location) - Send a redirect status code and set a Location header.
  • absoluteUrl - The absolute URL for the request.

The Request interface

The Request interface represents the HTTP request. It has the following properties and methods:

  • headers - An instance of Headers.
  • path - The path of the request, for example /foo.html.
  • method - For example, POST.
  • requestTarget - The full requestTarget from the first line of the HTTP request.
  • body - This might represent the body, but is initially just empty. It's up to middlewares to do something with raw body and parse it.
  • rawBody() - This function uses the raw-body function to parse the body from the request into a string or Buffer. You can only do this once, so a middleware should use this function to populate body.
  • query - An object containing the query parametes.
  • type - The Content-Type without additional parameters.
  • accepts - Uses the accepts package to do content-negotiation.
  • is(contentType) - Returns true or false if the Content-Type of the request matches the argument. If your Content-Type is application/hal+json it will return true for application/hal+json, hal+json and json.
  • origin - The 'origin' for the request, for example: http://my-api:8008.
  • absoluteUrl - The absolute URL for the request.
  • ip() - Returns the ip address of the client. This may be ipv4 or ipv6. If CURVEBALL_TRUSTPROXY is set in the environment and truthy, this will use the information from the X-Forwarded-For header (if available).

The Response interface

The Response interface represents a HTTP response. It has the following properties and methods:

  • headers - An instance of Headers.
  • status - The HTTP status code, for example 200 or 404.
  • body - The response body. Can be a string, a buffer or an Object. If it's an object, the server will serialize it as JSON.
  • type - The Content-Type without additional parameters.
  • sendInformational(status, headers?) - Sends a 100 Continue, 102 Processing or 103 Early Hints - response with optional headers.
  • push(callback: Middleware) - Do a HTTP/2 push.
  • is(contentType) - Returns true or false if the Content-Type of the response matches the argument. If your Content-Type is application/hal+json it will return true for application/hal+json, hal+json and json.
  • redirect(status, location) - Send a redirect status code and set a Location header.
  • origin - The 'origin' for the request, for example: http://my-api:8008.

The Headers interface

The Headers interface represents HTTP headers for both the Request and Response.

It has the following methods:

  • set(name, value) - Sets a HTTP header.
  • get(name) - Returns the value of a HTTP header, or null.
  • has(name) - Returns true or false if the header exists.
  • delete(name) - Deletes a HTTP header.
  • append(name, value) - Adds a HTTP header, but doesn't erase an existing one with the same name.
  • getAll() - Returns all HTTP headers as a key-value object.

Other features

Use the checkConditional function to verify the following headers:

  • If-Match
  • If-None-Match
  • If-Modified-Since
  • If-Unmodified-Since.

Signature:

checkConditionial(req: RequestInterface, lastModified: Date | null, etag: string | null): 200 | 304 : 412;

This function returns 200 if the conditional passed. If it didn't, it will return either 304 or 412. The former means you'll want to send a 304 Not Modified back, the latter 412 Precondition Failed.

200 does not mean you have to return a 200 OK status, it's just an easy way to indicate that all all conditions have passed.

changelog

Changelog

1.0.2 (2024-11-06)

  • Fixed another bug related to CURVEBALL_TRUSTPROXY

1.0.1 (2024-10-30)

  • Actually check CURVEBALL_TRUSTPROXY.

1.0.0 (2024-01-15)

  • Finally! Curveball v1. Only took 6 years.
  • CommonJS support has been dropped. The previous version of this library supported both CommonJS and ESM. The effort of this no longer feels worth it. ESM is the future, so we're dropping CommonJS.
  • Now requires Node 18.

0.21.1 (2023-02-17)

  • Export server version string

0.21.0 (2023-02-14)

  • ESM support.
  • Dropped Node 14 support. Use Node 16 or later.

0.20.0 (2022-09-03)

  • Most of the internal plumbing has moved to @curveball/kernel. @curveball/core still contains all of the Node-specific code, but this allows curveball to run on Bun.

0.20.0-alpha.0 (2022-09-01)

  • Added support for fetch(), which lets you make requests in an a Curveball application using the native Request and Response objects.

0.19.0 (2022-04-25)

  • Now requires Node 14.
  • Application, Context, Request and Response now have a origin property. This defaults defaults to http://localhost. This can be overridden by setting Application.origin, or setting a CURVEBALL_ORIGIN environment variable. PUBLIC_URI also works, but it's mainly a fallback for earlier examples and recommendations.
  • Request and Context now have a absoluteUrl property. This is calculated based on the request path and the origin.
  • BC Break: Due to the new origin property, all Request and Response classes now have an extra constructor argument. This means if you ever manually constructed any of these, there's a small change you'll need to make. Typescript should point all these problems!
  • If CURVEBALL_TRUSTPROXY is set, request.ip() will trust proxies by default, and return the ip of the real client instead of the proxy.

0.18.0 (2022-04-16)

Identical release as the previous alpha.

0.18.0-alpha.0 (2022-04-09)

  • The Context interface has been removed, and the BaseContext class is renamed to Context. This is a BC break, but should only be an issue if you used the Context interface directly. BaseContext is still exported but simply aliased to Context. This alias will be removed from a future version. This change should make ite asier to use interface declaration merging to extend Context.
  • The ws dependency has been updated to version 8. There are some breaking changes in this release. The most likely you'll hit is that incoming messages are now of type Buffer instead of string. Check out the ws changelog for more details.

0.17.0 (2022-02-08)

  • listen() now starts a WebSocket service on the same port by default.
  • listenWs() is now deprecated, and will be removed in a future version.
  • JSON is now pretty-printed by default.

Happy birthday Mom!

0.16.4 (2022-01-05)

  • Update all dependencies and ensure compatibility with latest Typescript changes.

0.16.3 (2021-05-06)

  • Updated lint rules
  • Make file update
  • Updated dependencies

0.16.2 (2021-02-18)

  • Releasing on Github packages.

0.16.1 (2021-02-01)

  • Request.body is no longer optional, which will help with typing. It can still be explicitly set to null.

0.16.0 (2021-01-30)

  • This release is identical to the last

0.16.0-beta.0 (2021-01-10)

  • BC Break: Request.body is now typed as unknown instead of any. This forces users to either validate the body, or cast to any.
  • It's now possible to write directly to response streams by setting response.body to a callback.

0.15.0 (2020-12-05)

  • Curveball now required Node 12.
  • esModuleInterop flag is no longer required to use curveball.

0.14.3 (2020-09-23)

  • 155 - listen and listenWs now both have a second host argument to bind

    to a specific interface. (@Nicholaiii)
  • 145 - request.headers and response.head function to get a list of header

    values for a given header name. (@Nicholaiii)ersnow have agetMany()`

0.14.2 (2020-07-14)

  • Republish of 1.14.1, which was missing some changes.

0.14.1 (2020-07-14)

  • types ws package is now non-devDependency

0.14.0 (2020-07-13)

  • Native Websocket support. If enabled, ctx will now have a webSocket property.

0.13.0 (2020-06-16)

  • Removed Request and Response interfaces again. They actually made it more difficult to extend.

0.12.0 (2020-03-22)

  • Both Request and Response are now typescript interfaces. This will allow plugins to extends them via interface declaration merging.
  • Everything is now compiled with the typescript 'strict' mode, which caused some internal refactoring.

0.11.2 (2020-03-09)

  • Added utilities to check If-Match, If-None-Match, If-Modified-Since, If-Unmodified-Since.
  • Typescript target is now es2019 instead of esnext to ensure that older Node.js versions are supported.
  • Added a workaround to make sure the package works around a bug in @types/node@13.

0.11.1 (2020-03-03)

  • Set Content-Type to text/plain for errors that fall without being caught by an exception handling middleware.

0.11.0 (2020-02-26)

  • Context is no longer a class, it's an interface. It's default implementation is now BaseContext. This allows plugins to modify the interface and add new features.

0.10.0 (2020-01-05)

  • Added a redirect() function to Context and Response objects, making it easier to set a status-code and location header in one step.
  • Support for more Prefer parameters: depth-noroot, safe, transclude.

0.9.4 (2019-12-21)

  • Fix a bug in HTTP/2 Push. Resources with query parameters in their path were not pushed correctly.

0.9.3 (2019-12-19)

  • The is() function will now also match wildcards, such as image/*.

0.9.2 (2019-11-04)

  • rawBody() had an incorrect type. It should always return a Buffer if the first argument is omitted.

0.9.1 (2019-09-19)

  • The server now sets a application/hal+json content-type if nothing else was set. This fixes a regression from 0.9.0.

0.9.0 (2019-09-13)

  • Request and Response types are now abstract classes instead of interfaces. This removes a bunch of duplication.
  • Request objects now have a prefer() method for quickly reading out the values from the RFC7240 Prefer header.

0.8.6 (2019-03-30)

  • Correctly set status for HTTP exceptions in sub-requests.
  • Fixed a regression from 0.8.3.

0.8.3 (2019-03-29)

  • Correctly set status for HTTP exceptions in sub-requests.

0.8.2 (2019-03-29)

  • Subrequests should behave as regular requests and catch any exceptions.
  • Updated all dependecies.
  • Stricter typescript rules.

0.8.1 (2018-11-01)

  • Now exporting an invokeMiddleware function that can be used to chain and call mutltiple middlewares.
  • Application will now by default throw a NotFound exception if nothing handled a HTTP request.

0.8.0 (2018-10-12)

  • It's now possible to pass objects as Middlewares. If an object has a member thats the middlewareCall symbol, it will call that instead.
  • The package now exports a invokeMiddleware function, which is a convenience method to call many middlewares.
  • 70: It's possible to set Response.body to a stream.Readable object.

  • 91: Bugfix: The accept() function ignored changes made my middlewares to

    Accept header.

0.7.0 (2018-10-04)

  • The Context object now has an ip method that can be used to get the ip address of the client that's connecting.
  • The Request and Response objects now have an is() method that can be used to easily check the Content-Type header of the object. For example Request.is('json') will return true for application/hal+json.
  • The Headers object now has a has() method.

0.6.0 (2018-09-05)

  • Request and Response object are now generic. Response<T> implies the body property has type T.
  • ctx.status is now writable.

0.5.0 (2018-08-31)

  • 74: Added method, path, status, accepts, push,

    sendInformational, and query to Context object. These properties and methods all forward to the request or response object.
  • 78: By default the Application will return with a 404 response, unless a

    middleware updates the status or a body was set.
  • Tests will now error when a node version under 8.11.2 is used. They broke before as well, but it's more explicit now about why.

0.4.3 (2018-07-09)

  • Application.buildContextFromHttp is now public.

0.4.2 (2018-07-04)

  • 71: Fixed error messages when a HTTP/2 client disables or refuses a push

    late in the process.
  • 72: Refactored node-specific code into its own directory.

0.4.1 (2018-07-01)

  • 57: Response.type is now settable.

0.4.0 (2018-07-01)

  • 4: Support for HTTP/2 push via the Response.push() method.

  • 62: It's now possible to do internal sub-requests without going through the

    HTTP stack, with Application.subRequest().
  • Added MemoryRequest and MemoryResponse.
  • 56: Response.body may now be null.

  • Renamed package to @curveball/core.

0.3.1 (2018-06-29)

  • Added License, Code of Conduct.
  • 52: Support for Buffer and arbitrary objects in response.body. The

    latter will automatically get converted to JSON.

0.3.0 (2018-06-26)

  • 5: Support for informational status codes such as 100 Continue and `103

    Early Hints` for both HTTP/1 and HTTP/2.
  • 28: HTTP2 support.

  • 34: Application is now the default export.

  • 47: Application.callback now returns a callback instead of implementing

    it. This makes it a bit easier to deal with this scope and is also consistent with Koa.
  • 48: Added a setter for Response.status().

  • Now exporting the Middleware type.

0.2.0 (2018-06-25)

  • 19: Added Request.rawBody() method.

  • 33: Added Request.accept() method.

  • 35: Added Request.type and Response.type.

  • 36: Added Request.query.

  • 37: Response.body now has type any.

  • 38: Added Context.state.

  • 39: Added Application.callback.

0.1.2 (2018-06-24)

  • Set script and types correctly in package.json.

0.1.1 (2018-06-24)

  • Fixed npm package distribution. Was shipping the wrong files.

0.1.0 (2018-06-24)

  • Created Request, Response, Application, Context, Headers classes.
  • Basic framework works

0.0.1 (2018-06-23)

  • First published on npm.js to claim package name.