Last active
February 18, 2025 15:49
-
Star
(453)
You must be signed in to star a gist -
Fork
(126)
You must be signed in to fork a gist
-
-
Save joshnuss/37ebaf958fe65a18d4ff to your computer and use it in GitHub Desktop.
Express.js role-based permissions middleware
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// the main app file | |
import express from "express"; | |
import loadDb from "./loadDb"; // dummy middleware to load db (sets request.db) | |
import authenticate from "./authentication"; // middleware for doing authentication | |
import permit from "./authorization"; // middleware for checking if user's role is permitted to make request | |
const app = express(), | |
api = express.Router(); | |
// first middleware will setup db connection | |
app.use(loadDb); | |
// authenticate each request | |
// will set `request.user` | |
app.use(authenticate); | |
// setup permission middleware, | |
// check `request.user.role` and decide if ok to continue | |
app.use("/api/private", permit("admin")); | |
app.use(["/api/foo", "/api/bar"], permit("manager", "employee")); | |
// setup requests handlers | |
api.get("/private/whatever", (req, res) => res.json({whatever: true})); | |
api.get("/foo", (req, res) => res.json({currentUser: req.user})); | |
api.get("/bar", (req, res) => res.json({currentUser: req.user})); | |
// setup permissions based on HTTP Method | |
// account creation is public | |
api.post("/account", (req, res) => res.json({message: "created"})); | |
// account update & delete (PATCH & DELETE) are only available to managers | |
api.patch("/account", permit('manager'), (req, res) => res.json({message: "updated"})); | |
api.delete("/account", permit('manager'), (req, res) => res.json({message: "deleted"})); | |
// viewing account "GET" available to manager and employee | |
api.get("/account", permit('manager', 'employee'), (req, res) => res.json({currentUser: req.user})); | |
// mount api router | |
app.use("/api", api); | |
// start 'er up | |
app.listen(process.env.PORT || 3000); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// middleware for authentication | |
export default async function authorize(request, _response, next) { | |
const apiToken = request.headers['x-api-token']; | |
// set user on-success | |
request.user = await request.db.users.findByApiKey(apiToken); | |
// always continue to next middleware | |
next(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// dummy middleware for db (set's request.db) | |
export default function loadDb(request, _response, next) { | |
// dummy db | |
request.db = { | |
users: { | |
findByApiKey: async token => { | |
switch { | |
case (token == '1234') { | |
return {role: 'manager', id: 1234}; | |
case (token == '5678') { | |
return {role: 'employee', id: 5678}; | |
default: | |
return null; // no user | |
} | |
} | |
} | |
}; | |
next(); | |
} |
I got an error. Please advice.
import authenticate from "./authentication.js"; // middleware for doing authentication ^^^^^^
SyntaxError: Cannot use import statement outside a module at Object.compileFunction (vm.js:344:18)Take a look at https://stackoverflow.com/questions/58384179/syntaxerror-cannot-use-import-statement-outside-a-module or update your syntax to use require. For example, const {authorize} = require('./authentication.js')
Thank you.
Thank you , this is very helpful
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Take a look at https://stackoverflow.com/questions/58384179/syntaxerror-cannot-use-import-statement-outside-a-module or update your syntax to use require. For example, const {authorize} = require('./authentication.js')