A router for Single-Function APIs in ⚡ Azure Functions.
Azure Monofunction is a router for Single-Function APIs (SFA) that makes possible for you to develop monolithic APIs and still use the power of serverless, like cost-per-consumption and automatic scalability.
See all features.
Azure Monofunction was tested for the environments below. Even we believe it may works in older versions or other platforms, it is not intended to.
See tested environments
Environment | Tested version |
---|---|
OS | Ubuntu 20.04 |
Node.js | 12.16.3 |
Package Manager | npm 6.14.5 |
Platforms | Azure Functions Host v2 and v3 |
$ npm install --save azure-monofunction
-
Create a HTTP trigger in you function app. It can have any name, like "monofunction".
-
Change the
route
property for the trigger binding infunction.json
to a wildcard, like{*segments}
. You can copy the function.json template. -
Create the function
index.js
at the function folder and start develop following the basic usage below.
The most simple usage
const app = require('azure-monofunction');
const routes = [{
path: '/example/:param',
methods: ['GET', 'PUT'],
run: async (context) => {
context.res.body = { it: 'works' };
},
}]
app.addRoutes(routes);
module.exports = app.listen();
The anatomy of a simple route is an URL path
, its HTTP methods
to match and some middleware to run
.
An alternative to the addRoutes
method that supports a list of route objects is the method route
that adds a single route to the router.
app.route(['GET','POST'], '/example/path', async (context) => {
// ...middleware logic
});
Route paths must always start with /
and be followed by valid URL paths.
You can define dynamic parameters prefixing a path level with :
.
For example, the URL /users/123
will match the route /users/:id
.
The parameter will be available at the context.params
object.
Example:
app.route(['GET'], '/users/:id', async (context) => {
const userId = context.params.id;
});
HTTP verbs must always be defined as arrays of strings both in route objects and route
method.
The supported HTTP verbs are: ['GET', 'POST', 'PATCH', 'PUT', 'DELETE']
.
Additionally, you can add routes with verbs corresponding methods: get()
, post()
, patch()
, put()
, delete()
.
app.addRoutes([{
methods: ['GET', 'POST'],
path,
run,
}]);
// is the same of:
app.route(['GET', 'POST'], path, run);
// and the same of:
app.get(path, run); // feels like Express 🙂
Middlewares works with the power of Azure Functions Middlewares and are asynchronous functions that takes a required context
argument and an optional STOP_SIGNAL
argument.
Request object is available through context.req
property and response should be set using the context.res
property. Other Azure Functions context properties are also available.
When defining a middleware for a route with the run
property/argument, you can use a single middleware function or an array of middlewares.
Example:
const headerMiddleware = async (context) => {
context.res.headers['X-User-ID'] = context.params.id;
context.anything = true;
};
const resourceMiddleware = async (context) => {
context.res.body = { added: context.anything };
};
app.get('/posts/:id', [headerMiddleware, resourceMiddleware]);
// output of /posts request should be
// HTTP 200 /posts/123
// X-User-ID: 123
// { "added": true }
It is strongly recommended you read Azure Functions Middlewares docs, but, if you can't, please note:
- Always use asynchronous functions as middlewares.
- Do not return anything inside your middleware function, unless you want to throw an error.
- You can pass values to the next middlewares using the
context
object reference.- Return the
STOP_SIGNAL
argument in the middleware if you need to prevent any following middleware to be executed. This is useful for Content-type negotiation or Authorization.- See other context-related docs in Accessing and modifying the context at the Azure Functions Middlewares reference.
- Common community middlewares are available under the Azure Functions Middlewares project.
Using a single-function API leads you back to the need of middleware reusability and composition.
This monolithic approach is more maintainable than microservices approach, but requires more organization.
When dealing with many routes and its middlewares, you certainly will fall to a controller separation pattern.
Controllers are often separated by resource entities or related services.
It is also good to separate your routes inside a route file for better reading.
Example:
Function directory tree
.
├── controllers
│ └── user.controller.js
├── monofunction
│ ├── function.json
│ ├── index.js
│ └── routes.js
└── host.json
/controllers/user.controller.js
async function getUser({ req, res, params }) {
context.res.body = await db.getUser(params.id);
}
async function createUser({ req }) {
context.res.status = await db.createUser(req.body);
}
module.exports = {
getUser,
createUser,
};
/monofunction/routes.js
const userController = require('../controllers/user.controller');
module.exports = [{
path: '/users/:id',
methods: ['GET'],
run: userController.getUser,
}, {
path: '/users',
methods: ['POST'],
run: userController.createUser,
}];
/monofunction/index.js
const app = require('azure-monofunction');
const routes = require('./routes');
app.addRoutes(routes);
module.exports = app.listen();
If a middleware returns a value or throws an error, an error will also be forwarded to Azure Functions Host execution.
If you want to catch this unhandled errors, you can use the onError
handler.
app.onError((error, context) => {
// error argument contains the original value
// context is preserved until here
});
Sometimes you need to run common middlewares for all the routes, regardless its resources, just like headers validation, authorization rules and content-type negotiation and parsing.
You can add middlewares for all the routes ("global middlewares") using the use()
method.
app.use(async (context) => {
context.res.headers['X-Powered-By'] = '⛈ Azure Monofunction';
});
app.addRoutes(routes);
// now all the routes will respond with a 'X-Powered-By' header
You can also add a conditional global middleware calling the useIf()
method from Azure Functions Middlewares.
const isChromeAgent = (context) => {
return context.req.headers['User-Agent'].indexOf('Chrome') !== -1;
}
app.useIf(isChromeAgent, async (context) => {
context.res.headers['Content-Type'] = 'text/html';
});
// now if a route was called from Chrome browser, the response will be set to HTML COntent-Type
Find useful built middlewares in Common community middlewares of Azure Functions Middlewares.
If no route was matched during a request, Azure Monofunction will throw a not found error.
But if you want to handle this not found event with a fallback route, you can use the on404()
handler.
app.on404((context) => {
// add your fallback logic here, like:
context.res.status = 404;
});
You can add custom route metadata in route object's meta
property that will be available in context.meta
property:
const routes = [{
path: '/route',
methods,
meta: {
something: 'value',
},
run: async (context) => {
context.res.body = context.meta;
// body will be { "something": "value" }
},
}];
This is useful when you need to recover this meta in other middlewares, specially conditional middlewares, like an authorization middleware:
const hasAuth = (context) => context.meta && context.meta.auth;
app.useIf(hasAuth, async (context, STOP_SIGNAL) => {
if (!context.req.headers.Authorization) {
context.res.status = 401;
return STOP_SIGNAL;
}
});
app.addRoutes([{
path: '/resource',
methods: ['POST'],
meta: {
auth: true
},
run,
}, {
path: '/resource',
methods: ['GET'],
run,
}]);
// POST /resource without an Authorization header will return HTTP 401
// but GET /resource will not
You can log everything that is done by Azure Monofunction setting the debug
property to true
.
app.debug = true;
You can also use a different logger than console/context setting it in the logger
property.
app.logger = MyCustomLogger;
ℹ Note that your logger need to have a
log(message, ...args)
method.
Azure Functions Host has a route prefix for all the requests. This defaults to /api
but you can customize it in host.json
:
{
"extensions": {
"http": {
"routePrefix": "api",
}
}
}
If you customize the route prefix, Azure Monofunction will try to guess it from extensions.http.routePrefix
configuration defined in host.json
file.
If in a parallel universe you did not defined the route prefix in host.json
but your function app has a route prefix different than /api
, you need to specify that in the Azure Monofunction's routePrefix
property.
app.routePrefix = '/notapi';
Make sure if you need to change it you did this changing before adding any route.
Azure Monofunction is not intended to be extensible, but the middleware approach is extensible itself.
If you want to publish a middleware (or an evaluation function) you developed and think it will be useful for any other developer, see Writing and publishing common middlewares in Azure Functions Middlwares.
If you need help or have a problem with this project and you not found you problem in FAQ above, start an issue.
We will not provide a SLA to your issue, so, don't expect it to be answered in a short time.
path
_string_
The path should be matched for the route.
methods
_string[]_
The HTTP verbs that should be matched for the route.
run
_AsyncFunctionGenerator|AsyncFunctionGenerator[]_
A single middleware function or an array of middleware functions.
A middleware should be in form of async function (context, STOP_SIGNAL?):any
, as documented in asyncMiddleware
specification.
meta
_object_={}
Custom route metadata to be available at
context.meta
property.
debug
_boolean_=false
Determines if Azure Monofunctions operations should be logged or not.
logger
_{ log: function }_=console
The logger object containing a
log(message, ...args)
function that will be used for logging messages.
routePrefix
_string_="/api"
The route prefix that will be used in route matching.
addRoutes()
function(routes):void
Adds a list of routes to the monofunction router.
Arguments
Argument | Type | Required | Default | Description |
---|---|---|---|---|
routes | AzureMonofunctionRoute[] |
true | A list of valid routes. |
get()
, post()
, patch()
, put()
, delete()
function(path, middlewares):void
Adds a single route to the monofunction router for HTTP verb corresponding to the method name.
It is an alias for the method route()
, but with the argument methods
already defined.
listen()
function():AzureFunction
Returns the Azure Functions entrypoint
async (context) => {}
that will be called by the function HTTP trigger and will execute the entire router.
Returns
AzureFunction
: the Azure Functions entrypoint.
onError()
function(catchCallback):void
Registers an error callback to be called when a middleware throws an error.
Arguments
Argument | Type | Required | Default | Description |
---|---|---|---|---|
catchCallback | function |
true | A callback function that takes two arguments (error, context) |
Callbacks
function (error, context):any
Argument | Type | Required | Default | Description |
---|---|---|---|---|
error | `Error | any` | true | |
context | Context ℹ |
true | The Azure Function context object. |
Returns: anything returned by the callback will be ignored.
on404()
function(notFoundHandler):void
Registers a fallback handler to be called when no route matches the current URL.
Arguments
Argument | Type | Required | Default | Description |
---|---|---|---|---|
notFoundHandler | function |
true | A callback function that takes one argument (context) |
Callbacks
function (context):void
Argument | Type | Required | Default | Description |
---|---|---|---|---|
context | Context ℹ |
true | The Azure Function context object. |
Returns: anything returned by the callback will be ignored.
route()
function(methods, path, middlewares):void
Adds a single route to the monofunction router.
Arguments
Argument | Type | Required | Default | Description |
---|---|---|---|---|
methods | `Array<'GET' | 'POST' | 'PATCH' | 'PUT' |
path | string |
true | The URL path for the route. | |
middlewares | `AsyncGeneratorFunction | AsyncGeneratorFunction[]` | true | default |
use()
function(asyncMiddleware, phase?):void
Adds a global middleware to be executed for every matched route.
Identical to use()
method from Azure Functions Middlewares.
useIf()
function(expression, asyncMiddleware, phase?):void
Adds a global middleware to be conditionally executed for every matched route if the specified expression returns
true
.
Identical to useIf()
method from Azure Functions Middlewares.
Azure Monofunction borned to help developers use the power of serverless in Azure Functions but with minimal complexity with function management.
When using serverles, developers often end up with a microservices architecture, even with functions in the same Function App. Each function requires a HTTP binding, your own endpoint and an endpoint maps only to a single resource.
This complexity leads developers to create many triggers and functions to simple resource APIs, like CRUD APIs: you will ever need at least two functions (one for /resource
and one for /resource/:id
) and at least five if
clauses in these functions to make it possible.
Then, in the end you always end up with two options: mix logic and confuse next API maintainers or deal with a lot of functions for basic operations that could be aggregated.
You probably experienced this, as we experienced at NOALVO, and you sure were not satisfied with any of these two approaches.
If you want to keep up with Azure Functions powers, like cost-per-consumption and elastic, automatic scalability, now you can build a monolithic API architecture, just like in Express, Koa or HAPI, using Azure Monofunction.
Azure Monofunction was inspired specially from Express monolithic APIs.
Curiosity: Azure is the cloud ☁, Functions is the flash ⚡, Azure Functions Middlewares is the thunder 🌩 and Azure Monofunction is the thunderstorm ⛈.
- Multilevel route matching
- Dynamic route params
- Route HTTP verbs
- Route middleware
- Route metadada
- Multiple middlewares (cascade) per route
- Global standard and conditional middlewares
- Error handler middleware
- Not found/404 handler middleware
Help us spreading the word or consider making a donation.
Add your company name to the Who is using secion
Make a pull request or start an issue to add your company's name.
We follow Contributor Covenant Code of Conduct. If you want to contribute to this project, you must accept and follow it.
This project adheres to Semantic Versioning 2.0.0.
If you are not solving an issue or fixing a bug, you can help developing the roadmap below.
See the roadmap
- Improve documentation
- Conditional middlewares for routes
- Async error and not found handlers
Licensed under the MIT License.