multidir: Simple tool for running commands in multiple directories
Modules
- Multidir
- Commands
- Multidir.Commands.Discover
- Multidir.Commands.List
- Multidir.Commands.Register
- Multidir.Commands.Run
- Multidir.Commands.Where
- Multidir.Iterate
- Multidir.ReadConfig
- Multidir.RunTask
- Multidir.Selection
- Commands
- TOML
- Decode
- TOML.Decode.Extra
- Decode
Downloads
- multidir-0.1.0.0.tar.gz [browse] (Cabal source package)
- Package description (as included in the package)
Maintainer's Corner
For package maintainers and hackage trustees
Candidates
- No Candidates
Versions [RSS] | 0.1.0.0 |
---|---|
Change log | CHANGELOG.md |
Dependencies | base (>=4.7 && <5), cond (>=0.5 && <0.6), containers (>=0.7 && <0.8), directory (>=1.3 && <1.4), filepath (>=1.5 && <1.6), filepattern (>=0.1.3 && <0.2), multidir , optparse-applicative (>=0.18.1 && <0.19), process (>=1.6 && <1.7), text (>=2.1 && <2.2), toml-reader (>=0.2.1 && <0.3) [details] |
License | BSD-3-Clause |
Copyright | James Cranch 2025 |
Author | James Cranch |
Maintainer | cranch@cantab.net |
Category | Command Line Tools |
Home page | https://github.com/jcranch/multidir#readme |
Bug tracker | https://github.com/jcranch/multidir/issues |
Source repo | head: git clone https://github.com/jcranch/multidir |
Uploaded | by jcranch at 2025年04月05日T17:01:26Z |
Distributions | |
Executables | muld |
Downloads | 0 total (0 in the last 30 days) |
Rating | (no votes yet) [estimated by Bayesian average] |
Your Rating |
|
Status | Docs not available [build log] All reported builds failed as of 2025年04月05日 [all 2 reports] |
Readme for multidir-0.1.0.0
[back to package description]Multidir ("muld
"): run tasks in multiple directories
This application exists to run tasks in several directories sequentially. The motivating use is to view the status of, or fetch remote changes to, multiple version control repositories.
The design differs from some other superficially similar applications in that it is designed to be flexible: if you maintain a set of repositories using several different version control systems, requiring completely different commands to achieve analogous ends in each directory, one can easily do so (and, in particular, there is no built-in git-centric behaviour).
Getting started (without explanations)
- Clone the repository, and run
stack install
in it. - Copy
tasks.toml
andrecipes.toml
fromexamples/
to~/.config/muld/
. - Run
muld discover
on the directory containing all your projects. - Inspect
~/.config/muld/projs.toml
by hand to remove anything unwanted, and to add extra tags as desired. - Now you can run
muld status
, andmuld fetch
to fetch changes.
Steps 1 and 2 above can be replaced by:
- Run
cabal update
andcabal install multidir
. - Copy and paste
tasks.toml
andrecipes.toml
from the repository on Github to~/.config/muld
.
Getting started (with explanations)
You can build it from source by cloning the repository and running
stack install
in it. This of course requires you to have installed
Haskell/stack.
Configuration consists of providing three TOML files:
-
projs.toml
lists the directories to be acted upon (together with tags which classify them). -
tasks.toml
lists the tasks which may be run upon those directories, and explains how to do them; differing implementations can be given for differing tags. -
recipes.toml
is used for autodiscovery; it contains instructions on what makes a directory suitable for discovery.
Vaguely useful samples of recipes.toml
and tasks.toml
can be found
in the examples/
directory and copied in. Once recipes.toml
is in
place, one can generate a projs.toml
by running muld discover
(see
below).
These configuration files are each searched for in the following order:
- Their locations can be supplied on the command-line (see below);
- Failing that, their locations are deduced from the environment
variables
$MULD_TASK_FILE
$MULD_PROJ_FILE
and$MULD_RECIPE_FILE
, if set; - Failing that, their locations are taken as
tasks.toml
,projs.toml
andrecipes.toml
in$XDG_CONFIG_HOME/muld
(if that environment variable is set); - Failing that, their locations are taken as
tasks.toml
,projs.toml
andrecipes.toml
in$HOME/.config/muld
(if that environment variable is set); - Failing that, an error is raised.
For most users, steps 3 and 4 have the same results.
Command-line syntax
The following commands are built in. In every case there is
command-line assistance (type muld COMMAND --help
):
-
muld discover
finds all subdirectories of the given directory matching the patterns inrecipes.toml
, and adds them toprojs.toml
. -
muld register
adds the given directory toprojs.toml
. -
muld list
lists the directories inprojs.toml
. -
muld run
runs a supplied command in every directory. -
muld where
list the locations of config files (useful for debugging, perhaps).
In addition, every task in tasks.toml
gives another command which
can be called; that task is run in every directory.
Selecting directories
Those commands listed above which operate on a collection of
directories (list
, run
and the task commands) can be put to work
on a user-defined set of directories. By default, it is those with the
default
tag, but this can be adjusted using the -s
flag.
The syntax is as follows:
-
muld list -s important
will list those repositories with theimportant
tag; -
muld list -s {/home/fred}
will list those repositories which are under/home/fred
; -
these can be combined with the binary operators
&
(and),|
(or) and,\
(difference). Hencemuld list -s important\scary
will list the repositories which areimportant
but notscary
; -
parentheses are permitted for complex expressions;
-
for ease, if a specification starts with an operator, it is treated as though
default
was specified first. Somuld list -s &interesting
lists the directories which are bothdefault
andinteresting
.
There is an all
tag, which by default is added to all
repositories. One can remove the all
tag from a repository, if
desired, but maybe that's not a very good idea.
Defining tasks
User-defined tasks are in tasks.toml
. They are listed by name, and
by what tag they operate on (allowing the same task to be run
differently for different types of directory).
The rule is that, for each directory, the tags are gone through in order; the first tag with an implementation for that task is the one which is used.
Here's an example:
[fetch]
description = "Downloads changes"
[fetch.git]
run = "git -c color.ui=always fetch"
[fetch.jj]
run = "TERM=xterm-color jj git fetch"
This defines a task which can be run on any directory with tags git
or jj
.
Defining recipes
Recipes for registering directories are in recipes.toml
. Each recipe
consists of:
-
the files required to trigger that recipe;
-
the tags added if that recipe is triggered;
-
the
discover
flag (which defaults toTrue
) which says whether this recipe is enough on its own to discover a directory (or merely add extra tags to it if discovered for other reasons).
General advice
In most setups, the default shell is not a login shell (dash
rather
than bash
under Debian, for example), and has not had read .bashrc
or anything. Hence one might need to be careful in defining tasks:
-
specifying paths exactly if they're in
$PATH
in a login shell but not in the default shell; -
reassuring processes that they're okay to assume colour features for output (for example, prefacing jujutsu commands with
TERM=xterm-color
, and replacinggit
withgit -c color.ui=always
).
Prior art
There are many other applications which do vaguely similar jobs. Examples include:
-
Mani is heavyweight, fairly git-centric, and written in go.
-
Myrepos is nice, but old and unloved; it is written in javascript for node.js.
Development
The projects in TODO.md are probably all achievable, and I will do them when I can be bothered (or when someone else can be bothered).
I'd be happy to take suggestions for extra recipes or extra implementations of tasks. I'm happy to take suggestions of extra tasks (and if they're a bit specialist in nature, I may start a new directory containing suggestions for more niche tasks).
Several of the other available applications support parallelism, which we don't (at least not yet, and it's not high on my list of priorities).
Bug reports are very welcome (just email me); there probably are some!