- Server
new Server([options])- Server properties
server.after(method, [dependencies])server.auth.default(options)server.auth.scheme(name, scheme)server.auth.strategy(name, scheme, [mode], [options])server.auth.test(strategy, request, next)server.bind(context)server.cache(options)server.connection([options])server.decorate(type, property, method)server.dependency(dependencies, [after])server.expose(key, value)server.expose(obj)server.ext(event, method, [options])server.handler(name, method)server.inject(options, callback)server.log(tags, [data, [timestamp]])server.lookup(id)server.match(method, path, [host])server.method(name, method, [options])server.method(methods)server.path(relativeTo)server.register(plugins, [options], callback)server.render(template, context, [options], callback)server.route(options)server.select(labels)server.start([callback])server.state(name, [options])server.stop([options], [callback])server.table([host])server.views(options)- Server events
- Plugins
- Requests
- Reply interface
The Server object is the main application container. The server manages all incoming connections
along with all the facilities provided by the framework. A server can contain more than one
connection (e.g. listen to port 80 and 8080).
Creates a new Server object where:
options- optional configuration:-
app- application-specific configuration which can later be accessed viaserver.settings.app. Note the difference betweenserver.settings.appwhich is used to store static configuration values andserver.appwhich is meant for storing run-time state. Defaults to{}. -
cache- sets up server-side caching. Every server includes a default cache for storing application state. By default, a simple memory-based cache is created which has limited capacity and capabilities. hapi uses catbox for its cache which includes support for common storage solutions (e.g. Redis, MongoDB, Memcached, and Riak). Caching is only utilized if methods and plugins explicitly store their state in the cache. The server cache configuration only defines the storage container itself.cachecan be assigned:- a prototype function (usually obtained by calling
require()on a catbox strategy such asrequire('catbox-redis')). - a configuration object with the following options:
engine- a prototype function or catbox engine object.name- an identifier used later when provisioning or configuring caching for server methods or plugins. Each cache name must be unique. A single item may omit thenameoption which defines the default cache. If every cache includes aname, a default memory cache is provisions as well.shared- iftrue, allows multiple cache uses to share the same segment (e.g. multiple methods using the same cache storage container). Default tofalse.- other options passed to the catbox strategy used.
- an array of the above object for configuring multiple cache instances, each with a unique
name. When an array of objects is provided, multiple cache connections are established
and each array item (except one) must include a
name.
- a prototype function (usually obtained by calling
-
connections- sets the default connections configuration which can be overridden by each connection where:-
app- application-specific connection configuration which can be accessed viaconnection.settings.app. Provides a safe place to store application configuration without potential conflicts with the framework internals. Should not be used to configure plugins which should useplugins[name]. Note the difference betweenconnection.settings.appwhich is used to store configuration values andconnection.appwhich is meant for storing run-time state. -
load- connection load limits configuration where:maxHeapUsedBytes- maximum V8 heap size over which incoming requests are rejected with an HTTP Server Timeout (503) response. Defaults to0(no limit).maxRssBytes- maximum process RSS size over which incoming requests are rejected with an HTTP Server Timeout (503) response. Defaults to0(no limit).maxEventLoopDelay- maximum event loop delay duration in milliseconds over which incoming requests are rejected with an HTTP Server Timeout (503) response. Defaults to0(no limit).
-
plugins- plugin-specific configuration which can later be accessed viaconnection.settings.plugins. Provides a place to store and pass connection-specific plugin configuration.pluginsis an object where each key is a plugin name and the value is the configuration. Note the difference betweenconnection.settings.pluginswhich is used to store configuration values andconnection.pluginswhich is meant for storing run-time state. -
router- controls how incoming request URIs are matched against the routing table:isCaseSensitive- determines whether the paths '/example' and '/EXAMPLE' are considered different resources. Defaults totrue.stripTrailingSlash- removes trailing slashes on incoming paths. Defaults totrue.
-
routes- sets the default configuration for every route a route public interface object. -
state- sets the default configuration for every state (cookie) set explicitly viaserver.state()or implicitly (without definition) using the state configuration object.
-
-
debug- determines which logged events are sent to the console (this should only be used for development and does not affect which events are actually logged internally and recorded). Set tofalseto disable all console logging, or to an object with:log- a string array of server log tags to be displayed viaconsole.error()when the events are logged viaserver.log()as well as internally generated server logs. For example, to display all errors, set the option to['error']. To turn off all console debug messages set it tofalse. Defaults to uncaught errors thrown in external code (these errors are handled automatically and result in an Internal Server Error response) or runtime errors due to developer error.request- a string array of request log tags to be displayed viaconsole.error()when the events are logged viarequest.log()as well as internally generated request logs. For example, to display all errors, set the option to['error']. To turn off all console debug messages set it tofalse. Defaults to uncaught errors thrown in external code (these errors are handled automatically and result in an Internal Server Error response) or runtime errors due to developer error.
-
files- file system related settings:etagsCacheMaxSize- sets the maximum number of file etag hash values stored in the etags cache. Defaults to10000.
-
load- process load monitoring where:sampleInterval- the frequency of sampling in milliseconds. Defaults to0(no sampling).
-
mime- options passed to the mimos module when generating the mime database used by the server and accessed viaserver.mime. -
minimal- iftrue, does not load the inert (file and directory support), h2o2 (proxy support), and vision (views support) plugins automatically. The plugins can be loaded manually after construction. Defaults tofalse(plugins loaded). -
plugins- plugin-specific configuration which can later be accessed viaserver.settings.plugins.pluginsis an object where each key is a plugin name and the value is the configuration. Note the difference betweenserver.settings.pluginswhich is used to store static configuration values andserver.pluginswhich is meant for storing run-time state. Defaults to{}.
-
Note that the options object is deeply cloned and cannot contain any values that are unsafe to
perform deep copy on.
var Hapi = require('hapi');
var server = new Hapi.Server({
cache: require('catbox-redis'),
load: {
sampleInterval: 1000
}
});Provides a safe place to store server-specific run-time application data without potential conflicts with the framework internals. The data can be accessed whenever the server is accessible. Initialized with am empty object.
var Hapi = require('hapi');
server = new Hapi.Server();
server.app.key = 'value';
var handler = function (request, reply) {
return reply(request.server.app.key);
};An array containing the server's connections. When the server object is returned from
server.select(), the connections array only includes the connections
matching the selection criteria.
var server = new Hapi.Server();
server.connection({ port: 80, labels: 'a' });
server.connection({ port: 8080, labels: 'b' });
// server.connections.length === 2
var a = server.select('a');
// a.connections.length === 1When the server contains exactly one connection, info is an object containing information about
the sole connection where:
id- a unique connection identifier (using the format '{hostname}:{pid}:{now base36}').created- the connection creation timestamp.started- the connection start timestamp (0when stopped).port- the connection port based on the following rules:- the configured port value before the server has been started.
- the actual port assigned when no port is configured or set to
0after the server has been started.
host- the host name the connection was configured to. Defaults to the operating system hostname when available, otherwise'localhost'.address- the active IP address the connection was bound to after starting. Set toundefineduntil the server has been started or when using a non TCP port (e.g. UNIX domain socket).protocol- the protocol used:'http'- HTTP.'https'- HTTPS.'socket'- UNIX domain socket or Windows named pipe.
uri- a string representing the connection (e.g. 'http://example.com:8080' or 'socket:/unix/domain/socket/path'). Contains theurisetting if provided, otherwise constructed from the available settings. If noportis available or set to0, theuriwill not include a port component.
When the server contains more than one connection, each server.connections
array member provides its own connection.info.
var server = new Hapi.Server();
server.connection({ port: 80 });
// server.info.port === 80
server.connection({ port: 8080 });
// server.info === null
// server.connections[1].info.port === 8080An object containing the process load metrics (when load.sampleInterval is enabled):
eventLoopDelay- event loop delay milliseconds.heapUsed- V8 heap usage.rss- RSS memory usage.
var Hapi = require('hapi');
var server = new Hapi.Server({ load: { sampleInterval: 1000 } });
console.log(server.load.rss);When the server contains exactly one connection, listener is the node HTTP server object of the
sole connection.
When the server contains more than one connection, each server.connections
array member provides its own connection.listener.
var Hapi = require('hapi');
var SocketIO = require('socket.io');
var server = new Hapi.Server();
server.connection({ port: 80 });
var io = SocketIO.listen(server.listener);
io.sockets.on('connection', function(socket) {
socket.emit({ msg: 'welcome' });
});An object providing access to the server methods where each server method name is an object property.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.method('add', function (a, b, next) {
return next(null, a + b);
});
server.methods.add(1, 2, function (err, result) {
// result === 3
});Provides access to the server MIME database used for setting content-type information. The object
must not be modified directly but only through the mime server setting.
var Hapi = require('hapi');
var options = {
mime: {
override: {
'node/module': {
source: 'steve',
compressible: false,
extensions: ['node', 'module', 'npm'],
type: 'node/module'
}
}
}
};
var server = new Hapi.Server(options);
// server.mime.path('code.js').type === 'application/javascript'
// server.mime.path('file.npm').type === 'node/module'An object containing the values exposed by each plugin registered where each key is a plugin name
and the values are the exposed properties by each plugin using
server.expose(). Plugins may set the value of the
server.plugins[name] object directly or via the server.expose() method.
exports.register = function (server, options, next) {
server.expose('key', 'value');
// server.plugins.example.key === 'value'
return next();
};
exports.register.attributes = {
name: 'example'
};The realm object contains server-wide or plugin-specific state that can be shared across various
methods. For example, when calling server.bind(), the active realm
settings.bind property is set which is then used by routes and extensions added at the same level
(server root or plugin). Realms are a limited version of a sandbox where plugins can maintain state
used by the framework when adding routes, extensions, and other properties.
modifiers- when the server object is provided as an argument to the pluginregister()method,modifiersprovides the registration preferences passed theserver.register()method and includes:route- routes preferences:prefix- the route path prefix used by any calls toserver.route()from the server.vhost- the route virtual host settings used by any calls toserver.route()from the server.
plugin- the active plugin name (empty string if at the server root).plugins- plugin-specific state to be shared only among activities sharing the same active state.pluginsis an object where each key is a plugin name and the value is the plugin state.settings- settings overrides:files.relativeTobind
The server.realm object should be considered read-only and must not be changed directly except
for the plugins property can be directly manipulated by the plugins (each setting its own under
plugins[name]).
exports.register = function (server, options, next) {
console.log(server.realm.modifiers.route.prefix);
return next();
};The root server object containing all the connections and the root server methods (e.g. start(),
stop(), connection()).
The server configuration object after defaults applied.
var Hapi = require('hapi');
var server = new Hapi.Server({
app: {
key: 'value'
}
});
// server.settings.app === { key: 'value' }The hapi module version number.
var Hapi = require('hapi');
var server = new Hapi.Server();
// server.version === '8.0.0'Adds a method to be called after all the plugin dependencies have been registered and before the server starts (only called if the server is started) where:
after- the method with signaturefunction(plugin, next)where:server- server object theafter()method was called on.next- the callback function the method must call to return control over to the application and complete the registration process. The function signature isfunction(err)where:err- internal error which is returned back via theserver.start()callback.
dependencies- a string or array of string with the plugin names to call this method after theirafter()methods. There is no requirement for the other plugins to be registered. Setting dependencies only arranges the after methods in the specified order.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.after(function () {
// Perform some pre-start logic
});
server.start(function (err) {
// After method already executed
});Sets a default strategy which is applied to every route where:
options- a string with the default strategy name or an object with a specified strategy or strategies using the same format as the routeauthhandler options.
The default does not apply when the route config specifies auth as false, or has an
authentication strategy configured. Otherwise, the route authentication config is applied to the
defaults. Note that the default only applies at time of route configuration, not at runtime.
Calling default() after adding a route will have no impact on routes added prior.
The default auth strategy configuration can be accessed via connection.auth.settings.default.
var server = new Hapi.Server();
server.connection({ port: 80 });
server.auth.scheme('custom', scheme);
server.auth.strategy('default', 'custom');
server.auth.default('default');
server.route({
method: 'GET',
path: '/',
handler: function (request, reply) {
return reply(request.auth.credentials.user);
}
});Registers an authentication scheme where:
name- the scheme name.scheme- the method implementing the scheme with signaturefunction(server, options)where:server- a reference to the server object the scheme is added to.options- optional scheme settings used to instantiate a strategy.
The scheme method must return an object with the following keys:
authenticate(request, reply)- required function called on each incoming request configured with the authentication scheme where:request- the request object.reply- the reply interface the authentication method must call when done authenticating the request where:reply(err, response, result)- is called if authentication failed where:err- any authentication error.response- any authentication response action such as redirection. Ignored iferris present, otherwise required.result- an object containing:credentials- the authenticated credentials.artifacts- optional authentication artifacts.
reply.continue(result)- is called if authentication succeeded where:result- same object asresultabove.
payload(request, reply)- optional function called to authenticate the request payload where:request- the request object.reply(err, response)- is called if authentication failed where:err- any authentication error.response- any authentication response action such as redirection. Ignored iferris present, otherwise required.
reply.continue()- is called if payload authentication succeeded.
response(request, reply)- optional function called to decorate the response with authentication headers before the response headers or payload is written where:request- the request object.reply(err, response)- is called if an error occurred where:err- any authentication error.response- any authentication response to send instead of the current response. Ignored iferris present, otherwise required.
reply.continue()- is called if the operation succeeded.
options- an optional object with the following keys:payload- iftrue, requires payload validation as part of the scheme and forbids routes from disabling payload auth validation. Defaults tofalse.
When the scheme authenticate() method implementation calls reply() with an error condition,
the specifics of the error affect whether additional authentication strategies will be attempted
if configured for the route. If the err returned by the reply() method includes a message, no
additional strategies will be attempted. If the err does not include a message but does include
a scheme name (e.g. Boom.unauthorized(null, 'Custom')), additional strategies will be attempted
in order of preference.
When the scheme payload() method returns an error with a message, it means payload validation
failed due to bad payload. If the error has no message but includes a scheme name (e.g.
Boom.unauthorized(null, 'Custom')), authentication may still be successful if the route
auth.payload configuration is set to 'optional'.
var server = new Hapi.Server();
server.connection({ port: 80 });
var scheme = function (server, options) {
return {
authenticate: function (request, reply) {
var req = request.raw.req;
var authorization = req.headers.authorization;
if (!authorization) {
return reply(Boom.unauthorized(null, 'Custom'));
}
return reply(null, { credentials: { user: 'john' } });
}
};
};
server.auth.scheme('custom', scheme);Registers an authentication strategy where:
name- the strategy name.scheme- the scheme name (must be previously registered usingserver.auth.scheme()).mode- iftrue, the scheme is automatically assigned as a required strategy to any route without anauthconfig. Can only be assigned to a single server strategy. Value must betrue(which is the same as'required') or a valid authentication mode ('required','optional','try'). Defaults tofalse.options- scheme options based on the scheme requirements.
var server = new Hapi.Server();
server.connection({ port: 80 });
server.auth.scheme('custom', scheme);
server.auth.strategy('default', 'custom');
server.route({
method: 'GET',
path: '/',
config: {
auth: 'default',
handler: function (request, reply) {
return reply(request.auth.credentials.user);
}
}
});Tests a request against an authentication strategy where:
strategy- the strategy name registered withserver.auth.strategy().request- the request object.next- the callback function with signaturefunction(err, credentials)where:err- the error if authentication failed.credentials- the authentication credentials object if authentication was successful.
Note that the test() method does not take into account the route authentication configuration. It
also does not perform payload authentication. It is limited to the basic strategy authentication
execution. It does not include verifying scope, entity, or other route properties.
var server = new Hapi.Server();
server.connection({ port: 80 });
server.auth.scheme('custom', scheme);
server.auth.strategy('default', 'custom');
server.route({
method: 'GET',
path: '/',
handler: function (request, reply) {
request.server.auth.test('default', request, function (err, credentials) {
if (err) {
return reply({ status: false });
}
return reply({ status: true, user: credentials.name });
});
}
});Sets a global context used as the default bind object when adding a route or an extension where:
context- the object used to bindthisin handler and extension methods.
When setting context inside a plugin, the context is applied only to methods set up by the plugin. Note that the context applies only to routes and extensions added after it has been set.
var handler = function (request, reply) {
return reply(this.message);
};
exports.register = function (server, options, next) {
var bind = {
message: 'hello'
};
server.bind(bind);
server.route({ method: 'GET', path: '/', handler: handler });
return next();
};Provisions a cache segment within the server cache facility where:
options- catbox policy configuration where:expiresIn- relative expiration expressed in the number of milliseconds since the item was saved in the cache. Cannot be used together withexpiresAt.expiresAt- time of day expressed in 24h notation using the 'HH:MM' format, at which point all cache records expire. Uses local time. Cannot be used together withexpiresIn.generateFunc- a function used to generate a new cache item if one is not found in the cache when callingget(). The method's signature isfunction(id, next)where: -id- theidstring or object provided to theget()method. -next- the method called when the new item is returned with the signaturefunction(err, value, ttl)where: -err- an error condition. -value- the new value generated. -ttl- the cache ttl value in milliseconds. Set to0to skip storing in the cache. Defaults to the cache global policy.staleIn- number of milliseconds to mark an item stored in cache as stale and attempt to regenerate it whengenerateFuncis provided. Must be less thanexpiresIn.staleTimeout- number of milliseconds to wait before checking if an item is stale.generateTimeout- number of milliseconds to wait before returning a timeout error when thegenerateFuncfunction takes too long to return a value. When the value is eventually returned, it is stored in the cache for future requests.cache- the cache name configured in 'server.cache`. Defaults to the default cache.segment- string segment name, used to isolate cached items within the cache partition. When called within a plugin, defaults to '!name' where 'name' is the plugin name. Required when called outside of a plugin.shared- iftrue, allows multiple cache provisions to share the same segment. Default tofalse.
var server = new Hapi.Server();
server.connection({ port: 80 });
var cache = server.cache({ segment: 'countries', expiresIn: 60 * 60 * 1000 });
cache.set('norway', { capital: 'oslo' }, function (err) {
cache.get('norway', function (err, value, cached, log) {
// value === { capital: 'oslo' };
});
});Adds an incoming server connection where:
host- the public hostname or IP address. Used only to setserver.info.hostandserver.info.uri. If not configured, defaults to the operating system hostname and if not available, to'localhost'.address- sets the host name or IP address the connection will listen on. If not configured, defaults tohostif present, otherwise to all available network interfaces (i.e.'0.0.0.0'). Set to127.0.0.1orlocalhostto restrict connection to only those coming from the same machine.port- the TCP port the connection will listen to. Defaults to an ephemeral port (0) which uses an available port when the server is started (and assigned toserver.info.port). Ifportis a string containing a '/' character, it is used as a UNIX domain socket path and if it starts with '\.\pipe' as a Windows named pipe.uri- the full public URI without the path (e.g. 'http://example.com:8080'). If present, used as the connectioninfo.uriotherwise constructed from the connection settings.listener- optional node.js HTTP (or HTTPS)http.Serverobject or any compatible object. If thelistenerneeds to be manually started, setautoListentofalse.autoListen- indicates that theconnection.listenerwill be started manually outside the framework. Cannot be specified with aportsetting. Defaults totrue.cache- caching headers configuration:statuses- an array of HTTP response status codes (e.g.200) which are allowed to include a valid caching directive. Defaults to[200].
labels- a string or string array of labels used toserver.select()specific connections matching the specified labels. Defaults to an empty array[](no labels).tls- used to create an HTTPS connection. Thetlsobject is passed unchanged as options to the node.js HTTPS server as described in the node.js HTTPS documentation.- Any connections configuration server defaults can be included to override and customize the individual connection.
Returns a server object with the new connection selected.
Note that the options object is deeply cloned (with the exception of listener which is
shallowly copied) and cannot contain any values that are unsafe to perform deep copy on.
var Hapi = require('hapi');
var server = new Hapi.Server();
var web = server.connection({ port: 8000, host: 'example.com', labels: ['web'] });
var admin = server.connection({ port: 8001, host: 'example.com', labels: ['admin'] });
// server.connections.length === 2
// web.connections.length === 1
// admin.connections.length === 1Extends various framework interfaces with custom methods where:
type- the interface being decorated. Supported types:'reply'- adds methods to the reply interface.'server'- adds methods to the Server object.
property- the object decoration key name.method- the extension function.
Note that decorations apply to the entire server and all its connections regardless of current selection.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.decorate('reply', 'success', function () {
return this.response({ status: 'ok' });
});
server.route({
method: 'GET',
path: '/',
handler: function (request, reply) {
return reply.success();
}
});Used within a plugin to declares a required dependency on other plugins where:
dependencies- a single string or array of plugin name strings which must be registered in order for this plugin to operate. Plugins listed must be registered before the server is started. Does not provide version dependency which should be implemented using npm peer dependencies.after- an optional function called after all the specified dependencies have been registered and before the server starts. The function is only called if the server is started. If a circular dependency is detected, an exception is thrown (e.g. two plugins each has anafterfunction to be called after the other). The function signature isfunction(server, next)where:server- the server thedependency()method was called on.next- the callback function the method must call to return control over to the application and complete the registration process. The function signature isfunction(err)where:err- internal error condition, which is returned back via theserver.start()callback.
exports.register = function (server, options, next) {
server.dependency('yar', after);
return next();
};
var after = function (server, next) {
// Additional plugin registration logic
return next();
};Used within a plugin to expose a property via server.plugins[name] where:
key- the key assigned (server.plugins[name][key]).value- the value assigned.
exports.register = function (server, options, next) {
server.expose('util', function () { console.log('something'); });
return next();
};Merges a deep copy of an object into to the existing content of server.plugins[name] where:
obj- the object merged into the exposed properties container.
exports.register = function (server, options, next) {
server.expose({ util: function () { console.log('something'); } });
return next();
};Registers an extension function in one of the available extension points where:
event- the event name.method- a function or an array of functions to be executed at a specified point during request processing. The required extension function signature isfunction(request, reply)where:request- the request object.reply- the reply interface which is used to return control back to the framework. To continue normal execution of the request lifecycle,reply.continue()must be called. To abort processing and return a response to the client, callreply(value)where value is an error or any other valid response.this- the object provided viaoptions.bindor the current active context set withserver.bind().
options- an optional object with the following:before- a string or array of strings of plugin names this method must execute before (on the same event). Otherwise, extension methods are executed in the order added.after- a string or array of strings of plugin names this method must execute after (on the same event). Otherwise, extension methods are executed in the order added.bind- a context object passed back to the provided method (viathis) when called.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onRequest', function (request, reply) {
// Change all requests to '/test'
request.setUrl('/test');
return reply.continue();
});
var handler = function (request, reply) {
return reply({ status: 'ok' });
};
server.route({ method: 'GET', path: '/test', handler: handler });
server.start();
// All requests will get routed to '/test'Registers a new handler type to be used in routes where:
name- string name for the handler being registered. Cannot override the built-in handler types (directory,file,proxy, andview) or any previously registered type.method- the function used to generate the route handler using the signaturefunction(route, options)where:route- the route public interface object.options- the configuration object provided in the handler config.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ host: 'localhost', port: 8000 });
// Defines new handler for routes on this server
server.handler('test', function (route, options) {
return function (request, reply) {
return reply('new handler: ' + options.msg);
}
});
server.route({
method: 'GET',
path: '/',
handler: { test: { msg: 'test' } }
});
server.start();The method function can have a defaults object or function property. If the property is set to
an object, that object is used as the default route config for routes using this handler. If the
property is set to a function, the function uses the signature function(method) and returns the
route default configuration.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ host: 'localhost', port: 8000 });
var handler = function (route, options) {
return function (request, reply) {
return reply('new handler: ' + options.msg);
}
};
// Change the default payload processing for this handler
handler.defaults = {
payload: {
output: 'stream',
parse: false
}
};
server.handler('test', handler);When the server contains exactly one connection, injects a request into the sole connection simulating an incoming HTTP request without making an actual socket connection. Injection is useful for testing purposes as well as for invoking routing logic internally without the overhead or limitations of the network stack. Utilizes the shot module for performing injections, with some additional options and response properties:
options- can be assign a string with the requested URI, or an object with:method- the request HTTP method (e.g.'POST'). Defaults to'GET'.url- the request URL. If the URI includes an authority (e.g.'example.com:8080'), it is used to automatically set an HTTP 'Host' header, unless one was specified inheaders.headers- an object with optional request headers where each key is the header name and the value is the header content. Defaults to no additions to the default Shot headers.payload- an optional string or buffer containing the request payload (object must be manually converted to a string first). Defaults to no payload. Note that payload processing defaults to'application/json'if no 'Content-Type' header provided.credentials- an optional credentials object containing authentication information. Thecredentialsare used to bypass the default authentication strategies, and are validated directly as if they were received via an authentication scheme. Defaults to no credentials.simulate- an object with options used to simulate client request stream conditions for testing:error- iftrue, emits an'error'event after payload transmission (if any). Defaults tofalse.close- iftrue, emits a'close'event after payload transmission (if any). Defaults tofalse.end- iffalse, does not end the stream. Defaults totrue.
callback- the callback function with signaturefunction(res)where:res- the response object where:statusCode- the HTTP status code.headers- an object containing the headers set.payload- the response payload string.rawPayload- the raw response payload buffer.raw- an object with the injection request and response objects:req- the request object.res- the response object.
result- the raw handler response (e.g. when not a stream) before it is serialized for transmission. If not available, set topayload. Useful for inspection and reuse of the internal objects returned (instead of parsing the response string).
When the server contains more than one connection, each server.connections
array member provides its own connection.inject().
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var handler = function (request, reply) {
return reply('Success!');
};
server.route({ method: 'GET', path: '/', handler: handler });
server.inject('/', function (res) {
console.log(res.result);
});Logs server events that cannot be associated with a specific request. When called the server emits
a 'log' event which can be used by other listeners or plugins to record the
information or output to the console. The arguments are:
tags- a string or an array of strings (e.g.['error', 'database', 'read']) used to identify the event. Tags are used instead of log levels and provide a much more expressive mechanism for describing and filtering events. Any logs generated by the server internally include the'hapi'tag along with event-specific information.data- an optional message string or object with the application data being logged.timestamp- an optional timestamp expressed in milliseconds. Defaults toDate.now()(now).
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.on('log', function (event, tags) {
if (tags.error) {
console.log(event);
}
});
server.log(['test', 'error'], 'Test event');When the server contains exactly one connection, looks up a route configuration where:
id- the route identifier as set in the route options.
returns the route public interface object if found, otherwise null.
var server = new Hapi.Server();
server.connection();
server.route({
method: 'GET',
path: '/',
config: {
handler: function (request, reply) { return reply(); },
id: 'root'
}
});
var route = server.lookup('root');When the server contains more than one connection, each server.connections
array member provides its own connection.lookup() method.
When the server contains exactly one connection, looks up a route configuration where:
method- the HTTP method (e.g. 'GET', 'POST').path- the requested path (must begin with '/').host- optional hostname (to match against routes withvhost).
returns the route public interface object if found, otherwise null.
var server = new Hapi.Server();
server.connection();
server.route({
method: 'GET',
path: '/',
config: {
handler: function (request, reply) { return reply(); },
id: 'root'
}
});
var route = server.match('get', '/');When the server contains more than one connection, each server.connections
array member provides its own connection.match() method.
Registers a server method. Server methods are functions registered with the server and used throughout the application as a common utility. Their advantage is in the ability to configure them to use the built-in cache and share across multiple request handlers without having to create a common module.
Methods are registered via server.method(name, method, [options]) where:
name- a unique method name used to invoke the method viaserver.methods[name]. When configured with caching enabled,server.methods[name].cache.drop(arg1, arg2, ..., argn, callback)can be used to clear the cache for a given key. Supports using nested names such asutils.users.getwhich will automatically create the missing path underserver.methodsand can be accessed for the previous example viaserver.methods.utils.users.get.method- the method function with the signature is one of:function(arg1, arg2, ..., argn, next)where:arg1,arg2, etc. - the method function arguments.next- the function called when the method is done with the signaturefunction(err, result, ttl)where:err- error response if the method failed.result- the return value.ttl-0if result is valid but cannot be cached. Defaults to cache policy.
function(arg1, arg2, ..., argn)where:arg1,arg2, etc. - the method function arguments.- the
callbackoption is set tofalse. - the method must returns a value (result,
Error, or a promise) or throw anError.
options- optional configuration:bind- a context object passed back to the method function (viathis) when called. Defaults to active context (set viaserver.bind()when the method is registered.cache- the same cache configuration used inserver.cache().callback- iffalse, expects themethodto be a synchronous function. Note that using a synchronous function with caching will convert the method interface to require a callback as an additional argument with the signaturefunction(err, result, cached, report)since the cache interface cannot return values synchronously. Defaults totrue.generateKey- a function used to generate a unique key (for caching) from the arguments passed to the method function (the callback argument is not passed as input). The server will automatically generate a unique key if the function's arguments are all of types'string','number', or'boolean'. However if the method uses other types of arguments, a key generation function must be provided which takes the same arguments as the function and returns a unique string (ornullif no key can be generated).
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
// Simple arguments
var add = function (a, b, next) {
return next(null, a + b);
};
server.method('sum', add, { cache: { expiresIn: 2000 } });
server.methods.sum(4, 5, function (err, result) {
console.log(result);
});
// Object argument
var addArray = function (array, next) {
var sum = 0;
array.forEach(function (item) {
sum += item;
});
return next(null, sum);
};
server.method('sumObj', addArray, {
cache: { expiresIn: 2000 },
generateKey: function (array) {
return array.join(',');
}
});
server.methods.sumObj([5, 6], function (err, result) {
console.log(result);
});
// Synchronous method with cache
var addSync = function (a, b) {
return a + b;
};
server.method('sumSync', addSync, { cache: { expiresIn: 2000 }, callback: false });
server.methods.sumSync(4, 5, function (err, result) {
console.log(result);
});Registers a server method function as described in
server.method() using a configuration object
where:
methods- an object or an array of objects where each one contains:name- the method name.method- the method function.options- optional settings.
var add = function (a, b, next) {
next(null, a + b);
};
server.method({
name: 'sum',
method: add,
options: {
cache: {
expiresIn: 2000
}
}
});Sets the path prefix used to locate static resources (files and view templates) when relative paths are used where:
relativeTo- the path prefix added to any relative file path starting with'.'.
Note that setting a path within a plugin only applies to resources accessed by plugin methods.
If no path is set, the connection files.relativeTo configuration is used. The path only applies
to routes added after it has been set.
exports.register = function (server, options, next) {
server.path(__dirname + '../static');
server.route({ path: '/file', method: 'GET', handler: { file: './test.html' } });
next();
};Registers a plugin where:
plugins- an object or array of objects where each one is either:- a plugin registration function.
- an object with the following:
register- the plugin registration function.options- optional options passed to the registration function when called.
options- optional registration options (different from the options passed to the registration function):select- a string or array of string labels used to pre-select connections for plugin registration.routes- modifiers applied to each route added by the plugin:prefix- string added as prefix to any route path (must begin with'/'). If a plugin registers a child plugin theprefixis passed on to the child or is added in front of the child-specific prefix.vhost- virtual host string (or array of strings) applied to every route. The outer-mostvhostoverrides the any nested configuration.
callback- the callback function with signaturefunction(err)where:err- an error returned from the registration function. Note that exceptions thrown by the registration function are not handled by the framework.
server.register({
register: require('plugin_name'),
options: {
message: 'hello'
}
}, function (err) {
if (err) {
console.log('Failed loading plugin');
}
});Utilizes the server views manager to render a template where:
template- the template filename and path, relative to the views manager templates path (pathorrelativeTo).context- optional object used by the template to render context-specific result. Defaults to no context ({}).options- optional object used to override the views manager configuration.callback- the callback function with signaturefunction (err, rendered, config)where:err- the rendering error if any.rendered- the result view string.config- the configuration used to render the template.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.views({
engines: { html: require('handlebars') },
path: __dirname + '/templates'
});
var context = {
title: 'Views Example',
message: 'Hello, World'
};
server.render('hello', context, function (err, rendered, config) {
console.log(rendered);
});Adds a connection route where:
options- a route configuration object or an array of configuration objects.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.route({ method: 'GET', path: '/', handler: function (request, reply) { return reply('ok'); } });
server.route([
{ method: 'GET', path: '/1', handler: function (request, reply) { return reply('ok'); } },
{ method: 'GET', path: '/2', handler: function (request, reply) { return reply('ok'); } }
]);Selects a subset of the server's connections where:
labels- a single string or array of strings of labels used as a logical OR statement to select all the connections with matching labels in their configuration.
Returns a server object with connections set to the requested subset. Selecting again on a
selection operates as a logic AND statement between the individual selections.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80, labels: ['a', 'b'] });
server.connection({ port: 8080, labels: ['a', 'c'] });
server.connection({ port: 8081, labels: ['b', 'c'] });
var a = server.select('a'); // 80, 8080
var ac = a.select('c'); // 8080Starts the server connections by listening for incoming requests on the configured port of each
listener (unless the connection was configured with autoListen set to false), where:
callback- optional callback when server startup is completed or failed with the signaturefunction(err)where:err- any startup error condition.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.start(function (err) {
console.log('Server started at: ' + server.info.uri);
});HTTP state management uses client cookies to persist a state across multiple requests. Registers a cookie definitions where:
name- the cookie name string.options- are the optional cookie settings:ttl- time-to-live in milliseconds. Defaults tonull(session time-life - cookies are deleted when the browser is closed).isSecure- sets the 'Secure' flag. Defaults tofalse.isHttpOnly- sets the 'HttpOnly' flag. Defaults tofalse.path- the path scope. Defaults tonull(no path).domain- the domain scope. Defaults tonull(no domain).autoValue- if present and the cookie was not received from the client or explicitly set by the route handler, the cookie is automatically added to the response with the provided value. The value can be a function with signaturefunction(request, next)where:request- the request object.next- the continuation function using thefunction(err, value)signature.
encoding- encoding performs on the provided value before serialization. Options are:'none'- no encoding. When used, the cookie value must be a string. This is the default value.'base64'- string value is encoded using Base64.'base64json'- object value is JSON-stringified than encoded using Base64.'form'- object value is encoded using the x-www-form-urlencoded method.'iron'- Encrypts and sign the value using iron.
sign- an object used to calculate an HMAC for cookie integrity validation. This does not provide privacy, only a mean to verify that the cookie value was generated by the server. Redundant when'iron'encoding is used. Options are:integrity- algorithm options. Defaults torequire('iron').defaults.integrity.password- password used for HMAC key generation.
password- password used for'iron'encoding.iron- options for'iron'encoding. Defaults torequire('iron').defaults.ignoreErrors- iffalse, errors are ignored and treated as missing cookies.clearInvalid- iftrue, automatically instruct the client to remove invalid cookies. Defaults tofalse.strictHeader- iffalse, allows any cookie value including values in violation of RFC 6265. Defaults totrue.passThrough- overrides the default proxylocalStatePassThroughsetting.
State defaults can be modified via the server connections.routes.state configuration option.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
// Set cookie definition
server.state('session', {
ttl: 24 * 60 * 60 * 1000, // One day
isSecure: true,
path: '/',
encoding: 'base64json'
});
// Set state in route handler
var handler = function (request, reply) {
var session = request.state.session;
if (!session) {
session = { user: 'joe' };
}
session.last = Date.now();
return reply('Success').state('session', session);
};Registered cookies are automatically parsed when received. Parsing rules depends on the route
state.parse configuration. If an incoming registered cookie fails parsing,
it is not included in request.state, regardless of the state.failAction setting. When
state.failAction is set to 'log' and an invalid cookie value is received, the server will emit
a 'request-internal' event. To capture these errors subscribe to the 'request-internal' events
and filter on 'error' and 'state' tags:
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.on('request-internal', function (request, event, tags) {
if (tags.error && tags.state) {
console.error(event);
}
});Stops the server's connections by refusing to accept any new connections (existing connections will continue until closed or timeout), where:
options- optional object with:timeout- overrides the timeout in millisecond before forcefully terminating a connection. Defaults to5000(5 seconds).
callback- optional callback method with signaturefunction()which is called once all the connections have ended and it is safe to exit the process.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.stop({ timeout: 60 * 1000 }, function () {
console.log('Server stopped');
});Returns a copy of the routing table where:
host- optional host to filter routes matching a specific virtual host. Defaults to all virtual hosts.
The return value is an array where each item is an object containing:
info- theconnection.infothe connection the table was generated for.labels- the connection labels.table- an array of routes where each route contains:settings- the route config with defaults applied.method- the HTTP method in lower case.path- the route path.
Note that if the server has not been started and multiple connections use port 0, the table items
will override each other and will produce an incomplete result.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80, host: 'example.com' });
server.route({ method: 'GET', path: '/example', handler: function (request, reply) { return reply(); } });
var table = server.table();When calling connection.table() directly on each connection, the return value is the same as the
array table item value of an individual connection:
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80, host: 'example.com' });
server.route({ method: 'GET', path: '/example', handler: function (request, reply) { return reply(); } });
var table = server.connections[0].table();
/*
[
{
method: 'get',
path: '/example',
settings: { ... }
}
]
*/Initializes the server views manager where:
options- a configuration object with the following:engines- required object where each key is a file extension (e.g. 'html', 'hbr'), mapped to the npm module used for rendering the templates. Alternatively, the extension can be mapped to an object with the following options:module- the npm module used for rendering the templates. The module object must contain:compile()- the rendering function. The required function signature depends on thecompileModesettings. If thecompileModeis'sync', the signature iscompile(template, options), the return value is a function with signaturefunction(context, options), and the method is allowed to throw errors. If thecompileModeis'async', the signature iscompile(template, options, callback)wherecallbackhas the signaturefunction(err, compiled)wherecompiledis a function with signaturefunction(context, options, callback)andcallbackhas the signaturefunction(err, rendered).
- any of the
viewsoptions listed below (exceptdefaultExtension) to override the defaults for a specific engine.
defaultExtension- defines the default filename extension to append to template names when multiple engines are configured and not explicit extension is provided for a given template. No default value.path- the root file path used to resolve and load the templates identified when callingreply.view(). Defaults to current working directory.partialsPath- the root file path where partials are located. Partials are small segments of template code that can be nested and reused throughout other templates. Defaults to no partials support (empty path).helpersPath- the directory path where helpers are located. Helpers are functions used within templates to perform transformations and other data manipulations using the template context or other inputs. Each '.js' file in the helpers directory is loaded and the file name is used as the helper name. The files must export a single method with the signaturefunction(context)and return a string. Sub-folders are not supported and are ignored. Defaults to no helpers support (empty path). Note that jade does not support loading helpers this way.relativeTo- a base path used as prefix forpathandpartialsPath. No default.layout- if set totrueor a layout filename, layout support is enabled. A layout is a single template file used as the parent template for other view templates in the same engine. Iftrue, the layout template name must be 'layout.ext' where 'ext' is the engine's extension. Otherwise, the provided filename is suffixed with the engine's extension and loaded. Disablelayoutwhen using Jade as it will handle including any layout files independently. Defaults tofalse.layoutPath- the root file path where layout templates are located (using therelativeToprefix if present). Defaults topath.layoutKeyword- the key used by the template engine to denote where primary template content should go. Defaults to'content'.encoding- the text encoding used by the templates when reading the files and outputting the result. Defaults to'utf8'.isCached- if set tofalse, templates will not be cached (thus will be read from file on every use). Defaults totrue.allowAbsolutePaths- if set totrue, allows absolute template paths passed toreply.view(). Defaults tofalse.allowInsecureAccess- if set totrue, allows template paths passed toreply.view()to contain '../'. Defaults tofalse.compileOptions- options object passed to the engine's compile function. Defaults to empty options{}.runtimeOptions- options object passed to the returned function from the compile operation. Defaults to empty options{}.contentType- the content type of the engine results. Defaults to'text/html'.compileMode- specify whether the enginecompile()method is'sync'or'async'. Defaults to'sync'.context- a global context used with all templates. The global context option can be either an object or a function that takes no arguments and returns a context object. When rendering views, the global context will be merged with any context object specified on the handler or usingreply.view(). When multiple context objects are used, values from the global context always have lowest precedence.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.views({
engines: {
html: require('handlebars'),
jade: require('jade')
},
path: '/static/templates'
});When server.views() is called within a plugin, the views manager is only
available to plugins methods.
The server object inherits from Events.EventEmitter and emits the following events:
'log'- events logged withserver.log()and server events generated internally by the framework.'start'- emitted when the server is started usingserver.start().'stop'- emitted when the server is stopped usingserver.stop().'request'- events generated byrequest.log(). Does not include any internally generated events.'request-internal'- request events generated internally by the framework (multiple events per request).'request-error'- emitted whenever an Internal Server Error (500) error response is sent. Single event per request.'response'- emitted after a response to a client request is sent back. Single event per request.'tail'- emitted when a request finished processing, including any registered tails. Single event per request.
Note that the server object should not be used to emit application events as its internal implementation is designed to fan events out to the various plugin selections and not for application events.
When provided (as listed below) the event object include:
timestamp- the event timestamp.request- if the event relates to a request, therequest id.server- if the event relates to a server, theserver.info.uri.tags- an array of tags (e.g.['error', 'http']).data- optional event-specific information.internal-trueis the event was generated internally by the framework.
The 'log' event includes the event object and a tags object (where each tag is a key with the
value true):
server.on('log', function (event, tags) {
if (tags.error) {
console.log('Server error: ' + (event.data || 'unspecified'));
}
});The 'request' and 'request-internal' events include the request object, the
event object, and a tags object (where each tag is a key with the value true):
server.on('request', function (request, event, tags) {
if (tags.received) {
console.log('New request: ' + event.id);
}
});The 'request-error' event includes the request object and the causing error
err object:
server.on('request-error', function (request, err) {
console.log('Error response (500) sent for request: ' + request.id + ' because: ' + err.message);
});The 'response' and 'tail' events include the request object:
server.on('response', function (request) {
console.log('Response sent for request: ' + request.id);
});The following logs are generated automatically by the framework. Each event can be identified by the combination of tags used.
Emitted by the server 'request-internal' event:
received- a new request received. Includes information about the request.auth{strategy}- the request successfully authenticated with the listed strategy.authunauthenticated- no authentication scheme included with the request.authunauthenticatedresponse{strategy}- the authentication strategy listed returned a non-error response (e.g. a redirect to a login page).authunauthenticatederror{strategy}- the request failed to pass the listed authentication strategy.authunauthenticatedtry{strategy}- the request failed to pass the listed authentication strategy in'try'mode and will continue.authscopeerror{strategy}- the request authenticated but failed to meet the scope requirements.authentityusererror{strategy}- the request authenticated but included an application entity when a user entity was required.authentityapperror{strategy}- the request authenticated but included a user entity when an application entity was required.handler- the route handler executed. Includes the execution duration.handlererror- the route handler returned an error. Includes the execution duration and the error message.handlermethod{method} - a string-shortcut handler method was executed (when cache enabled). Includes information about the execution including cache performance.premethod{method} - a string-shortcut pre method was executed (when cache enabled). Includes information about the execution including cache performance.pre- a pre method was executed. Includes the execution duration and assignment key.preerror- a pre method was executed and returned an error. Includes the execution duration, assignment key, and error.internalerror- an HTTP 500 error response was assigned to the request.internalimplementationerror- a function provided by the user failed with an exception during request execution.requestclosederror- the request closed prematurely.requesterror- the request stream emitted an error. Includes the error.requestservertimeouterror- the request took too long to process by the server. Includes the timeout configuration value and the duration.tailadd- a request tail was added. Includes the tail name and id.tailremove- a request tail was removed. Includes the tail name and id.tailremovelast- the last request tail was removed. Includes the tail name and id.tailremoveerror- failed to remove a request tail (already removed). Includes the tail name and id.stateerror- the request included an invalid cookie or cookies. Includes the cookies and error details.stateresponseerror- the response included an invalid cookie which prevented generating a valid header. Includes the error.payloaderror- failed processing the request payload. Includes the error.response- the response was sent successfully.responseerror- failed writing the response to the client. Includes the error.responseerrorclose- failed writing the response to the client due to prematurely closed connection.responseerroraborted- failed writing the response to the client due to prematurely aborted connection.validationerror{input}- input (i.e. payload, query, params, headers) validation failed. Includes the error.validationresponseerror- response validation failed. Includes the error message.
Emitted by the server 'log' event:
load- logs the current server load measurements when the server rejects a request due to high load. The event data contains the metrics.internalimplementationerror- a function provided by the user failed with an exception during request execution. The log appears under the server logs when the exception cannot be associated with the request that generated it.
Plugins provide a way to organize the application code by splitting the server logic into smaller components. Each plugin can manipulate the server and its connections through the standard server interface, but with the added ability to sandbox certain properties (such as the views manager and relative file paths).
A plugin is a function with the signature function(server, options, next) where:
server- the server object the plugin is being registered to.options- an options object passed to the plugin during registration.next- a callback method the function must call to return control back to the framework to complete the registration process with signaturefunction(err)where:err- any plugin registration error.
The plugin function must include an attributes function property with the following:
name- required plugin name string. The name is used as a unique key. Published plugins should use the same name as the name field in the 'package.json' file. Names must be unique within each application.version- optional plugin version. The version is only used informatively to enable other plugins to find out the versions loaded. The version should be the same as the one specified in the plugin's 'package.json' file.multiple- iftrue, allows the plugin to be registered multiple times with the same server. Defaults tofalse.
var register = function (server, options, next) {
server.route({
method: 'GET',
path: '/test',
handler: function (request, reply) {
return reply('ok');
}
});
return next();
};
register.attributes = {
name: 'test',
version: '1.0.0'
};Alternatively, the name and version can be included via the pkg attribute containing the
'package.json' file for the module which already has the name and version included:
register.attributes = {
pkg: require('./package.json')
};Incoming requests are handled by the server via routes. Each route describes an HTTP endpoint with
a path, method, and other properties. The route logic is divided between static configuration,
prerequisite functions and a route handler function. Routes are added via the
server.route() method.
Each incoming request passes through a pre-defined list of steps, along with optional extensions:
'onRequest'extension point- always called
- the request object passed to the extension functions is decorated with the
request.setUrl()andrequest.setMethod()methods. Calls to these methods will impact how the request is routed and can be used for rewrite rules. request.routeis not yet populated at this point.
- Lookup route using request path
- Parse cookies
'onPreAuth'extension point- Authenticate request
- Read and parse payload
- Authenticate request payload
'onPostAuth'extension point- Validate path parameters
- Process query extensions (e.g. JSONP)
- Validate query
- Validate payload
'onPreHandler'extension point- Route prerequisites
- Route handler
'onPostHandler'extension point- The response object contained in
request.responsemay be modified (but not assigned a new value). To return a different response type (for example, replace an error with an HTML response), return a new response viareply(response).
- The response object contained in
- Validate response payload
'onPreResponse'extension point- always called (except when
reply.close()is called or the client terminates the connection prematurely). - The response contained in
request.responsemay be modified (but not assigned a new value). To return a different response type (for example, replace an error with an HTML response), return a new response viareply(response). Note that any errors generated afterreply(response)is called will not be passed back to the'onPreResponse'extension method to prevent an infinite loop.
- always called (except when
- Send response (may emit
'request-error'event) - Emits
'response'event - Wait for tails
- Emits
'tail'event
The route configuration object supports the following options:
-
path- (required) the absolute path used to match incoming requests (must begin with '/'). Incoming requests are compared to the configured paths based on the connectionrouterconfiguration option. The path can include named parameters enclosed in{}which will be matched against literal values in the request as described in Path parameters. -
method- (required) the HTTP method. Typically one of 'GET', 'POST', 'PUT', 'PATCH', 'DELETE', or 'OPTIONS'. Any HTTP method is allowed, except for 'HEAD'. Use'*'to match against any HTTP method (only when an exact match was not found, and any match with a specific method will be given a higher priority over a wildcard match). Can be assigned an array of methods which has the same result as adding the same route with different methods manually. -
vhost- an optional domain string or an array of domain strings for limiting the route to only requests with a matching host header field. Matching is done against the hostname part of the header only (excluding the port). Defaults to all hosts. -
handler- (required) the function called to generate the response after successful authentication and validation. The handler function is described in Route handler. If set to a string, the value is parsed the same way a prerequisite server method string shortcut is processed. Alternatively,handlercan be assigned an object with a single key using the name of a registered handler type and value with the options passed to the registered handler. -
config- additional route options.
Note that the options object is deeply cloned (with the exception of bind which is shallowly
copied) and cannot contain any values that are unsafe to perform deep copy on.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
// Handler in top level
var status = function (request, reply) {
return reply('ok');
};
server.route({ method: 'GET', path: '/status', handler: status });
// Handler in config
var user = {
cache: { expiresIn: 5000 },
handler: function (request, reply) {
return reply({ name: 'John' });
}
};
server.route({ method: 'GET', path: '/user', config: user });Each route can be customize to change the default behavior of the request lifecycle using the following options:
-
app- application-specific configuration. Should not be used by plugins which should useplugins[name]instead. -
auth- authentication configuration. Value can be:falseto disable authentication if a default strategy is set.- a string with the name of an authentication strategy registered with
server.auth.strategy(). - an object with:
mode- the authentication mode. Defaults to'required'if a server authentication strategy is configured, otherwise defaults to no authentication. Available values:'required'- authentication is required.'optional'- authentication is optional (must be valid if present).'try'- same as'optional'but allows for invalid authentication.
strategies- a string array of strategy names in order they should be attempted. If only one strategy is used,strategycan be used instead with the single string value. Defaults to the default authentication strategy which is available only when a single strategy is configured.payload- if set, the payload (in requests other than 'GET' and 'HEAD') is authenticated after it is processed. Requires a strategy with payload authentication support (e.g. Hawk). Cannot be set to a value other than'required'when the scheme sets theoptions.payloadtotrue. Available values:false- no payload authentication. This is the default value.'required'- payload authentication required. This is the default value when the scheme setsoptions.payloadtotrue.'optional'- payload authentication performed only when the client includes payload authentication information (e.g.hashattribute in Hawk).
scope- the application scope required to access the route. Value can be a scope string or an array of scope strings. The authenticated credentials objectscopeproperty must contain at least one of the scopes defined to access the route. Set tofalseto remove scope requirements. Defaults to no scope required.entity- the required authenticated entity type. If set, must match theentityvalue of the authentication credentials. Available values:any- the authentication can be on behalf of a user or application. This is the default value.user- the authentication must be on behalf of a user.app- the authentication must be on behalf of an application.
-
bind- an object passed back to the providedhandler(viathis) when called. -
cache- if the route method is 'GET', the route can be configured to include caching directives in the response using the following options:privacy- determines the privacy flag included in client-side caching using the 'Cache-Control' header. Values are:'default'- no privacy flag. This is the default setting.'public'- mark the response as suitable for public caching.'private'- mark the response as suitable only for private caching.
expiresIn- relative expiration expressed in the number of milliseconds since the item was saved in the cache. Cannot be used together withexpiresAt.expiresAt- time of day expressed in 24h notation using the 'HH:MM' format, at which point all cache records for the route expire. Cannot be used together withexpiresIn.
-
cors- the Cross-Origin Resource Sharing protocol allows browsers to make cross-origin API calls. CORS is required by web applications running inside a browser which are loaded from a different domain than the API server. CORS headers are disabled by default. To enable, setcorstotrue, or to an object with the following options:origin- a strings array of allowed origin servers ('Access-Control-Allow-Origin'). The array can contain any combination of fully qualified origins along with origin strings containing a wildcard '*' character, or a single'*'origin string. Defaults to any origin['*'].matchOrigin- iftrue, matches the value of the incoming 'Origin' header to the list oforiginvalues ('*' matches anything) and if a match is found, uses that as the value of the 'Access-Control-Allow-Origin' response header. When false, theoriginconfig is returned as-is. Defaults totrue.isOriginExposed- iffalse, prevents the connection from returning the full list of non-wildcardoriginvalues if the incoming origin header does not match any of the values. Has no impact ifmatchOriginis set tofalse. Defaults totrue.maxAge- number of seconds the browser should cache the CORS response ('Access-Control-Max-Age'). The greater the value, the longer it will take before the browser checks for changes in policy. Defaults to86400(one day).headers- a strings array of allowed headers ('Access-Control-Allow-Headers'). Defaults to['Authorization', 'Content-Type', 'If-None-Match'].additionalHeaders- a strings array of additional headers toheaders. Use this to keep the default headers in place.methods- a strings array of allowed HTTP methods ('Access-Control-Allow-Methods'). Defaults to['GET', 'HEAD', 'POST', 'PUT', 'DELETE', 'OPTIONS'].additionalMethods- a strings array of additional methods tomethods. Use this to keep the default methods in place.exposedHeaders- a strings array of exposed headers ('Access-Control-Expose-Headers'). Defaults to['WWW-Authenticate', 'Server-Authorization'].additionalExposedHeaders- a strings array of additional headers toexposedHeaders. Use this to keep the default headers in place.credentials- iftrue, allows user credentials to be sent ('Access-Control-Allow-Credentials'). Defaults tofalse.override- iffalse, preserves existing CORS headers set manually before the response is sent. Defaults totrue.
-
files- defines the behavior for serving static resources using the built-in route handlers for files and directories:relativeTo- determines the folder relative paths are resolved against when using the file and directory handlers.
-
handler- an alternative location for the routehandleroption. -
id- an optional unique identifier used to look up the route usingserver.lookup(). -
json- optional arguments passed toJSON.stringify()when converting an object or error response to a string payload. Supports the following:replacer- the replacer function or array. Defaults to no action.space- number of spaces to indent nested object keys. Defaults to no indentation.suffix- string suffix added after conversion to JSON string. Defaults to no suffix.
-
jsonp- enables JSONP support by setting the value to the query parameter name containing the function name used to wrap the response payload. For example, if the value is'callback', a request comes in with'callback=me', and the JSON response is'{ "a":"b" }', the payload will be'me({ "a":"b" });'. Does not work with stream responses. -
payload- determines how the request payload is processed:output- the type of payload representation requested. The value must be one of:'data'- the incoming payload is read fully into memory. Ifparseistrue, the payload is parsed (JSON, form-decoded, multipart) based on the 'Content-Type' header. Ifparseis false, the rawBufferis returned. This is the default value except when a proxy handler is used.'stream'- the incoming payload is made available via aStream.Readableinterface. If the payload is 'multipart/form-data' andparseistrue, fields values are presented as text while files are provided as streams. File streams from a 'multipart/form-data' upload will also have a propertyhapicontainingfilenameandheadersproperties.'file'- the incoming payload in written to temporary file in the directory specified by the server'spayload.uploadssettings. If the payload is 'multipart/form-data' andparseistrue, fields values are presented as text while files are saved. Note that it is the sole responsibility of the application to clean up the files generated by the framework. This can be done by keeping track of which files are used (e.g. using therequest.appobject), and listening to the server'response'event to perform any needed cleaup.
parse- can betrue,false, orgunzip; determines if the incoming payload is processed or presented raw.trueandgunzipincludes gunzipping when the appropriate 'Content-Encoding' is specified on the received request. If parsing is enabled and the 'Content-Type' is known (for the whole payload as well as parts), the payload is converted into an object when possible. If the format is unknown, a Bad Request (400) error response is sent. Defaults totrue, except when a proxy handler is used. The supported mime types are:- 'application/json'
- 'application/x-www-form-urlencoded'
- 'application/octet-stream'
- 'text/*'
- 'multipart/form-data'
allow- a string or an array of strings with the allowed mime types for the endpoint. Defaults to any of the supported mime types listed above. Note that allowing other mime types not listed will not enable them to be parsed, and that if parsing mode is'parse', the request will result in an error response.override- a mime type string overriding the 'Content-Type' header value received. Defaults to no override.maxBytes- limits the size of incoming payloads to the specified byte count. Allowing very large payloads may cause the server to run out of memory. Defaults to1048576(1MB).client- payload reception timeout in milliseconds. Sets the maximum time allowed for the client to transmit the request payload (body) before giving up and responding with a Request Timeout (408) error response. Set tofalseto disable. Defaults to10000(10 seconds).uploads- the directory used for writing file uploads. Defaults toos.tmpDir().failAction- determines how to handle payload parsing errors. Allowed values are:'error'- return a Bad Request (400) error response. This is the default value.'log'- report the error but continue processing the request.'ignore'- take no action and continue processing the request.
-
plugins- plugin-specific configuration.pluginsis an object where each key is a plugin name and the value is the plugin configuration. -
pre- an array with route prerequisites methods which are executed in serial or in parallel before the handler is called. -
response- validation rules for the outgoing response payload (response body). Can only validate object response:schema- the default response object validation rules (for all non-error responses) expressed as one of:true- any payload allowed (no validation performed). This is the default.false- no payload allowed.- a Joi validation object.
- a validation function using the signature
function(value, options, next)where:value- the object containing the response object.options- the server validation options.next(err)- the callback function called when validation is completed.
status- HTTP status-code-specific validation rules. Thestatuskey is set to an object where each key is a 3 digit HTTP status code and the value has the same definition asschema. If a response status code is not present in thestatusobject, theschemadefinition is used, expect for errors which are not validated by default.sample- the percent of responses validated (0 - 100). Set to0to disable all validation. Defaults to100(all responses).failAction- defines what to do when a response fails validation. Options are:error- return an Internal Server Error (500) error response. This is the default value.log- log the error but send the response.
modify- iftrue, applies the validation rule changes to the response. Defaults tofalse.options- options to pass to Joi. Useful to set global options such asstripUnknownorabortEarly(the complete list is available here). Defaults to no options.
-
security- sets common security headers (disabled by default). To enable setsecuritytotrueor to an object with the following options:hsts- controls the 'Strict-Transport-Security' header. If set totruethe header will be set tomax-age=15768000, if specified as a number the maxAge parameter will be set to that number. Defaults totrue. You may also specify an object with the following fields:maxAge- the max-age portion of the header, as a number. Default is15768000.includeSubdomains- a boolean specifying whether to add theincludeSubdomainsflag to the header.
xframe- controls the 'X-Frame-Options' header. When set totruethe header will be set toDENY, you may also specify a string value of 'deny' or 'sameorigin'. To use the 'allow-from' rule, you must set this to an object with the following fields:rule- either 'deny', 'sameorigin', or 'allow-from'source- whenruleis 'allow-from' this is used to form the rest of the header, otherwise this field is ignored. Ifruleis 'allow-from' butsourceis unset, the rule will be automatically changed to 'sameorigin'.
xss- boolean that controls the 'X-XSS-PROTECTION' header for IE. Defaults totruewhich sets the header to equal '1; mode=block'. NOTE: This setting can create a security vulnerability in versions of IE below 8, as well as unpatched versions of IE8. See here and here for more information. If you actively support old versions of IE, it may be wise to explicitly set this flag tofalse.noOpen- boolean controlling the 'X-Download-Options' header for IE, preventing downloads from executing in your context. Defaults totruesetting the header to 'noopen'.noSniff- boolean controlling the 'X-Content-Type-Options' header. Defaults totruesetting the header to its only and default option, 'nosniff'.
-
state- HTTP state management (cookies) allows the server to store information on the client which is sent back to the server with every request (as defined in RFC 6265).statesupports the following options:parse- determines if incoming 'Cookie' headers are parsed and stored in therequest.stateobject. Defaults totrue.failAction- determines how to handle cookie parsing errors. Allowed values are:'error'- return a Bad Request (400) error response. This is the default value.'log'- report the error but continue processing the request.'ignore'- take no action.
-
validate- request input validation rules for various request components. When using a Joi validation object, the values of the other inputs (i.e.headers,query,params,payload, andauth) are made available under the validation context (accessible in rules asJoi.ref('$query.key')). Note that validation is performed in order (i.e. headers, params, query, payload) and if type casting is used (converting a string to number), the value of inputs not yet validated will reflect the raw, unvalidated and unmodified values. Thevalidateobject supports:-
headers- validation rules for incoming request headers. Values allowed:true- any headers allowed (no validation performed). This is the default.false- no headers allowed (this will cause all valid HTTP requests to fail).- a Joi validation object.
- a validation function using the signature
function(value, options, next)where:value- the object containing the request headers.options- the server validation options.next(err, value)- the callback function called when validation is completed.
-
params- validation rules for incoming request path parameters, after matching the path against the route and extracting any parameters then stored inrequest.params. Values allowed:true- any path parameters allowed (no validation performed). This is the default.false- no path variables allowed.- a Joi validation object.
- a validation function using the signature
function(value, options, next)where:value- the object containing the path parameters.options- the server validation options.next(err, value)- the callback function called when validation is completed.
-
query- validation rules for an incoming request URI query component (the key-value part of the URI between '?' and '#'). The query is parsed into its individual key-value pairs (using the qs module) and stored inrequest.queryprior to validation. Values allowed:true- any query parameters allowed (no validation performed). This is the default.false- no query parameters allowed.- a Joi validation object.
- a validation function using the signature
function(value, options, next)where:value- the object containing the query parameters.options- the server validation options.next(err, value)- the callback function called when validation is completed.
-
payload- validation rules for an incoming request payload (request body). Values allowed:true- any payload allowed (no validation performed). This is the default.false- no payload allowed.- a Joi validation object.
- a validation function using the signature
function(value, options, next)where:value- the object containing the payload object.options- the server validation options.next(err, value)- the callback function called when validation is completed.
-
errorFields- an optional object with error fields copied into every validation error response. -
failAction- determines how to handle invalid requests. Allowed values are:'error'- return a Bad Request (400) error response. This is the default value.'log'- log the error but continue processing the request.'ignore'- take no action.- a custom error handler function with the signature
'function(request, reply, source, error)` where:
request- the request object.reply- the continuation reply interface.source- the source of the invalid field (e.g.'path','query','payload').error- the error object prepared for the client response (including the validation function error undererror.data).
-
options- options to pass to Joi. Useful to set global options such asstripUnknownorabortEarly(the complete list is available here). Defaults to no options.
-
-
timeout- define timeouts for processing durations:server- response timeout in milliseconds. Sets the maximum time allowed for the server to respond to an incoming client request before giving up and responding with a Service Unavailable (503) error response. Disabled by default (false).socket- by default, node sockets automatically timeout after 2 minutes. Use this option to override this behavior. Defaults toundefinedwhich leaves the node default unchanged. Set tofalseto disable socket timeouts.
The following documentation options are also available when adding new routes (they are not available when setting defaults):
description- route description used for generating documentation (string).notes- route notes used for generating documentation (string or array of strings).tags- route tags used for generating documentation (array of strings).
When route information is returned or made available as a property, it is an object with the following:
method- the route HTTP method.path- the route path.vhost- the route vhost option if configured.realm- the active realm associated with the route.settings- the route options object with all defaults applied.
Parameterized paths are processed by matching the named parameters to the content of the incoming
request path at that path segment. For example, '/book/{id}/cover' will match '/book/123/cover' and
request.params.id will be set to '123'. Each path segment (everything between the opening '/'
and the closing '/' unless it is the end of the path) can only include one named parameter. A
parameter can cover the entire segment ('/{param}') or part of the segment ('/file.{ext}').
An optional '?' suffix following the parameter name indicates an optional parameter (only allowed
if the parameter is at the ends of the path or only covers part of the segment as in
'/a{param?}/b'). For example, the route '/book/{id?}' matches '/book/' with the value of
request.params.id set to an empty string ''.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var getAlbum = function (request, reply) {
return reply('You asked for ' +
(request.params.song ? request.params.song + ' from ' : '') +
request.params.album);
};
server.route({
path: '/{album}/{song?}',
method: 'GET',
handler: getAlbum
});In addition to the optional ? suffix, a parameter name can also specify the number of matching
segments using the * suffix, followed by a number greater than 1. If the number of expected parts
can be anything, then use * without a number (matching any number of segments can only be used in
the last path segment).
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var getPerson = function (request, reply) {
var nameParts = request.params.name.split('/');
return reply({ first: nameParts[0], last: nameParts[1] });
};
server.route({
path: '/person/{name*2}', // Matches '/person/john/doe'
method: 'GET',
handler: getPerson
});The router iterates through the routing table on each incoming request and executes the first (and only the first) matching route. Route matching is done based on the combination of the request path and the HTTP verb (e.g. 'GET, 'POST'). The query is excluded from the routing logic. Requests are matched in a deterministic order where the order in which routes are added does not matter.
Routes are matched based on the specificity of the route which is evaluated at each segment of the
incoming request path. Each request path is split into its segment (the parts separated by '/').
The segments are compared to the routing table one at a time and are matched against the most
specific path until a match is found. If no match is found, the next match is tried.
When matching routes, string literals (no path parameter) have the highest priority, followed by
mixed parameters ('/a{p}b'), parameters ('/{p}'), and then wildcard (/{p*}).
Note that mixed parameters are slower to compare as they cannot be hashed and require an array iteration over all the regular expressions representing the various mixed parameter at each routing table node.
If the application needs to override the default Not Found (404) error response, it can add a catch-all route for a specific method or all methods. Only one catch-all route can be defined per server connection.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var handler = function (request, reply) {
return reply('The page was not found').code(404);
};
server.route({ method: '*', path: '/{p*}', handler: handler });The route handler function uses the signature function(request, reply) where:
request- is the incoming request object (this is not the node.js request object).reply- the reply interface the handler must call to set a response and return control back to the framework.
var handler = function (request, reply) {
return reply('success');
};The framework comes with a few built-in handler types available by setting the route handler
config to an object containing one of these keys:
-
file- generates a static file endpoint for serving a single file.filecan be set to:- a relative or absolute file path string (relative paths are resolved based on the
route
filesconfiguration). - a function with the signature
function(request)which returns the relative or absolute file path. - an object with the following options:
path- a path string or function as described above.filename- an optional filename to specify if sending a 'Content-Disposition' header, defaults to the basename ofpathmode- specifies whether to include the 'Content-Disposition' header with the response. Available values:false- header is not included. This is the default value.'attachment''inline'
lookupCompressed- iftrue, looks for the same filename with the '.gz' suffix for a pre-compressed version of the file to serve if the request supports content encoding. Defaults tofalse.
- a relative or absolute file path string (relative paths are resolved based on the
route
-
directory- generates a directory endpoint for serving static content from a directory. Routes using the directory handler must include a path parameter at the end of the path string (e.g./path/to/somewhere/{param}where the parameter name does not matter). The path parameter can use any of the parameter options (e.g.{param}for one level files only,{param?}for one level files or the directory root,{param*}for any level, or{param*3}for a specific level). If additional path parameters are present, they are ignored for the purpose of selecting the file system resource. The directory handler is an object with the following options:path- (required) the directory root path (relative paths are resolved based on the routefilesconfiguration). Value can be:- a single path string used as the prefix for any resources requested by appending the request path parameter to the provided string.
- an array of path strings. Each path will be attempted in order until a match is found (by following the same process as the single path string).
- a function with the signature
function(request)which returns the path string or an array of path strings. If the function returns an error, the error is passed back to the client in the response.
index- optional boolean, determines if 'index.html' will be served if found in the folder when requesting a directory. Defaults totrue.listing- optional boolean, determines if directory listing is generated when a directory is requested without an index document. Defaults tofalse.showHidden- optional boolean, determines if hidden files will be shown and served. Defaults tofalse.redirectToSlash- optional boolean, determines if requests for a directory without a trailing slash are redirected to the same path with the missing slash. Useful for ensuring relative links inside the response are resolved correctly. Disabled when the server configrouter.stripTrailingSlashistrue.Defaults tofalse.lookupCompressed- optional boolean, instructs the file processor to look for the same filename with the '.gz' suffix for a pre-compressed version of the file to serve if the request supports content encoding. Defaults tofalse.defaultExtension- optional string, appended to file requests if the requested file is not found. Defaults to no extension.
-
proxy- generates a reverse proxy handler with the following options:host- the upstream service host to proxy requests to. The same path on the client request will be used as the path on the host.port- the upstream service port.protocol- The protocol to use when making a request to the proxied host:'http''https'
uri- an absolute URI used instead of the incoming host, port, protocol, path, and query. Cannot be used withhost,port,protocol, ormapUri.passThrough- iftrue, forwards the headers sent from the client to the upstream service being proxied to, headers sent from the upstream service will also be forwarded to the client. Defaults tofalse.localStatePassThrough- iffalse, any locally defined state is removed from incoming requests before being passed upstream. This is a security feature to prevent local state (e.g. authentication cookies) from leaking upstream to other servers along with the cookies intended for those servers. This value can be overridden on a per state basis via theserver.state()passThroughoption. Defaults tofalse.acceptEncoding- iffalse, does not pass-through the 'Accept-Encoding' HTTP header which is useful when using anonResponsepost-processing to avoid receiving an encoded response (e.g. gzipped). Can only be used together withpassThrough. Defaults totrue(passing header).rejectUnauthorized- sets therejectUnauthorizedproperty on the https agent making the request. This value is only used when the proxied server uses TLS/SSL. When set it will override the node.jsrejectUnauthorizedproperty. Iffalsethen ssl errors will be ignored. Whentruethe server certificate is verified and an 500 response will be sent when verification fails. This shouldn't be used alongside theagentsetting as theagentwill be used instead. Defaults to the https agent default value oftrue.xforward- iftrue, sets the 'X-Forwarded-For', 'X-Forwarded-Port', 'X-Forwarded-Proto' headers when making a request to the proxied upstream endpoint. Defaults tofalse.redirects- the maximum number of HTTP redirections allowed, to be followed automatically by the handler. Set tofalseor0to disable all redirections (the response will contain the redirection received from the upstream service). If redirections are enabled, no redirections (301, 302, 307, 308) will be passed along to the client, and reaching the maximum allowed redirections will return an error response. Defaults tofalse.timeout- number of milliseconds before aborting the upstream request. Defaults to180000(3 minutes).mapUri- a function used to map the request URI to the proxied URI. Cannot be used together withhost,port,protocol, oruri. The function signature isfunction(request, callback)where:request- is the incoming request object.callback- isfunction(err, uri, headers)where:err- internal error condition.uri- the absolute proxy URI.headers- optional object where each key is an HTTP request header and the value is the header content.
onResponse- a custom function for processing the response from the upstream service before sending to the client. Useful for custom error handling of responses from the proxied endpoint or other payload manipulation. Function signature isfunction(err, res, request, reply, settings, ttl)where: -err- internal or upstream error returned from attempting to contact the upstream proxy. -res- the node response object received from the upstream service.resis a readable stream (use the wreck modulereadmethod to easily convert it to a Buffer or string). -request- is the incoming request object. -reply- the reply interface function. -settings- the proxy handler configuration. -ttl- the upstream TTL in milliseconds ifproxy.ttlit set to'upstream'and the upstream response included a valid 'Cache-Control' header with 'max-age'.ttl- if set to'upstream', applies the upstream response caching policy to the response using theresponse.ttl()method (or passed as an argument to theonResponsemethod if provided).agent- a node http(s) agent to be used for connections to upstream server.maxSockets- sets the maximum number of sockets available per outgoing proxy host connection.falsemeans use the wreck module default value (Infinity). Does not affect non-proxy outgoing client connections. Defaults toInfinity.
-
view- generates a template-based response. Theviewoption can be set to one of:- a string with the template file name.
- an object with the following keys:
template- a string with the template file name.context- an optional template context object. Defaults to an object with the following key:payload- maps torequest.payload.params- maps torequest.params.query- maps torequest.query.pre- maps torequest.pre.
options- optional object used to override the server's views manager configuration.
It is often necessary to perform prerequisite actions before the handler is called (e.g. load
required reference data from a database). The route pre option allows defining such pre-handler
methods. The methods are called in order. If the pre array contains another array, those methods
are called in parallel. pre can be assigned a mixed array of:
- arrays containing the elements listed below, which are executed in parallel.
- objects with:
method- the function to call (or short-hand method string as described below). the function signature is identical to a route handler as describer in Route handler.assign- key name to assign the result of the function to withinrequest.pre.failAction- determines how to handle errors returned by the method. Allowed values are:'error'- returns the error response back to the client. This is the default value.'log'- logs the error but continues processing the request. Ifassignis used, the error will be assigned.'ignore'- takes no special action. Ifassignis used, the error will be assigned.
- functions - same as including an object with a single
methodkey. - strings - special short-hand notation for registered
server methods using the format 'name(args)' (e.g.
'user(params.id)') where:- 'name' - the method name. The name is also used as the default value of
assign. - 'args' - the method arguments (excluding
next) where each argument is a property of the request object.
- 'name' - the method name. The name is also used as the default value of
Note that prerequisites do not follow the same rules of the normal
reply interface. In all other cases, calling reply() with or without a value
will use the result as the response sent back to the client. In a prerequisite method, calling
reply() will assign the returned value to the provided assign key. If the returned value is an
error, the failAction setting determines the behavior. To force the return value as the response
and ends the request lifecycle, use the reply().takeover() method.
The reason for the difference in the reply interface behavior is to allow reusing handlers and prerequisites methods interchangeably. By default, the desired behavior for a prerequisite is to retain the result value and pass it on to the next step. Errors end the lifecycle by default. While less consistent, this allows easier code reusability.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var pre1 = function (request, reply) {
return reply('Hello');
};
var pre2 = function (request, reply) {
return reply('World');
};
var pre3 = function (request, reply) {
return reply(request.pre.m1 + ' ' + request.pre.m2);
};
server.route({
method: 'GET',
path: '/',
config: {
pre: [
[
// m1 and m2 executed in parallel
{ method: pre1, assign: 'm1' },
{ method: pre2, assign: 'm2' }
],
{ method: pre3, assign: 'm3' },
],
handler: function (request, reply) {
return reply(request.pre.m3 + '\n');
}
}
});The request object is created internally for each incoming request. It is different from the
node.js request object received from the HTTP server callback (which is available in
request.raw.req). The request object methods and properties change throughout the
request lifecycle.
Each request object includes the following properties:
app- application-specific state. Provides a safe place to store application data without potential conflicts with the framework. Should not be used by plugins which should useplugins[name].auth- authentication information:isAuthenticated-trueis the request has been successfully authenticated, otherwisefalse.credentials- thecredentialobject received during the authentication process. The presence of an object does not mean successful authentication.artifacts- an artifact object received from the authentication strategy and used in authentication-related actions.mode- the route authentication mode.error- the authentication error is failed and mode set to'try'.session- an object used by the'cookie'authentication scheme.
domain- the node domain object used to protect against exceptions thrown in extensions, handlers and route prerequisites. Can be used to manually bind callback functions otherwise bound to other domains.headers- the raw request headers (referencesrequest.raw.headers).id- a unique request identifier (using the format '{now}:{connection.info.id}:{5 digits counter}').info- request information:received- request reception timestamp.responded- request response timestamp (0is not responded yet).remoteAddress- remote client IP address.remotePort- remote client port.referrer- content of the HTTP 'Referrer' (or 'Referer') header.host- content of the HTTP 'Host' header (e.g. 'example.com:8080').hostname- the hostname part of the 'Host' header (e.g. 'example.com').
method- the request method in lower case (e.g.'get','post').mime- the parsed content-type header. Only available when payload parsing enabled and no payload error occurred.orig- an object containing the values ofparams,query, andpayloadbefore any validation modifications made. Only set when input validation is performed.params- an object where each key is a path parameter name with matching value as described in Path parameters.paramsArray- an array containing all the pathparamsvalues in the order they appeared in the path.path- the request URI's path component.payload- the request payload based on the routepayload.outputandpayload.parsesettings.plugins- plugin-specific state. Provides a place to store and pass request-level plugin data. Thepluginsis an object where each key is a plugin name and the value is the state.pre- an object where each key is the name assigned by a route prerequisites function. The values are the raw values provided to the continuation function as argument. For the wrapped response object, useresponses.response- the response object when set. The object can be modified but must not be assigned another object. To replace the response with another from within an extension point, usereply(response)to override with a different response.preResponses- same asprebut represented as the response object created by the pre method.query- an object containing the query parameters.raw- an object containing the Node HTTP server objects. Direct interaction with these raw objects is not recommended.req- the node.js request object.res- the node.js response object.
route- the route public interface.server- the server object.session- Special key reserved for plugins implementing session support. Plugins utilizing this key must check fornullvalue to ensure there is no conflict with another similar server.state- an object containing parsed HTTP state information (cookies) where each key is the cookie name and value is the matching cookie content after processing using any registered cookie definition.url- the parsed request URI.
Available only in 'onRequest' extension methods.
Changes the request URI before the router begins processing the request where:
url- the new request path value.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onRequest', function (request, reply) {
// Change all requests to '/test'
request.setUrl('/test');
return reply.continue();
});Available only in 'onRequest' extension methods.
Changes the request method before the router begins processing the request where:
method- is the request HTTP method (e.g.'GET').
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onRequest', function (request, reply) {
// Change all requests to 'GET'
request.setMethod('GET');
return reply.continue();
});Always available.
Logs request-specific events. When called, the server emits a 'request' event which can be used
by other listeners or plugins. The arguments are:
tags- a string or an array of strings (e.g.['error', 'database', 'read']) used to identify the event. Tags are used instead of log levels and provide a much more expressive mechanism for describing and filtering events.data- an optional message string or object with the application data being logged.timestamp- an optional timestamp expressed in milliseconds. Defaults toDate.now()(now).
Any logs generated by the server internally will be emitted only on the 'request-internal'
channel and will include the event.internal flag set to true.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.on('request', function (request, event, tags) {
if (tags.error) {
console.log(event);
}
});
var handler = function (request, reply) {
request.log(['test', 'error'], 'Test event');
return reply();
};Always available.
Returns an array containing the events matching any of the tags specified (logical OR) where:
tags- is a single tag string or array of tag strings. If notagsspecified, returns all events.internal- filters the events to only those with a matchingevent.internalvalue. Iftrue, only internal logs are included. Iffalse, only user event are included. Defaults to all events (undefined).
request.getLog();
request.getLog('error');
request.getLog(['error', 'auth']);
request.getLog(['error'], true);
request.getLog(false);Available until immediately after the 'response' event is emitted.
Adds a request tail which has to complete before the request lifecycle is complete where:
name- an optional tail name used for logging purposes.
Returns a tail function which must be called when the tail activity is completed.
Tails are actions performed throughout the request lifecycle, but which may end after a response is sent back to the client. For example, a request may trigger a database update which should not delay sending back a response. However, it is still desirable to associate the activity with the request when logging it (or an error associated with it).
When all tails completed, the server emits a 'tail' event.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
var get = function (request, reply) {
var dbTail = request.tail('write to database');
db.save('key', 'value', function () {
dbTail();
});
return reply('Success!');
};
server.route({ method: 'GET', path: '/', handler: get });
server.on('tail', function (request) {
console.log('Request completed including db activity');
});The request object supports the following events:
'peek'- emitted for each chunk of payload data read from the client connection. The event method signature isfunction(chunk, encoding).'finish'- emitted when the request payload finished reading. The event method signature isfunction ().'disconnect'- emitted when a request errors or aborts unexpectedly.
var Crypto = require('crypto');
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onRequest', function (request, reply) {
var hash = Crypto.createHash('sha1');
request.on('peek', function (chunk) {
hash.update(chunk);
});
request.once('finish', function () {
console.log(hash.digest('hex'));
});
request.once('disconnect', function () {
console.error('request aborted');
});
return reply.continue();
});The various request lifecycle events (e.g. extensions, authentication, route prerequisites, handlers) provide a reply interface as one of the function arguments. The reply interface acts as both a callback interface to return control to the framework and a response generator.
When reply() is called with an error or result response, that value is used as the response sent
to the client. When reply() is called within a prerequisite, the value is saves for future use
and is not used as the response. In all other places except for the handler, calling reply() will
be considered an error and will abort the request lifecycle, jumping directly
to the 'onPreResponse' event.
To return control to the framework within an extension or other
places other than the handler, without setting a response, the method
reply.continue() must be called.
Concludes the handler activity by setting a response and returning control over to the framework where:
err- an optional error response.result- an optional response payload.
Since an request can only have one response regardless if it is an error or success, the reply()
method can only result in a single response value. This means that passing both an err and
result will only use the err. There is no requirement for either err or result to be (or
not) an Error object. The framework will simply use the first argument if present, otherwise the
second. The method supports two arguments to be compatible with the common callback pattern of
error first.
Both err and result can be set to:
nullundefined- string
- number
- boolean
BufferobjectErrorobjectStreamobject- Promise object
- any other object or array
var handler = function (request, reply) {
return reply('success');
};If the input is not an Error object, the method returns a response object
which provides a set of methods to customize the response (e.g. HTTP status code, custom headers,
etc.). If the input is an Error object, the method returns back the error wrapped in a
Boom object.
Note that when used to return both an error and credentials in the authentication methods,
reply() must be called with three arguments function(err, null, data) where data is the
additional authentication information.
The response flow control rules apply.
// Detailed notation
var handler = function (request, reply) {
var response = reply('success');
response.type('text/plain');
response.header('X-Custom', 'some-value');
};
// Chained notation
var handler = function (request, reply) {
return reply('success')
.type('text/plain')
.header('X-Custom', 'some-value');
};Note that if result is a Stream with a statusCode property, that status code will be used as
the default response code.
Any value provided to reply() (including no value) will be used as the response sent back to the
client. This means calling reply() with a value in an
extension methods or authentication function will be considered
an error and will terminate the request lifecycle. With the exception of the
handler function, all other methods provide the reply.continue() method which instructs the
framework to continue processing the request without setting a response.
Every response includes the following properties:
statusCode- the HTTP response status code. Defaults to200(except for errors).headers- an object containing the response headers where each key is a header field name. Note that this is an incomplete list of headers to be included with the response. Additional headers will be added once the response is prepare for transmission.source- the value provided using the reply interface.variety- a string indicating the type ofsourcewith available values:'plain'- a plain response such as string, number,null, or simple object (e.g. not aStream,Buffer, or view).'buffer'- aBuffer.'view'- a view generated withreply.view().'file'- a file generated withreply.file()of via the directory handler.'stream'- aStream.'promise'- a Promise object.
app- application-specific state. Provides a safe place to store application data without potential conflicts with the framework. Should not be used by plugins which should useplugins[name].plugins- plugin-specific state. Provides a place to store and pass request-level plugin data. Thepluginsis an object where each key is a plugin name and the value is the state.settings- response handling flags:charset- the 'Content-Type' HTTP header 'charset' property. Defaults to'utf-8'.encoding- the string encoding scheme used to serial data into the HTTP payload whensourceis a string or marshals into a string. Defaults to'utf8'.passThrough- iftrueandsourceis aStream, copies thestatusCodeandheadersof the stream to the outbound response. Defaults totrue.stringify- options used forsourcevalue requiring stringification. Defaults to no replacer and no space padding.ttl- if set, overrides the route cache expiration milliseconds value set in the route config. Defaults to no override.varyEtag- iftrue, a suffix will be automatically added to the 'ETag' header at transmission time (separated by a'-'character) when the HTTP 'Vary' header is present.
The response object provides the following methods:
bytes(length)- sets the HTTP 'Content-Length' header (to avoid chunked transfer encoding) where:length- the header value. Must match the actual payload size.
charset(charset)- sets the 'Content-Type' HTTP header 'charset' property where:charset- the charset property value.code(statusCode)- sets the HTTP status code where:statusCode- the HTTP status code.
created(uri)- sets the HTTP status code to Created (201) and the HTTP 'Location' header where:uri- an absolute or relative URI used as the 'Location' header value.encoding(encoding)- sets the string encoding scheme used to serial data into the HTTP payload where:encoding- the encoding property value (see node Buffer encoding).etag(tag, options)- sets the representation entity tag where:tag- the entity tag string without the double-quote.options- optional settings where:weak- iftrue, the tag will be prefixed with the'W/'weak signifier. Weak tags will fail to match identical tags for the purpose of determining 304 response status. Defaults tofalse.vary- iftrueand content encoding is set or applied to the response (e.g 'gzip' or 'deflate'), the encoding name will be automatically added to the tag at transmission time (separated by a'-'character). Ignored whenweakistrue. Defaults totrue.
header(name, value, options)- sets an HTTP header where:name- the header name.value- the header value.options- optional settings where:append- iftrue, the value is appended to any existing header value usingseparator. Defaults tofalse.separator- string used as separator when appending to an exiting value. Defaults to','.override- iffalse, the header value is not set if an existing value present. Defaults totrue.
location(uri)- sets the HTTP 'Location' header where:uri- an absolute or relative URI used as the 'Location' header value.
redirect(uri)- sets an HTTP redirection response (302) and decorates the response with additional methods listed below, where:uri- an absolute or relative URI used to redirect the client to another resource.
replacer(method)- sets theJSON.stringify()replacerargument where:method- the replacer function or array. Defaults to none.
spaces(count)- sets theJSON.stringify()spaceargument where:count- the number of spaces to indent nested object keys. Defaults to no indentation.
state(name, value, [options])- sets an HTTP cookie where:name- the cookie name.value- the cookie value. If noencodingis defined, must be a string.options- optional configuration. If the state was previously registered with the server usingserver.state(), the specified keys inoptionsoverride those same keys in the server definition (but not others).
suffix(suffix)- sets a string suffix when the response is process viaJSON.stringify().ttl(msec)- overrides the default route cache expiration rule for this response instance where:msec- the time-to-live value in milliseconds.
type(mimeType)- sets the HTTP 'Content-Type' header where:value- is the mime type. Should only be used to override the built-in default for each response type.
unstate(name, [options])- clears the HTTP cookie by setting an expired value where:name- the cookie name.options- optional configuration for expiring cookie. If the state was previously registered with the server usingserver.state(), the specified keys inoptionsoverride those same keys in the server definition (but not others).
vary(header)- adds the provided header to the list of inputs affected the response generation via the HTTP 'Vary' header where:header- the HTTP request header name.
When using the redirect() method, the response object provides these additional methods:
temporary(isTemporary)- sets the status code to302or307(based on therewritable()setting) where:isTemporary- iffalse, sets status to permanent. Defaults totrue.
permanent(isPermanent)- sets the status code to301or308(based on therewritable()setting) where:isPermanent- iftrue, sets status to temporary. Defaults tofalse.
rewritable(isRewritable)- sets the status code to301/302for rewritable (allows changing the request method from 'POST' to 'GET') or307/308for non-rewritable (does not allow changing the request method from 'POST' to 'GET'). Exact code based on thetemporary()orpermanent()setting. Arguments:isRewritable- iffalse, sets to non-rewritable. Defaults totrue.
| Permanent | Temporary | |
|---|---|---|
| Rewritable | 301 | 302(1) |
| Non-rewritable | 308(2) | 307 |
Notes:
- Default value.
- Proposed code, not supported by all clients.
The response object supports the following events:
'peek'- emitted for each chunk of data written back to the client connection. The event method signature isfunction(chunk, encoding).'finish'- emitted when the response finished writing but before the client response connection is ended. The event method signature isfunction ().
var Crypto = require('crypto');
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onPreResponse', function (request, reply) {
var response = request.response;
if (response.isBoom) {
return reply();
}
var hash = Crypto.createHash('sha1');
response.on('peek', function (chunk) {
hash.update(chunk);
});
response.once('finish', function () {
console.log(hash.digest('hex'));
});
return reply.continue();
});hapi uses the boom error library for all its internal
error generation. boom provides an expressive interface to return HTTP errors. Any error
returned via the reply interface is converted to a boom object and defaults
to status code 500 if the error is not a boom object.
When the error is sent back to the client, the responses contains a JSON object with the
statusCode, error, and message keys.
var Hapi = require('hapi');
var Boom = require('boom');
var server = new Hapi.Server();
server.route({
method: 'GET',
path: '/badRequest',
handler: function (request, reply) {
return reply(Boom.badRequest('Unsupported parameter'));
}
});
server.route({
method: 'GET',
path: '/internal',
handler: function (request, reply) {
return reply(new Error('unexpect error'));
}
});Errors can be customized by changing their output content. The boom error object includes the
following properties:
isBoom- iftrue, indicates this is aBoomobject instance.message- the error message.output- the formatted response. Can be directly manipulated after object construction to return a custom error response. Allowed root keys:statusCode- the HTTP status code (typically 4xx or 5xx).headers- an object containing any HTTP headers where each key is a header name and value is the header content.payload- the formatted object used as the response payload (stringified). Can be directly manipulated but any changes will be lost ifreformat()is called. Any content allowed and by default includes the following content:statusCode- the HTTP status code, derived fromerror.output.statusCode.error- the HTTP status message (e.g. 'Bad Request', 'Internal Server Error') derived fromstatusCode.message- the error message derived fromerror.message.
- inherited
Errorproperties.
It also supports the following method:
reformat()- rebuildserror.outputusing the other object properties.
var Boom = require('boom');
var handler = function (request, reply) {
var error = Boom.badRequest('Cannot feed after midnight');
error.output.statusCode = 499; // Assign a custom error code
error.reformat();
error.output.payload.custom = 'abc_123'; // Add custom key
return reply(error);
});When a different error representation is desired, such as an HTML page or a different payload
format, the 'onPreResponse' extension point may be used to identify errors and replace them with
a different response object.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.views({
engines: {
html: require('handlebars')
}
});
server.ext('onPreResponse', function (request, reply) {
var response = request.response;
if (!response.isBoom) {
return reply.continue();
}
// Replace error with friendly HTML
var error = response;
var ctx = {
message: (error.output.statusCode === 404 ? 'page not found' : 'something went wrong')
};
return reply.view('error', ctx);
});When calling reply(), the framework waits until process.nextTick() to continue processing the
request and transmit the response. This enables making changes to the returned
response object before the response is sent. This means the framework
will resume as soon as the handler method exits. To suspend this behavior, the returned
response object supports the following methods:
hold()- puts the response on hold untilresponse.send()is called. Available only afterreply()is called and untilresponse.hold()is invoked once.send()- resume the response which will be transmitted in the next tick. Available only afterresponse.hold()is called and untilresponse.send()is invoked once.
var handler = function (request, reply) {
var response = reply('success').hold();
setTimeout(function () {
response.send();
}, 1000);
};Returns control back to the framework without setting a response. If called in the handler, the
response defaults to an empty payload with status code 200. The data argument is only used for
passing back authentication data and is ignored elsewhere.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.ext('onRequest', function (request, reply) {
// Change all requests to '/test'
request.setUrl('/test');
return reply.continue();
});Transmits a file from the file system. The 'Content-Type' header defaults to the matching mime type based on filename extension.:
path- the file path.options- optional settings:filename- an optional filename to specify if sending a 'Content-Disposition' header, defaults to the basename ofpathmode- specifies whether to include the 'Content-Disposition' header with the response. Available values:false- header is not included. This is the default value.'attachment''inline'
lookupCompressed- iftrue, looks for the same filename with the '.gz' suffix for a pre-compressed version of the file to serve if the request supports content encoding. Defaults tofalse.
No return value.
The response flow control rules do not apply.
var handler = function (request, reply) {
return reply.file('./hello.txt');
};Concludes the handler activity by returning control over to the router with a templatized view response where:
template- the template filename and path, relative to the templates path configured via the server views manager.context- optional object used by the template to render context-specific result. Defaults to no context{}.options- optional object used to override the server's views manager configuration for this response. Cannot overrideisCached,partialsPath, orhelpersPathwhich are only loaded at initialization.
Returns a response object.
The response flow control rules apply.
var Hapi = require('hapi');
var server = new Hapi.Server();
server.connection({ port: 80 });
server.views({
engines: { html: require('handlebars') },
path: __dirname + '/templates'
});
var handler = function (request, reply) {
var context = {
title: 'Views Example',
message: 'Hello, World'
};
return reply.view('hello', context);
};
server.route({ method: 'GET', path: '/', handler: handler });templates/hello.html
<!DOCTYPE html>
<html>
<head>
<title>{{title}}</title>
</head>
<body>
<div>
<h1>{{message}}</h1>
</div>
</body>
</html>Concludes the handler activity by returning control over to the router and informing the router
that a response has already been sent back directly via request.raw.res and that no further
response action is needed. Supports the following optional options:
end- iffalse, the router will not callrequest.raw.res.end())to ensure the response was ended. Defaults totrue.
No return value.
The response flow control rules do not apply.
Proxies the request to an upstream endpoint where:
options- an object including the same keys and restrictions defined by the routeproxyhandler options.
No return value.
The response flow control rules do not apply.
var handler = function (request, reply) {
return reply.proxy({ host: 'example.com', port: 80, protocol: 'http' });
};Redirects the client to the specified uri. Same as calling reply().redirect(uri).
Returns a response object.
The response flow control rules apply.
var handler = function (request, reply) {
return reply.redirect('http://example.com');
};Changing to a permanent or non-rewriterable redirect is also available see response object redirect for more information.