Skip to content
/ mici Public

A lightweight CLI framework that automatically discovers and executes your commands based on filesystem hierarchy.

License

Notifications You must be signed in to change notification settings

rwxdash/mici

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

48 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

mici

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.

Quick Start

Make sure Rust is installed. See #install for more details.

# Install mici using Cargo
# This will install `mici` as executable.
cargo install mici

Run 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 hello

Why mici?

Traditional 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

How it works

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.

Install

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.

From crates.io

cargo install mici

From source

git clone [email protected]:rwxdash/mici.git
cd ./mici

cargo install --path .

Uninstall

Simply run:

cargo uninstall mici

What's to come

There are some major stories to complete before I call this project version 1.0. Here's what I have in my mind so far:

TODOs

  • 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
  • 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

Contributions and Code of Conduct

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.

License

Distributed under the Apache License 2.0.

See the LICENSE file for more information.


mici: Your filesystem is the best argument parser.

About

A lightweight CLI framework that automatically discovers and executes your commands based on filesystem hierarchy.

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published