The product owner comes and says: We need to rewrite everything from Node to Deno…

Content

In the fast-paced world of web development, innovation is constant. This article explores Deno, comparing it to Node.js, examining their similarities, differences, and the possibility of Deno replacing Node.js. Let's journey from Node to Deno, embracing change and discovering new possibilities.

Let's start with the riddle, who is shown in this photo? 🙂

The man, creator of several well-known technologies that use JS.
The man, creator of several well-known technologies that use JS.

Most likely, you have worked with his product. 😉

Answer

Ryan Dahl is a software engineer and the original developer of the Node.js (~2009 year). In January 2012, Dahl announced that he turned over the reins to NPM creator.

In 2018 he made a presentation '10 things I regret about Node.js' at the JSConf conference. Also, at this presentation, he announced Deno — a secure runtime for JavaScript and TypeScript.

In this article, we will take a look at Deno and how it compares to Node.js to help understand what they have in common, how they differ and is it real that Deno will kill Node.js?

Project

Let's imagine a situation: A product owner comes to us and tells us to rewrite the project from Node to Deno.

But why? Who knows, it’s just an example. 🙂

For now, let's imagine that there is no other way around this decision and we need to do it. But then what? Do we really need to rewrite most of the code in the project?

Let's check this with an example of a small project.

  • Project structure:

    ├── node_modules
    ├── public
    ├── src/
    │   ├── api/
    │   │   ├── /* apis */
    │   │   └── api.ts
    │   ├── common/
    │   │   ├── enums/
    │   │   │   ├── /* enums */
    │   │   │   └── index.ts
    │   │   ├── interfaces/
    │   │   │   ├── /* interfaces */
    │   │   │   └── index.ts
    │   │   └── types/
    │   │       ├── /* types */
    │   │       └── index.ts
    │   ├── helpers/
    │   │   ├── /* helpers */
    │   │   └── index.ts
    │   ├── repositories/
    │   │   ├── /* repositories */
    │   │   └── repositories.ts
    │   ├── services/
    │   │   ├── /* services */
    │   │   └── services.ts
    │   └── server.ts
    ├── .env
    ├── .eslintrc.yml
    ├── package-lock.json
    ├── package-lock.json
    └── tsconfig.json
  • Dependencies:

    // package.json
    
    {
      "private": true,
      "scripts": {
        "lint:js": "eslint --ext .js,.ts src",
        "lint": "npm run lint:js",
        "start": "nodemon --exec ts-node --files -r dotenv/config ./src/server.ts"
      },
      "dependencies": {
        "axios": "0.21.1",
        "dotenv": "8.2.0",
        "koa": "2.13.1",
        "koa-bodyparser": "4.3.0",
        "koa-router": "10.0.0",
        "koa-static": "5.0.0"
      },
      "devDependencies": {
        "@types/jest": "26.0.22",
        "@types/koa": "2.13.1",
        "@types/koa-bodyparser": "4.3.0",
        "@types/koa-router": "7.4.2",
        "@types/koa-static": "4.0.1",
        "@types/node": "14.14.41",
        "@typescript-eslint/eslint-plugin": "4.22.0",
        "@typescript-eslint/parser": "4.22.0",
        "eslint": "7.24.0",
        "jest": "26.6.3",
        "nodemon": "2.0.7",
        "prettier": "2.2.1",
        "ts-jest": "26.5.5",
        "ts-node": "9.1.1",
        "typescript": "4.2.4"
      }
    }
  • Root file:

    // src/server.ts
    
    import { resolve } from 'path'
    import Koa from 'koa'
    import serve from 'koa-static'
    import Router from 'koa-router'
    import bodyParser from 'koa-bodyparser'
    import { ENV } from './common/enums'
    import { initRepositories } from './repositories/repositories'
    import { initServices } from './services/services'
    import { initApis } from './api/api'
    
    const app = new Koa()
    
    app.use(bodyParser())
    
    const repositories = initRepositories()
    
    const services = initServices({
      repositories,
    })
    
    const apiRouter = initApis({
      Router,
      services,
    })
    
    app.use(apiRouter.routes())
    
    app.use(serve(resolve(__dirname, '../public')))
    
    app.listen(ENV.APP.SERVER_PORT)
    
    console.log(`Listening to connections on port — ${ENV.APP.SERVER_PORT}`)

Nothing special. The project is written using a node-framework - Koa (next generation of the Express node-framework).

What it can do:

  • Serve static content

    > curl "<http://localhost:3000>"
    
    <!DOCTYPE html>
    <html lang="en">
      <head>
        <meta charset="UTF-8" />
        <meta http-equiv="X-UA-Compatible" content="IE=edge" />
        <meta name="viewport" content="width=device-width, initial-scale=1.0" />
        <title>Document</title>
      </head>
      <body>
        <h1>Hello world 😛</h1>
      </body>
    </html>
  • Posts api (wrapper for the jsonplaceholder service).

    > curl "<http://localhost:3000/api/v1/posts/1>"
    
    {"userId":1,"id":1,"title":"sunt aut facere repellat provident occaecati excepturi optio reprehenderit","body":"quia et suscipit\\nsuscipit recusandae consequuntur expedita et cum\\nreprehenderit molestiae ut ut quas totam\\nnostrum rerum est autem sunt rem eveniet architecto"}
  • Books CRUD api (all data is stored in the json format).

    > curl "<http://localhost:3000/api/v1/books/1>"
    
    {"id":"1","name":"DDD"}

That's all. Perfect!

Let's move on to the most interesting part.

From Node to Deno

Before we start, let's take a look at the definition of these two technologies that we can find on the technology homepage.

Node — is a JavaScript runtime.

Deno — is a secure runtime for JavaScript and TypeScript.

Both definitions contain runtime and JavaScript. Does it mean that we can run the same code on both platforms? Not at all.

Let's try to run the code that we wrote on NodeJS base using Deno (if you don't have Deno installed yet, you can find how to do it here):

> deno run src/server.ts

Aaaand we get an error:

error: Cannot resolve module "file:///src/services/services" from "file:///src/server.ts".
  at file:///src/server.ts:8:0

But interestingly, we did not receive an error saying that we use TypeScript code. This is because Deno supports TypeScript out of the box!

If we tried to run code written with TypeScript in NodeJS without using ts-node (or any other additional package), we would immediately get errors about the unknown syntax.

Okay, let's try to fix the differences between Node and Deno.

First of all, let's install an official extension to help us develop on Deno and add this to the editor settings:

// .vscode/settings.json

{
  "deno.enable": true,
  "deno.lint": true,
  "deno.unstable": true
}

Then, let's remove the things that are completely useless for Deno that are in NodeJS.

> rm -rf node_modules package-lock.json package.json

Stop what? Where do we store our packages now?

Do not worry, we will store the dependencies we need here:

// src/dependencies.ts

import '<https://deno.land/x/dotenv/load.ts>'
import * as Oak from '<https://deno.land/x/oak/mod.ts>'

export { Oak }

Two questions immediately arise: what is dependencies.ts and why do we use a link to import packages?

Let's start from the end. Two reasons that Ryan talked about in his presentation and that he tried to fix this in Deno are node_modules and package.json.

Why:

  • package.json:

    • Allow Node's require() to inspect package.json files for "main";

    • Included NPM in the Node distribution, which much made it the defacto standard;

    • It's unfortunate that there is centralized (privately controlled even) repository for modules;

    • Allowing package.json gave rise to the concept of a "module" as a directory of files;

    • This is no a strictly necessary abstraction - and one that doesn't exist on the web;

    • package.json now includes all sorts of unnecessary information. License? Repository? Description? It's boilerplate noise;

    • If only relative files and URLs were used when importing, the path defines the version. There is no need to list dependencies.

  • node_modules:

    • It massively complicates the module resolution algorithm;

    • vendored-by-default has good intentions, but in practice just using $NODE_PATH wouldn't have precluded that;

    • Deviates greatly from browser semantics;

    • It's my fault and I'm very sorry;

    • Unfortunately it's impossible to undo now.

(The reasons are taken from Dahl's presentation). When the program is launched for the first time, modules are downloaded and cached in the system. And when reused, they will be taken from there. While reusing modules, they will be taken from the cache.

Now about dependencies.ts. It's just a file that I created to store all third-party project's dependencies (the name can be whatever). Third-party modules can be included in any part of the program but this is not a good practice. It is better to keep all the modules in one place.

Let's try to run the code again. We get the following error:

error: Is a directory (os error 21)
  at file:///src/server.ts:6:0

On the sixth line we have this:

// src/server.ts

import { ENV } from './common/enums'

// src/common/enums/index.ts

export * from './api'
export * from './app'
export * from './file'
export * from './http'

Everything looks fine. What is the problem?

It would seem that we can miss the file name if the file is called index.

This is possible but only in NodeJS. In Deno we must always explicitly specify the file with its extension.

These are two more things Dahl regrets about Node:

  • index.js:

    • I thought it was cute because there was index.html;

    • It needlessly complicated the module loading system;

    • It became especially unnecessary after require supported package.json.

  • modules without the extension:

    • Needlessly less explicit;

    • Not how browser JavaScript works. You cannot omit the .js in a script tag src attribute;

    • The module loader has to query the file system at multiple locations trying to guess what the user intended.

Let's fix this and a few other errors that we will run into.

  1. Resole magic names

    // src/common/enums/index.ts
    
    export * from './api/index.ts'
    export * from './app/index.ts'
    export * from './file/index.js'
    export * from './http/index.ts'
  2. Resolve magic variables/functions/modules that are only available in Node

    error: Uncaught ReferenceError: __dirname is not defined
    
    - app.use(serve(resolve(__dirname, '../public')));
    + await Oak.send(ctx, ctx.request.url.pathname, {
    +   root: 'public',
    +   index: 'index.html',
    + });
    error: Uncaught ReferenceError: require is not defined
    
    - const contentType = require('./content-type.enum');
    + import contentType from './content-type.enum.ts';
    
    error: Uncaught ReferenceError: module is not defined
    
    - module.exports = {
    + export {
  3. Resolve take variables from env

    
    // src/common/enums/app/env.enum.ts
    
    - const { PORT, PLACEHOLDER_API_URL } = process.env;
    
    const ENV = {
      APP: {
    -    SERVER_PORT: <string>PORT,
    +    SERVER_PORT: Number(Deno.env.get('PORT')),
      },
      API: {
        V1_PATH: '/api/v1',
      },
      API_URL: {
    -    PLACEHOLDER_API: <string>PLACEHOLDER_API_URL,
    +    PLACEHOLDER_API: <string>Deno.env.get('PLACEHOLDER_API_URL'),
      },
    } as const;
  4. Resolve writeFile/readFile helpers

    
    // src/helpers/fs/read-file/read-file.helper.helper.ts
    
    - import fs from 'fs/promises';
    
    - const readFile = (path: string): Promise<Buffer> => {
    -   return fs.readFile(path);
    + const readFile = (path: string): Promise<string> => {
    +   return Deno.readTextFile(path);
    };
    
    export { readFile };
    
    // src/helpers/fs/write-file/write-file.helper.helper.ts
    - import fs from 'fs/promises';
    
    - const writeFile = (path: string, data: string | Uint8Array): Promise<void> => {
    -   return fs.writeFile(path, data);
    + const writeFile = (path: string, data: Uint8Array): Promise<void> => {
    +   return Deno.writeFile(path, data);
    };
    
    export { writeFile };
  5. Resolve Paths

    
    src/repositories/book/books.repository.ts
    
    ...
    
    - const booksDataPath = path.resolve(__dirname, './books.json');
    + const booksDataPath = new URL('./books.json', import.meta.url).pathname;
    
    class Books implements IRepository<Book> {
    
      ...
    
      private _saveBooks(books: Book[]): Promise<void> {
    -    return writeFile(booksDataPath, JSON.stringify(books));
    +    return writeFile(
    +      booksDataPath,
    +      new TextEncoder().encode(JSON.stringify(books)),
    +    );
      }
    }
  6. Fetch out of the box

    
    // src/services/http/http.service.ts
    
    class Http {
    -  #http: AxiosInstance;
    -
    -  constructor() {
    -    this.#http = axios.create({});
    -  }
    
      public load<T = unknown>(
        url: string,
    -    options: AxiosRequestConfig = {
    +    options: RequestInit = {
          method: HttpMethod.GET,
        },
      ): Promise<T> {
    -    return this.#http
    -      .request<T>({ url, ...options })
    -      .then(Http.getData)
    -      .catch(Http.catchError);
    +    return fetch(url, options)
    +      .then(this._checkStatus)
    +      .then((res) => this._parseJSON<T>(res))
    +      .catch(this._throwError);
      }
    
    -  static getData<T>(response: AxiosResponse<T>): T {
    -    return response.data;
    +  private _checkStatus(response: Response): Response | never {
    +    if (!response.ok) {
    +      throw new Error(response.statusText);
    +    }
    +
    +    return response;
      }
    
    -  static catchError(err: AxiosError<unknown>): never {
    -    const { response } = err;
    +  private _parseJSON<T>(response: Response): Promise<T> {
    +    return response.json();
      }
    
    -    throw new Error(response?.statusText);
    +  private _throwError(err: Error): never {
    +    throw err;
      }
    }

Of course there were more changes, but these are the most interesting things that deserve attention.

Root file using Deno:

// src/server.ts

import { Oak } from './dependencies.ts'
import { ENV } from './common/enums/index.ts'
import { initRepositories } from './repositories/repositories.ts'
import { initServices } from './services/services.ts'
import { initApis } from './api/api.ts'

const app = new Oak.Application()

const repositories = initRepositories()

const services = initServices({
  repositories,
})

initApis({
  Router: Oak.Router,
  services,
  app,
})

app.use(async (ctx) => {
  await Oak.send(ctx, ctx.request.url.pathname, {
    root: 'public',
    index: 'index.html',
  })
})

app.listen({
  port: ENV.APP.SERVER_PORT,
})

console.log(`Listening to connections on port — ${ENV.APP.SERVER_PORT}`)

Almost the same, isn't it? 🙂

Permissions

Finally, let's run our refactored app!

We are still getting the error:

error: Uncaught PermissionDenied: Requires env access to "PORT", run again with the --allow-env flag
    SERVER_PORT: Deno.env.get('PORT'),

But this time it is an error that was not received before.

Another issue which Ryan regrets is security in NodeJS:

  • V8 by itself is a very good security sandbox;

  • Had I put more thought into how that could be maintained for certain applications, Node could have had some nice security guarantees not available in any other language;

  • Example: Your linter shouldn't get complete access to your computer and network.

Every time we run a program on Deno, we need to specify the appropriate permissions that it will possess.

To run our app we need to use these permissions:

> deno run --allow-env --allow-read --allow-write --allow-net src/server.ts

By this link you can find a list of all permissions.

Let's try to run it again:

> deno run --allow-env --allow-read --allow-write --allow-net src/server.ts

Listening to connections on port — 3000

Let's try to call the APIs:

> curl "<http://localhost:3000/api/v1/posts/1>"

{"userId":1,"id":1,"title":"sunt aut facere repellat provident occaecati excepturi optio reprehenderit","body":"quia et suscipit\\nsuscipit recusandae consequuntur expedita et cum\\nreprehenderit molestiae ut ut quas totam\\nnostrum rerum est autem sunt rem eveniet architecto"}
> curl "<http://localhost:3000/api/v1/books/1>"

{"id":"1","name":"DDD"}

Everything works! 🔥

But look at the code differences:

A pull request in which approximately 8,000 lines were removed, showing changes during migration from Node.js to Deno.
A pull request in which approximately 8,000 lines were removed, showing changes during migration from Node.js to Deno.

Amazing, isn't it?

Okay-okay, most of the changes (almost all) are due to dependencies, since now we only have 2 dependencies.

Easter eggs

Do you like easter eggs? Hope so 🙂

Let's see something:

const checkIsSameStr = (stringA: string, stringB: string): boolean => {
  const unifyStr = (str: string) => str.toLowerCase().split('').sort().join('')

  return unifyStr(stringA) === unifyStr(stringB)
}

const isEasterEgg = checkIsSameStr('Node', 'Deno') // true

// Koa - the middleware framework for Node
// Oak - the middleware framework for Deno
const isEasterEgg = checkIsSameStr('Koa', 'Oak') // true

Not sure if this was done on purpose (I hope so), but there is something similar here, isn't there? 😉

(There are a number of other packages that are named similarly.)

Conclusions

Until recently Node had almost no competitors (io.js ?) and was almost the only platform where we could run JavaScript on the server.

But now, Node has a worthy competitor, Deno — a secure runtime for JavaScript and TypeScript, who will step on its heels every day.

Competition is usually always good!

This article does not cover all topics such as code linting, code formatting, code testing, etc. By the way, most of these things Deno has out of the box 😉

There is no need to run and rewrite everything but at least every JS developer should take a look and try Deno. People who have already worked with Node shouldn't take a lot of effort to make friends with this beautiful technology.

Webmentions

If you liked this article and think others should read it, please share it. Leave comments on platforms such as dev.to, twitter.com, etc., and they will magically appear here ✨

  • 0 reposts
  • 0 comments

These are webmentions via the IndieWeb and webmention.io. Mention this post from your site:

Not an Easter Egg (but actually it is, yes 😁)