Skip to content

Shell tasks management tool for CLI and non-CLI environments

License

Notifications You must be signed in to change notification settings

omercsp/taskrunner

Repository files navigation

Test

What is this?

Taskrunner (henceforth, TR) allows users to run tasks with relation to the current working directory. Somewhat similar to VScode tasks, TR tasks are not bound to a specific platform and can be invoked from anywhere. A trivial environment for TR is a shell console, but any environment with ability to run executables is suitable.

One can think of TR is as a commands repository for a directory or a way to enable dynamic PATH, where commands becomes available and unavailable according to the working directory.

TR was written as a successor to runevn, a tool I wrote for running containers commands in similar manner which was extremely popular between myself and a guy I knew from work. runenv is great for sparing the user from remembering container settings for a given command, but is limited for a single task, isn't built for tasks that aren't container based and has limited features.

Why do I need it?

Assume working on a project in a directory project-dir, where a long and complicated commands like: my-command -vv --with-this-flags --and-this-flag=5 THIS_ARG AND_THIS_ARG are often required. Remembering and repeating it can be be annoying and time consuming.

One could rely on history, but commands history browsing can be as annoying as typing it. Besides, after a long period of time, the required entry it might not even be there, and if another user needs it, he need to figure it out by his own.

Using shell script is a possible solution - place the logic in a script like my-task.sh and run it instead. This would work, of course, but now we need to remember my-task.sh and where it resides. Unless placed in the search PATH, it's not just a matter of of running my-task but path/to/my-task. And again, if a another user needs to run this command by herself, she has to have knowledge of this script in advance.

An alias approach can also be tried: Alias my-command-alias to our command. Again, this works but suffers from drawbacks: you are either bound to have the my-command available in PATH or have the entire project pinned to a given directory. In addition, aliases are user level entities: A different user or placing the project in a different machine will require redefining the alias for that login. And once again - we need to know and remember what aliases are relevant - there's no way to differentiate between project and global aliases.

The situation might get more complicated if other commands are needed as well - with other arguments, other working directories, maybe some environment variables, etc.

With TR, all these problems are solved. All that is required is a project level configuration file defining a task named my-task to run the horrible my-command... and the command becomes available all over the project hierarchy, to all users with access to the project. task list can be used to see what tasks are available and task info to understand what they are doing. As long as TR is installed, moving the project to another machine will provide all the tasks with it.

Installation and requirements

TR is python based tool, verified for python 3.10 and above. Older versions might work as well, but are not actively supported and packaged so manuall install is required.

To install run pip3 install pytaskrunner.

For installation from source:

  1. Clone the repository
  2. Install the requirements using pip pip3 install <path to cloned repository>

TR should work for any environment with compatible python, however it is developed on and maintained for Linux machines. CI tests are ran for MacOS as well without container support. It is known to work on Cygwin systems (again, without containers support).

Basic configuration and usage

TR uses yaml or json configuration files (By default, named .tasks.yaml) to describe tasks. These tasks are available when the current work directory is anywhere in the directory hierarchy under the directory the configuration file was placed in - TR recursively looks for .tasks.yaml from the current working directory up to root.

Once set, use the following commands:

  1. Running tasks - by running task run <TASK>
  2. Getting info on a task - by running task info <TASK>
  3. Listing available tasks - by running task list

Configuration file search order

TR searches for configuration files in the following order:

  1. The current working directory, and recursively up to the root directory.
  2. The user's home directory, in the .config directory. In any of these directories, the first configuration file found is used. TR searchs for the following files (in order): .tasks.yaml, .tasks.yml, .tasks.json and tasks.json.

Configuration files may include other files, as well as the default configuration file located in the user's home directory. By default, the default configuration file is not included. This behaviour can be disabled by setting use_default_include to true.

Example - building projects

[source code projects are used in the following examples, but any sort of task is valid - scp some files, start a service through systemctl, open a video with vlc, etc.]:

Assume a cmake based project in a directory named cmake-project with lots of sub directories. It might look something like this:

user@host $ tree cmake-project
.
├── build
├── subdir0
├── subdir1
│   └── subdir3
│       └── subdir4
└── subdir5
    └── subdir6

Often running basic cmake commands in such environments requires:

  1. Entering the project's build directory
  2. Running cmake ..

Not very complicated, but can get tedious when actively developing cmake files all over the tree. By adding the following .tasks.yaml file into cmake-project directory, rerunning the relevant cmake command becomes trivial one-liner:

tasks:
  cmake:
    short_desc: Run cmake
    description: Run 'cmake ..' for the cmake-project
    commands:
      - cmake ..
    cwd: '{{taskRoot}}/build'

Once set, running task run cmake from anywhere under cmake-project will run the cmake .. with cmake-project/build as the working directory.

In the tasks section our task, named cmake is defined:

  • short_desc and description are for documentation and self-explanatory.
  • commands sets a list of commands for this task to execute. In this case, a single cmake .. command is set.
  • cwd sets the working directory for this task. Its commands are executed with this setting value as their working directory. The {{taskRoot}} is an automatic variable with the value of the location of the configuration file. Using it in cwd allows moving the project around without the need to update it. In our case, if the configuration file is placed in cmake-project root, say, /home/user/projects/cmake-project, the expanded value is /home/user/projects/cmake-project/build.

To list the available tasks, run task list:

user@host $ task list
Name                    Flags Description
----                    ----- -----------
cmake                         Run cmake

And for the task full details, run task info cmake, an a -x flag values expansion; in our case, the working directory:

user@host $ task info -x cmake
Task name:              cmake
Short description:      Run cmake
Description:            Run 'cmake ..' for the cmake-project
Hidden                  No
Use shell:              No
Working directory:      /home/user/projects/cmake-project/build
Command:                cmake ..

Now lets assume there's a need to occasionally run another task, similar to the one already defined. For example, with some cmake definition like -DSOMEVAR=somevalue. And another task that actually builds the project with cmake --build .. -j8 , both, like the first task, need to be invoked from the build directory (Again, if you don't care about cmake and source building, don't worry about what each command does. Anything can be used here). Instead of remembering each command or placing these in shell scripts and then trying to remember where they are, using TR tasks simplify the workflow:

tasks:
  cmake:
    short_desc: Run cmake
    description: Run 'cmake ..' for the cmake-project
    commands:
      - cmake ..
    cwd: '{{taskRoot}}/build'
  cmake-with-def:
    short_desc: Run cmake with cmake define flag
    description: Run 'cmake ..' for the cmake-project, with a cmake define flag
    commands:
      - cmake -DSOMEVAR=somevalue ..
    cwd: '{{taskRoot}}/build'
  build:
    short_desc: Build the project
    description: Build the project using cmake's build command
    commands:
      - cmake --build ..
    cwd: '{{taskRoot}}/build'

And now all commands are available, from any directory under cmake-project:

user@host $ task list
Name                    Flags Description
----                    ----- -----------
cmake                         Run cmake
cmake-with-def                Run cmake with cmake define flag
build                         Build the project

user@host $ task info build
Task name:              build
Short description:      Build the project
Description:            Build the project using cmake's build command
Hidden:                 No
Abstract:               No
Use shell:              No
Working directory:      {{taskRoot}}/build
Command:                cmake --build ..

user@host $ task run build
...

The above example can be further simplified by using variables and inheritance.

Multiple projects - same semantics, different details

Extending the example above, assume multiple projects, each with its own way of doing things. Let's say (based on true events), that other than the notorious cmake-project we have 2 additional projects named make-project and scons-project (and ninja-project and meson project etc...).

user@host $ tree projects
projects
├── cmake-project
├── make-project
└── scons-project

As these new projects name imply, they are built around make and scons build systems so their build process them is different than our old cmake-project yet still we want to 'build' each one. We can try to remember each build command by our own and make sure we are in the correct project, or we can use TR: After adding a tasks file on each of these projects root directories each with its own build task, (like the build task exampled for cmake-project) the user can just run task run build in any of these projects, and the correct build command will be invoked. If her environment is consistent, she might add an alias like (Bash style) alias build='task run build' for even easier build - and whenever inside any of the source projects directories, running build invokes the correct command for the current project.

Advanced usage

This section will describe some of advanced configuration and usage topics. For a detailed list of all configuration settings, refer to the configuration documentation

Variables

TR allows definitions of variables to be used through out the configuration.

  • Variables are evaluated as strings.
  • A variable VAR can be referred as {{VAR}}
  • Variables can be used in the following task properties: commands, environment variables, current working directory, container image, container working directory and container volumes.
  • A variable can be defined using other variables

There are 3 types of variables with different priorities:

  1. Global variables - Defined in global level using variables as key word. They are available for every task with the scope of the tasks file.
  2. Task variables - Defined in task definitions level using variables as key word. They are available for the task they are defined in and those inherit from it. Inherited variables are overridden by the inheriting task variables. All task variables override global variables with the same name.
  3. System variables - set by TR and will override any variables with the same name. The following variables are defined:
    • cwd - for the current working directory
    • taskRoot - for the path of the directory the found configuration file was found. This is helpful when a task needs to refer to a directory relatively to the project (see previous examples for its usage).
    • cliArgs - refers for arguments given by the user through CLI. See section about passing command CLI arguments

TR auto variables override any user variable, so avoid setting variables with reserved names.

The following example show the different types of variables definition and usage.

variables:
  ENV_NAME: global_env_name
  ENV_VALUE: global_env_value
  var0: global_var0_value
  var1: global_var1_value
tasks:
  task0:
    variables:
      var1: task0_var1_value
      var2: task0_var2_value
    env:
      '{{ENV_NAME}}': '{{ENV_VALUE}}'
    commands:
      - printenv {{ENV_NAME}}
      - echo {{var0}}
      - echo {{var1}}
      - echo {{var2}}
  task1:
    base: task0
    variables:
      ENV_VALUE: task1_env_value
      var2: task1_var2_value
    env:
      '{{ENV_NAME}}': '{{ENV_VALUE}}'

In the above example, a few global vars are defined. task0 overrides the global var1 and sets its own var2. task1 overrides the inherited task0 var2. Note the environment variable setting using the TR variables. See the inheritance section for more deatils about task inheritance.

Environment variables

Environment variables can be referred with {{$VAR}}, where VAR is the variable name. Environment variables can be used whereever TR variables can be used. Note the difference between environment variables used in the task setting and environment variables that are set for a given task using the env keyword.

When a task is invoked its commands are invoked with the current set of system environment variables merged with task specific variables if such exist either from the task configuration or from CLI. Task variables take precedence over system variables. This behavior can be modified by setting the task's env_inherit. Set to false, TR ignores the system environment variables, and passes the task commands only the task environment variables. By default env_inherit is set to true.

Container based tasks

TR includes special support for running tasks inside a container. The main container setting is c_image defining a container image to use. The following task runs make inside a container with a volume, CWD set, tty allocated, interactive mode, wrapped in a /usr/bin/sh -c:

build:
  short_desc: Build
  c_image: localhost/media-builder:latest
  c_volumes:
    - '{cwd}:{cwd}'
  c_tty: true
  c_interactive: true
  c_shell: true
  c_cwd: '{{taskRoot}}'
  commands:
    - ls -l

The actual command the task invokes is /usr/bin/docker -i -t --rm -w {{taskRoot}} -v {{cwd}}:{{cwd}} localhost/media-builder:latest /usr/bin/sh -c "ls -l".

Container support is synthetic sugar, and putting the complete container command as a commands entry instead of setting a task container settings is completely valid and equivalent for having TR assemble the command instead, but usually it's much easier to let TR handle these settings.

For a detailed list of all container configuration settings, refer to the configuration documentation.

Inheritance

A task might inherit another task settings by using the base settings. If task a inherits task b, all of b's settings are inherited. Setting redefined in task a will override inherited setting. The following example demonstrates how to utilize task inheritance for creating multiple tasks with similar characteristics that differ in a small details (working directory):

tasks:
  base-task:
    abstract: true
    commands:
      - very_complicated command --with --lots=of --flags and arguments
    env:
      and: some
      environment: variables
  task-1:
    base: base-task
    cwd: /opt/task-1_dir
  task-2:
    base: base-task
    cwd: /opt/task-2_dir

Configuration files inclusion

Every configuration file can include multiple files using a global include setting. Global settings, variables and tasks are all included and overridden if exist in a following include file, and finally in the original configuration file.

A special file located in ${HOME}/.config/tasks.json is always included if it exists. This behavior can be disabled by setting use_default_include to false.

Passing command line arguments to commands

Command line arguments are transferred to a task run with the -- convention: text written after the 'dash dash' token is transferred as an arguments. The arguments aren't passed to the commands automatically. In order for a command to use CLI arguments, it must be explicitly use it with the {{cliArgs}} variable. The allows fine grain control of which commands and where inside the command the CLI arguments are used. In fact, since arguments are translates to a TR variable, {{cliArgs}} can be used in every setting with variables support. Running task run ls -- -l somefile.txt in the following task will run ls -l somefile.txt:

ls:
  commands:
    - ls {{cliArgs}}
  short_desc: Clean build

TR CLI options

TR allows overriding almost any configuration setting using the CLI. In general these should be avoided unless debugging or tweaking a task temporarily. For example, if one wants to run a task but just tweak the command it runs (retaining other settings like container settings, environment variable, etc.), running task run -c "my new command" <TASK> will do the trick.

Run task -h and task <CMD> -h for the full list of CLI options.

Bash completion

TR uses argcomplete for bash auto complete. See argcomplete documentation for more details.

Logging

TR Logging can be enabled with the --log_file <FILE> CLI option. Use -v to increase its verbosity.

Status

While tested and works, TR is still a work in progress. Breaking changes are expected. Version and schema validations will issue error messages if incompatibility is detected.

About

Shell tasks management tool for CLI and non-CLI environments

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published