Research
Security News
Quasar RAT Disguised as an npm Package for Detecting Vulnerabilities in Ethereum Smart Contracts
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
Nodecaf is an Express framework for developing REST APIs in a quick and convenient manner.
Docs for version v0.6.x.
Nodecaf is an Express framework for developing REST APIs in a quick and convenient manner. Using Nodecaf you'll get:
If you are unfamiliar with Express, checkout their routing docs so that you can better grasp Nodecaf features and whatnot.
--no-optional
is meant to ignore DTrace package which is a compiled dependency of bunyan. Chek out their issue.
npm i --no-optional -g nodecaf
.package.json
).npm i --no-optional nodecaf
.nodecaf init
.lib/main.js
const { AppServer } = require('nodecaf');
const api = require('./api');
module.exports = function init(conf){
let app = new AppServer(conf);
// Expose things to all routes putting them on the 'shared' object.
let shared = {};
app.expose(shared);
// You can intercept all error that escape the route handlers.
app.onRuoteError = function(input, err, send){
// Any error that is not handled here will just become a harmless 500.
};
// Perform your server initialization logic.
app.beforeStart = async function(){
};
// Perform your server finalization logic.
app.afterStop = async function(){
};
// Load your routes and API definitions.
app.api(api);
// Don't forget to return your app.
return app;
}
lib/api.js
module.exports = function({ post, get, del, head, patch, put }){
// Use express routes and a list of functions (async or regular no matter).
get('/foo/:f/bar/:b', Foo.read, Bar.read);
post('/foo/:f/bar', Foo.read, Bar.write);
// ...
};
If you have found any problems with this module, please:
~bug
.We will make sure to take a look when time allows us.
If you wish to get that awesome feature or have some advice for us, please:
~proposal
.If you have spotted any enhancements to be made and is willing to get your hands dirty about it, fork us and submit your merge request so we can collaborate effectively.
Beyond all the cool features of Express has to offer, check out how to use all the awesome goodies Nodecaf can give you.
In this manual we address as handler args the keys in the object passed as the only argument of any route handler function. The code below shows all handler args exposed by Nodecaf:
function({ req, res, next, query, params, body, flash, conf, log, error }){
// Do your stuff.
}
Quick reference:
req
, res
, next
: Basically the good old parameters used regularly in Express.query
, parameters
, body
: Shortcuts to the homonymous properties of req
.
They contain respectively the query string, the URL parameters, and the request
body data.flash
: Is a shortcut to Express req.locals
. Keys inserted in this a object
are preserved for the lifetime of a request and can be accessed on all handlers
of a route chain.conf
: This object contain the entire
application configuration data.log
: A bunyan logger instance. Use it to log custom events of
your application.error
: A function to output REST errors and abort the
handler chain execution.Nodecaf allow you to read a configuration file in the TOML format (we plan to add more in the future) and use it's data on all routes and server configuration.
Use this feature to manage:
generate a project with configuration file already plugged in
To setup a config file for an existing project, open the binary for your server
on bin/proj-name.js
. Then add a confPath
key to the run parameter object
whose value must be a string path pointing to your conf file.
The data in the config file can be accessed on lib/main.js
through the first
parameter of the exported init
function:
module.exports = function init(conf){
console.log(conf);
}
You can also use the config data through it's handler arg on all route handlers as follows:
post('/foo', function({ conf }){
console.log(conf.myField);
});
To setup Nodecaf logger you must add a log
key to the application server config
object like this:
conf.log = { file: 'path/to/conf/file.log' }; // Use this for a file
conf.log = { stream: process.stdout }; // Use this for a write stream
let app = new AppServer(conf);
You can also setup a file log on your settings file to be automatically
transferred to the conf
argument of init
.
[log]
file = "path/to/conf/file.log"
On your route handlers, use the log
handler arg as a
bunyan instance:
function({ log }){
log.info('hi');
log.warn({lang: 'fr'}, 'au revoir');
}
Nodecaf will automatically log some useful server events as described in the table below:
Level | Event |
---|---|
warn | An error happened inside a route after the headers were already sent |
error | An error happened inside a route and was not caught |
fatal | An error happened that crashed the server process |
debug | A request has arrived |
Nodecaf brings a useful feature of accepting async functions as route handlers with zero configuration. The real deal is that all rejections/error within your async handler will be gracefully handled by the same routine the deals with regular functions. Allowing you to avoid callback hell without creating bogus adapters for your promises.
get('/my/thing',
function({ res, next }){
res.send('My regular function works!');
next();
},
async function({ res }){
await myAsyncThing();
res.end('My async function works too!');s
}
);
In Nodecaf, all uncaught synchronous errors happening inside route handler code is automatically converted into a harmless RESTful 500.
post('/my/thing', function(){
throw new Error('Should respond with a 500');
});
To support the callback error pattern, use the error
handler arg.
const fs = require('fs');
post('/my/thing', function({ error, res }){
fs.readFile('./my/file', 'utf8', function(err, contents){
if(err)
return error(err, 'Optional message to replace the original');
res.end(contents);
});
});
To use other HTTP status codes you can send a string in the first parameter of
error
. The supported error names are the following:
NotFound
: 404Unauthorized
: 401ServerFault
: 500InvalidActionForState
: 405InvalidCredentials
: 400InvalidContent
: 400post('/my/thing', function({ error }){
try{
doThing();
}
catch(e){
error('NotFound', 'Optional message for the JSON response');
}
});
You can always deal with uncaught exceptions on all routes through a default
global error handler. In your lib/main.js
add a onRuoteError
function
property to the app
.
app.onRuoteError = function(input, err, send){
if(err instanceof MyDBError)
send('ServerFalut', 'Sorry! Database is sleeping.');
else if(err instanceof ValidationError)
send('InvalidContent', err.data);
}
send
function will instruct Nodecaf to output the given error.input
arg contain all handler args for the request.Error
the normal 500 behavior will
take place.Nodecaf provides you with an assertion module containing functions to generate the most common REST outputs based on some condition. Checn an example to trigger a 404 in case a database record doesn't exist.
let { exist } = require('nodecaf').assertions;
get('/my/thing/:id', function({ params, db }){
let thing = await db.getById(params.id);
exist(thing, 'thing not found');
doStuff();
});
If the record is not found, the exist
call will stop the route execution right
away and generate a RESTful NotFound
error.
Along with exist
, the following assertions with similar behavior are provided:
valid
: InvalidContent
authorized
: Unauthorized
authn
: InvalidCredentials
able
: InvalidActionForState
To use it with callback style functions, pass the error
handler arg as the
third parameter.
let { exist } = require('nodecaf').assertions;
post('/my/file/:id', function({ error, res, params }){
fs.readFile('./my/file/' + params.id, 'utf8', function(err, contents){
exist(!err, 'File not found', error);
res.end(contents);
});
});
Nodecaf makes it simple to share global objects (eg.: database connections,
instanced libraries) across all route handlers. In your lib/main.js
you can
expose an object of which all keys will become handler args.
app.expose({
db: myDbConnection,
libX: new LibXInstance()
});
Then in all routes you can:
get('/my/thing', function({ db, libX }){
// use your global stuff
});
In production it's generally desirable to have an HTTPS setup for both client to
API and API to API communications. You can enable SSL for your server by adding
a ssl
key to your config, containing both the path for your key and cert.
[ssl]
key = "/path/to/key.pem"
cert = "/path/to/cert.pem"
When SSL is enabled the default server port becomes 443.
Nodecaf allows you to descibe your api and it's functionality, effectively turning your code in the single source of truth. The described API can later be used to generate an Open API compatible document.
In lib/api.js
describe your API as whole through the info
parameter:
module.exports = function({ get, info }){
info({
description: 'My awesome API that foos the bars and bazes the bahs'
});
get('/my/thing/:id', function(){
// ...
});
}
The info
funciton expects an object argument on the OpenAPI
Info Object
format. If not defined the title
and version
keys will default to your server's.
Describe your API endpoints by chaining a desc
method to each route definition.
module.exports = function({ get }){
get('/my/thing/:id', function(){
// ...
}).desc('Retrieves a thing from the database\n' +
`Searches the database for the thing with the given :id. Returns a
NotFound error in case no thing is found.`);
}
The desc
method takes a single string argument and uses it's first line (before \n
)
to set the
Operation object's
summary
property and the rest of the text to set the description
(CommonMark).
Nodecaf also provides some CLI tools to ease your development. To install the cli
commands, run: npm i --no-optional -g nodecaf
.
Check below the commands we provide.
nodecaf init
Generates a skelleton Nodecaf project file structure on the current
directory.
You must already have a well-formed package.json on the target directory.
Options
-p --path [directory]
: Project root directory (defaults to working dir)-c --confPath [file]
: Generate a config file and plug it in the structure-n --name [string]
: A name/title for the generated app structurenodecaf openapi
Generates a Open API compliant
document of a given Nodecaf API.
Options
-o --outFile file
: A file path to save the generated document (required)-p --path directory
: Project root directory (defaults to working dir)--apiPath file
: A path to your project's API file (defaults to lib/api.js
)-t --type (json | yaml)
: The type of document to be generated (defaults to JSON)-c --confPath [file]
: The config file to be considered[v0.6.0] - 2019-06-24
nodecat openapi
)del
commandroute
method was renamed to api
FAQs
Nodecaf is a light framework for developing RESTful Apps in a quick and convenient manner.
The npm package nodecaf receives a total of 567 weekly downloads. As such, nodecaf popularity was classified as not popular.
We found that nodecaf demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Research
Security News
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
Security News
Research
A supply chain attack on Rspack's npm packages injected cryptomining malware, potentially impacting thousands of developers.
Research
Security News
Socket researchers discovered a malware campaign on npm delivering the Skuld infostealer via typosquatted packages, exposing sensitive data.