A simple way to replace npm scripts in Deno

vonheikemen

Heiker

Posted on May 14, 2020

A simple way to replace npm scripts in Deno

Update 2022-04-13:

Since version 1.20 deno now includes a task runner. You can find the details in the release note: new subcommand deno task.

Feel free to ignore the rest of this article.

Sooo I tried deno, did the traditional "hello world" and 5 seconds later I felt the pain of not having the npm cli and the package.json file. I can't npm start anymore and it bothers me more than it should. Say whatever you want about the node_module folder and npm, Inc. but npm scripts are the bee's knees.

Anyway, as I was saying, Deno's simplicity just punched me in the face. Now this is usually the part were I search for a third party library to fill the void, but not today. Today I just want to run a couple of commands in a sequence, that's not hard, I can do that with some functions. So I created this file in the root of the project.

// Taskfile.js

function run([name, ...args], tasks) {
  name in tasks 
    ? tasks[name](...args) 
    : console.log(`Task "${name}" not found`);
}

async function exec(args) {
  const proc = await Deno.run({ cmd: args }).status();

  if(proc.success == false) {
    Deno.exit(proc.code);
  }

  return proc;
}
Enter fullscreen mode Exit fullscreen mode

The function run is the one that decides which task is going to be executed, and it will be kind enough to tell you if the task you asked for can't be found. But the interesting part of this story is exec, that's the one that's going to execute the external commands I need using Deno.run. Now we can define our tasks.

run(Deno.args, {
  start() {
    exec(['deno', 'run', './src/mod.ts']);
  },
  async install() {
    await exec(['echo', 'Installing stuff....']);
    // do other things
  },
  echo(str) {
    return exec(['echo', str, "\nThis is javascript, y'all"]);
  },
  async test() {
    // need more power? You got it
    if(/* some condition */) {
       await this.echo('Awesome');
    }

    // do other things
  }
});
Enter fullscreen mode Exit fullscreen mode

You can make this even better if you use a dependency to parse the arguments. Deno has one on their standard library.

This is how you run it.

deno run --allow-run ./Taskfile.js start
Enter fullscreen mode Exit fullscreen mode

Now simplicity strikes again. That is quite a command, no one wants to type that, lucky for us we have the shell on our side. To solve this we can make an alias.

alias deno-task='deno run --allow-run ./Taskfile.js'
Enter fullscreen mode Exit fullscreen mode

On windows using powershell.

Set-Alias -Name deno-task -Value deno run --allow-run ./Taskfile.js
Enter fullscreen mode Exit fullscreen mode

You can even improve this with some shell kung-fu.

alias deno-root-task='deno run --allow-run $(git rev-parse --show-toplevel)/Taskfile.js'
Enter fullscreen mode Exit fullscreen mode

If you are in a folder controlled by git, this command can be used to make sure you execute the Taskfile of the root of the project.

If anyone knows how to make an alias in cmd.exe, please put it in the comments.

Now we are done, we can use deno-task start to start our application/script or use any other custom commands to automate what we need. There is a bunch of things that this script doesn't do but if npm scripts were enough for you so will this.

edit: I wrote another post about this exact same thing but using a "shell function": Extending the deno cli using a shell function


Thank you for your time. If you find this article useful and want to support my efforts, consider leaving a tip in ko-fi.com/vonheikemen.

buy me a coffee

💖 💪 🙅 🚩
vonheikemen
Heiker

Posted on May 14, 2020

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related