Hi! I've been fighting this for a bit, but I'm trying to gather a list of all of the endpoints in our express applications. I've tried 'express-list-router' and the code below, but none give the full paths of the endpoint and the allowed methods to use them. I've read the documentation, but can't seem to find the properties/methods to access this information. Thank you in advance for your time and assistance!

It is not possible to get a fill list with the built-in router. Since paths are regular expression-based, there is no easy way in which to reverse them into a path.
Hmmm, I'm auditing a public facing application. Tons of repos and submodules, it would be almost laughable to do it manually. Is my only option then to catch them downstream via: req.method and req.originalUrl?
**Creative and/or potentially crazy solutions are welcome :)
There is no general solution that would guarantee you are not missing anything. Is it possible to share the app at all? We could at least see what kind of structure the program is using to see if it's possible at all. What you posted at first is a pretty simplistic method, and if that's not giving what you're looking for, it's hard to really understand why without seeing something.
O I think I can share my sandbox app! (Way less code anyway.) Two main datapoints I'm trying to capture:
Sandbox:
var expressListRoutes = require('express-list-routes'),
express = require('express'),
router = express.Router();
var app = express();
app.use('/api/v1', router);
app.get('/', function(req, res){
res.send('hello world');
});
router.route('/user')
.post(function(test){})
.get(function(test){})
.put(function(test){});
// expressListRoutes({ prefix: '/api/v1' }, 'API:', router );
var route, routes = [];
app._router.stack.forEach(function(middleware){
if(middleware.route){ // routes registered directly on the app
routes.push(middleware.route);
} else if(middleware.name === 'router'){ // router middleware
middleware.handle.stack.forEach(function(handler){
route = handler.route;
route && routes.push(route);
});
}
});
routes.forEach(function(temp){
var methods = "";
for(var method in temp.methods){
methods += method + ", ";
}
console.log(temp.path + ": " + methods);
});
Output:
/user: post, get, put,
/: get,
Expected Output:
/api/v1/user: post, get, put,
/: get,
Here is a very, very hacky way for that app. I can't stress how hacky this is, but here it is:
function print (path, layer) {
if (layer.route) {
layer.route.stack.forEach(print.bind(null, path.concat(split(layer.route.path))))
} else if (layer.name === 'router' && layer.handle.stack) {
layer.handle.stack.forEach(print.bind(null, path.concat(split(layer.regexp))))
} else if (layer.method) {
console.log('%s /%s',
layer.method.toUpperCase(),
path.concat(split(layer.regexp)).filter(Boolean).join('/'))
}
}
function split (thing) {
if (typeof thing === 'string') {
return thing.split('/')
} else if (thing.fast_slash) {
return ''
} else {
var match = thing.toString()
.replace('\\/?', '')
.replace('(?=\\/|$)', '$')
.match(/^\/\^((?:\\[.*+?^${}()|[\]\\\/]|[^.*+?^${}()|[\]\\\/])*)\$\//)
return match
? match[1].replace(/\\(.)/g, '$1').split('/')
: '<complex:' + thing.toString() + '>'
}
}
app._router.stack.forEach(print.bind(null, []))
That produces:
POST /api/v1/user
GET /api/v1/user
PUT /api/v1/user
GET /
Hey, I'll take hacky over nothing any day . :)
I'll have to try this against our other apps and get back to you! Thanks for looking into this so quickly!!
It's been working perfectly for our simpler projects, pretty confident it'll work for our more convoluted apps as well, but I will keep you posted. :)
I wanted to check in with you before I did it, but there's this stackoverflow post of others who have tried to get this working, would it be ok if I posted your hacky albeit effective solution?
It's the only I found that realy works so far ... good job
Thanks, @dougwilson! This is the only solution working. I hope the next version do not break this code :D
Same topic discussed and addressed on StackOveflow.
I've found this package, which seems to work properly: https://github.com/AlbertoFdzM/express-list-endpoints !
I wrote a package to list middleware and routes mounted on an app quite a while ago: https://github.com/ErisDS/middleware-stack-printer
Maybe its useful for someone else.
Why don't you try something like swagger using tsoa(it automatically generates a swagger.json file) which will have all the api liat including what parameters it takes.
You can also hoat the same using swagge-ui-express locally. That will act as a better documents and will always auto update if you set up your run scripts correctly.
Just one note, tsoa requires Typescript.
@surendra-y - the problem I've had with swagger & friends (and tsoa looks similar) is that they all depend on some sort of non-authoritative, duplicate source of route information: annotations, jsdoc, etc. Which only works long term if you're disciplined enough where you probably don't need it in the first place. The great thing about doing it via reflection is:
That said, if whatever middleware does this also spits out the result in OpenAPI format - even better!
Here's how I parsed out the express routes to a json array.
const endpoints = app._router.stack.filter(x=> x.route && x.route.path && Object.keys(x.route.methods) != 0).map(layer => ({ method :layer.route.stack[0].method.toUpperCase(), path: layer.route.path}));
Result
[{
method: "GET",
path: "/api/example
}, {...}]
@StuAtGit, maybe this will help you?
@wesleytodd - 馃憤
That's a very clever solution. The kind of thing that seems obvious once you see, but as seen in this thread, not until then ;)
I use this package, which gives both terminal and web view
https://www.npmjs.com/package/express-routes-catalogue
@dougwilson
Seems like that there are also some parasites on the internet, who steal your idea and sell them as their own...
https://medium.com/@stupid_arnob/get-all-api-path-with-method-in-a-single-api-request-f6116254ea1a
Hi if someone is still searching, I rewrote the solution of @dougwilson in Typescript and made it output an array of strings of your routes so you can do with them what you wish.
function getRoutesOfLayer(path: string, layer: any): string[] {
if (layer.method) {
return [layer.method.toUpperCase() + ' ' + path];
}
else if (layer.route) {
return getRoutesOfLayer(path + split(layer.route.path), layer.route.stack[0]);
}
else if (layer.name === 'router' && layer.handle.stack) {
let routes: string[] = [];
layer.handle.stack.forEach(function(stackItem: any) {
routes = routes.concat(getRoutesOfLayer(path + split(layer.regexp), stackItem));
});
return routes;
}
return [];
}
function split (thing: any): string {
if (typeof thing === 'string') {
return thing;
} else if (thing.fast_slash) {
return '';
} else {
var match = thing.toString()
.replace('\\/?', '')
.replace('(?=\\/|$)', '$')
.match(/^\/\^((?:\\[.*+?^${}()|[\]\\\/]|[^.*+?^${}()|[\]\\\/])*)\$\//)
return match
? match[1].replace(/\\(.)/g, '$1')
: '<complex:' + thing.toString() + '>';
}
}
And then you can call it like this:
function getRoutes(app: Application): string[] {
let routes: string[] = [];
app._router.stack.forEach(function(layer: any) {
routes = routes.concat(getRoutesOfLayer('', layer));
});
return routes;
}
We are always listening 馃榾
Actually I wrote it as an middleware and in a singleton pattern. This is useful if you dont add routes on runtime, every route has only one Routehandler. So you can save the routes into a Map one time and can have a fast lookup. I used it to match the routes to the RouteHandlers and now I can lookup what route a routehandler has.
Most helpful comment
Here is a very, very hacky way for that app. I can't stress how hacky this is, but here it is:
That produces: