Debugging

Debugging

In this guide, we will explore multiple ways of debugging your AdonisJS applications, from using the VSCode debugger to using Dump and Die and viewing the debug logs of the framework.

Debug using the VSCode debugger

Debugging AdonisJS applications using the VSCode debugger is straightforward. You only need to create a .vscode/launch.json file and use the Node.js debugger.

In the following example, we define a configuration to start the AdonisJS development server in the debug mode and then attach the VSCode debugger to it.

See also: VSCode Debugging Docs

.vscode/launch.json
{
"version": "0.2.0",
"configurations": [
{
"type": "node",
"request": "launch",
"name": "Dev server",
"program": "${workspaceFolder}/ace.js",
"args": ["serve", "--hmr"],
"skipFiles": ["<node_internals>/**"]
}
]
}

To start debugging:

  • Open the Command Palette with (CMD + Shift + P).
  • Search for Debug: Select and Start Debugging. You will find a list of launch options from the .vscode/launch.json file.
  • Select the Dev server option to run the HTTP server with the VSCode debugger.

Debugging tests

You may define another launch option to run tests in the debug mode.

.vscode/launch.json
{
"version": "0.2.0",
"configurations": [
{
"type": "node",
"request": "launch",
"name": "Dev server",
"program": "${workspaceFolder}/ace.js",
"args": ["serve", "--hmr"],
"skipFiles": ["<node_internals>/**"]
},
{
"type": "node",
"request": "launch",
"name": "Tests",
"program": "${workspaceFolder}/ace.js",
"args": ["test", "--watch"],
"skipFiles": ["<node_internals>/**"]
}
]
}

Debugging all other Ace commands

Defining individual launch options for every ace command is not a practical option. Therefore, you can define an attach configuration within the .vscode/launch.json file.

In attach mode, VSCode will attach its debugger with an already running Node.js process, given the process was started with the --inspect flag from within the VSCode integrated terminal.

Let's start by modifying the .vscode/launch.json file and adding the following configuration to it.

.vscode/launch.json
{
"version": "0.2.0",
"configurations": [
{
"type": "node",
"request": "attach",
"name": "Attach Program",
"port": 9229,
"autoAttachChildProcesses": true,
"skipFiles": ["<node_internals>/**"]
},
{
"type": "node",
"request": "launch",
"name": "Dev server",
"program": "${workspaceFolder}/ace.js",
"args": ["serve", "--hmr"],
"skipFiles": ["<node_internals>/**"]
},
{
"type": "node",
"request": "launch",
"name": "Tests",
"program": "${workspaceFolder}/ace.js",
"args": ["test", "--watch"],
"skipFiles": ["<node_internals>/**"]
}
]
}

To start debugging in attach mode:

  • Open the Command Palette with (CMD + Shift + P).
  • Search for Debug: Select and Start Debugging. You will find a list of launch options from the .vscode/launch.json file.
  • Select the Attach Program option.
  • Run an Ace command with the --inspect flag. For example:
    node --inspect ace migration:run

Debugging Edge templates

You can debug Edge templates similar to your application code written in TypeScript. However, with Edge, you cannot use the breakpoints provided by VSCode. Instead, you must use the @debugger tag to define an in-code breakpoint.

The debugger will show the compiled output for Edge templates.

@debugger

Dump and Die

Dump and Die (known as dd) is similar to the most loved debugging technique, console.log. However, the dd helper will halt the execution by throwing an exception and displaying the output inside the browser or the terminal.

The output is rendered as an HTML document when you use the dd helper during an HTTP request. Otherwise, the output is displayed within the terminal.

start/routes.ts
import User from '#models/user'
import router from '@adonisjs/core/services/router'
import { dd } from '@adonisjs/core/services/dumper'
router.get('/users', async () => {
const users = await User.all()
/**
* Visit the "/users" endpoint to view the dumped values
*/
dd(users)
return users
})

The output of dd slightly differs from what you see when using console.log.

  • You can see the source code location where the value was dumped.
  • You can view static properties of a class and prototype properties of an object.
  • By default, nested values up to 10 levels deep are displayed.
  • Support for multiple themes for the HTML output. You can choose between nightOwl, catppuccin, and minLight.

Edge helpers for debugging

You can use the dd helper within Edge templates via the @dd tag. Additionally, you can use the @dump helper, which doesn't throw an exception and continues rendering the rest of the template.

{{-- Dump template state and die --}}
@dd(state)
{{-- Dump template state and continue rendering --}}
@dump(state)

When using the @dump helper make sure there is an EdgeJS Stack named "dumper" on the page. The script and styles used by the @dump helper will be written to this stack for inclusion in the final HTML output.

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
@stack('dumper')
</head>
<body>
@dump(state)
</body>
</html>

Dumper settings

You can configure the dumper settings inside the config/app.ts file. This file should export a dumper configuration object, as shown below.

config/app.ts
/**
* The global configuration used by the "dd" helper. You can
* separately configure the settings for both the "console"
* and the "html" printers.
*/
export const dumper = dumperConfig({
/**
* Settings for the console printer
*/
console: {
depth: 10,
/**
* Objects that should not be further expanded. The
* array accepts an array of object constructor
* names.
*/
collapse: ['DateTime', 'Date'],
inspectStaticMembers: true,
},
/**
* Settings for the HTML printer
*/
html: {
depth: 10,
inspectStaticMembers: true,
},
})
showHidden

When set to true, the non-enumerable properties of an object will be processed. Default: false

depth

The depth at which to stop parsing nested values. The depth is shared among all tree-like data structures. For example, Objects, Arrays, Maps, and Sets. Default: 5

inspectObjectPrototype

Inspect prototype properties of an object. The non-enumerable properties of the prototype are included by default. Default: 'unless-plain-object'.

  • When set to true, the prototype properties will be processed for all the objects.
  • The prototype properties are never processed when set to false.
  • When set to 'unless-plain-object', the prototype properties of class instances will be processed.
inspectArrayPrototype

Inspect prototype properties of an Array. Default: false.

inspectStaticMembers

Inspect static members of a class. Even though functions and classes are technically the same, this config only applies to functions defined using the [class] keyword. Default: false.

maxArrayLength

Maximum number of members to process for Arrays, Maps, and Sets. Default: 100.

maxStringLength

Maximum number of characters to display for a string. Default: 1000.

collapse

An array of object constructor names that should not be further inspected.

Framework debug logs

You can view the framework debug logs using the NODE_DEBUG environment variable. The NODE_DEBUG flag is supported by the Node.js runtime, and you may use it to view logs for one or more modules using the module name.

For example, you can view logs for all AdonisJS packages using the NODE_DEBUG="adonisjs:*" value.

NODE_DEBUG="adonisjs:*" node ace serve --hmr

Similarly, you may use the NODE_DEBUG environment variable to view logs from the Node.js native modules like fs, net, module, and so on.

NODE_DEBUG="adonisjs:*,net,fs" node ace serve --hmr