Beyond console.log()
There is more to debugging JavaScript than console.log to output values. It might seem obvious Iâm going to pimp the debugger, but actually no.
It seems like itâs cool to tell people doing JavaScript that they should be using the browserâs debugger, and thereâs certainly a time and a place for that. But a lot of the time you just want to see whether a particular bit of code executes or what a variable is, without disappearing into the RxJS codebase or the bowels of a Promise library.
Nevertheless, while console.log
has its place, a lot of people donât realise that the console
itself has a lot of options beyond the basic log
. Appropriate use of these functions can make debugging easier, faster, and more intuitive.
console.log()
There is a surprising amount of functionality in good old console.log that people donât expect. While most people use it as console.log(object)
, you can also do console.log(object, otherObject, string)
and it will log them all out neatly. Occasionally handy.
More than that, thereâs another format: console.log(msg, values)
. This works a lot like something like sprintf
in C or PHP.
console.log('I like %s but I do not like %s.', 'Skittles', 'pus');
Will output exactly as youâd expect.
> I like Skittles but I do not like pus.
Common placeholders are %o
(thatâs a letter o, not a zero) which takes an object, %s
which takes a string, and %d
which is for a decimal or integer.
Another fun one is %c
but your mileage may vary on this. Itâs actually a placeholder for CSS values.
console.log('I am a %cbutton', 'color: white; background-color: orange; padding: 2px 5px; border-radius: 2px');
The values will run onto anything that follows, thereâs no âend tagâ, which is a bit weird. But you can mangle it a bit like this.
Itâs not elegant, nor is it particularly useful. Thatâs not really a button, of course.
Again, is it useful? Ehhhhh.
console.dir()
For the most part, console.dir()
functions very much like log()
, though it looks a teeny bit different.
Dropping down the little arrow will show the same exact object details as above, which can also be seen from the console.log
version. Where things diverge more drastically, and get more interesting, is when you look at elements.
let element = document.getElementById('2x-container');
This is the output from logging that input:
Iâve popped open a few elements. This is clearly showing the DOM, and we can navigate through it. But console.dir(element)
gives us a surprisingly different output.
This is a way more objecty way of looking at the element. There may be times when thatâs what you actually want, something more like inspecting the element.
console.warn()
Probably the most obvious direct replacement for log()
, you can just use console.warn()
in exactly the same way. The only real difference is the output is a bit yellow. Specifically the output is at a warning level not an info level, so the browser will treat it slightly differently. This has the effect of making it a bit more obvious in a cluttered output.
Thereâs a bigger advantage, though. Because the output is a warning rather than an info, you can filter out all the console.log
and leave only console.warn
. This is particularly helpful in those occasionally chatty apps that constantly output a bunch of useless nonsense to the browser. Clearing the noise can let you see your output much more easily.
console.table()
Itâs surprising that this isnât better known, but the console.table()
function is intended to display tabular data in a way thatâs much neater than just dumping out the raw array of objects.
As an example, hereâs a list of data.
const transactions = [{
id: "7cb1-e041b126-f3b8",
seller: "WAL0412",
buyer: "WAL3023",
price: 203450,
time: 1539688433
},
{
id: "1d4c-31f8f14b-1571",
seller: "WAL0452",
buyer: "WAL3023",
price: 348299,
time: 1539688433
},
{
id: "b12c-b3adf58f-809f",
seller: "WAL0012",
buyer: "WAL2025",
price: 59240,
time: 1539688433
}];
If we use console.log
to dump out the above we get some pretty unhelpful output:
ⶠ(3) [{â¦}, {â¦}, {â¦}]
The little arrow lets you click down and open up the array, sure, but itâs not really the âat a glanceâ that weâd like.
The output from console.table(data)
though, is a lot more helpful.
The optional second argument is the list of columns you want. Obviously defaults to all columns, but we can also do this.
> console.table(data, ["id", "price"]);
We get this output, showing only the id and the price. Useful for overly large objects with largely irrelevant detail. The index column is auto-created and doesnât go away as far as I can tell.
Something to note here is that this is out of order â the arrow on the far right column header shows why. I clicked on that column to sort by it. Very handy for finding the biggest or smallest of a column, or just getting a different look at the data. That functionality has nothing to do with only showing some columns, by the way. Itâs always available.
console.table()
only has the ability to handle a maximum of 1000 rows, so it might not be suitable to all datasets.
console.assert()
A function whose usefulness is often missed, assert()
is the same as log()
but only in the case where the first argument is falsey. It does nothing at all if the first argument is true.
This can be particularly useful for cases where you have a loop (or several different function calls) and only one displays a particular behaviour. Essentially itâs the same as doing this.
if (object.whatever === 'value') {
console.log(object);
}
To clarify, when I say âthe same asâ I should better say that itâs the opposite of doing that. So youâd need to invert the conditional.
So letâs assume that one of our values above is coming through with a null
or 0
in its timestamp, which is screwing up our code formatting the date.
console.assert(tx.timestamp, tx);
When used with any of the valid transaction objects it just skips on past. But the broken one triggers our logging, because the timestamp is 0 or null, which is falsey.
Sometimes we want more complex conditionals. For example, weâve seen issues with the data for user WAL0412
and want to display out only transactions from them. This is the intuitive solution.
console.assert(tx.buyer === 'WAL0412', tx);
This looks right, but wonât work. Remember, the condition has to be false⦠we want to assert, not filter.
console.assert(tx.buyer !== 'WAL0412', tx);
This will do what we want. Any transaction where the buyer is not WAL0412 will be true on that conditional, leaving only the ones that are. Or⦠arenât not.
Like a few of these, console.assert()
isnât always particularly useful. But it can be an elegant solution in specific cases.
console.count()
Another one with a niche use, count simply acts as a counter, optionally as a named counter.
for(let i = 0; i < 10000; i++) {
if(i % 2) {
console.count('odds');
}
if(!(i % 5)) {
console.count('multiplesOfFive');
}
if(isPrime(i)) {
console.count('prime');
}
}
This is not useful code, and a bit abstract. Also Iâm not going to demonstrate the isPrime
function, just pretend it works.
What weâll get should be essentially a list of
odds: 1
odds: 2
prime: 1
odds: 3
multiplesOfFive: 1
prime: 2
odds: 4
prime: 3
odds: 5
multiplesOfFive: 2
...
And so on. This is useful for cases where you may have been just dumping out the index, or you would like to keep one (or more) running counts.
You can also use console.count()
just like that, with no arguments. Doing so calls it default
.
Thereâs also a related console.countReset()
that you can use to reset the counter if you like.
console.trace()
This is harder to demo in a simple bit of data. Where it excels is when youâre trying to figure out inside a class or library which actual caller is causing the problem.
For example, there might be 12 different components calling a service, but one of them doesnât have a dependency set up properly.
export default class CupcakeService {
constructor(dataLib) {
this.dataLib = dataLib;
if(typeof dataLib !== 'object') {
console.log(dataLib);
console.trace();
}
} ...}
Using console.log()
alone here would tell us what the dataLib is being passed in as, but not where. The stacktrace, though, will tell us very clearly that the problem is Dashboard.js
, which we can see is new CupcakeService(false)
and causing the error. And now we get bullied into using TypeScript.
console.time()
A dedicated function for tracking time taken for actions, console.time() is better way to track the microtime taken for JavaScript executions.
function slowFunction(number) {
var functionTimerStart = new Date().getTime();
// something slow or complex with the numbers.
// Factorials, or whatever.
var functionTime = new Date().getTime() - functionTimerStart;
console.log(`Function time: ${ functionTime }`);
}var start = new Date().getTime();
for (i = 0; i < 100000; ++i) {
slowFunction(i);
}
var time = new Date().getTime() - start;
console.log(`Execution time: ${ time }`);
This is an old school method. I should also point to the console.log above. A lot of people donât realise you can use template strings and interpolation there, but you can. From time to time it helps.
So letâs modernise the above.
const slowFunction = number => {
console.time('slowFunction');
// something slow or complex with the numbers.
// Factorials, or whatever.
console.timeEnd('slowFunction');
}console.time();
for (i = 0; i < 100000; ++i) {
slowFunction(i);
}console.timeEnd();
We now no longer need to do any math or set temporary variables.
console.group()
Now weâre probably in the most complex and advanced area of the console output. Group lets you⦠well, group things. In particular it lets you nest things. Where it excels is in terms of showing structure that exists in code.
// this is the global scope
let number = 1;console.group('OutsideLoop');
console.log(number);
console.group('Loop');for (let i = 0; i < 5; i++) {
number = i + number;
console.log(number);
}console.groupEnd();
console.log(number);
console.groupEnd();
console.log('All done now');
This is kind of rough again. You can see the output here.
Not really useful for much, but you could potentially see how some of these are combined.
class MyClass {
constructor(dataAccess) {
console.group('Constructor');
console.log('Constructor executed');
console.assert(typeof dataAccess === 'object',
'Potentially incorrect dataAccess object');
this.initializeEvents();
console.groupEnd();
} initializeEvents() {
console.group('events');
console.log('Initialising events');
console.groupEnd();
}
}let myClass = new MyClass(false);
This is a lot of work and a lot of code for debugging info that might not be all that useful. But itâs nevertheless an interesting idea, and you can see how much clearer it can make the context of your logging.
There is a final point to make on these, which is console.groupCollapsed
. Functionally this is the same as console.group
but the block starts off closed. Itâs not as well supported, but if you have a huge block of nonsense you might want hidden by default itâs an option.
Conclusion
Thereâs not really much of a conclusion to make here. All of these tools are potentially useful, in cases where you might just want a tiny bit more than console.log(pet)
but donât really need a debugger.
Probably the most useful is console.table
, but all the others have their place as well. Iâm a fan of console.assert
for cases where we want to debug something out, but only under a specific condition.