skills/dart/dart-build-cli-app/SKILL.md
Build Dart command-line applications with proper entrypoint structure, exit codes, and cross-platform scripting. Use when creating CLI utilities, automation scripts, or standalone Dart executables.
npx skillsauth add dhruvanbhalara/skills dart-build-cli-appInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Initialize new CLI projects using the official Dart template to ensure standard directory structures.
dart create -t cli <project_name> to scaffold a console application with basic argument parsing.main()) exclusively in the bin/ directory.lib/src/ and expose public APIs via lib/<project_name>.dart.dart format . --set-exit-if-changed. This returns exit code 1 if formatting violations exist.Implement the args package to manage command-line arguments, flags, and subcommands.
ArgParser directly to define flags (addFlag) and options (addOption).git): Implement CommandRunner and extend Command for each subcommand.CommandRunner.argParser and command-specific arguments on the individual Command.argParser.UsageException to gracefully handle invalid arguments and display the automatically generated help text.Use the io and stack_trace packages to build reliable, production-ready CLI tools.
io package's ExitCode enum to return standard POSIX exit codes (e.g., ExitCode.success.code, ExitCode.usage.code).sharedStdIn from the io package if multiple asynchronous listeners need sequential access to standard input.Chain.capture() from the stack_trace package to track asynchronous stack chains.Trace.terse or Chain.terse to strip noisy core library frames and present readable errors to the user.Use test_process and test_descriptor to write high-fidelity integration tests for your CLI.
test_descriptor (d.dir, d.file).await d.Descriptor.create().TestProcess.start('dart', ['run', 'bin/cli.dart', ...args]).StreamQueue matchers (e.g., emitsThrough, emits).await process.shouldExit(0).await d.Descriptor.validate().Select the appropriate compilation target based on your distribution requirements.
dart run bin/cli.dart. This uses the JIT compiler for rapid iteration.dart build cli. This runs build hooks and outputs to build/cli/_/bundle/.dart compile exe bin/cli.dart -o <output_path>. This bundles the Dart runtime and machine code into a single file.dart compile aot-snapshot bin/cli.dart. Run the resulting .aot file using dartaotruntime.Dart supports cross-compiling to Linux from macOS, Windows, or Linux hosts.
Use the --target-os and --target-arch flags with dart compile exe or dart compile aot-snapshot.
--target-os=linux (Only Linux is currently supported as a cross-compilation target)--target-arch=arm64 (64-bit ARM)--target-arch=x64 (x86-64)--target-arch=arm (32-bit ARM)--target-arch=riscv64 (64-bit RISC-V)Example: dart compile exe --target-os=linux --target-arch=arm64 bin/cli.dart
Command in lib/src/commands/.name and description properties.argParser.addFlag() or argParser.addOption().run() method with the core logic.CommandRunner instance in bin/cli.dart using addCommand().dart run bin/cli.dart help <command_name> to verify help text generation.dart format . --set-exit-if-changed to ensure code formatting.dart analyze to ensure no static analysis errors.dart test to pass all integration tests.dart compile exe bin/cli.dart -o build/cli-hostdart compile exe --target-os=linux --target-arch=x64 bin/cli.dart -o build/cli-linux-x64import 'dart:io';
import 'package:args/command_runner.dart';
import 'package:stack_trace/stack_trace.dart';
class CommitCommand extends Command {
@override
final String name = 'commit';
@override
final String description = 'Record changes to the repository.';
CommitCommand() {
argParser.addFlag('all', abbr: 'a', help: 'Commit all changed files.');
}
@override
Future<void> run() async {
final commitAll = argResults?['all'] as bool? ?? false;
print('Committing... (All: $commitAll)');
}
}
void main(List<String> args) {
Chain.capture(() async {
final runner = CommandRunner('dgit', 'Distributed version control.')
..addCommand(CommitCommand());
await runner.run(args);
}, onError: (error, chain) {
if (error is UsageException) {
stderr.writeln(error.message);
stderr.writeln(error.usage);
exit(64); // ExitCode.usage.code
} else {
stderr.writeln('Fatal error: $error');
stderr.writeln(chain.terse);
exit(1);
}
});
}
import 'package:test/test.dart';
import 'package:test_process/test_process.dart';
import 'package:test_descriptor/test_descriptor.dart' as d;
void main() {
test('CLI formats output correctly and modifies filesystem', () async {
// 1. Setup mock filesystem
await d.dir('project', [
d.file('config.json', '{"key": "value"}')
]).create();
// 2. Spawn the CLI process
final process = await TestProcess.start(
'dart',
['run', 'bin/cli.dart', 'process', '--path', '${d.sandbox}/project']
);
// 3. Validate stdout stream
await expectLater(process.stdout, emitsThrough('Processing complete.'));
// 4. Validate exit code
await process.shouldExit(0);
// 5. Validate filesystem mutations
await d.dir('project', [
d.file('config.json', '{"key": "value"}'),
d.file('output.log', 'Success')
]).validate();
});
}
development
Perform REST API networking operations (GET, POST, PUT, DELETE) using the lightweight and robust standard `http` package, including platform configurations and background parsing models.
development
Configure internationalization and localization support using Flutter's built-in l10n system, App Resource Bundle (ARB) files, and ICU formatting syntax.
development
Create model classes with fromJson/toJson using dart:convert and Dart 3 pattern matching. Use when manually mapping JSON to classes, parsing HTTP responses, or choosing between manual and code-generated serialization.
data-ai
Diagnose and fix Flutter layout constraint violations (RenderFlex overflow, unbounded height/width, ParentData misuse). Use when encountering layout exceptions, yellow-black overflow stripes, or red error screens.