Intro
This article is about making a npm CLI package to install / manage a self-made framework.
The first thing I did was making 2 repositories on git:
Workspace
I added them to a vscode workspace for simplicity. You might also want to add a 3rd folder to it, where you will test your CLI
In the CLI, run the following commands:
npm init -y
npm install commander axios tar
The first command creates the package.json, and the second one adds a necessary package.
Then add the following to your package.json:
"bin": {
"<your-command>": "./bin/<your-command>.js"
},
Then add a folder called bin
with a file called <your-command>.js
inside it.
<your-command>
Make sure to replace
<your-command>
with the name you want to use, like angular hasng
, and I’ll usekazamata
.
Now add the following code to <your-command>.js
:
const { Command } = require('commander');
const axios = require('axios');
const tar = require('tar');
const fs = require('fs');
const path = require('path');
const program = new Command();
// Function to download and extract a GitHub release
async function downloadRelease(repo, dest) {
const url = `https://api.github.com/repos/${repo}/releases/latest`;
const response = await axios.get(url);
const tarballUrl = response.data.tarball_url;
const writer = fs.createWriteStream(dest);
const responseTar = await axios.get(tarballUrl, { responseType: 'stream' });
responseTar.data.pipe(writer);
return new Promise((resolve, reject) => {
writer.on('finish', resolve);
writer.on('error', reject);
});
}
// Function to extract the tarball
async function extractTarball(tarballPath, extractPath) {
await tar.x({
file: tarballPath,
cwd: extractPath,
strip: 1
});
}
// Ensure the directory exists before extracting the tarball
async function ensureDirectoryExists(directory) {
return fs.promises.mkdir(directory, { recursive: true });
}
program
.command('new <app-name>')
.option('--demo-app', 'Include demo app')
.action(async (appName, options) => {
const repo = options.demoApp ? 'your-username/your-demo-repo' : 'your-username/your-base-repo';
const dest = path.resolve(__dirname, `${appName}.tgz`);
const extractPath = path.resolve(process.cwd(), appName);
console.log(`Creating a new project in ${extractPath}...`);
try {
await ensureDirectoryExists(extractPath);
await downloadRelease(repo, dest);
await extractTarball(dest, extractPath);
fs.unlinkSync(dest);
console.log('Project created successfully!');
} catch (error) {
console.error('Error creating project:', error);
}
});
program
.command('generate <type> <name>')
.action((type, name) => {
console.log(`Generating ${type} named ${name}...`);
// Add your generation logic here
});
program.parse(process.argv);```
>[!important] `your-username/your-demo-repo`
>
>
>Don't forget to replace `your-username/your-demo-repo` with the actual path. For example, this is my repo:
>
>https://github.com/oldmartijntje/Project-Kazamata will become `oldmartijntje/Project-Kazamata`
npm link
and then in the repo where you want to install it, try:
and now it should be installed (if you made a release on your github repo.)
## uploading to npm
This tutorial shows how to upload a package to npm, it will work the same for your CLI tool, just from another path.
![[23.12 Angular Packages and creation#publish-to-npm|publish to npm]]