High-level declarative HTTP library for Effect-TS built on top of @effect/platform.
- ⭐ Client derivation. Write the api specification once, get the type-safe client with runtime validation for free.
- 🌈 OpenAPI derivation.
/docs
endpoint with OpenAPI UI out of box. - 🔋 Batteries included server implementation. Automatic runtime request and response validation.
- 🔮 Example server derivation. Automatic derivation of example server implementation.
- 🐛 Mock client derivation. Test safely against a specified API.
Under development. Please note that currently any release might introduce breaking changes and the internals and the public API are still evolving and changing.
Note
This is an unofficial community package. You might benefit from checking the @effect/platform
and @effect/rpc
packages as they are the official Effect packages. The effect-http
package strongly
relies on @effect/platform
, and knowledge of it can be beneficial for understanding what
the effect-http
does under the hood.
- Quickstart
- Request validation
- Headers
- Security
- Responses
- Testing the server
- Error handling
- Grouping endpoints
- Descriptions in OpenApi
- Representations
- API on the client side
- Compatibility
Install
effect-http
- platform-agnostic, this one is enough if you intend to use it in browser onlyeffect-http-node
- if you're planning to run a HTTP server on a node
pnpm add effect-http effect-http-node
Note that effect
, @effect/platform
and @effect/platform-node
are requested as peer dependencies.
You very probably have them already. If not, install them using
pnpm add effect @effect/platform @effect/platform-node
The @effect/platform-node
is needed only for the node version.
Bootstrap a simple API specification.
import { Schema } from "@effect/schema";
import { Api } from "effect-http";
const User = Schema.struct({
name: Schema.string,
id: pipe(Schema.number, Schema.int(), Schema.positive()),
});
const GetUserQuery = Schema.struct({ id: Schema.NumberFromString });
const api = Api.api({ title: "Users API" }).pipe(
Api.get("getUser", "/user", {
response: User,
request: { query: GetUserQuery },
}),
);
Create the app implementation.
import { Effect, pipe } from "effect";
import { RouterBuilder } from "effect-http";
const app = pipe(
RouterBuilder.make(api),
RouterBuilder.handle("getUser", ({ query }) =>
Effect.succeed({ name: "milan", id: query.id }),
),
RouterBuilder.build,
);
Now, we can generate an object providing the HTTP client interface using Client.make
.
import { Client } from "effect-http";
const client = Client.make(api, { baseUrl: "http://localhost:3000" });
Spawn the server on port 3000,
import { NodeRuntime } from "@effect/platform-node"
import { NodeServer } from "effect-http-node";
app.pipe(NodeServer.listen({ port: 3000 }), NodeRuntime.runMain);
and call it using the client
.
const response = pipe(
client.getUser({ query: { id: 12 } }),
Effect.flatMap((user) => Effect.log(`Got ${user.name}, nice!`)),
Effect.scoped,
);
Also, check the auto-generated OpenAPI UI running on localhost:3000/docs. How awesome is that!
Each endpoint can declare expectations on the request format. Specifically,
body
- request bodyquery
- query parametersparams
- path parametersheaders
- request headers
They are specified in the input schemas object (3rd argument of Api.get
, Api.post
, ...).
import { Schema } from "@effect/schema";
import { Api } from "effect-http";
const Stuff = Schema.struct({ value: Schema.number });
const StuffBody = Schema.struct({ field: Schema.array(Schema.string) });
const StuffQuery = Schema.struct({ value: Schema.string });
const StuffParams = Schema.struct({ param: Schema.string });
export const api = Api.api({ title: "My api" }).pipe(
Api.post("stuff", "/stuff/:param", {
response: Stuff,
request: {
body: StuffBody,
query: StuffQuery,
params: StuffParams,
},
}),
);
(This is a complete standalone code example)
Optional parameter is denoted using a question mark in the path
match pattern. In the request param schema, use Schema.optional(<schema>)
.
In the following example the last :another
path parameter can be
ommited on the client side.
import { Schema } from "@effect/schema";
import { pipe } from "effect";
import { Api } from "effect-http";
const Stuff = Schema.struct({ value: Schema.number });
const StuffParams = Schema.struct({
param: Schema.string,
another: Schema.optional(Schema.string),
});
export const api = pipe(
Api.api({ title: "My api" }),
Api.get("stuff", "/stuff/:param/:another?", {
response: Stuff,
request: {
params: StuffParams,
},
}),
);
Request headers are part of input schemas along with the request body or query parameters.
Their schema is specified similarly to query parameters and path parameters, i.e. using
a mapping from header names onto their schemas. The example below shows an API with
a single endpoint /hello
which expects a header X-Client-Id
to be present.
import { NodeRuntime } from "@effect/platform-node";
import { Schema } from "@effect/schema";
import { pipe } from "effect";
import { Api, ExampleServer, RouterBuilder } from "effect-http";
import { NodeServer } from "effect-http-node";
const api = Api.api().pipe(
Api.get("hello", "/hello", {
response: Schema.string,
request: {
headers: Schema.struct({ "X-Client-Id": Schema.string }),
},
}),
);
pipe(
ExampleServer.make(api),
RouterBuilder.build,
NodeServer.listen({ port: 3000 }),
NodeRuntime.runMain,
);
(This is a complete standalone code example)
Server implementation deals with the validation the usual way. For example, if we try to call the endpoint without the header we will get the following error response.
{
"error": "Request validation error",
"location": "headers",
"message": "x-client-id is missing"
}
And as usual, the information about headers will be reflected in the generated OpenAPI UI.
Important note. You might have noticed the details
field of the error response
describes the missing header using lower-case. This is not an error but rather a
consequence of the fact that HTTP headers are case-insensitive.
Don't worry, this is also encoded into the type information and if you were to implement the handler, both autocompletion and type-checking would hint the lower-cased form of the header name. Take a look at examples/headers.ts to see a complete example API implementation with in-memory rate-limiting and client identification using headers.
To define which security mechanisms should be used for a specific endpoint, fill option.security
field in endpoint constructor.
const api = Api.api().pipe(
Api.post("security", "/testSecurity", { response: Schema.string }, {
description: '',
security: {
myAwesomeBearerAuth: { // myAwesomeBearerAuth - arbitrary name for the security scheme
type: "http",
options: {
scheme: "bearer",
bearerFormat: "JWT"
},
// Schema<any, string> for decoding-encoding the significant part
// "Authorization: Bearer <significant part>"
schema: Schema.Secret
}
}
}));
Currently, only the "http"
type is supported. bearer
and basic
constructors placed at the SecurityScheme
module.
Encoded security token placed in the second parameter of the handler.
const app = pipe(
RouterBuilder.make(api),
RouterBuilder.handle("security", ({}, security) => {
security.myAwesomeBearerAuth.token; // Secret
}),
RouterBuilder.build
)
In case several security schemes are specified, tokens will be Either<ParseError, MyType>
type
with the guarantee that at least one of them is of the Right<MyType>
type
const app = pipe(
RouterBuilder.make(api),
RouterBuilder.handle("security", ({}, security) => {
security.myAwesomeBearerAuth.token; // Either<ParseError, Secret>
security.myAwesomeBasicAuth.token; // Either<ParseError, Secret>
}),
RouterBuilder.build
)
On the client side security token must be passed into the appropriate security scheme
client.security({}, {
myAwesomeBearerAuth: bearerToken, // Secret
})
Response can be specified using a Schema.Schema<A, I>
which automatically
returns status code 200 and includes only default headers.
If you want a response with custom headers and status code, use the full response schema instead. The following example will enforce (both for types and runtime) that returned status, content and headers conform the specified response.
const api = Api.api().pipe(
Api.post("hello", "/hello", {
response: {
status: 200,
content: Schema.number,
headers: Schema.struct({ "My-Header": Schema.string }),
},
}),
);
It is also possible to specify multiple full response schemas.
const api = Api.api().pipe(
Api.post("hello", "/hello", {
response: [
{
status: 201,
content: Schema.number,
},
{
status: 200,
content: Schema.number,
headers: Schema.struct({ "My-Header": Schema.string }),
},
{
status: 204,
headers: Schema.struct({ "X-Another": Schema.NumberFromString }),
},
],
}),
);
The server implemention is type-checked against the api responses and one of the specified response objects must be returned.
Note: the status
needs to be as const
because without it Typescript
will infere the number
type.
const app = RouterBuilder.make(api).pipe(
RouterBuilder.handle("hello", () =>
Effect.succeed({
headers: { "my-header": 12 },
content: 12,
status: 200 as const,
}),
),
RouterBuilder.build,
);
You need to install effect-http-node
.
While most of your tests should focus on the functionality independent
of HTTP exposure, it can be beneficial to perform integration or
contract tests for your endpoints. The NodeTesting
module offers a
NodeTesting.make
combinator that generates a testing client from
the Server. This derived testing client has a similar interface
to the one derived by Client.make
.
Now, let's write an example test for the following server.
const api = Api.api().pipe(
Api.get("hello", "/hello", {
response: Schema.string,
}),
);
const app = RouterBuilder.make(api).pipe(
RouterBuilder.handle("hello", ({ query }) =>
Effect.succeed(`${query.input + 1}`),
),
RouterBUilder.build,
);
The test might look as follows.
import { NodeTesting } from 'effect-http-node';
test("test /hello endpoint", async () => {
const response = await NodeTesting.make(app, api).pipe(
Effect.flatMap((client) => client.hello({ query: { input: 12 } })),
Effect.scoped,
Effect.runPromise,
);
expect(response).toEqual("13");
});
In comparison to the Client
we need to run our endpoint handlers
in place. Therefore, in case your server uses DI services, you need to
provide them in the test code. This contract is type safe and you'll be
notified by the type-checker if the Effect
isn't invoked with all
the required services.
Validation of query parameters, path parameters, body and even responses is handled for you out of box. By default, failed validation will be reported to clients in the response body. On the server side, you get warn logs with the same information.
On top of the automatic input and output validation, handlers can fail for variety of different reasons.
Suppose we're creating user management API. When persisting a new user, we want
to guarantee we don't attempt to persist a user with an already taken name.
If the user name check fails, the API should return 409 CONFLICT
error because the client
is attempting to trigger an operatin conflicting with the current state of the server.
For these cases, effect-http
provides error types and corresponding creational
functions we can use in the error rail of the handler effect.
- 400
ServerError.badRequest
- client make an invalid request - 401
ServerError.unauthorizedError
- invalid authentication credentials - 403
ServerError.forbiddenError
- authorization failure - 404
ServerError.notFoundError
- cannot find the requested resource - 409
ServerError.conflictError
- request conflicts with the current state of the server - 415
ServerError.unsupportedMediaTypeError
- unsupported payload format - 429
ServerError.tooManyRequestsError
- the user has sent too many requests in a given amount of time
- 500
ServerError.internalServerError
- internal server error - 501
ServerError.notImplementedError
- functionality to fulfill the request is not supported - 502
ServerError.badGatewayError
- invalid response from the upstream server - 503
ServerError.serviceunavailableError
- server is not ready to handle the request - 504
ServerError.gatewayTimeoutError
- request timeout from the upstream server
Let's see it in action and implement the mentioned user management API. The API will look as follows.
import { Schema } from "@effect/schema";
import { Context, Effect, pipe } from "effect";
import { Api, RouterBuilder, ServerError } from "effect-http";
import { NodeServer } from "effect-http-node";
const api = Api.api({ title: "Users API" }).pipe(
Api.post("storeUser", "/users", {
response: Schema.string,
request: {
body: Schema.struct({ name: Schema.string }),
},
}),
);
Now, let's implement a UserRepository
interface abstracting the interaction with
our user storage. I'm also providing a mock implementation which will always return
the user already exists. We will plug the mock user repository into our server
so we can see the failure behavior.
interface UserRepository {
userExistsByName: (name: string) => Effect.Effect<boolean>;
storeUser: (user: string) => Effect.Effect<void>;
}
const UserRepository = Context.GenericTag<UserRepository>("UserRepository");
const mockUserRepository = UserRepository.of({
userExistsByName: () => Effect.succeed(true),
storeUser: () => Effect.unit,
});
const { userExistsByName, storeUser } = Effect.serviceFunctions(UserRepository);
And finally, we have the actual App
implementation.
const app = RouterBuilder.make(api).pipe(
RouterBuilder.handle("storeUser", ({ body }) =>
pipe(
userExistsByName(body.name),
Effect.filterOrFail(
(alreadyExists) => !alreadyExists,
() => ServerError.conflictError(`User "${body.name}" already exists.`),
),
Effect.andThen(storeUser(body.name)),
Effect.map(() => `User "${body.name}" stored.`),
)),
RouterBuilder.build,
);
To run the server, we will start the server using NodeServer.listen
and provide
the mockUserRepository
service.
app.pipe(
NodeServer.listen({ port: 3000 }),
Effect.provideService(UserRepository, mockUserRepository),
NodeRuntime.runMain
);
Try to run the server and call the POST /user
.
Server
$ pnpm tsx examples/conflict-error-example.ts
22:06:00 (Fiber #0) DEBUG Static swagger UI files loaded (1.7MB)
22:06:00 (Fiber #0) INFO Listening on :::3000
22:06:01 (Fiber #8) WARN POST /users client error 409
Client (using httpie cli)
$ http localhost:3000/users name="patrik"
HTTP/1.1 409 Conflict
Connection: keep-alive
Content-Length: 68
Content-Type: application/json; charset=utf-8
Date: Sat, 15 Apr 2023 16:36:44 GMT
ETag: W/"44-T++MIpKSqscvfSu9Ed1oobwDDXo"
Keep-Alive: timeout=5
X-Powered-By: Express
User "patrik" already exists.
To create a new group of endpoints, use Api.apiGroup("group name")
. This combinator
initializes new ApiGroup
object. You can pipe it with combinators like Api.get
,
Api.post
, etc, as if were defining the Api
. Api groups can be combined into an
Api
using a Api.addGroup
combinator which merges endpoints from the group
into the api in the type-safe manner while preserving group names for each endpoint.
This enables separability of concers for big APIs and provides information for generation of tags for the OpenAPI specification.
import { NodeRuntime } from "@effect/platform-node";
import { Schema } from "@effect/schema";
import { Api, ExampleServer, RouterBuilder } from "effect-http";
import { NodeServer } from "effect-http-node";
const ExampleResponse = Schema.struct({ name: Schema.string });
const testApi = Api.apiGroup("test").pipe(
Api.get("test", "/test", { response: ExampleResponse }),
);
const userApi = Api.apiGroup("Users").pipe(
Api.get("getUser", "/user", { response: ExampleResponse }),
Api.post("storeUser", "/user", { response: ExampleResponse }),
Api.put("updateUser", "/user", { response: ExampleResponse }),
Api.delete("deleteUser", "/user", { response: ExampleResponse }),
);
const categoriesApi = Api.apiGroup("Categories").pipe(
Api.get("getCategory", "/category", { response: ExampleResponse }),
Api.post("storeCategory", "/category", { response: ExampleResponse }),
Api.put("updateCategory", "/category", { response: ExampleResponse }),
Api.delete("deleteCategory", "/category", { response: ExampleResponse }),
);
const api = Api.api().pipe(
Api.addGroup(testApi),
Api.addGroup(userApi),
Api.addGroup(categoriesApi),
);
ExampleServer.make(api).pipe(
RouterBuilder.build,
NodeServer.listen({ port: 3000 }),
NodeRuntime.runMain,
);
(This is a complete standalone code example)
The OpenAPI UI will group endpoints according to the api
and show
corresponding titles for each group.
The schema-openapi library which is
used for OpenApi derivation from the Schema
takes into account
description
annotations and propagates them into the specification.
Some descriptions are provided from the built-in @effect/schema/Schema
combinators.
For example, the usage of Schema.Int.pipe(Schema.positive())
will result in "a positive number"
description in the OpenApi schema. One can also add custom description using the
Schema.description
combinator.
On top of types descriptions which are included in the schema
field, effect-http
also checks top-level schema descriptions and uses them for the parent object which
uses the schema. In the following example, the "User" description for the response
schema is used both as the schema description but also for the response itself. The
same holds for the id
query paremeter.
For an operation-level description, call the API endpoint method (Api.get
,
Api.post
etc) with a 4th argument and set the description
field to the
desired description.
import { NodeRuntime } from "@effect/platform-node";
import { Schema } from "@effect/schema";
import { Effect, pipe } from "effect";
import { Api, RouterBuilder } from "effect-http";
import { NodeServer } from "effect-http-node";
const User = pipe(
Schema.struct({
name: Schema.string,
id: pipe(Schema.number, Schema.int(), Schema.positive()),
}),
Schema.description("User"),
);
const GetUserQuery = Schema.struct({
id: pipe(Schema.NumberFromString, Schema.description("User id")),
});
const api = Api.api({ title: "Users API" }).pipe(
Api.get(
"getUser",
"/user",
{
response: User,
request: {
query: GetUserQuery,
},
},
{ description: "Returns a User by id" },
),
);
const app = RouterBuilder.make(api).pipe(
RouterBuilder.handle("getUser", ({ query }) =>
Effect.succeed({ name: "mike", id: query.id }),
),
RouterBuilder.build,
);
app.pipe(NodeServer.listen({ port: 3000 }), NodeRuntime.runMain);
By default, the effect-http
client and server will attempt the serialize/deserialize
messages as JSONs. This means that whenever you return something from a handler, the
internal logic will serialize it as a JSON onto a string and send the response along
with content-type: application/json
header.
This behaviour is a result of a default Representation.json.
The default representation of the content can be changed by specifying representations
field in the response
API specification.
For example, the following API specification states that the response of /test
endpoint
will be always a string represent as a plain text. Therefore, the HTTP message
will contain content-type: text/plain
header.
export const api = Api.api().pipe(
Api.get("myHandler", "/test", {
response: {
content: Schema.string,
status: 200,
representations: [Representation.plainText],
},
}),
);
The representations
is a list and if it contains multiple possible represetations
of the data it internal server logic will respect incomming Accept
header to decide
which representation to use.
The following example uses plainText
and json
representations. The order of
representations is respected by the logic that decides which representation should
be used, and if there is no representation matching the incomming Accept
media type,
it will choose the first representation in the list.
import { NodeRuntime } from "@effect/platform-node";
import { Schema } from "@effect/schema";
import { Effect } from "effect";
import { Api, Representation, RouterBuilder } from "effect-http";
import { NodeServer } from "effect-http-node";
import { PrettyLogger } from "effect-log";
export const api = Api.api({ title: "Example API" }).pipe(
Api.get("root", "/", {
response: {
content: Schema.unknown,
status: 200,
representations: [Representation.plainText, Representation.json],
},
}),
);
export const app = RouterBuilder.make(api).pipe(
RouterBuilder.handle("root", () =>
Effect.succeed({ content: { hello: "world" }, status: 200 as const }),
),
RouterBuilder.build,
);
const program = app.pipe(
NodeServer.listen({ port: 3000 }),
Effect.provide(PrettyLogger.layer()),
);
NodeRuntime.runMain(program);
Try running the server above and call the root path with different
Accept
headers. You should see the response content-type reflecting
the incomming Accept
header.
# JSON
curl localhost:3000/ -H 'accept: application/json' -v
# Plain text
curl localhost:3000/ -H 'accept: text/plain' -v
While effect-http
is intended to be primarly used on the server-side, i.e.
by developers providing the HTTP service, it is possible to use it also to
model, use and test against someone else's API. Out of box, you can make
us of the following combinators.
Client
- client for the real integration with the API.MockClient
- client for testing against the API interface.ExampleServer
- server implementation derivation with example responses.
effect-http
has the ability to generate an example server
implementation based on the Api
specification. This can be
helpful in the following and probably many more cases.
- You're in a process of designing an API and you want to have something to share with other people and have a discussion over before the actual implementation starts.
- You develop a fullstack application with frontend first approach you want to test the integration with a backend you haven't implemeted yet.
- You integrate a 3rd party HTTP API and you want to have an ability to perform integration tests without the need to connect to a real running HTTP service.
Use ExampleServer.make
combinator to generate a RouterBuilder
from an Api
.
import { NodeRuntime } from "@effect/platform-node";
import { Schema } from "@effect/schema";
import { Effect, pipe } from "effect";
import { Api, ExampleServer, RouterBuilder } from "effect-http";
import { NodeServer } from "effect-http-node";
const MyResponse = Schema.struct({
name: Schema.string,
value: Schema.number,
});
const api = Api.api().pipe(Api.get("test", "/test", { response: MyResponse }));
pipe(
ExampleServer.make(api),
RouterBuilder.build,
NodeServer.listen({ port: 3000 }),
NodeRuntime.runMain,
);
(This is a complete standalone code example)
Go to localhost:3000/docs and try calling
endpoints. The exposed HTTP service conforms the api
and will return
only valid example responses.
To performed quick tests against the API interface, effect-http
has
the ability to generate a mock client which will return example or
specified responses. Suppose we are integrating a hypothetical API
with /get-value
endpoint returning a number. We can model such
API as follows.
import { Schema } from "@effect/schema";
import { pipe } from "effect";
import { Api } from "effect-http";
const api = Api.api().pipe(
Api.get("getValue", "/get-value", { response: Schema.number }),
);
In a real environment, we will probably use the derived client
using MockClient.make
. But for tests, we probably want a dummy
client which will return values conforming the API. For such
a use-case, we can derive a mock client.
const client = MockClient.make(api);
Calling getValue
on the client will perform the same client-side
validation as would be done by the real client. But it will return
an example response instead of calling the API. It is also possible
to enforce the value to be returned in a type-safe manner
using the option argument. The following client will always
return number 12
when calling the getValue
operation.
const client = MockClient.make(api, { responses: { getValue: 12 } });
This library is tested against nodejs 20.9.0.