install
source · Clone the upstream repo
git clone https://github.com/kevmoo/dash_skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/kevmoo/dash_skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/.agent/skills/dart-cli-app-best-practices" ~/.claude/skills/kevmoo-dash-skills-dart-cli-app-best-practices && rm -rf "$T"
manifest:
.agent/skills/dart-cli-app-best-practices/SKILL.mdsource content
Dart CLI Application Best Practices
1. When to use this skill
Use this skill when:
- Creating a new Dart CLI application.
- Refactoring an existing CLI entrypoint (
).bin/ - Reviewing CLI code for quality and standards.
- Setting up executable scripts for Linux/Mac.
2. Best Practices
Entrypoint Structure (bin/
)
bin/Keep the contents of your entrypoint file (e.g.,
bin/my_app.dart) minimal.
This improves testability by decoupling logic from the process runner.
DO:
// bin/my_app.dart import 'package:my_app/src/entry_point.dart'; Future<void> main(List<String> arguments) async { await runApp(arguments); }
DON'T:
- Put complex logic directly in
.bin/my_app.dart - Define classes or heavy functions in the entrypoint.
Put an executable
entry in pubspec.yaml
executablepubspec.yamlList your executables in
pubspec.yaml to make them available for global
activation and clean invocation via dart run.
DO: Add an
executables section mapping the command name to the Dart file in
bin/ (excluding the .dart extension).
executables: my_app: # Maps to bin/my_app.dart custom_name: main # Maps to bin/main.dart
Then run via
dart run my_app or dart run custom_name.
CONSIDER #!
for other scripts on *nix systems
#!This is NOT a hard and fast rule, but it is something to consider.
For CLI tools intended to be run directly on Linux and Mac, add a shebang and ensure the file is executable.
DO:
- Add
to the first line.#!/usr/bin/env dart - Run
to make it executable.chmod +x bin/my_script.dart
#!/usr/bin/env dart void main() => print('Ready to run!');
Process Termination (exitCode
)
exitCodeProperly handle process termination to allow for debugging and correct status reporting.
DO:
- Use the
setter to report failure.exitCode - Allow
to complete naturally.main - Use standard exit codes (sysexits) for clarity (e.g.,
for bad usage,64
for configuration errors).78- See
package:io
class or FreeBSD sysexits man page.ExitCode
- See
import 'dart:io'; void main() { if (someFailure) { exitCode = 64; // DO! return; } }
AVOID:
- Calling
directly, as it terminates the VM immediately, preventing "pause on exit" debugging andexit(code)
blocks from running.finally
Exception Handling
Uncaught exceptions automatically set a non-zero exit code, but you should handle expected errors gracefully.
Example:
Future<void> main(List<String> arguments) async { try { await runApp(arguments); } catch (e, stack) { print('App crashed!'); print(e); print(stack); exitCode = 1; // Explicitly fail } }
Cross-Platform Compatibility (Windows Support)
When writing CLI applications and tests, ensure compatibility with Windows:
- Paths: Never hardcode path separators like
. Use/
'spackage:path
orp.join
to construct paths portably.p.normalize - File Permissions: When testing file permission errors, remember that
is not available on Windows. Usechmod
on Windows or appropriate mock libraries. Never skip tests on Windows simply because of permission commands if a Windows equivalent exists.icacls
3. Recommended Packages
Use these community-standard packages owned by the Dart team to solve common CLI problems:
| Category | Recommended Package | Usage |
|---|---|---|
| Stack Traces | | detailed, cleaner stack traces |
| Arg Parsing | | standard flag/option parsing |
| Testing | | integration testing for CLI apps |
| Testing | | file system fixtures for tests |
| Networking | | standard HTTP client (remember user-agent!) |
| ANSI Output | | handling ANSI colors and styles |
4. Interesting community packages
| Category | Recommended Package | Usage |
|---|---|---|
| Configuration | | strongly typed config objects |
| CLI Generation | | generate arg parsers from classes |
| Version Info | | automatic version injection |
| Configuration | | precise YAML parsing with line numbers |
5. Conventions
- File Caching: Write cached files to
..dart_tool/[pkg_name]/ - User-Agent: Always set a User-Agent header in HTTP requests, including version info.