In this article, we are going to try setting up a development environment for Node.js service using Typescript, Docker, and VS Code debugger. Keep in mind that this setup may not suit all your needs and represents something that worked for me a couple of times. In some next articles, Iāll cover more detailed way how to set up other tools, like code formatting and file structure that works for me.
Prereqs š§±
Prerequisites for following this article are to haveĀ npm,Ā node,Ā VSCode,Ā andĀ docker & docker-composeĀ installed. Also, you should create a directory and initialize a project using theĀ npm initĀ command.
Inside the project directory we need the next packages as devDependencies for a starting point:
- esbuild ā you can opt to use different build tool
- esbuild-node-tsc
- nodemon
- typescript
- @types/node
Install them by issuing the command:
npm i --save-dev esbuild esbuild-node-tsc typescript nodemon @types/node
This is how our directory should look like

TSpilationā”
Next, we want to set up TS by creatingĀ tsconfig.json

TheĀ tsconfig.jsonĀ file can be configured per your preferences but whatās important is to haveĀ sourceMapĀ set toĀ trueĀ as well as specifiedĀ outDir.
Also, create the emptyĀ app.tsĀ file inside the srcĀ folder.
Next, we want to try out this config by creating a build script and running it. Add scripts section insideĀ package.json file and create build script like this.
etscĀ command is a command for esbuild-node-tsc. It usesĀ esbuildĀ as the main bundler and sets a default esbuild config for Node.js & TS setup using the previously configured tsconfig.json. More on how it works you can find atĀ esbuild-node-tscĀ github. This is completely up to you which build tool you want to use, for the sake of simplicity I have choosen theĀ etscĀ but you can use Webpack, plain esbuild, or whatever fits your needs and preferences the best. Just donāt forget to set your tool to generate source maps as those are going to be crucial for the debugger that we are going to set later.

After running the npm run build you should haveĀ dist (or however you named the outDir)Ā created withĀ app.jsĀ andĀ app.js.mapĀ files inside

Perpetuum mobileš
So far so good. The next thing we want to do is set up theĀ nodemonĀ just to make our life easier.
Create nodemon.jsonĀ file inside the root project directory and put the next configuration inside. Also, we want to update theĀ scriptsĀ section insideĀ package.jsonĀ to run the app in development mode.

This configuration is quite straightforward. The only field thatās maybe less known isĀ the legacy watchĀ which isĀ needed for running the app inside of the container environment.
Also, I hope you noticed that the exec command after building runs the node app using two parametersĀ --enable-source-maps --inspect=0.0.0.0:9229
Ā . First of those enables source maps for our debugging when printing the stack trace. Also, keep in mind that this flagĀ --enable-source-maps
Ā is still experimental and should not be used as is inside production. There are also someĀ issuesĀ reported regarding this feature. The other one enables a debugger to be attached at port 9229 and we need this one for later when we are going to attach VS Code debugger to it.
If you now runĀ npm run start:dev
Ā and visitĀ http://localhost:8080/Ā you should see the message our service returns.
Dockerize it! š³
Now we are ready to dockerize this app before we start rocking it seamlessly.
CreateĀ Dockerfile,Ā .dockerignore, andĀ docker-compose.ymlĀ files in the root directory of the project.

First, we tell Docker, insideĀ .dockerignore, that it should ignoreĀ node_modulesĀ andĀ dist directory when performing the copy.
Next inside Dockerfile we configure it to use node v16 as the base image, and set WORKDIR which is a dir where the next commands are going to be invoked. Then we copyĀ package.jsonĀ andĀ package-lock.jsonĀ files and run an npm install command to install the dependencies. After dependencies installation is finished we copy all other files inside the image (without node_modules and dis which are ignored), expose the 8080 port, and set the default command to be invoked at container start.

This docker-compose setup is also straightforward. We tell the docker-compose which directory to use when searching for Docker file withĀ build: .
Ā field, and which command to run as well as which ports to expose. Here we need to expose both ports for the app (8080) and the debugger (9229).
Quite an important part of the above config is the volumes we create. The first one is for the whole project directory which will enable us to edit files inside the container through VSCode and let the nodemon do its job. The second one lets us install dependencies locally, without the need of entering the container CLI and running the npm i command.
Letās launch itā¦
Go inside the root project directory and run the commandĀ docker-compose up
Ā .
Your app should start running and you can see the logs. Try editing some of the source files to see if the nodemon does itās job.
Disinsectionšš

It is time for us to debug this piece of software. You can always go with console.log but I think I donāt need to explain why thatās a bad practice.
For this setup, we are going to use VS Code debugger. It may be intimidating at the beginning of use but you will see how simple it is to run it and youāll forget about console methods soon after.
Inside the root project directory create a new dir namedĀ .vscodeĀ and inside of it create fileĀ launch.json.

With this configuration, we tell VS Code debugger to attach to the currently running process, at the port of 9229. Besides that, we configure the remote root directory of the container toĀ usr/src/app
, which matches the one from Dockerfile, as well as local root which matches our local directory. Very important is this outFiles config which tells VS Code debugger that the build codeĀ .jsĀ files are inside ofĀ distĀ directory so it knows where to find source and source maps and connect them to theĀ .tsĀ files.
Now go to your source .ts files, and place a breakpoint at the line of your choice, preferably some part of the code that is executed on each request.
Go to the Debug tab inside VS Code and run the debugger we configured previously.
Run the browser and visitĀ http://localhost:8080/Ā and switch back to VS Code. You should see something like what is shown in the next image.

On the left, you can see the JS scopes currently active.
One more note: If you are using WSL, keep in mind that you want your VS Code to be up and running in remote WSL environment when starting a Debugger.
I hope you enjoyed this one and found it useful. There are possibly things that could be done or explained better but as I believe you should dig into it by yourself and not use this one as the only source of truth. If you have any suggestions or find any mistakes please inform me about it. š