pogo
Server framework for Deno
Pogo is an easy to use, safe, and expressive framework for writing web servers and applications. It is inspired by hapi.
Supports Deno v0.19.0 and higher.
Contents
Why?
- Designed to encourage reliable and testable applications.
- Routes are simple, pure functions that return values directly.
- Automatic JSON responses from objects.
Usage
import pogo from 'https://deno.land/x/pogo/main.js';
const server = pogo.server({ port : 3000 });
server.route({
method : 'GET',
path : '/',
handler() {
return 'Hello, world!';
}
});
server.start();
Adding routes
Adding routes is easy, just call server.route()
and pass it a single route or an array of routes. You can call server.route()
multiple times. You can even chain calls to server.route()
, because it returns the server instance.
Add routes in any order you want to, it’s safe! Pogo orders them internally by specificity, such that their order of precedence is stable and predictable and avoids ambiguity or conflicts.
server.route({ method : 'GET', path : '/hi', handler : () => 'Hello!' });
server.route({ method : 'GET', path : '/bye', handler : () => 'Goodbye!' });
server
.route({ method : 'GET', path : '/hi', handler : () => 'Hello!' });
.route({ method : 'GET', path : '/bye', handler : () => 'Goodbye!' });
server.route([
{ method : 'GET', path : '/hi', handler : () => 'Hello!' },
{ method : 'GET', path : '/bye', handler : () => 'Goodbye!' }
});
You can also configure the route to handle multiple methods by using an array, or *
to handle all possible methods.
server.route({ method : ['GET', 'POST'], path : '/hi', handler : () => 'Hello!' });
server.route({ method : '*', path : '/hi', handler : () => 'Hello!' });
Writing tests
When it comes time to write tests for your app, Pogo has you covered with server.inject()
.
By injecting a request into the server directly, we can completely avoid the need to listen on an available port, make HTTP connections, and all of the problems and complexity that come along with that. You should focus on writing your application logic and server.inject()
makes that easier.
The server still processes the request using the same code paths that a normal HTTP request goes through, so you can rest assured that your tests are meaningful and realistic.
const response = await server.inject({
method : 'GET',
url : '/users'
});
API
server = pogo.server(option)
Returns a server instance, which can then be used to add routes and start listening for requests.
option
Type: object
hostname
Type: string
Default: 'localhost'
Specifies which domain or IP address the server will listen on when server.start()
is called. Use 0.0.0.0
to listen on any hostname.
port
Type: number
Example: 3000
Specifies which port number the server will listen on when server.start()
is called. Use 0
to listen on any available port.
Server
The server
object returned by pogo.server()
represents your web server. When you start the server, it begins listening for HTTP requests, processes those requests when they are received, and makes the content within each request available to the route handlers that you specify.
server.inject(request)
Performs a request directly to the server without using the network. Useful when writing tests, to avoid conflicts from multiple servers trying to listen on the same port number.
Returns a Promise
for a Response
instance.
request
Type: object
method
Any valid HTTP method, such as GET
or POST
. Used to lookup the route handler.
url
Any valid URL path. Used to lookup the route handler.
server.route(option)
Adds a route to the server so that the server knows how to respond to requests for the given HTTP method and path, etc.
option
Type: object
or array<object>
method
Type: string
or array<string>
Example: GET
Any valid HTTP method, or *
to match all methods. Used to limit which requests will trigger the route handler.
path
Type: string
Example: '/users/{userId}'
Any valid URL path. Used to limit which requests will trigger the route handler.
Supports path parameters with dynamic values, which can be accessed in the handler as request.params
.
handler(request, h)
Type: function
request
is a Request instance with properties forheaders
,method
,url
, and more.h
is a Response Toolkit instance, which has utility methods for modifying the response.
The implementation for the route that handles requests. Called when a request is received that matches the method
and path
specified in the route options.
The handler must return one of:
- A
string
, which will be sent as HTML. - An
object
, which will be stringified and sent as JSON. - A
Uint8Array
, which will be sent as-is (raw bytes). - A Response instance, which will send the
response.body
, if any. - Any object that implements the
Reader
interface, such as aFile
orBuffer
instance. - A
Promise
for any of the above types.
An appropriate Content-Type
header will be set automatically based on the response body before the response is sent. You can use response.type()
to override the default behavior.
server.start()
Begins listening on the hostname
and port
specified in the server options.
Returns a Promise
that resolves when the server is listening.
Request
The request
object passed to route handlers represents an HTTP request that was sent to the server. It is similar to an instance of Deno’s ServerRequest
class, with some additions.
It provides properties and methods for inspecting the content of the request.
request.body()
Returns a Promise
for a Uint8Array
containing the raw bytes of the request body.
Note that calling this method will cause the entire body to be read into memory, which is convenient, but may be inappropriate for requests with a very large body. See request.bodyStream()
to get the body as a stream, which will improve latency and lower memory usage.
request.bodyStream()
Returns an AsyncGenerator
object, which is an async iterable and async iterator that provides the raw bytes of the request body in chunks. Useful for requests with a very large body, because streaming is low-latency and memory efficient.
See request.body()
to get the entire body all at once.
request.headers
Type: Headers
Contains the HTTP headers that were sent in the request, such as Accept
, User-Agent
, and others.
request.host
Type: string
Example: 'localhost:3000'
The value of the HTTP Host
header, which is a combination of the hostname and port at which the server received the request, separated by a :
colon. Useful for returning different content depending on which URL your visitors use to access the server. Shortcut for request.url.host
.
To get the hostname, which does not include the port number, see request.hostname
.
request.hostname
Type: string
Example: 'localhost'
The hostname part of the HTTP Host
header. That is, the domain or IP address at which the server received the request, without the port number. Useful for returning different content depending on which URL your visitors use to access the server. Shortcut for request.url.hostname
.
To get the host, which includes the port number, see request.host
.
request.href
Type: string
Example: 'http://localhost:3000/page.html?query'
The full URL associated with the request, represented as a string. Shortcut for request.url.href
.
To get this value as a parsed object instead, use request.url
.
request.method
Type: string
Example: 'GET'
The HTTP method associated with the request, such as GET
or POST
.
request.origin
Type: string
Example: 'http://localhost:3000'
The scheme and host parts of the request URL. Shortcut for request.url.origin
.
request.params
Type: object
Contains the dynamic variables of the path
in the route configuration, where each key is a variable name and the value is the corresponding part of the request path.
request.path
Type: string
Example: /page.html
The path part of the request URL, excluding the query. Shortcut for request.url.pathname
.
request.raw
Type: ServerRequest
The original request object from Deno’s http
module, upon which many of the other request properties are based.
You probably don’t need this. It is provided as an escape hatch, but using it is not recommended.
request.referrer
Type: string
The value of the HTTP Referer
header, which is useful for determining where the request came from. However, not all user agents send a referrer and the value can be influenced by various mechanisms, such as Referrer-Policy
. As such, it is recommended to use the referrer as a hint, rather than relying on it directly.
Note that this property uses the correct spelling of “referrer”, unlike the header. It will be an empty string if the header is missing.
request.response
Type: Response
The response that will be sent for the request. To create a new response, see h.response()
.
request.search
Type: string
Example: '?query'
The query part of the request URL, represented as a string. Shortcut for request.url.search
.
To get this value as a parsed object instead, use request.searchParams
.
request.searchParams
Type: URLSearchParams
The query part of the request URL, represented as an object that has methods for working with the query parameters. Shortcut for request.url.searchParams
.
To get this value as a string instead, use request.search
.
request.server
Type: Server
The server that is handling the request.
request.url
Type: URL
The full URL associated with the request, represented as an object that contains properties for various parts of the URL,
To get this value as a string instead, use request.href
. In some cases, the URL object itself can be used as if it were a string, because it has a smart .toString()
method.
Response
The response
object represents an HTTP response to the associated request
that is passed to route handlers. You can access it as request.response
or create a new response with the Response Toolkit by calling h.response()
. It has utility methods that make it easy to modify the headers, status code, and other attributes.
response.body
Type: string
| object
| Uint8Array
| Reader
The body that will be sent in the response. Can be updated by returning a value from the route handler or by creating a new response with h.response()
and giving it a value.
response.code(statusCode)
Sets the response status code. When possible, it is better to use a more specific method instead, such as response.created()
or response.redirect()
.
Tip: Use Deno’s status
constants to define the status code.
import { Status as status } from 'https://deno.land/std/http/http_status.ts';
const handler = (request) => {
return request.response.code(status.Teapot);
};
response.created(url)
Sets the response status to 201 Created
and sets the Location
header to the value of url
.
Returns the response so other methods can be chained.
response.header(name, value)
Sets a response header. Always replaces any existing header of the same name. Headers are case insensitive.
Returns the response so other methods can be chained.
response.headers
Type: Headers
Contains the HTTP headers that will be sent in the response, such as Location
, Vary
, and others.
response.location(url)
Sets the Location
header on the response to the value of url
. When possible, it is better to use a more specific method instead, such as response.created()
or response.redirect()
.
Returns the response so other methods can be chained.
response.permanent()
Only available after calling the response.redirect()
method.
Sets the response status to 301 Moved Permanently
or 308 Permanent Redirect
based on whether the existing status is considered rewritable (see “method handling” on Redirections in HTTP for details).
Returns the response so other methods can be chained.
response.redirect(url)
Sets the response status to 302 Found
and sets the Location
header to the value of url
.
Also causes some new response methods to become available for customizing the redirect behavior:
Returns the response so other methods can be chained.
response.rewritable(isRewritable)
Only available after calling the response.redirect()
method.
Sets the response status to 301 Moved Permanently
or 302 Found
based on whether the existing status is a permanent or temporary redirect code. If isRewritable
is false
, then the response status will be set to 307 Temporary Redirect
or 308 Permanent Redirect
.
Returns the response so other methods can be chained.
response.status
Type: number
Example: 418
The status code that will be sent in the response. Defaults to 200
, which means the request succeeded. 4xx and 5xx codes indicate an error.
response.temporary()
Only available after calling the response.redirect()
method.
Sets the response status to 302 Found
or 307 Temporary Redirect
based on whether the existing status is considered rewritable (see “method handling” on Redirections in HTTP for details).
Returns the response so other methods can be chained.
response.type(mediaType)
Sets the Content-Type
header on the response to the value of mediaType
.
Overrides the media type that is set automatically by the framework.
Returns the response so other methods can be chained.
Response Toolkit
The response toolkit is an object that is passed to route handlers, with utility methods that make it easy to modify the response. For example, you can use it to set headers or a status code.
By convention, this object is assigned to a variable named h
in code examples.
h.redirect(url)
Creates a new response with a redirect status. Shortcut for h.response().redirect(url)
. See response.redirect()
for details.
Returns the response so other methods can be chained.
h.response(body)
Creates a new response with an optional body. This is the same as returning the body directly from the route handler, but it is useful in order to begin a chain with other response methods.
Returns the response so other methods can be chained.
Contributing
See our contributing guidelines for more details.
- Fork it.
- Make a feature branch:
git checkout -b my-new-feature
- Commit your changes:
git commit -am 'Add some feature'
- Push to the branch:
git push origin my-new-feature
- Submit a pull request.
License
Go make something, dang it.