Your go to library for building Azure Functions with TypeScript!
A magical TypeScript utility belt for Azure Functions developers who are tired of writing the same boilerplate code over and over again. This library will make your functions more readable, maintainable, and less prone to "It works on my machine" syndrome.
npm install azure-functions-essentials
or if you're feeling yarn-ish
yarn add azure-functions-essentials
Chain your function logic like a boss. Found a guard or an input that should be reused in other functions? Declare as a const and reuse the logic!
import { startChain, guard, funcResult, inputFactory } from 'azure-functions-essentials';
// Create an input binding for user lookup with your fancy database call
const userLookup = inputFactory<string, User>('user', userId => getUserFromDatabase(userId));
// Make sure they have the magic password
const keyGuard = guard(req => req.headers.get('x-api-key') === 'secret' || funcResult('Forbidden', 'Nice try, hacker!'));
// Make sure the user is a ninja (Guard functor pattern)
const ninjaGuard = (user: User) => guard(() => user.isNinja || funcResult('Forbidden', 'Only π₯·s are allowed!'));
app.post('super-secret-endpoint', {
handler: startChain()
.useGuard(keyGuard) // Use a guard (or several?)
.parseBody(myZodSchema) // Parse the body and (optionally) validate with Zod
.useInputBinding(({ body }) => userLookup.create(body.user.id)) // Initialize the input
.useGuard(({ context }) => ninjaGuard(userLookup.get(context))) // Use input results in the chain
// Handle the request with all goodies available
.handle((req, body, ctx) => {
const user = userLookup.get(ctx); // Get the user from our input
// Your incredibly important business logic here
// Return a result with the funcResult helper
return funcResult('OK', {
message: `You're in ${user.name}! Here's your cookie πͺ`,
userData: user,
requestData: body,
});
}),
});
For more details and the list of built-in guards, check out the Guards Documentation.
Keep the bad guys out:
const isAdminGuard = guard((req, ctx) => {
const user = ctx.extraInputs.get('user');
return user.role === 'admin' || funcResult('Forbidden', 'You shall not pass! π§ββοΈ');
});
Because more inputs means more fun:
const userLookup = inputFactory<string, User>('user', async userId => {
// Imagine some fancy database call here
return { id: userId, name: 'Function Fanatic', role: 'wizard' };
});
// Later in your chain...
.useInputBinding(({ request }) => userLookup.create(request.params.userId))
Parse query params without pulling your hair out:
import { getQuery, getQueryFlag } from 'azure-functions-essentials';
const limit = getQuery(request, 'limit'); // Will throw if missing
const page = getQuery(request, 'page', true); // Optional, returns null if missing
const includeDeleted = getQueryFlag(request, 'includeDeleted'); // Boolean flags made easy
- Less Code: Why write 100 lines when 10 will do?
- Type Safety: TypeScript all the things!
- Testability: Every component is designed to be easily testable
- Chain of Responsibility: Handle authentication, validation, and business logic in a clean, readable way
- Consistent Responses: No more forgetting to set the right status code
For more examples of our wizardry, check out these magical spells:
Guards can directly check headers, query parameters, or any request property:
// Simple header-based authentication guard
const apiKeyGuard = guard((req, ctx) => {
const apiKey = req.headers.get('x-api-key');
return apiKey === process.env.API_KEY || funcResult('Unauthorized', 'Invalid API key');
});
// Usage in chain
startChain()
.useGuard(apiKeyGuard)
.handle((req, ctx) => {
// Only executed if API key is valid
return funcResult('OK', 'Access granted!');
});
Guards can also be created on-the-fly with the functor pattern, allowing you to pass in parameters:
type BodyType = { name: string; age: number };
// Guard factory that validates a specific property in the body
const validateBody = (body: BodyType) => guard((req, ctx) => body.age < 18 || funcResult('BadRequest', `Age must be at least 18 years old`));
// Usage with parsed body
startChain()
.parseBody<BodyType>()
.useGuard(({ body }) => validateBody(body)) // Pass the body to validateBody
.handle((req, body, ctx) => {
// Only executed if the body validator is passed
return funcResult('OK', `Welcome, ${body.name}!`);
});
In some scenarios, it is enough for just one of the guards to pass. For example, admins are allowed to edit any resource, while regular users can only edit their own resources.
const combinedGuard = anyGuard(isAdminGuard, hasPermissionGuard('CAN_EDIT'), isResourceOwnerGuard);
const userSchema = z.object({
name: z.string().min(2),
email: z.string().email("That doesn't look like an email to me π€¨"),
age: z.number().min(18).optional(),
});
startChain()
.parseBody(userSchema)
.handle((req, body, ctx) => {
// body is typed and validated!
});
Pull requests are welcome! Fork it and PR away!
MIT License - Use it, abuse it, but please give credit where it's due.
May your functions be stateless and your deployments be seamless!