24 stable releases

3.6.0 Dec 1, 2024
3.5.7 Nov 26, 2024
3.0.6 Oct 31, 2024
2.1.0 Sep 5, 2024
1.1.0 Sep 10, 2022

#33 in Build Utils

Download history 17/week @ 2024-09-12 15/week @ 2024-09-19 6/week @ 2024-09-26 2/week @ 2024-10-03 559/week @ 2024-10-24 715/week @ 2024-10-31 353/week @ 2024-11-07 472/week @ 2024-11-14 385/week @ 2024-11-21 147/week @ 2024-11-28 26/week @ 2024-12-05 46/week @ 2024-12-12

82 downloads per month

MIT license

280KB
6.5K SLoC

monorail

A tool for effective polyglot, multi-project monorepo development.

Build Status Cargo

monorail is optimized to support the needs of single or multi-project/multi-language monorepos. Using a lightweight graph configuration, it provides APIs for analyzing changes, parallel command execution, logging, and more. It is designed for flexibility and speed, to output structured text for use in other systems, and compose with existing compilers, interpreters, and tools without the need for bespoke rules or plugins.

See the tutorial below for a practical walkthrough of how monorail works.

Installation

UNIX/Linux variants

At present, only source builds are supported. Packages for popular managers will be provided at a later time.

Ensure that Rust 1.63+ is installed and available on your system path:

Run cargo install --path .

Windows

At this time, Windows is unsupported. However, there are no technical limitations preventing its support in the future.

Overview

monorail is a tool best classified as a "development orchestrator". It is designed to be generic and minimally opinionated, allowing you to flexibly define the components of your repository in virtually any way. Similarly, it makes no requirements about the tooling you use; shell scripts, interpreters, compilers, or even a large monorepo build system such as bazel are equally supported. At its core, monorail is a means of fine-tuning how and when your preferred tools are invoked, making it an effective abstraction for building efficient parallelized workflows for development and CI/CD.

monorail is internally driven by three things:

  1. A graph representation of your repository, built from a list of target entries and each target's uses list.
  2. A change detection provider, which provides a view of changes in the repository
  3. A scheduler and parallel execution engine for user-defined executables

Changes are mapped to affected targets, and a graph traversal powers various dependency-related tasks such as target grouping for parallel execution, "depended upon by" analysis, and so forth.

Monorail has a small lexicon:

  • change: a created, updated, or deleted filesystem path as reported by a change provider
  • checkpoint: a location in change provider history that marks the beginning of an interval of changes
  • target: a unique container that can be referenced by change detection and command execution
  • uses: a set of paths that a target depends upon
  • ignores: a set of paths that should not affect a target during change detection
  • command: executable code written in any language
  • sequence: a named list of commands

See the Tutorial for a practical walkthrough of using monorail to build a toy monorepo, or Monorail Example Repo for a pre-built example.

Roadmap

monorail is suitable for production use, but remains under active development. The following features are either planned, or in progress:

  • Remote execution
  • Signal capture and command interruption
  • Fine-grained command execution scheduling control
  • Various convenience APIs
  • Additional change provider backends
  • Expanded examples for various languages and CI/CD setups

Concepts

Each of the core sets of APIs is linked below. For each section, a description and example use of related APIs is provided. For complete documentation, provide the --help flag to any command or subcommand.

Base options

Verbosity

Some APIs support multiple levels of workflow and debug logging. Enable them with monorail -v <command>, monorail -vv <command>, or monorail -vvv <command>.

Config file location

Some APIs require a configuration file. Manually specifying the file is done with monorail -f </path/to/Monorail.json>.

Config

monorail uses a JSON configuration file that describes the repository target mapping, target dependencies, and various other options. This file is fully documented in the Monorail.reference.js file at the root of this repository. Its default name is Monorail.json, located in the root of your repository. While this location and name can be customized, all subsequent documentation will refer to it by this name.

Note: while writing JSON configuration manually is supported, it's highly recommended to use the config generate API to write your configuration files in more expressive languages and allow monorail to synchronize the output with your source file.

APIs

config generate

Assigns a source file and its output as the source of truth for configuration. This enables configuration to be written in any language and synchronized automatically by monorail. For example, here is a simple source file written in jsonnet (but, you could write the same in Python or any other language):

Monorail.jsonnet

local paths = {
  rust: "rust"
};
{
  source: {
    path: std.thisFile
  },
  targets: [
    {
      path: paths.rust
    }
  ]
}

This jsonnet code emits a JSON object that conforms to the Monorail.json format (documented in Monorail.reference.js), and includes one mandatory field when using this API: source.path.

Run this script to generate the JSON, and pipe it to this API:

jsonnet Monorail.jsonnet | monorail config generate

This creates two new files: a valid Monorail.json and Monorail.lock. The former is what was provided as output from the piped command, plus additional data used for integrity checks along with the latter. The generated lockfile must be checked into version control, or integrity checks in other clones of the repo will fail.

Now, when an API that uses the config file is used, an integrity check verifies that the source and output files are synchronized. For example, manually adding a target to this generated Monorail.json (or modifying the checksum in Monorail.lock) will cause the following error from any API that uses the config:

monorail target show
{
  "timestamp": "2024-11-19T15:30:49.294100+00:00",
  "kind": "error",
  "type": "generic",
  "message": "Generated configuration has been modified since the last `config generate`, or the lockfile checksum has been edited"
}

Similarly, adding a target to the source file without running monorail config generate results in the following:

monorail target show
{
  "timestamp": "2024-11-19T15:32:55.408455+00:00",
  "kind": "error",
  "type": "generic",
  "message": "Source configuration has been modified since the last `config generate`"
}

See monorail config generate -h for more information.

config show

Parses the configuration file and displays it, including defaults for fields not specified in the file.

Targets

A target is a core abstraction of monorail. High-level, it is simply a label for a filesystem path. You choose which paths in your repository to operate over, and assign each of those those paths a target in Monorail.json. When files are added, modified, or deleted within a target, the target will be considered "changed".

Targets must be:

  • a directory containing at least one file in its tree
  • unique

For example, this simple directory structure has one target and the file within it is part of that target:

repo/
  rust/
    Cargo.toml
  Monorail.json

Monorail.json

{
  "targets": [{"path": "rust"}]
}

Nesting targets

Targets may be nested within other targets, creating a hierarchy. For example, this assigns a target to a subdirectory of an existing target:

repo/
  rust/
    app1/
      Cargo.toml
    Cargo.toml
  Monorail.json

Monorail.json

{
  "targets": [{"path": "rust"}, {"path": "rust/app1"}]
}

In this example, rust/app1 is now its own target. It is also part of the rust target, so changes within this directory will affect the rust target as well.

Target dependencies

Targets can declare dependencies on other paths within the repository. When a change is detected in a dependency path, all dependent targets are affected.

Valid dependencies are any of the following:

  • targets
  • paths within targets
  • paths outside of targets

Dependencies are declared per-target with the uses array. For example:

repo/
  one/
    foo.txt
  two/
    bar.txt
  not_a_target/
    zap.txt
  Monorail.json

Monorail.json

{
  "targets": [
    {
      "path": "one",
      "uses": ["not_a_target"]
    }, 
    {
      "path": "two",
      "uses": ["one"]
    }
  ]
}

Changes to target one will affect target two, and changes to not_a_target will affect target one.

Graph cycles

However, targets may not declare dependencies that would create a circular reference. For example:

repo/
  one/
    foo.txt
  two/
    bar.txt
  Monorail.json

Monorail.json

{
  "targets": [
    {
      "path": "one",
      "uses": ["two"]
    }, 
    {
      "path": "two",
      "uses": ["one"]
    }
  ]
}

This configuration will produce a graph cycle error when the target graph is used.

Visualization

The target graph can be visualized using common tools, such as graphviz. This visualization is generated with the target render API. For example:

monorail target render

This will generate a target.dot file in your current directory. You can choose the output file location by providing the -f flag to render. In general, when passing this file to dot the sfdp layout is likely to produce the best visual results, but you are free to use others. For example, using the output from monorail target render, with dot installed:

dot -Ksfdp -Tpng target.dot -o target.png

APIs

target render

target render [--output-file, -f] [--type, -t]

Generate a visual representation of the target graph, and write it to an output file. Use monorail target render -h for details.

target show

target show [--target-groups, -g] [--commands, -c]

Display configured targets, target groupings, and target commands. Use monorail target show -h for details.

Commands

A command is an executable program called in the run API. Commands can be written in any language, and need only exist within the repository and have executable permissions. Common commands are those that install toolchains, lint and compile code, execute test binaries, build containers, deploy to servers, and so forth. Commands must perform a task, then exit with either a zero exit code to indicate success, or non-zero exit code to indicate an error. Persistent processes that do not naturally exit, such as servers, should be backgrounded in a Command using tools like foreman, docker-compose, etc. as appropriate.

Commands are invoked for one or more targets, and each command may optionally be parameterized with runtime arguments on a per-target basis using Args.

Command output to stderr and stdout is captured and streamed to a log tailing socket (if active), and streamed to compressed zstd archives.

Defining commands

Defining a command can be done automatically by accepting monorail defaults, or manually by specifying a label and location in Monorail.json.

Automatic

By default, targets store their commands in a monorail/cmd directory relative to the target's path. The stem of the file becomes the command name. For example:

repo/
  one/
    monorail/
      cmd/
        build.sh

This defines the build command for the one target.

Manual

The default command directory, command name, and/or executable location can be customized on a per-target basis in Monorail.json. Refer to Monorail.reference.js for more details.

Manually specifying commands is primarily useful when a command can be parameterized and reused by multiple targets, allowing the executable to exist in one place and be referenced by targets.

Running commands

Commands are executed by providing a list of command names to run, either implictly in parallel by a graph-guided analysis of changes, or explicitly in series for a provided set of targets. If an invoked command is not defined for a target, this is noted in the results of the run but is by default not considered an error. This behavior can be overriden by specifying the --fail-on-undefined flag on run.

When executing multiple commands, each command must fully succeed for all involved targets before the next is invoked. Otherwise, the run will end with an error.

On completion, run returns a structure containing the results of the run. Its return code is non-zero if any errors were encountered during execution, or any commands returned non-zero exit codes.

Implicitly

Commands may be run for changed targets by using a dependency graph derived from uses declarations in Monorail.json. When commands are run this way, each command is run sequentially, in the order specified to run. For each command, a graph traversal produces groupings of unrelated targets (called "target groups"), and within each target group, the command is executed in parallel. For example:

monorail run -c lint build

This will first run the lint command for each target group in parallel, and then do the same for build.

Explicitly

Commands may be run for specific targets by providing a list of targets to the run command. By default, commands are executed sequentially for each provided target. For example:

monorail run -c lint build -t target1 target2

This will run the lint command for target1, then target2, and then build for target1, then target2, if they have this command defined.

However, providing the --deps flag will include dependencies for the provided targets, and enable target group parallelism. For example:

Monorail.json

{
  "targets": [
    {
      "path": "target1",
      "uses": [
        "dep1"
      ]
    },
    {
      "path": "target2"
    },
    {
      "path": "dep1"
    }
  ]
}
monorail run -c build -t target1 target2 --deps

This will run build first for dep1, and then target1 and target2 in parallel.

Arguments

Commands can be provided with positional arguments at runtime. Arguments are provided with either --args or --argmaps flags passed to monorail run. In your command executables, capture these positional arguments as you would in any program in that language. As is standard, index 0 of the positional arguments is always an absolute path to the executable, so any additional arguments begin at index 1.

Args

The simplest way to provide arguments is with --args. For example:

monorail run -c build -t target1 --args foo bar baz

This provides foo, bar, and baz to the build command of target1 as positional arg indexes 1, 2, and 3 respectively. As mentioned, this is the least useful way to provide arguments; --args can only be used when exactly one command and one target are provided.

Argmaps

For more flexibility, --argmaps allows for providing a list of named files containing argmap JSON, which map command names to lists of arguments. This JSON has the following structure:

{
  "<command1 name>": [
    "arg1",
    "arg2",
    "...",
    "argN"
  ],
  "<command2 name>": [
    "..."
  ]
}

Each target may define or reference argmap files (by default, stored in the targets monorail/argmap directory; this can be adjusted in Monorail.json per-target), and they are loaded when requested during monorail run. For example:

monorail run -c build --argmaps ci

This will load the ci.json argmap file for every target involved in this run. If a target does not have that file, execution continues normally without error.

There is, however, another argmap that is loaded by default (again, if defined for a given target): The base argmap. If provided, it is automatically loaded before any other argument mappings provided to monorail run. This is especially useful for parameterizing reusable commands. An example of this pattern is shown in https://github.com/pnordahl/monorail-example in the rust crate targets, to re-use the same test command file for each crate.

Automatic loading of the base argmap can be disabled with monorail run --no-base-argmaps.

Argument merging

Whether arguments come from --args or --argmaps, or both, they are merged from left-to-right in the following order:

  • The base argmap
  • --argmaps
  • --args

For example:

monorail run -c build -t target1 --args baz --argmaps ci

target1/monorail/argmap/base.json

{
  "build": ["foo"]
}

target1/monorail/argmap/ci.json

{
  "build": ["bar"]
}

This would provide the arguments ["foo", "bar", "baz"] to the build command of target1.

Sequences

A sequence is an array of commands, and is specified in Monorail.json. It's useful for bundling together related commands into a convenient alias, and when used with run simply expands into the list of commands it references. For example:

{
  "targets": [],
  "sequences": {
    "validate": ["check", "build", "unit-test", "integration-test"]
  }
}
monorail run -s validate

This is equivalent to monorail run -c check build unit-test integration-test.

APIs

run

run [--target, -t <1> <2> ... <M>] <--commands, -c <1> <2> ... <N> | --sequences, -s <1> <2> ... <P>> | [--args, -a <1> <2> ... <Q>>] | [--argmaps, -m <1> <2> ... <R>>]

Runs commands or sequences for targets. At least one command or sequence is required. For more details, see monorail run -h.

Logs

Commands can emit output to stdout and stderr, and this output is captured by monorail. This output can be routed to an active log tailing socket, and is also stored on disk in compressed archives on a per command + target basis.

Showing logs

Logs generated by the run API are kept alongside the results of that run in the monorail output directory. These logs can be retrieved with the log show API. For example, this will retrieve all stdout an stderr logs from the most recent run:

monorail log show --stdout --stderr

This command will decompress each file and print it to stdout. Each file is preceded with a header indicating which command, target, and type (stdout, stderr) the output is for.

In the future, the ability to query specific historical logs beyond the most recent will be provided.

Tailing logs

In addition to historical logs, run can optionally stream command output to a socket. As commands are often parallelized, this socket is interleaved with blocks of log output from each command. Each block is preceded by a header containing the command, target, and type. For example:

monorail log tail --stdout --stderr

When this is running in a separate shell process, run commands will connect to it and send logs. When those logs are received, they are printed by the tail process.

APIs

log show

log show [--target, -t <1> <2> ... <M>] [--commands, -c <1> <2> ... <N>] <--stdout | --stderr>

Retrieves compressed historical logs for the provided stream types, targets and/or commands, and prints them. For more details, see monorail log show -h.

log tail

log tail [--target, -t <1> <2> ... <M>] [--commands, -c <1> <2> ... <N>] <--stdout | --stderr>

Starts a socket server with filters for the provided stream types, targets and/or commands, and prints them as received. This must be run as a separate process. For more details, see monorail log tail -h.

Results

In addition to logs, the result of each run API call is stored in the monorail out directory.

APIs

result show

Retrieves run results and displays them.

In the future, the ability to query specific historical results beyond the most recent will be provided.

Checkpoint

The checkpoint is what enables the change detection and incremental execution features of monorail. It is used as a starting point for change detection, allowing for a subset of changes in a repo to be queried. When the checkpoint is not present all targets are considered to be changed.

The checkpoint is essentially a marker in history, and may also include files not present in history. Effective use of monorail involves regular manipulation of this checkpoint. For example, as files within targets are modified, each target accumulates in the list of changed targets. Over time, this list can become unwieldy or wasteful to execute commands for repeatedly. When this occurs, the solution is to run the set of commands that would "validate" the changed targets, e.g. "if all of these commands pass, I'm done with that target", and then update the checkpoint. For example, this is a useful pattern:

monorail run -s validate && monorail checkpoint update -p

This runs the validate sequence, which you might define with a full suite of checks like check, build, unit-test, and integration-test, for all changed targets. If that fully succeeds for all targets involved, we then update the checkpoint to the latest point in history and include any pending edits on the local filesystem. Now, it will appear as if no targets are changed. Then, you can repeat this process as you make more changes. This ensures that your run invocations are only operating on the latest changes, and not on older ones that you've already checked.

Updating the checkpoint

The checkpoint can be updated to any value supported by the change provider backend.

Git backend

For example, git checkpoint ids are SHAs pointing to git objects, such as a commit or tag. When you update the checkpoint, you can optionally provide one as the id or allow monorail to select the SHA for the current HEAD. For example:

monorail checkpoint update

This will update the checkpoint to the SHA pointed to by the current HEAD reference.

monorail checkpoint update -i $(git rev-parse HEAD~1)

This will update the checkpoint to the SHA of the commit previous to the current HEAD reference.

Pending

As you work in a repository, many changes are not yet part of version control history, and these files and their checksums can be captured and stored in the checkpoint at any time. For example:

monorail checkpoint update --pending

For the git change provider, this will update the checkpoint with the current HEAD reference and include all uncommitted, unstaged, and staged changes in the checkpoint.

APIs

checkpoint show

Displays the current checkpoint.

checkpoint update

checkpoint update [--id, i <id>] [--pending, -p]

Updates the checkpoint id and pending array. For more details, see monorail checkpoint update -h.

checkpoint delete

Deletes the checkpoint, reverting the repository to its default change detection state (all targets changed).

Analyze

Many of the features of monorail involve an analysis of changes, targets, and the target graph. For example, this analysis is internally used by run to perform graph-guided command execution, and it's often useful to first see this analysis before running commands.

All targets

By default, analyze will act over all targets in the graph. For example:

monorail analyze --target-groups

This will display the currently affected targets and how they fall into target groupings. You can also specify --changes and --change-targets for a more detailed breakdown how each change was seen by monorail, but this is typically not useful beyond debugging of monorail itself.

APIs

analyze

analyze [[--changes] [--change-targets] [--target-groups] | --all] [--targets, -t <1> <2> ... <N>

Display an analysis containing changes, change targets, and/or target groups. Can be optionally scoped to subtrees of targets intead of the entire graph.

Out

Monorail stores various artifacts and tracking information in a directory, by default monorail-out. This location can be customized in Monorail.json by setting the out_dir field.

Deleting the output directory

While the out directory contains caches, run results, and the checkpoint, it can be deleted freely. For example:

monorail out delete --all

This will delete the checkpoint, all results, and all logs.

In the future, more selective deletion will be supported.

APIs

out delete

Development setup

monorail is written in Rust, so working on it is straightforward; this will build the project and run the tests:

cargo build
cargo test -- --nocapture

Dependencies

~15–27MB
~370K SLoC