$ npm install @feathersjs/express --save
The @feathersjs/express
module contains Express framework integrations for Feathers:
- The Express framework bindings to make a Feathers application Express compatible
- An Express based transport to expose services through a REST API
- An Express error handler for Feathers errors
const express = require('@feathersjs/express');
Very Important: This chapter assumes that you are familiar with Express and describes how to set up an Express server and REST API. See the REST client chapter how to use this server on the client.
express(app) -> app
is a function that turns a Feathers application into a fully Express (4+) compatible application that additionally to Feathers functionality also lets you use the Express API.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
// Create an app that is a Feathers AND Express application
const app = express(feathers());
Note that @feathersjs/express
(express
) also exposes the standard Express middleware:
express.json
- A JSON body parserexpress.urlencoded
- A URL encoded form body parserexpress.static
- To statically host files in a folderexpress.Router
- Creates an Express router object
If no Feathers application is passed, express() -> app
returns a plain Express application just like a normal call to Express would.
app.use(path, service|mw|[mw]) -> app
registers either a service object, an Express middleware or an array of Express middleware on the given path. If a service object is passed it will use Feathers registration mechanism, for a middleware function Express.
// Register a service
app.use('/todos', {
async get(id) {
return { id };
}
});
// Register an Express middleware
app.use('/test', (req, res) => {
res.json({
message: 'Hello world from Express middleware'
});
});
// Register multiple Express middleware functions
app.use('/test', (req, res, next) => {
res.data = 'Step 1 worked';
next();
}, (req, res) => {
res.json({
message: 'Hello world from Express middleware ' + res.data
});
});
app.listen(port) -> HttpServer
will first call Express app.listen and then internally also call the Feathers app.setup(server).
// Listen on port 3030
const server = app.listen(3030);
server.on('listening', () => console.log('Feathers application started'));
app.setup(server) -> app
is usually called internally by app.listen
but in the cases described below needs to be called explicitly.
When registering an application as a sub-app, app.setup(server)
has to be called to initialize the sub-apps services.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const api = express(feathers())
.configure(express.rest())
.use('/service', myService);
const mainApp = express().use('/api/v1', api);
const server = mainApp.listen(3030);
// Now call setup on the Feathers app with the server
api.setup(server);
ProTip: We recommend avoiding complex sub-app setups because websockets and Feathers built in authentication are not fully sub-app aware at the moment.
HTTPS requires creating a separate server in which case app.setup(server)
also has to be called explicitly. In a generated application src/index.js
should look like this:
const https = require('https');
const logger = require('./logger');
const app = require('./app');
const port = app.get('port');
const server = https.createServer({
key: fs.readFileSync('privatekey.pem'),
cert: fs.readFileSync('certificate.pem')
}, app).listen(443);
// Call app.setup to initialize all services and SocketIO
app.setup(server);
process.on('unhandledRejection', (reason, p) =>
logger.error('Unhandled Rejection at: Promise ', p, reason)
);
server.on('listening', () =>
logger.info('Feathers application started on http://%s:%d', app.get('host'), port)
);
The vhost Express middleware can be used to run a Feathers application on a virtual host but again requires app.setup(server)
to be called explicitly.
const vhost = require('vhost');
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const app = express(feathers());
app.use('/todos', todoService);
const host = express().use(vhost('foo.com', app));
const server = host.listen(8080);
// Here we need to call app.setup because .listen on our virtual hosted
// app is never called
app.setup(server);
express.rest
registers a Feathers transport mechanism that allows you to expose and consume services through a RESTful API. This means that you can call a service method through the GET
, POST
, PUT
, PATCH
and DELETE
HTTP methods:
Service method | HTTP method | Path |
---|---|---|
.find() | GET | /messages |
.get() | GET | /messages/1 |
.create() | POST | /messages |
.update() | PUT | /messages/1 |
.patch() | PATCH | /messages/1 |
.remove() | DELETE | /messages/1 |
To expose services through a RESTful API we will have to configure express.rest
and provide our own body parser middleware (usually the standard Express 4 body-parser) to make REST .create
, .update
and .patch
calls parse the data in the HTTP body. If you would like to add other middleware before the REST handler, call app.use(middleware)
before registering any services.
Important: The body-parser middleware has to be registered before any service. Otherwise the service method will throw a
No data provided
orFirst parameter for 'create' must be an object
error.
Configures the transport provider with a standard formatter sending JSON response via res.json.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
// Create an Express compatible Feathers application
const app = express(feathers());
// Turn on JSON parser for REST services
app.use(express.json())
// Turn on URL-encoded parser for REST services
app.use(express.urlencoded({ extended: true }));
// Set up REST transport
app.configure(express.rest())
The default REST response formatter is a middleware that formats the data retrieved by the service as JSON. If you would like to configure your own formatter
middleware pass a formatter(req, res)
function. This middleware will have access to res.data
which is the data returned by the service. res.format can be used for content negotiation.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const app = express(feathers());
// Turn on JSON parser for REST services
app.use(express.json())
// Turn on URL-encoded parser for REST services
app.use(express.urlencoded({ extended: true }));
// Set up REST transport
app.configure(express.rest(function(req, res) {
// Format the message as text/plain
res.format({
'text/plain': function() {
res.end(`The Message is: "${res.data.text}"`);
}
});
}))
Custom Express middleware that only should run before or after a specific service can be passed to app.use
in the order it should run:
const todoService = {
async get(id) {
return {
id,
description: `You have to do ${id}!`
};
}
};
app.use('/todos', logRequest, todoService, updateData);
Important: Custom middleware will only run for REST requests and not when used with other transports like Socket.io or Primus.
Middleware that runs after the service has the service call information available as
res.data
- The data that will be sentres.hook
- The hook context of the service method call
For example updateData
could look like this:
function updateData(req, res, next) {
res.data.updateData = true;
next();
}
If you run res.send
in a custom middleware after the service and don't call next
, other middleware (like the REST formatter) will be skipped. This can be used to e.g. render different views for certain service method calls, for example to export a file as CSV:
const json2csv = require('json2csv');
const fields = [ 'done', 'description' ];
app.use('/todos', todoService, function(req, res) {
const result = res.data;
const data = result.data; // will be either `result` as an array or `data` if it is paginated
const csv = json2csv({ data, fields });
res.type('csv');
res.end(csv);
});
All middleware registered after the REST transport will have access to the req.feathers
object to set properties on the service method params
:
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const bodyParser = require('body-parser');
const app = express(feathers());
app.configure(express.rest())
.use(bodyParser.json())
.use(bodyParser.urlencoded({extended: true}))
.use(function(req, res, next) {
req.feathers.fromMiddleware = 'Hello world';
next();
});
app.use('/todos', {
async get(id, params) {
console.log(params.provider); // -> 'rest'
console.log(params.fromMiddleware); // -> 'Hello world'
return {
id, params,
description: `You have to do ${id}!`
};
}
});
app.listen(3030);
You can see the parameters set by running the example and visiting http://localhost:3030/todos/test
.
Avoid setting req.feathers = something
directly since it may already contain information that other Feathers plugins rely on. Adding individual properties or using Object.assign(req.feathers, something)
is the more reliable option.
Very important: Since the order of Express middleware matters, any middleware that sets service parameters has to be registered before your services (in a generated application before
app.configure(services)
or inmiddleware/index.js
).
ProTip: Although it may be convenient to set
req.feathers.req = req;
to have access to the request object in the service, we recommend keeping your services as provider independent as possible. There usually is a way to pre-process your data in a middleware so that the service does not need to know about the HTTP request or response.
params.query
will contain the URL query parameters sent from the client. For the REST transport the query string is parsed using the qs module. For some query string examples see the database querying chapter.
Important: Only
params.query
is passed between the server and the client, other parts ofparams
are not. This is for security reasons so that a client can't set things likeparams.user
or the database options. You can always map fromparams.query
to otherparams
properties in a before hook.
For example:
GET /messages?read=true&$sort[createdAt]=-1
Will set params.query
to
{
"read": "true",
"$sort": { "createdAt": "-1" }
}
ProTip: Since the URL is just a string, there will be no type conversion. This can be done manually in a hook or with the query-types Express middleware to convert Boolean and Numeric types.
Note: If an array in your request consists of more than 20 items, the qs parser implicitly converts it to an object with indices as keys. To extend this limit, you can set a custom query parser:
app.set('query parser', str => qs.parse(str, {arrayLimit: 1000}))
For any service method call made through REST params.provider
will be set to rest
. In a hook this can for example be used to prevent external users from making a service method call:
app.service('users').hooks({
before: {
remove(context) {
// check for if(context.params.provider) to prevent any external call
if(context.params.provider === 'rest') {
throw new Error('You can not delete a user via REST');
}
}
}
});
See the routing section.
express.notFound()
returns middleware that returns a NotFound
(404) Feathers error. It should be used as the last middleware before the error handler. The following options are available:
verbose
: Set totrue
if the URL should be included in the error message (default:false
)
// Return errors that include the URL
app.use(express.notFound({ verbose: true });
app.use(errorHandler());
express.errorHandler
is an Express error handler middleware that formats any error response to a REST call as JSON (or HTML if e.g. someone hits our API directly in the browser) and sets the appropriate error code.
ProTip: You can still use any other Express compatible error middleware with Feathers. In fact, the
express.errors
is just a slightly customized one. Very Important: Just as in Express, the error handler has to be registered after all middleware and services.
Set up the error handler with the default configuration.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const app = express(feathers());
// before starting the app
app.use(express.errorHandler())
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const app = express(feathers());
// Just like Express your error middleware needs to be
// set up last in your middleware chain.
app.use(express.errorHandler({
html: function(error, req, res, next) {
// render your error view with the error object
res.render('error', error);
}
}));
app.use(errorHandler({
html: {
404: 'path/to/notFound.html',
500: 'there/will/be/robots.html'
}
}));
ProTip: If you want to have the response in json format be sure to set the
Accept
header in your request toapplication/json
otherwise the default error handler will return HTML.
The following options can be passed when creating a new localstorage service:
html
(Function|Object) [optional] - A custom formatter function or an object that contains the path to your custom html error pages.logger
(Function|false) (default:console
) - Set a logger object to log the error (it will be logger withlogger.error(error)
ProTip:
html
can also be set tofalse
to disable html error pages altogether so that only JSON is returned.
express.authenticate(...strategies)
allows to protect an Express middleware with an authentication service that has strategies registered that can parse HTTP headers. It will set the authentication information on the req
object (e.g. req.user
). The following example protects the /hello
endpoint with the JWT strategy (so the Authorization: Bearer <JWT>
header needs to be set) and uses the user email to render the message:
const { authenticate } = require('@feathersjs/express');
app.use('/hello', authenticate('jwt'), (req, res) => {
const { user } = req;
res.render(`Hello ${user.email}`);
});
// When using with the non-default authentication service
app.use('/hello', authenticate({
service: 'v2/auth',
strategies: [ 'jwt', 'api-key' ]
}), (req, res) => {
const { user } = req;
res.render(`Hello ${user.email}`);
});
Express route placeholders in a service URL will be added to the services params.route
.
Important: See the FAQ entry on nested routes for more details on when and when not to use nested routes.
const feathers = require('@feathersjs/feathers');
const express = require('@feathersjs/express');
const app = express(feathers());
app.configure(express.rest())
.use(function(req, res, next) {
req.feathers.fromMiddleware = 'Hello world';
next();
});
app.use('/users/:userId/messages', {
async get(id, params) {
console.log(params.query); // -> ?query
console.log(params.provider); // -> 'rest'
console.log(params.fromMiddleware); // -> 'Hello world'
console.log(params.route.userId); // will be `1` for GET /users/1/messages
return {
id,
params,
read: false,
text: `Feathers is great!`,
createdAt: new Date().getTime()
};
}
});
app.listen(3030);