Claude-skill-registry cli-command-development
Creating new CLI commands and topics for the B2C CLI using oclif
install
source · Clone the upstream repo
git clone https://github.com/majiayu000/claude-skill-registry
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/cli-command-development" ~/.claude/skills/majiayu000-claude-skill-registry-cli-command-development && rm -rf "$T"
manifest:
skills/data/cli-command-development/SKILL.mdsource content
CLI Command Development
This skill covers creating new CLI commands and topics for the B2C CLI.
Command Organization
Commands live in
packages/b2c-cli/src/commands/. The directory structure maps directly to command names:
commands/ ├── code/ │ ├── deploy.ts → b2c code deploy │ ├── activate.ts → b2c code activate │ └── list.ts → b2c code list ├── ods/ │ ├── create.ts → b2c ods create │ └── list.ts → b2c ods list └── mrt/ └── env/ └── var/ └── set.ts → b2c mrt env var set
Command Class Hierarchy
Choose the appropriate base class based on what your command needs:
BaseCommand (logging, JSON output, error handling) └─ OAuthCommand (OAuth authentication) ├─ InstanceCommand (B2C instance: hostname, code version) │ ├─ CartridgeCommand (cartridge path + filters) │ ├─ JobCommand (job execution helpers) │ └─ WebDavCommand (WebDAV root directory) ├─ MrtCommand (Managed Runtime API) └─ OdsCommand (On-Demand Sandbox API)
Import from
@salesforce/b2c-tooling-sdk/cli:
import { InstanceCommand, CartridgeCommand, OdsCommand } from '@salesforce/b2c-tooling-sdk/cli';
Standard Command Template
/* * Copyright (c) 2025, Salesforce, Inc. * SPDX-License-Identifier: Apache-2 * For full license text, see the license.txt file in the repo root */ import {Args, Flags} from '@oclif/core'; import {InstanceCommand} from '@salesforce/b2c-tooling-sdk/cli'; import {getApiErrorMessage} from '@salesforce/b2c-tooling-sdk'; import {t} from '../../i18n/index.js'; interface MyCommandResponse { success: boolean; data: SomeType[]; } export default class MyCommand extends InstanceCommand<typeof MyCommand> { static description = t('commands.topic.mycommand.description', 'Human-readable description'); static enableJsonFlag = true; static examples = [ '<%= config.bin %> <%= command.id %> arg1', '<%= config.bin %> <%= command.id %> --flag value', '<%= config.bin %> <%= command.id %> --json', ]; static args = { name: Args.string({ description: 'Description of the argument', required: true, }), }; static flags = { myFlag: Flags.string({ char: 'm', description: 'Flag description', default: 'defaultValue', }), myBool: Flags.boolean({ description: 'Boolean flag', default: false, }), }; async run(): Promise<MyCommandResponse> { // Validation - call appropriate require* methods this.requireServer(); // Access parsed args and flags const {name} = this.args; const {myFlag, myBool} = this.flags; this.log(t('commands.topic.mycommand.working', 'Working on {{name}}...', {name})); // Implementation const {data, error, response} = await this.instance.ocapi.GET('/some/endpoint'); if (error) { this.error(t('commands.topic.mycommand.error', 'Failed: {{message}}', { message: getApiErrorMessage(error, response), })); } const result: MyCommandResponse = { success: true, data, }; // JSON mode returns the object directly (oclif handles serialization) if (this.jsonEnabled()) { return result; } // Human-readable output this.log('Success!'); return result; } }
Adding a New Topic
When creating a new command topic, add it to
packages/b2c-cli/package.json in the oclif section:
{ "oclif": { "topics": { "newtopic": { "description": "Commands for new functionality" }, "newtopic:subtopic": { "description": "Subtopic commands" } } } }
Flag Patterns
Common Flag Types
static flags = { // String with short alias and env var fallback server: Flags.string({ char: 's', description: 'Server hostname', env: 'SFCC_SERVER', }), // Integer with default timeout: Flags.integer({ description: 'Timeout in seconds', default: 60, }), // Boolean with --no-* variant wait: Flags.boolean({ description: 'Wait for completion', default: true, allowNo: true, // enables --no-wait }), // Comma-separated multiple values channels: Flags.string({ description: 'Site channels (comma-separated)', multiple: true, multipleNonGreedy: true, delimiter: ',', }), // Enum-like options format: Flags.string({ description: 'Output format', options: ['json', 'csv', 'table'], default: 'table', }), // Conditional flag secret: Flags.string({ description: 'Client secret (required for private clients)', dependsOn: ['client-id'], }), };
Table Output
Use
createTable for consistent tabular output:
import {createTable, TableRenderer, type ColumnDef} from '@salesforce/b2c-tooling-sdk/cli'; type MyData = {id: string; name: string; status: string}; const COLUMNS: Record<string, ColumnDef<MyData>> = { id: { header: 'ID', get: (item) => item.id, }, name: { header: 'Name', get: (item) => item.name, }, status: { header: 'Status', get: (item) => item.status, extended: true, // Only shown with --extended }, }; const DEFAULT_COLUMNS = ['id', 'name']; const tableRenderer = new TableRenderer(COLUMNS); // In run(): tableRenderer.render(data, DEFAULT_COLUMNS); // With --columns flag support: const columns = this.flags.columns ? tableRenderer.validateColumnKeys(this.flags.columns.split(',')) : DEFAULT_COLUMNS; tableRenderer.render(data, columns);
Internationalization
All user-facing strings use the
t() function:
import {t} from '../../i18n/index.js'; // Basic usage this.log(t('commands.topic.cmd.message', 'Default message')); // With interpolation this.log(t('commands.topic.cmd.working', 'Processing {{count}} items...', {count: 5})); // For errors this.error(t('commands.topic.cmd.error', 'Failed: {{message}}', {message: err.message}));
Keys follow the pattern:
commands.<topic>.<command>.<key>
Validation Methods
Base classes provide validation helpers:
// From OAuthCommand this.requireOAuthCredentials(); // Ensures clientId + clientSecret this.hasOAuthCredentials(); // Returns boolean // From InstanceCommand this.requireServer(); // Ensures hostname is set this.requireCodeVersion(); // Ensures code version is set this.requireWebDavCredentials(); // Ensures WebDAV auth (Basic or OAuth) // From MrtCommand this.requireMrtCredentials(); // Ensures MRT API credentials
Accessing Clients
InstanceCommand provides lazy-loaded clients:
// OCAPI client (OAuth authenticated) const result = await this.instance.ocapi.GET('/code_versions'); // WebDAV client (Basic or OAuth authenticated) await this.instance.webdav.put('path/to/file', buffer); // ODS client (from OdsCommand) const sandboxes = await this.odsClient.GET('/sandboxes'); // MRT client (from MrtCommand) const projects = await this.mrtClient.GET('/api/projects/');
Error Handling
// Simple error (exits with code 1) this.error('Something went wrong'); // Error with suggestions this.error('Config file not found', { suggestions: ['Run b2c auth login first', 'Check your dw.json file'], }); // Warning (continues execution) this.warn('Deprecated flag used'); // API errors - use getApiErrorMessage for clean messages import {getApiErrorMessage} from '@salesforce/b2c-tooling-sdk'; const {data, error, response} = await this.instance.ocapi.GET('/sites', {...}); if (error) { this.error(t('commands.topic.cmd.apiError', 'API error: {{message}}', { message: getApiErrorMessage(error, response), })); }
Important: Always destructure
response alongside error when making API calls. The getApiErrorMessage utility extracts clean messages from ODS, OCAPI, and SCAPI error patterns, and falls back to HTTP status (e.g., "HTTP 521 Web Server Is Down") for non-JSON responses like HTML error pages.
See API Client Development for supported error patterns.
Creating a Command Checklist
- Create file at
packages/b2c-cli/src/commands/<topic>/<command>.ts - Choose appropriate base class
- Define
,static description
,examples
,argsflags - Set
for JSON output supportstatic enableJsonFlag = true - Implement
method with proper return typerun() - Add topic to
if newpackage.json - Add i18n keys for all user-facing strings
- Update skill in
if existsplugins/b2c-cli/skills/b2c-<topic>/SKILL.md - Update CLI reference docs in
docs/cli/<topic>.md - Build and test:
pnpm run build && pnpm --filter @salesforce/b2c-cli run test