These tips and best practices are not just for development - but how to operate Node.js infrastructures, how you should do your day-to-day development and other useful pieces of advice.
\nUse ES2015
\nDuring the summer of 2015 the final draft of ES2015 (formerly ES6) was published. With this a number of new language features were added to the JavaScript language, including:
\n- \n
- arrow functions, \n
- template strings, \n
- rest operator, argument spreading, \n
- generators, \n
- promises, \n
- maps, sets, \n
- symbols, \n
and a lot more. For a comprehensive list of new features check out ES6 and Beyond by Kyle Simpson. Most of them are added to Node.js v4.
\nOn the client side, you can already use all of them with the help of Babel, a JavaScript compiler. Still, on the server side we prefer to use only the features that are added to the latest stable version, without compiling the source to save us from all the potential headaches.
\nFor more information on ES6 in Node.js, visit the official site: https://nodejs.org/en/docs/es6/.
\nCallback convention - with Promise support
\nFor the last years, we encouraged you to expose an error-first callback interface for your modules. With the generators functions already being available and with the upcoming async functions your modules (the ones published to NPM) should expose an error-first callback interface with Promise support.
\nWhy? To provide backward compatibility a callback interface has to be provided, and for future compatibility you will need the Promise support as well.
\nFor a demonstration of how to do it, take a look at the script below. In this example the readPackage
function reads the package.json
file and returns its' content by providing both a Promise and a callback interface.
Async patterns
\nFor a long time in Node.js, you had two choices to manage asynchronous flows: callbacks and streams. For callbacks you could use libraries like async and for streams through, bl or highland.
\nWith the introduction of Promises, generators and the async functions it is changing.
\n\n\nFor a more detailed history of asynchronous JavaScript check out The Evolution of Asynchronous JavaScript
\n
Error handling
\nError handling is a crucial part of the application to get right: knowing when to crash, or simply just log the error and continue/retry can be hard.
\nTo make it easier, we have to distinguish between programmer errors and operational errors.
\nProgrammer errors are basically bugs so you should crash the process immediately as you won't know in what state your application is.
\nOn the other hand, operational errors are problems with the system itself or a remote service, like request timeout or running out of memory. Based on the nature of the error you can try to solve with retrying, or if a file is missing you may have to create it first.
\nError handling in callbacks
\nIf an error occurs during an async operation, the error object will be passed as the first argument of the async function. You always have to check it and handle it.
\nThe code snippet in the Callback convention section above contains an example.
\nError handling in Promises
\nWhat's going to happen in the following snippet?
\n\n- \n
- It will throw an exception in line 3 \n
- The catch will handle it, and print it out to the stdout:
[Error: ops]
\n - The execution continues and in line 9 a new error will be thrown \n
- Nothing else \n
And really nothing else - the last error thrown will be a silent one. Pay extra attention to always add a catch as the last member of the promise chain. It will save you a lot of headaches. So it should look like this:
\n\nAnd now the output will be:
\n[Error: ops]\n[Error: ups]\n
\nUse JavaScript Standard Style
\nIn the past years, we had JSLint then JSHint, JSCS, ESLint - all excellent tools trying to automate as much code checking as possible.
\nRecently, when it comes to code style we use the JavaScript Standard Style by feross.
\n\nThe reason is simple: no configuration needed, just drop it in the project. Some rules that are incorporated (taken from the readme):
\n- \n
- 2 spaces – for indentation \n
- Single quotes for strings – except to avoid escaping \n
- No unused variables \n
- No semicolons \n
- Never start a line with
(
or[
\n- This is the only gotcha with omitting semicolons
\n - Space after keywords
if (condition) { ... }
\n - Space after function name
function name (arg) { ... }
\n - Always use
===
instead of==
– butobj == null
is allowed to checknull || undefined
. \n - Always handle the node.js
err
function parameter \n - Always prefix browser globals with
window
– exceptdocument
andnavigator
are okay\n- Prevents accidental use of poorly-named browser globals like
open
,length
,\nevent
, andname
.
\n - Prevents accidental use of poorly-named browser globals like
Also, if your editor of choice supports ESLint only, there is an ESLint ruleset as well for the Standard Style, the eslint-plugin-standard. With this plugin installed your .eslintrc
will something like this:
{\n "plugins": [\n "standard"\n ],\n}\n
\nThe Twelve-Factor Application
\nThe Twelve-Factor application manifesto describes best practices on how web applications should be written.
\n- \n
- One codebase tracked in revision control, many deploys \n
- Explicitly declare and isolate dependencies \n
- Store config in the environment \n
- Treat backing services as attached resources \n
- Strictly separate build and run stages \n
- Execute the app as one or more stateless processes \n
- Export services via port binding \n
- Scale out via the process model \n
- Maximize robustness with fast startup and graceful shutdown \n
- Keep development, staging, and production as similar as possible \n
- Treat logs as event streams \n
- Run admin/management tasks as one-off processes \n
Starting new projects
\nAlways start a new project with npm init
. This will generate a basic package.json
for your project.
If you want to skip the initial questions and go with the defaults, just run npm init --yes
.
Monitoring your applications
\nGetting notified as soon as something wrong happened or is going to happen in your system can save your business.
\nTo monitor your applications, you can use open-source software as well as SaaS products.
\nFor open-source, you can take a look at Zabbix, Collectd, ElasticSearch or Logstash.
\nIf you do not want to host them, you can try Trace, our Node.js and microservice monitoring solution.
\n\nUse a build system
\nAutomate everything you can. There is nothing more annoying and wasteful activity for a developer than to do grunt work.
\nNowadays the tooling around JavaScript evolved a lot - Grunt, Gulp, and Webpack, just to name a few.
\nAt RisingStack, most of the new projects use Webpack to aid in front-end development and gulp for other kinds of automated tasks. At first, Webpack can take more time to understand - for newcomers I highly recommend to check out the Webpack Cookbooks.
\nUse the latest LTS Node.js version
\nTo get both stability and new features we recommend to use the latest LTS (long-term support) version of Node.js - they are the ones with even release numbers. Of course, feel free to experiment with newer versions, called the Stable release line with the odd release numbers.
\nIf you are working on different projects with different Node.js version requirements then start using the Node Version Manager - nvm today.
\nFor more information on Node.js releases check out the official website: https://nodejs.org/en/blog/community/node-v5/.
\nUpdate dependencies on a weekly basis
\nMake it a habit to update dependencies on a weekly basis. For this, you can use npm outdated
or the ncu package.
Pick the right database
\nWhen talking about Node.js and databases the first technology that usually comes up is MongoDB. While there is nothing wrong with it, don't just jump into using it. Ask yourself and your team questions before doing so. To give some idea:
\n- \n
- Do you have structured data? \n
- Do you have to handle transactions? \n
- How long should you store the data? \n
You may only need Redis, or if you have structured data then you could go for PostgreSQL. If you start developing with SQL in Node.js, check out knex.
\nUse Semantic Versioning
\n\n\nSemantic Versioning is a formal convention for specifying compatibility using a three-part version number: major version; minor version; and patch.
\n
Major versions are bumped if an API change is not backward-compatible. Minor versions are bumped when new features are added, but the API change is backward compatible. Patch versions are bumped when only bug fixes happened.
\nLuckily, you can automate the release of your JavaScript modules with semantic-release.
\nKeep up
\nIt can be challenging to keep up with the latest news and developments in the JavaScript and Node.js world. To make it easier make sure to subscribe to the following media:
\n- \n
- Node.js Weekly Newsletter \n
- Microservice Weekly Newsletter \n
- Changelog Weekly - for Open-Source news \n
Learn more
\n\n\nWant to learn more about how you could implement Node.js at your company? Drop us a message at RisingStack.com!
\n