Skip to content

AlexPshul/azure-functions-essentials

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 

History

40 Commits
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

⚑ Azure Functions Essentials

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.

πŸš€ Installation

npm install azure-functions-essentials

or if you're feeling yarn-ish

yarn add azure-functions-essentials

✨ Features

πŸ”— Function Chains

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,
      });
    }),
});

πŸ›‘οΈ Guards

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! πŸ§™β€β™‚οΈ');
});

🎯 Input Bindings

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))

🧩 Query Parameters

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

πŸ€” Why Use This Library?

  1. Less Code: Why write 100 lines when 10 will do?
  2. Type Safety: TypeScript all the things!
  3. Testability: Every component is designed to be easily testable
  4. Chain of Responsibility: Handle authentication, validation, and business logic in a clean, readable way
  5. Consistent Responses: No more forgetting to set the right status code

πŸ“š Documentation

For more examples of our wizardry, check out these magical spells:

Creating Guards

Regular Guards

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!');
  });

Functor Pattern Guards

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}!`);
  });

Combined Guards with useAnyGuard

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);

Body Validation with Zod

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!
  });

🀝 Contributing

Pull requests are welcome! Fork it and PR away!

πŸ“ License

MIT License - Use it, abuse it, but please give credit where it's due.


Function wizard
May your functions be stateless and your deployments be seamless!

About

Tools that every NodeJS Azure Functions project needs!

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages