JSEPA-2: JavaScript coding style and guidelines
Introduction
This JSEPA aims to provide some general and simple guidelines for anyone. Some more specific recommandations are also yielded here.
General guidelines
The guidelines provided are here to homogenize the codebase of any application: every files, every lines, have to look like they were writen by one and only one person, regardless of the number of collaborators.
Tabs/space length
According to the knowledge we have at the time writing this text, there is no standard size for indentation. But one agreement emerged from all the sources we read: all the code must have and keep the same tab/space size.
Expanded syntax
It is better to use an expanded syntax in JavaScript:
function myFunc() {
console.log("My function here");
return 3;
}
function myFunc() { console.log("My function here"); return 3; }
As both syntaxes have the same result, it is preffered to use the expanded one, for readability and understanding the code.
Comments
A one-line comment in JavaScript is represented by a double slash //
followed by a single space and the content of the comment.
For a better understanding and reading, a one-line commment must be on one separated line of the code it is commenting:
A multi-line comment is represented by a block containing the content of the comment and surrounded by /*
(comment opener) and */
(comment closer).
For better reading, a multi-line comment must have its comment lines starting with an asterisk (*
), which must also be aligned with the above (previous) one. The multi-line comment must and with the closing tag (*/
) in a separated line of the last line of comment, and its asterisk must be aligned with the previous one as well.
// This variabe does x...
const myVar = 42;
// My comment
console.log("Hello");
// And an other one
document.location.href = "https://github.com";
const myVar = 42; // This variabe does x...
console.log("Hello"); // My comment
document.location.href = "https://github.com"; // And an other one
/* This variable does:
* x,
* y,
* z...
*/
const myVar = 42;
/* This variable does:
x,
* y,
z... */
const myVar = 42;
Docstring
A good code is a documented code, using docstrings in it is a must-have. Using JSDoc synthax is not standard, althrough it is very common and easy.
/**
* This class does things and can be used by A to to B.
*/
class MyClass {
/**
* Creates a MyClass instance.
* @param {number} x - The x to use.
* @param {number} y - The y to use.
*/
constructor(x, y) {
this.x = x;
this.y = y;
/**
* This attribute is used by the cass to to things.
* @member {number}
*/
this.thing = 3;
}
/**
* Method to retreive the x value of an object.
* @return {number} x
*/
getX() {
return this.x;
}
}
Coding syntax
Here are some examples of coding style to use for different situations. Those examples are here to improve the consistency fo the coding syntax, help for readability and understanding of the code.
By design the code should complies with the "use strict";
syntax:
// ✅
const myValue = 9;
for(let i = 0; i < myValue; i++) {
let result = myValue + i;
console.log(result);
}
// ❌
const myValue = 9
for(let i = 0; i < myValue; i++) {
let result = myValue + i
console.log(result)
}
The specification for control statements goes like so:
- No space between the keyword (
if
,while
...) and the opening parenthesis - 1 space between the closing parenthesis and the opening curly bracket
// ✅
if(condition) {
code();
}
// ❌
if (condition){
code();
}
Spacing the code with empty lines is mandatory for good-looking and readable code. Here are some simple rules for spacing:
- 1 empty line between two methods of a class
- 2 empty lines around a function
- 2 empty lines around a class
- 1 empty line after imports/includes (which should be the first lines of a script)
Info
The spacing specified here is very inspired by the PEP-8 convention coding style for Python.
// ✅ Good line spacing
class MyClass {
constructor() {
doThings();
}
myMethod() {
doThings();
}
}
function doThings() {
return 0;
}
// ❌ Bad line spacing
class MyClass {
constructor() {
doThings();
}
myMethod() {
doThings();
}
}
function doThings() {
return 0;
}
Strings
One must prefer to use template litterals instead of string concatenation for code readability:
const value = 42;
const name = "Bob";
// ✅ Template litteral
console.log(`The value is ${value} and the name is ${name}.`);
// ❌ String concatenation
console.log("The value is " + value + " and the name is " + name + ".");
Variables
Naming
The naming convention in JavaScript for variables is lowerCamelCase, and using a self-explaining name is a must-do (but don't overdo it). Using underscore is also not recommanded:
// ✅ Valid
const myAnimal = "dog";
// ❌ Invalid
const a = "cat";
// ❌ Invalid
const MY_ANIMAL = "bird";
// ❌ Invalid
const theNameOfMyFavouriteAnimalInTheHouseIsTheFollowing = "Oliver";
// ❌ Invalid
const my_animal = "mouse";
// ❌ Invalid
const _myAnimal = "horse";
// ❌ Invalid
const myAnimal__ = "tortoise";
It is also preffered to name the variabes according to its purpose:
Use one of the prefixes is
, are
or even has
to explicit the boolean variable:
let isValid = true;
let areCarsWorking = false;
let hasBeenHere = true;
Declaring
To declare variables, the intended way is to use either the const
or let
keyword. Each one have its own usecase:
const
: declaring a constant. It's value is not meant to change. Scope: block.let
: declaring a variable. Scope: block.
// ✅ Valid
const myAnimal = "dog";
// ✅ Valid
let myAnimal = "hamster";
// ❌ Invalid
var myAnimal = "cat";
Why the var
keyword is invalid?
Well, it is not invalid, but the main purpose of not using it is its scope. var
is function-scopped, while both const
and let
are block-scopped. In addition, var
is writeable, so there is no warranty for constants defined with var
, as per definition, are not read-only.
Functions
Declaring
Never declare a function with the new Function()
constructor as it uses plain text code evaluation wich may lead to security vulnerabilities:
// ❌ Don't do that
let a = document.getElementById("a-user-input").value;
let b = document.getElementById("b-user-input").value;
let operation = document.getElementById("operator-user-input").value;
const calculate = new Function("a", "b", `return a ${operator} b`);
Security advisory
Here the calculate()
function will compute the result of the provded operation with 2 numbers.
An attacker could input as the operator +(malicious_code)//
instead of just +
or -
to achieve execution of some malicious code.
For better security by design, one should code the above code like so:
// This is an example of code to avoid using code evaluation,
// under no circumstances this shoud be used as a vulnerability-proof code
let a = document.getElementById("a-user-input").value;
let b = document.getElementById("b-user-input").value;
let operation = document.getElementById("operator-user-input").value;
const calculate = (a=0, b=0, operation='+') => {
// Check if a and b are numbers ommited for size of code
switch(operation) {
case "+":
return a+b;
case "-":
return a-b;
default:
console.log("Operation not supported yet.");
return -1;
}
}
Async functions
To call an async function inside a non-async function one should use the instant call expression:
(async function myAsyncFunc() {
await doThis();
console.log("Done");
})();
// Or using an anonymous arrow function
(async () => {
await doThis();
console.log("Done");
})();
Parameters
Multiline signature
If a function needs a mutiline signature, all the arguments must be on one separated line of each other:
// ✅ Valid
function myFunc(
one,
two,
three,
four
) {
doThings();
}
// ❌ Invalid
function myFunc(one,
two,
three,
four) {
doThings();
}
// ❌ Invalid
function myFunc( one, two, three, four) {
doThings();
}
Default parameters
It is preffered to use the default parameter syntax instead of mutating:
Note
The order of default parameter is important too: the parameters with default values must be at the end of the list of all parameters.
// ✅ Valid
function myFunc(value1, value2=3) {
return value1 + value2;
}
// ❌ Invalid
function myFunc(value1, value2) {
value1 = value1 || 0;
if(value2 === void 0) {
value2 = 3;
}
return value1 + value2;
}
Parameters reassignment
Never reassign a parameter inside a function, this can lead to unwanted behavious when dealing with the arguments
object, and cause optimization issues, use variables instead:
// ✅ Preffered
function myFunc(a) {
let myNewA = a + "My value";
return myNewA;
}
// ❌ Don't do that
function myFunc(a) {
a += "My value";
return a;
}
Named parameters
To use named parameters, the prefered way is to go with objects literals and destructuring:
// Named params with default value
function add({a, b:0}={}) {
return a + b;
}
add({a: 1}); // 1
add({a: 1, b: 1}); // 2
add({a: 5, b: 3}); // 8
Arrow functions
Arrow functions can be used -- and this is the preffered way -- to keep the this
keyword:
Tip
This trick is in fact the recommanded way to keep the this
keyword in a class' method.
function myFunction() {
this.value = 0;
this.otherFunction = (x) => {
this.value = x;
}
}
Classes
Naming
Unlike variables, classes uses the PascalCase convention:
// ✅ Valid
class MyClass {
constructor() {
this.value = 1;
}
}
// ❌ Invalid
class myClass {
constructor() {
this.value = 1;
}
}
Methods
Methods chaining
Methods can return this
to encourage methods chaining:
class Dog {
constructor(name) {
this.name = name;
}
play() {
console.log("Waf!");
return this;
}
eat() {
console.log("Nom nom nom");
return this;
}
sleep() {
console.log("Zzzzzz");
return this;
}
}
const myDog = new Dog();
myDog.play()
.eat()
.sleep();
Static methods
If a method does not use this
, it should be marked as a static method:
class MyClass() {
static method() {
return 1;
}
}
Operations
Strict comparisons
To compare values of two variables, one must prefer to use strict comparison operations:
const name1 = "John";
const name2 = "James";
// ✅ Valid
if(name1 === name2) {
return true;
} else if(name1 !== name2) {
return false;
}
// ❌ Invalid
if(name1 == name2) {
return true;
} else if(name1 != name2) {
return false;
}
Boolean tests
For boolean tests, it it preffered to use the shortcodes of expressions instead of explicit testing the values:
// ✅ Valid
if(myVar) {
doThings();
} else {
doOtherThing();
}
// ❌ Invalid
if(myVar === true) {
doThings();
} else {
doOtherThing();
}
Simplify if possible
For commom operations, it is sometimes possible to simplify the code:
function myTest(myVar) {
if(myVar === true) {
return true;
} else {
return false;
}
}
function myTest(myVar) {
return myVar; // (1)
}
- At this point, this function is not realistic, but you get the idea for optimization
Code examples
The following projects are not affiliated with the JSEPA, but are good examples to follow:
- Feel free to push projects here.