0.7.0 • Published 7 months ago

gunshi v0.7.0

Weekly downloads
-
License
MIT
Repository
github
Last release
7 months ago

Version CI InstallSize JSR

Gunshi is a modern javascript command-line library

!TIP gunshi (軍師) is a position in ancient Japanese samurai battle in which a samurai devised strategies and gave orders. That name is inspired by the word "command".

✨ Features

Gunshi is designed to simplify the creation of modern command-line interfaces:

  • 📏 Simple: Run the commands with a simple API.
  • ⚙️ Declarative configuration: Configure the command modules declaratively.
  • 🛡️ Type Safe: Arguments parsing and options value resolution type-safely by args-tokens
  • 🧩 Composable: Sub-commands that can be composed with modularized commands.
  • Lazy & Async: Command modules lazy loading and asynchronously executing.
  • 📜 Auto usage generation: Automatic usage message generation with modularized commands.
  • 🎨 Custom usage generation: Usage message generation customizable.
  • 🌍 Internationalization: I18n out of the box and locale resource lazy loading.

💿 Installation

🐢 Node

# npm
npm install --save gunshi

## pnpm
pnpm add gunshi

## yarn
yarn add gunshi

🦕 Deno

deno add jsr:@kazupon/gunshi

🥟 Bun

bun add gunshi

🚀 Usage

📏 Simple API

Gunshi has a simple API that is a facade:

import { cli } from 'gunshi'

const args = process.argv.slice(2)
// run a simple command
cli(args, () => {
  // something logic ...
  console.log('Hello from Gunshi!', args)
})

⚙️ Declarative Configuration

Configure commands declaratively:

import { cli } from 'gunshi'

// define a command with declarative configuration, using commandable object
const command = {
  name: 'greet',
  description: 'A greeting command',
  options: {
    name: { type: 'string', short: 'n' },
    greeting: { type: 'string', short: 'g', default: 'Hello' },
    times: { type: 'number', short: 't', default: 1 }
  },
  usage: {
    options: {
      name: 'Name to greet',
      greeting: 'Greeting to use (default: "Hello")',
      times: 'Number of times to repeat the greeting (default: 1)'
    }
  },
  run: ctx => {
    const { name = 'World', greeting, times } = ctx.values
    for (let i = 0; i < times; i++) {
      console.log(`${greeting}, ${name}!`)
    }
  }
}

// run a command that is defined above
// (the 3rd argument of `cli` is the command option)
cli(process.argv.slice(2), command, {
  name: 'my-app',
  version: '1.0.0',
  description: 'My CLI application'
})

For more detailed examples, check out the playground/declarative in the repository.

🛡️ Type-Safe Arguments

Gunshi provides type-safe argument parsing with TypeScript:

import { cli } from 'gunshi'
import type { ArgOptions, Command, CommandContext } from 'gunshi'

// type-safe arguments parsing example
// this demonstrates how to define and use typed command options with `satisfies`

// define 'type-safe' command options with types
const options = {
  // define string option with short alias
  name: {
    type: 'string',
    short: 'n'
  },
  // define number option with default value
  age: {
    type: 'number',
    short: 'a',
    default: 25
  },
  // define boolean flag
  verbose: {
    type: 'boolean',
    short: 'v'
  }
} satisfies ArgOptions

// define 'type-safe' command
const command = {
  name: 'type-safe',
  options,
  run: (ctx: CommandContext<UserOptions, UserValues>) => {
    const { name, age, verbose } = ctx.values
    console.log(`Hello, ${name || 'World'}! You are ${age} years old.`)
  }
} satisfies Command<typeof options>

await cli(process.argv.slice(2), command)

For more detailed examples, check out the playground/type-safe in the repository.

🧩 Composable Sub-commands

Run a CLI with composable sub-commands:

import { cli } from 'gunshi'

// define 'create' command
const createCommand = {
  name: 'create',
  description: 'Create a new resource',
  options: {
    name: { type: 'string', short: 'n' }
  },
  run: ctx => {
    console.log(`Creating resource: ${ctx.values.name}`)
  }
}

// define 'list' command
const listCommand = {
  name: 'list',
  description: 'List all resources',
  run: () => {
    console.log('Listing all resources...')
  }
}

// prepare a Map of sub-commands
const subCommands = new Map()
subCommands.set('create', createCommand)
subCommands.set('list', listCommand)

// define the main ('resource-manager') command
const mainCommand = {
  name: 'resource-manager',
  description: 'Manage resources',
  run: () => {
    console.log('Use one of the sub-commands: create, list')
  }
}

// run the CLI with composable sub-commands
cli(process.argv.slice(2), mainCommand, {
  name: 'my-app',
  version: '1.0.0',
  subCommands
})

For more detailed examples, check out the playground/composable in the repository.

⏳ Lazy & Async Command Loading

Load commands lazily and execute them asynchronously:

import { cli } from 'gunshi'

// define a command that will be loaded lazily
const lazyCommand = async () => {
  // simulate async loading
  await new Promise(resolve => setTimeout(resolve, 1000))

  // return the actual command
  return {
    name: 'lazy',
    description: 'A command that is loaded lazily',
    run: async ctx => {
      // async execution
      await new Promise(resolve => setTimeout(resolve, 500))
      console.log('Command executed!')
    }
  }
}

// prepare a Map of sub-commands with lazy-loaded commands
const subCommands = new Map()
subCommands.set('lazy', lazyCommand)

// run the CLI with lazy-loaded commands
cli(
  process.argv.slice(2),
  { name: 'main', run: () => {} },
  {
    name: 'my-app',
    subCommands
  }
)

For more detailed examples, check out the playground/lazy-async in the repository.

📜 Auto Usage Generation

Gunshi automatically generates usage information:

import { cli } from 'gunshi'

const command = {
  name: 'app',
  description: 'My application',
  options: {
    path: { type: 'string', short: 'p' },
    recursive: { type: 'boolean', short: 'r' },
    operation: { type: 'string', short: 'o', required: true }
  },
  // define usage with object
  usage: {
    options: {
      path: 'File or directory path',
      recursive: 'Operate recursively on directories',
      operation: 'Operation to perform (list, copy, move, delete)'
    },
    examples: '# Example\n$ my-app --operation list --path ./src'
  },
  run: ctx => {
    // command implementation
  }
}

// run with --help to see the automatically generated usage information
cli(process.argv.slice(2), command, {
  name: 'my-app',
  version: '1.0.0'
})

For more detailed examples, check out the playground/auto-usage in the repository.

🎨 Custom Usage Generation

Customize the usage message generation:

import { cli } from 'gunshi'

// define custom header renderer
const customHeaderRenderer = ctx => {
  return Promise.resolve(`
╔═══════════════════════╗
║      ${ctx.env.name.toUpperCase()}      ║
╚═══════════════════════╝
${ctx.env.description}
Version: ${ctx.env.version}
`)
}

// define custom usage renderer
const customUsageRenderer = ctx => {
  const lines = []
  lines.push('USAGE:')
  lines.push(`  $ ${ctx.env.name} [options]`)
  lines.push('')
  lines.push('OPTIONS:')

  for (const [key, option] of Object.entries(ctx.options || Object.create(null))) {
    const shortFlag = option.short ? `-${option.short}, ` : '    '
    lines.push(`  ${shortFlag}--${key.padEnd(10)} ${ctx.translate(key)}`)
  }

  return Promise.resolve(lines.join('\n'))
}

// run with custom renderers
cli(
  process.argv.slice(2),
  { name: 'app', run: () => {} },
  {
    name: 'my-app',
    version: '1.0.0',
    description: 'My application',
    renderHeader: customHeaderRenderer,
    renderUsage: customUsageRenderer
  }
)

For more detailed examples, check out the playground/custom-usage in the repository.

🌍 Internationalization

Support internationalization:

import { cli } from 'gunshi'
import enUS from './locales/en-US.json' with { type: 'json' }

const command = {
  name: 'greeter',
  options: {
    name: { type: 'string', short: 'n' },
    formal: { type: 'boolean', short: 'f' }
  },
  // resource fetcher for translations
  resource: async ctx => {
    if (ctx.locale.toString() === 'ja-JP') {
      const resource = await import('./locales/ja-JP.json', { with: { type: 'json' } })
      return resource.default
    }

    // default to English
    return enUS
  },
  run: ctx => {
    const { name = 'World', formal } = ctx.values
    const greeting = formal ? ctx.translate('formal_greeting') : ctx.translate('informal_greeting')
    console.log(`${greeting}, ${name}!`)
  }
}

// run with locale support
cli(process.argv.slice(2), command, {
  name: 'my-app',
  version: '1.0.0',
  // set the locale via an environment variable
  // if Node v21 or later is used, you can use the built-in `navigator.language` instead)
  locale: new Intl.Locale(process.env.MY_LOCALE || 'en-US')
})

For more detailed examples, check out the playground/i18n in the repository.

💁‍♀️ Showcases

  • pnpmc: PNPM Catalogs Tooling

🙌 Contributing guidelines

If you are interested in contributing to gunshi, I highly recommend checking out the contributing guidelines here. You'll find all the relevant information such as how to make a PR, how to setup development) etc., there.

💖 Credits

This project is inspired and powered by:

  • citty, created by UnJS team and contributors
  • cline and claude 3.7 sonnet, examples and docs is generated

Thank you!

©️ License

MIT

0.7.0

7 months ago

0.6.2

7 months ago

0.6.1

7 months ago

0.6.0

7 months ago

0.5.4

8 months ago

0.5.3

8 months ago

0.5.2

8 months ago

0.5.1

8 months ago

0.5.0

8 months ago

0.4.0

8 months ago

0.3.1

8 months ago

0.3.0

8 months ago

0.2.2

8 months ago

0.2.1

8 months ago

0.2.0

8 months ago

0.1.1

8 months ago

0.0.0

8 months ago