Skip to content

Latest commit

 

History

History
261 lines (182 loc) · 7.15 KB

README.md

File metadata and controls

261 lines (182 loc) · 7.15 KB
     _             _
  __| | ___   ___ | |_
 / _` |/ _ \ / _ \| __|
| (_| | (_) | (_) | |_ _
 \__,_|\___/ \___/ \__(_)

Simple Zero Dependency Task Runner

This is a simple, zero dependency (except Python 3, which comes installed on most *nix operating systems. Python 3.8.0 or greater is required) task runner. Similar to make, but meant to be used for non-C style projects.

Python is ideal for this sort of thing, because it has a pretty comprehensive standard library; where most things you might need are built right in. However, if you have more complex needs, any python library can be used as a part of your tasks.

Installation

There are a few ways to install doot:

Install To Repository (recommended)

This installs doot directly to your repository, so that your colleagues don't have to do anything but use it. Run the following command in a terminal:

sh -c "$(curl -s https://raw.githubusercontent.com/synic/doot/main/install.sh)"

This will start an installation script that will download doot.py to a location of your choice (default is ./.doot/doot.py) and create an initial ./do script for you.

If you don't want to run the script, you can just download doot.py from the repository, put it in your repository, and then, in the top of your do file, before from doot import do, add the directory where doot.py resides to your python path:

#!/usr/bin/env python3

sys.path.append("./.doot")

from doot import do  # pyright: ignore

# define tasks here...

Submodule

I prefer this method, as doing it this way means your coworkers don't have to install anything etra to get it working (assuing they have Python installed, which is usually done by default on Mac and most Linux distros).

In the repository you want to use this in, run the following:

$ mkdir -p lib
$ git submodule add https://github.com/synic/doot lib/doot

Then, at the top of your do script, before the doot import, you can add the following:

if not os.path.isfile("./lib/doot/doot/__init__.py"):
    print("`doot` not found; run `git submodule update --init`")
    sys.exit(1)

sys.path.append("./lib/doot")

Install as a Library

Alternatively, you can install it as a library:

$ pip install git+https://github.com/synic/doot

Getting Started

In your project root directory, create a file (I usually call it do, but it can be anything you want):

#!/usr/bin/env python3

from doot import do

@do.task(allow_extra=True)
def bash(_, extra):
    """Bash shell on the web container."""
    do.run("docker exec -it api bash", extra)


@do.task()
def start():
    """Start all services."""
    do.run("docker-compose up -d")


@do.task()
def stop():
    """Stop all services."""
    do.run("docker-compose stop")


@do.task()
def dbshell():
    """Execute a database shell."""
    do.run("docker exec -it database psql -U myuser mydatabase")


@do.task()
def shell():
    """Open a django shell on the web container."""
    do.run("docker exec -it api django-admin shell")


@do.task(allow_extra=True)
def manage(_, extra):
    """Run a django management command."""
    do.run("docker exec -it api django-admin", extra)


@do.task(
    do.arg("-n", "--name", help="Container name", required=True),
    do.arg("-d", "--detach", help="Detach when running `up`", action="store_true"),
)
def reset_container(opt):
    """Reset a container."""
    do.run(f"docker-compose stop {opt.name}")
    do.run(f"docker-compose rm {opt.name}")

    extra = "-d" if opt.detach else ""
    do.run(f"docker-compose up {extra}")


if __name__ == "__main__":
    do.exec()

With this setup, you can run tasks like ./do manage, ./do shell, etc.

Running ./do -h will show output like this:

Usage: ./do [task]

Available tasks:

  bash                   Bash shell on the web container
  dbshell                Execute a database shell
  manage                 Run a django management command
  reset-container        Reset a container
  shell                  Open a django shell on the web container
  start                  Start all services
  stop                   Stop all services

Complex Example

To see a more complex example, look here

Doot Functions

do.task

This is a decorator that turns a function into a task. The task will have the same name as the function it decorates (changing _ to -), and the docstring will be the documentation that appears when you type ./do help or ./do help [task]. All underscores will be converted to hyphens in the resulting task name.

The task function can take 0, 1, or 2 arguments. The first argument, opt will contain any parsed argument values. For example:

@do.task(do.arg('-n', '--name'))
def hello(opt):
    print(f"Hello, {opt.name}")

The second argument only matters if you pass allow_extra=True. By default, if any command line arguments are passed that are not specified in do.arg arguments passed to do.task, an error message will be soon. If allow_extra=True is passed, then any extra arguments that were not specified will be passed as a list to the task function as the second argument.

For example, if you'd like to run Django management commands in a docker container by just running ./do manage [cmd]:

@do.task(allow_extra=True)
def manage(_, extra):
    """Run Django management commands."""
    do.run("docker exec -it api django-admin", extra)

Then when you run something like:

$ ./do manage makemigrations --name add_user_is_active_field accounts

The makemigrations --name add_user_is_active_field accounts will be passed through to django-admin on the container.

do.run

This runs a command on the host. Things like do.run('docker network add test'), or doot.run(["docker", "ps", "-q"])

The first argument of do.run, args, and any extra **kwargs are passed directly to subprocess.call. If you have more complex needs, you can use the subprocess module directly; the do.run function is just there for convenience.

do.arg

You can pass one or more do.arg arguments to the do.task decorator. These will set up arguments for your task, using the argparse module. They are passed directly to parser.add_argument, so they have the same parameters. See https://docs.python.org/3/library/argparse.html#the-add-argument-method for more information.

An example:

@do.task(do.arg("--name", dest="name", help="Your name"))
def hello(opt):
    do.info(f"Hello, {opt.name}!\n")

do.log, do.success, do.info, do.warn, do.error

These are logging statements. Each one has it's own color indicative of the type of message you want to show.

do.fatal

do.fatal(msg) will call do.error(msg) and then sys.exit(1) (you can specify the exit code by passing status, the default is 1).

@do.task(doot.arg("--name", required=True))
def hello(opt):
    if opt.name.lower() in ("tyler", "steve", "james"):
        do.fatal(f"Sorry, your name cannot be {opt.name}. Get a new one.")
    do.info(f"Hello, {opt.name}!\n")

Acknowledgements

This project was named after our beloved Doots. She will be missed.

Doots