This is a guide for writing consistent and aesthetically pleasing node.js code. It is inspired by what is popular within the community, and flavored with some personal opinions.
There is a .jshintrc which enforces these rules as closely as possible. You can either use that and adjust it, or use this script to make your own.
This guide was created by Felix Geisendörfer and is licensed under the CC BY-SA 3.0 license. You are encouraged to fork this repository and make adjustments according to your preferences.
- 2 Spaces for indentation
- Newlines
- No trailing whitespace
- Use Semicolons
- 80 characters per line
- Use single quotes
- Opening braces go on the same line
- Declare one variable per var statement
- Use lowerCamelCase for variables, properties and function names
- Use UpperCamelCase for class names
- Use UPPERCASE for Constants
- Write small functions
- Return early from functions
- Name your closures
- No nested closures
- Method chaining
- Object.freeze, Object.preventExtensions, Object.seal, with, eval
- Requires At Top
- Getters and setters
- Do not extend built-in prototypes
Use 2 spaces for indenting your code and swear an oath to never mix tabs and spaces - a special kind of hell is awaiting you otherwise.
Use UNIX-style newlines (\n), and a newline character as the last character
of a file. Windows-style newlines (\r\n) are forbidden inside any repository.
Just like you brush your teeth after every meal, you clean up any trailing whitespace in your JS files before committing. Otherwise the rotten smell of careless neglect will eventually drive away contributors and/or co-workers.
According to scientific research, the usage of semicolons is a core value of our community. Consider the points of the opposition, but be a traditionalist when it comes to abusing error correction mechanisms for cheap syntactic pleasures.
Limit your lines to 80 characters. Yes, screens have gotten much bigger over the last few years, but your brain has not. Use the additional room for split screen, your editor supports that, right?
Use single quotes, unless you are writing JSON.
Right:
var foo = 'bar';Wrong:
var foo = "bar";Your opening braces go on the same line as the statement. Also, notice the use of whitespace before and after the condition statement.
Right:
if (true) {
console.log('winning');
}Wrong:
if (true)
{
console.log('losing');
}
if (true){
console.log('losing');
}
if(true) {
console.log('losing');
}
if(true){
console.log('losing');
}PS: If you are using sublime-text, give it a try to the sublime conditions snippets.
Declare one variable per var statement, it makes it easier to re-order the lines. However, ignore Crockford when it comes to declaring variables deeper inside a function, just put the declarations wherever they make sense.
Right:
var keys = ['foo', 'bar'];
var values = [23, 42];
var object = {};
while (keys.length) {
var key = keys.pop();
object[key] = values.pop();
}Wrong:
var keys = ['foo', 'bar'],
values = [23, 42],
object = {},
key;
while (keys.length) {
key = keys.pop();
object[key] = values.pop();
}Variables, properties and function names should use lowerCamelCase. They
should also be descriptive. Single character variables and uncommon
abbreviations should generally be avoided.
Right:
var adminUser = db.query('SELECT * FROM users ...');Wrong:
var admin_user = db.query('SELECT * FROM users ...');Class names should be capitalized using UpperCamelCase.
Right:
function BankAccount() {
}Wrong:
function bank_Account() {
}PS: If you are using sublime-text, give it a try to the sublime functions snippets.
Constants should be declared as regular variables or static class properties, using all uppercase letters.
Right:
var SECOND = 1 * 1000;
function File() {
}
File.FULL_PERMISSIONS = 0777;Wrong:
const SECOND = 1 * 1000;
function File() {
}
File.fullPermissions = 0777;Use trailing commas and put short declarations on a single line. Only quote keys when your interpreter complains:
Right:
var a = ['hello', 'world'];
var b = {
good: 'code',
'is generally': 'pretty',
};Wrong:
var a = [
'hello', 'world'
];
var b = {"good": 'code'
, is generally: 'pretty'
};Programming is not about remembering stupid rules. Use the triple equality operator as it will work just as expected.
Right:
var a = 0;
if (a !== '') {
console.log('winning');
}Wrong:
var a = 0;
if (a == '') {
console.log('losing');
}Any non-trivial conditions should be assigned to a descriptively named variable or function:
Right:
var isValidPassword = password.length >= 4 && /^(?=.*\d).{4,}$/.test(password);
if (isValidPassword) {
console.log('winning');
}Wrong:
if (password.length >= 4 && /^(?=.*\d).{4,}$/.test(password)) {
console.log('losing');
}Keep your functions short. A good function fits on a slide that the people in the last row of a big room can comfortably read. So don't count on them having perfect vision and limit yourself to ~15 lines of code per function.
To avoid deep nesting of if-statements, always return a function's value as early as possible.
Right:
function isPercentage(val) {
if (val < 0) {
return false;
}
if (val > 100) {
return false;
}
return true;
}Wrong:
function isPercentage(val) {
if (val >= 0) {
if (val < 100) {
return true;
} else {
return false;
}
} else {
return false;
}
}Or for this particular example it may also be fine to shorten things even further:
function isPercentage(val) {
var isInRange = (val >= 0 && val <= 100);
return isInRange;
}Feel free to give your closures a name. It shows that you care about them, and will produce better stack traces, heap and cpu profiles.
Right:
req.on('end', function onEnd() {
console.log('winning');
});Wrong:
req.on('end', function() {
console.log('losing');
});Use closures, but don't nest them. Otherwise your code will become a mess.
Right:
setTimeout(function() {
client.connect(afterConnect);
}, 1000);
function afterConnect() {
console.log('winning');
}Wrong:
setTimeout(function() {
client.connect(function() {
console.log('losing');
});
}, 1000);One method per line should be used if you want to chain methods.
You should also indent these methods so it's easier to tell they are part of the same chain.
Right:
User
.findOne({ name: 'foo' })
.populate('bar')
.exec(function(err, user) {
return true;
});Wrong:
User
.findOne({ name: 'foo' })
.populate('bar')
.exec(function(err, user) {
return true;
});
User.findOne({ name: 'foo' })
.populate('bar')
.exec(function(err, user) {
return true;
});
User.findOne({ name: 'foo' }).populate('bar')
.exec(function(err, user) {
return true;
});
User.findOne({ name: 'foo' }).populate('bar')
.exec(function(err, user) {
return true;
});Use slashes for both single line and multi line comments. Try to write comments that explain higher level mechanisms or clarify difficult segments of your code. Don't use comments to restate trivial things.
Right:
// 'ID_SOMETHING=VALUE' -> ['ID_SOMETHING=VALUE', 'SOMETHING', 'VALUE']
var matches = item.match(/ID_([^\n]+)=([^\n]+)/));/**
* function description
*
* @param paramType id paramDescription
* @param paramType cb paramDescription
* @return returnType returnDescription
*/
function loadUser(id, cb) {
//...
}
var isSessionValid = (session.expires < Date.now());
if (isSessionValid) {
// ...
}Wrong:
// Execute a regex
var matches = item.match(/ID_([^\n]+)=([^\n]+)/);
// Usage: loadUser(5, function() { ... })
function loadUser(id, cb) {
// ...
}
// Check if the session is valid
var isSessionValid = (session.expires < Date.now());
// If the session is valid
if (isSessionValid) {
// ...
}Crazy shit that you will probably never need. Stay away from it.
Always put requires at top of file to clearly illustrate a file's dependencies. Besides giving an overview for others at a quick glance of dependencies and possible memory impact, it allows one to determine if they need a package.json file should they choose to use the file elsewhere.
Do not use setters, they cause more problems for people who try to use your software than they can solve.
Feel free to use getters that are free from side effects, like providing a length property for a collection class.
Do not extend the prototype of native JavaScript objects. Your future self will be forever grateful.
Right:
var a = [];
if (!a.length) {
console.log('winning');
}Wrong:
Array.prototype.empty = function() {
return !this.length;
}
var a = [];
if (a.empty()) {
console.log('losing');
}Opinionated Angular styleguide for teams by @toddmotto
A standardised approach for developing Angular applications in teams. This styleguide touches on concepts, syntax, conventions and is based on my experience writing about, talking about, and building Angular applications.
John Papa and I have discussed in-depth styling patterns for Angular and as such have both released separate styleguides. Thanks to those discussions, I've learned some great tips from John that have helped shape this guide. We've both created our own take on a styleguide. I urge you to check his out to compare thoughts.
See the original article that sparked this off
- Modules
- Controllers
- Services and Factory
- Directives
- Filters
- Routing resolves
- Publish and subscribe events
- Performance
- Angular wrapper references
- Comment standards
- Minification and annotation
- File Templates and Snippets
-
Definitions: Declare modules without a variable using the setter and getter syntax
// avoid var app = angular.module('app', []); app.controller(); app.factory(); // recommended angular .module('app', []) .controller() .factory();
-
Note: Using
angular.module('app', []);sets a module, whereasangular.module('app');gets the module. Only set once and get for all other instances. -
Methods: Pass functions into module methods rather than assign as a callback
// avoid angular .module('app', []) .controller('MainCtrl', function MainCtrl () { }) .service('SomeService', function SomeService () { }); // recommended function MainCtrl () { } function SomeService () { } angular .module('app', []) .controller('MainCtrl', MainCtrl) .service('SomeService', SomeService);
-
ES6 Classes are not hoisted, which will break your code if you rely on hoisting
-
This aids with readability and reduces the volume of code "wrapped" inside the Angular framework
-
IIFE scoping: To avoid polluting the global scope with our function declarations that get passed into Angular, ensure build tasks wrap the concatenated files inside an IIFE
(function () { angular .module('app', []); // MainCtrl.js function MainCtrl () { } angular .module('app') .controller('MainCtrl', MainCtrl); // SomeService.js function SomeService () { } angular .module('app') .service('SomeService', SomeService); // ... })();
-
controllerAs syntax: Controllers are classes, so use the
controllerAssyntax at all times<!-- avoid --> <div ng-controller="MainCtrl"> {{ someObject }} </div> <!-- recommended --> <div ng-controller="MainCtrl as vm"> {{ vm.someObject }} </div>
-
In the DOM we get a variable per controller, which aids nested controller methods, avoiding any
$parentcalls -
The
controllerAssyntax usesthisinside controllers, which gets bound to$scope// avoid function MainCtrl ($scope) { $scope.someObject = {}; $scope.doSomething = function () { }; } // recommended function MainCtrl () { this.someObject = {}; this.doSomething = function () { }; }
-
Only use
$scopeincontrollerAswhen necessary; for example, publishing and subscribing events using$emit,$broadcast,$onor$watch. Try to limit the use of these, however, and treat$scopeas a special use case -
Inheritance: Use prototypal inheritance when extending controller classes
function BaseCtrl () { this.doSomething = function () { }; } BaseCtrl.prototype.someObject = {}; BaseCtrl.prototype.sharedSomething = function () { }; AnotherCtrl.prototype = Object.create(BaseCtrl.prototype); function AnotherCtrl () { this.anotherSomething = function () { }; }
-
Use
Object.createwith a polyfill for browser support -
controllerAs 'vm': Capture the
thiscontext of the Controller usingvm, standing forViewModel// avoid function MainCtrl () { var doSomething = function () { }; this.doSomething = doSomething; } // recommended function MainCtrl () { var vm = this; var doSomething = function () { }; vm.doSomething = doSomething; }
Why? : Function context changes the
thisvalue, use it to avoid.bind()calls and scoping issues -
ES6: Avoid
var vm = this;when using ES6// avoid function MainCtrl () { let vm = this; let doSomething = arg => { console.log(vm); }; // exports vm.doSomething = doSomething; } // recommended function MainCtrl () { let doSomething = arg => { console.log(this); }; // exports this.doSomething = doSomething; }
Why? : Use ES6 arrow functions when necessary to access the
thisvalue lexically -
Presentational logic only (MVVM): Presentational logic only inside a controller, avoid Business logic (delegate to Services)
// avoid function MainCtrl () { var vm = this; $http .get('/users') .success(function (response) { vm.users = response; }); vm.removeUser = function (user, index) { $http .delete('/user/' + user.id) .then(function (response) { vm.users.splice(index, 1); }); }; } // recommended function MainCtrl (UserService) { var vm = this; UserService .getUsers() .then(function (response) { vm.users = response; }); vm.removeUser = function (user, index) { UserService .removeUser(user) .then(function (response) { vm.users.splice(index, 1); }); }; }
Why? : Controllers should fetch Model data from Services, avoiding any Business logic. Controllers should act as a ViewModel and control the data flowing between the Model and the View presentational layer. Business logic in Controllers makes testing Services impossible.
- All Angular Services are singletons, using
.service()or.factory()differs the way Objects are created.
Services: act as a constructor function and are instantiated with the new keyword. Use this for public methods and variables
```javascript
function SomeService () {
this.someMethod = function () {
};
}
angular
.module('app')
.service('SomeService', SomeService);
```
Factory: Business logic or provider modules, return an Object or closure
-
Always return a host Object instead of the revealing Module pattern due to the way Object references are bound and updated
function AnotherService () { var AnotherService = {}; AnotherService.someValue = ''; AnotherService.someMethod = function () { }; return AnotherService; } angular .module('app') .factory('AnotherService', AnotherService);
Why? : Primitive values cannot update alone using the revealing module pattern
-
Declaration restrictions: Only use
custom elementandcustom attributemethods for declaring your Directives ({ restrict: 'EA' }) depending on the Directive's role<!-- avoid --> <!-- directive: my-directive --> <div class="my-directive"></div> <!-- recommended --> <my-directive></my-directive> <div my-directive></div>
-
Comment and class name declarations are confusing and should be avoided. Comments do not play nicely with older versions of IE. Using an attribute is the safest method for browser coverage.
-
Templating: Use
Array.join('')for clean templating// avoid function someDirective () { return { template: '<div class="some-directive">' + '<h1>My directive</h1>' + '</div>' }; } // recommended function someDirective () { return { template: [ '<div class="some-directive">', '<h1>My directive</h1>', '</div>' ].join('') }; }
Why? : Improves readability as code can be indented properly, it also avoids the
+operator which is less clean and can lead to errors if used incorrectly to split lines -
DOM manipulation: Takes place only inside Directives, never a controller/service
// avoid function UploadCtrl () { $('.dragzone').on('dragend', function () { // handle drop functionality }); } angular .module('app') .controller('UploadCtrl', UploadCtrl); // recommended function dragUpload () { return { restrict: 'EA', link: function (scope, element, attrs) { element.on('dragend', function () { // handle drop functionality }); } }; } angular .module('app') .directive('dragUpload', dragUpload);
-
Naming conventions: Never
ng-*prefix custom directives, they might conflict future native directives// avoid // <div ng-upload></div> function ngUpload () { return {}; } angular .module('app') .directive('ngUpload', ngUpload); // recommended // <div drag-upload></div> function dragUpload () { return {}; } angular .module('app') .directive('dragUpload', dragUpload);
-
Directives and Filters are the only providers that have the first letter as lowercase; this is due to strict naming conventions in Directives. Angular hyphenates
camelCase, sodragUploadwill become<div drag-upload></div>when used on an element. -
controllerAs: Use the
controllerAssyntax inside Directives as well// avoid function dragUpload () { return { controller: function ($scope) { } }; } angular .module('app') .directive('dragUpload', dragUpload); // recommended function dragUpload () { return { controllerAs: 'vm', controller: function () { } }; } angular .module('app') .directive('dragUpload', dragUpload);
-
Global filters: Create global filters using
angular.filter()only. Never use local filters inside Controllers/Services// avoid function SomeCtrl () { this.startsWithLetterA = function (items) { return items.filter(function (item) { return /^a/i.test(item.name); }); }; } angular .module('app') .controller('SomeCtrl', SomeCtrl); // recommended function startsWithLetterA () { return function (items) { return items.filter(function (item) { return /^a/i.test(item.name); }); }; } angular .module('app') .filter('startsWithLetterA', startsWithLetterA);
-
This enhances testing and reusability
-
Promises: Resolve Controller dependencies in the
$routeProvider(or$stateProviderforui-router), not the Controller itself// avoid function MainCtrl (SomeService) { var _this = this; // unresolved _this.something; // resolved asynchronously SomeService.doSomething().then(function (response) { _this.something = response; }); } angular .module('app') .controller('MainCtrl', MainCtrl); // recommended function config ($routeProvider) { $routeProvider .when('/', { templateUrl: 'views/main.html', resolve: { // resolve here } }); } angular .module('app') .config(config);
-
Controller.resolve property: Never bind logic to the router itself. Reference a
resolveproperty for each Controller to couple the logic// avoid function MainCtrl (SomeService) { this.something = SomeService.something; } function config ($routeProvider) { $routeProvider .when('/', { templateUrl: 'views/main.html', controllerAs: 'vm', controller: 'MainCtrl' resolve: { doSomething: function () { return SomeService.doSomething(); } } }); } // recommended function MainCtrl (SomeService) { this.something = SomeService.something; } MainCtrl.resolve = { doSomething: function (SomeService) { return SomeService.doSomething(); } }; function config ($routeProvider) { $routeProvider .when('/', { templateUrl: 'views/main.html', controllerAs: 'vm', controller: 'MainCtrl' resolve: MainCtrl.resolve }); }
-
This keeps resolve dependencies inside the same file as the Controller and the router free from logic
-
$scope: Use the
$emitand$broadcastmethods to trigger events to direct relationship scopes only// up the $scope $scope.$emit('customEvent', data); // down the $scope $scope.$broadcast('customEvent', data);
-
$rootScope: Use only
$emitas an application-wide event bus and remember to unbind listeners// all $rootScope.$on listeners $rootScope.$emit('customEvent', data);
-
Hint: Because the
$rootScopeis never destroyed,$rootScope.$onlisteners aren't either, unlike$scope.$onlisteners and will always persist, so they need destroying when the relevant$scopefires the$destroyevent// call the closure var unbind = $rootScope.$on('customEvent'[, callback]); $scope.$on('$destroy', unbind);
-
For multiple
$rootScopelisteners, use an Object literal and loop each one on the$destroyevent to unbind all automaticallyvar unbind = [ $rootScope.$on('customEvent1'[, callback]), $rootScope.$on('customEvent2'[, callback]), $rootScope.$on('customEvent3'[, callback]) ]; $scope.$on('$destroy', function () { unbind.forEach(function (fn) { fn(); }); });
-
One-time binding syntax: In newer versions of Angular (v1.3.0-beta.10+), use the one-time binding syntax
{{ ::value }}where it makes sense// avoid <h1>{{ vm.title }}</h1> // recommended <h1>{{ ::vm.title }}</h1>
Why? : Binding once removes the watcher from the scope's
$$watchersarray after theundefinedvariable becomes resolved, thus improving performance in each dirty-check -
Consider $scope.$digest: Use
$scope.$digestover$scope.$applywhere it makes sense. Only child scopes will update$scope.$digest();
Why? :
$scope.$applywill call$rootScope.$digest, which causes the entire application$$watchersto dirty-check again. Using$scope.$digestwill dirty check current and child scopes from the initiated$scope
-
$document and $window: Use
$documentand$windowat all times to aid testing and Angular references// avoid function dragUpload () { return { link: function ($scope, $element, $attrs) { document.addEventListener('click', function () { }); } }; } // recommended function dragUpload ($document) { return { link: function ($scope, $element, $attrs) { $document.addEventListener('click', function () { }); } }; }
-
$timeout and $interval: Use
$timeoutand$intervalover their native counterparts to keep Angular's two-way data binding up to date// avoid function dragUpload () { return { link: function ($scope, $element, $attrs) { setTimeout(function () { // }, 1000); } }; } // recommended function dragUpload ($timeout) { return { link: function ($scope, $element, $attrs) { $timeout(function () { // }, 1000); } }; }
-
jsDoc: Use jsDoc syntax to document function names, description, params and returns
/** * @name SomeService * @desc Main application Controller */ function SomeService (SomeService) { /** * @name doSomething * @desc Does something awesome * @param {Number} x - First number to do something with * @param {Number} y - Second number to do something with * @returns {Number} */ this.doSomething = function (x, y) { return x * y; }; } angular .module('app') .service('SomeService', SomeService);
-
ng-annotate: Use ng-annotate for Gulp as
ng-minis deprecated, and comment functions that need automated dependency injection using/** @ngInject *//** * @ngInject */ function MainCtrl (SomeService) { this.doSomething = SomeService.doSomething; } angular .module('app') .controller('MainCtrl', MainCtrl);
-
Which produces the following output with the
$injectannotation/** * @ngInject */ function MainCtrl (SomeService) { this.doSomething = SomeService.doSomething; } MainCtrl.$inject = ['SomeService']; angular .module('app') .controller('MainCtrl', MainCtrl);
Use file templates or snippets to help follow consistent styles and patterns. Here are templates and/or snippets for some of the web development editors and IDEs.
[Style Y250]
-
Angular snippets that follow these styles and guidelines.
- Download the Sublime Angular snippets
- Place it in your Packages folder
- Restart Sublime
- In a JavaScript file type these commands followed by a
TAB
ngcontroller // creates an Angular controller ngdirective // creates an Angular directive ngfactory // creates an Angular factory ngmodule // creates an Angular module ngservice // creates an Angular service ngfilter // creates an Angular filter
For anything else, including API reference, check the Angular documentation.
