Languages
Although moon is currently focusing on the JavaScript ecosystem, our long-term vision is to be a multi-language task runner and monorepo management tool. To that end, we've designed our languages to work like plugins, where their functionality is implemented in isolation, and is opt-in.
We do not support third-party language plugins at this time, but are working towards it!
Enabling a language
moon supported languages are opt-in, and are not enabled by default. We chose this pattern to avoid unnecessary overhead, especially for the future when we have 10 or more built-in languages.
To enable a supported language, simply define a configuration block with the language's name in
.moon/toolchain.yml
. Even an empty block will enable the language.
# Enable Node.js
node: {}
# Enable Node.js with custom settings
node:
packageManager: 'pnpm'
# Enable Deno
deno: {}
For unsupported languages, use the system toolchain. Continue reading to learn more!
System language and toolchain
When working with moon, you'll most likely have tasks that run built-in system commands that do not
belong to any of the supported languages. For example, you may have a task that runs git
or
docker
commands, or common commands like rm
, cp
, mv
, etc.
For these cases, moon provides a special language/toolchain called system
, that is always enabled.
This toolchain is a catch-all, an escape-hatch, a fallback, and provides the following:
- Runs a system command or a binary found on
PATH
. - Wraps the execution in a shell.
To run system commands, set a task's toolchain
setting to "system".
tasks:
example:
command: 'git status'
toolchain: 'system'
Tier structure and responsibilities
As mentioned in our introduction, language support is divided up into tiers, where each tier introduces more internal integrations and automations, but requires more work to properly implement.
Internally each tier maps to a Rust crate, as demonstrated by the graph at the top of the article.
Tier 0 = Unsupported
The zero tier represents all languages not directly supported by moon. This tier merely exists as a mechanism for running non-supported language binaries via the system toolchain.
tasks:
example:
command: 'ruby'
toolchain: 'system'
Tier 1 = Language
The first tier is the language itself. This is the most basic level of support, and is the only tier that is required to be implemented for a language to be considered minimally supported. This tier is in charge of:
- Declaring metadata about the language. For example, the name of the binary, supported file extensions, available dependency/package/version managers, names of config/manifest/lock files, etc.
- Helpers for parsing lockfiles and manifest files, and interacting with the language's ecosystem (for example, Node.js module resolution).
- Mechanisms for detecting the language of a project based on config files and other criteria.
- Maps to a project's
language
setting.
language: 'javascript'
Tier 2 = Platform
The second tier requires the language functionality from tier 1, and eventually the toolchain functionality from tier 3, and provides interoperability with moon's internals. This is the most complex of all tiers, and the tier is in charge of:
- Determining when, where, and how to install dependencies for a project or the workspace.
- Loading project aliases and inferring implicit relationships between projects.
- Syncing a project and ensuring a healthy project state.
- Hashing efficiently for dependency installs and target runs.
- Prepending
PATH
with appropriate lookups to execute a task. - Running a target's command with proper arguments, environment variables, and flags.
- Maps to a project's
toolchain.default
or task'stoolchain
setting. - Supports a configuration block by name in
.moon/toolchain.yml
.
tasks:
example:
command: 'webpack'
toolchain: 'node'
node: {}
Tier 3 = Toolchain
The third tier is toolchain support via proto. This is the final tier, as the toolchain is unusable unless the platform has been entirely integrated, and as such, the platform depends on this tier. This tier handles:
- Downloading and installing a language into the toolchain.
- Installing and deduping project dependencies.
- Detecting appropriate versions of tools to use.
- Determining which binary to use and execute targets with.
- Supports a
version
field in the named configuration block in.moon/toolchain.yml
.
node:
version: '18.0.0'