mici is a lightweight CLI framework that automatically discovers and executes your commands based on filesystem hierarchy.
Define your commands as YAML files and mici handles the CLI and its execution for you.
Make sure Rust is installed. See #install for more details.
# Install mici using Cargo
# This will install `mici` as executable.
cargo install miciRun mici --help to see what's available.
# Initialize mici
mici init
# Create your first command and edit if needed
# at `~/.mici/jobs/commands/hello.yml`
# or run `mici edit hello` to quickly open in an editor.
mici new hello
# See what it is with --help
# Pager can be disabled with `disable_pager: true` in the `config.yml`.
mici hello --help
# Run it
mici helloTraditional CLI development:
- Write command parsers and argument handling
- Manage command registration and routing
- Rebuild and redeploy for every new command
- Maintain complex CLI application code
With mici:
- Drop YAML files in a directory structure
- Commands appear in your CLI automatically
- No rebuilds, no deployments, no CLI code
- Perfect for CI/CD replacement workflows
Create your commands as YAML files in a directory hierarchy.
~/.mici
├── config.yml # Configuration file
└── jobs
├── commands
│ ├── deploy
│ │ ├── terraform.yml # mici deploy terraform
│ │ └── frontend
│ │ ├── staging.yml # mici deploy frontend staging
│ │ └── production.yml # mici deploy frontend production
│ ├── database
│ │ ├── backup.yml # mici database backup
│ │ └── migrate.yml # mici database migrate
│ └── hello.yml # mici hello
│
└── scripts # Usable as:
├── hello.py # scripts/hello.py
└── run.sh # scripts/run.sh
Each YAML file has CI-like attributes - environment variables, confirmation prompts, parallel execution, and more; allowing mici to customize your run of that command and generate help documentation based on the available information.
For a full reference of how a mici command is structured, see examples/.../hello.yml.
version: "1.0"
name: "hello"
description: "A new mici command"
usage: "mici hello"
configuration:
confirm: false
environment:
VAR_ONE: "SOME_VALUE_123"
VAR_TWO: "SOME_VALUE_123"
IS_FORCED: "@{inputs.force}"
TOKEN: "${MY_PRIVATE_TOKEN}"
working_directory: null
inputs:
name:
type: string
description: "A name to say hello to!"
required: true
secret: false
short: -n
long: --name
default: "World"
force:
type: boolean
description: "Run this with force, maybe?"
short: -f
long: --force
steps:
- id: "say_hello"
name: "Say hello on terminal"
run:
shell: "bash"
working_directory: null
environment:
VAR_TWO: "ANOTHER_VALUE_456"
command: |
echo "Hello, @{inputs.name}!"That's it. Your filesystem is your CLI structure, and YAML is your command.
Cargo is a package manager for Rust. Make sure to have Rust toolset available on your computer first. See rustup installation guide for easy introduction.
Once you have Rust available, you can run any of the following commands to install mici.
cargo install micigit clone [email protected]:rwxdash/mici.git
cd ./mici
cargo install --path .Simply run:
cargo uninstall miciThere are some major stories to complete before I call this project version 1.0. Here's what I have in my mind so far:
- Implement tests and CI checks
- Implement the basic runner
- Handle step confirmation
- Basic execution of a simple command
- Implement environment variable substitution
- Make basic execution run on Windows
- Implement
script: "scripts/..."usage for steps
- Better logging throughout the program
- Error/Debug log level can be set through config file
- Can be set to None/Silent
- Implement tracing
- Add tracing for rust/otel # https://github.com/tokio-rs/tracing
- The default schema in new command should have proper default shell for linux/windows
- Implement expression evaluator for
when:in steps-
on_failure()# any previous step failed -
on_success()# all previous steps passed -
on_platform("linux")# linux/win/darwin -
depends_on("step_id")# or depends_on(["step1", "step2"]) -
${ENV_VAR} == "production" -
@{inputs.cleanup} -
@{inputs.branch} == "main" -
@{steps.<STEP_ID>.output} == "success"(Or shell code? not sure) - Accept operators and chains
- Implement variable masking for
secret: true
-
- Basic validation for commands
- Handle errors with miette # https://github.com/zkat/miette
- Might need a new error class
- Think about runner/step-execution isolation
- with chroot/containers/microvms
Code of conduct is simple. Be nice and thoughtful. That's all.
About the Contributions, the project is obviously open to contributions. However, since it's quite early and I'm still shaping what this tool can be, no major refactor or feature PRs, please.
I'd very much appreciate if you open an issue beforehand if you plan to contribute any change.
Distributed under the Apache License 2.0.
See the LICENSE file for more information.
mici: Your filesystem is the best argument parser.