Agent Rig System
Scannednpx machina-cli add skill YPares/rigup.nix/agent-rig-system --openclawAgent Rig System
Overview
A rig is a project-scoped collection of riglets that provide knowledge and tools for AI agents.
Core Concepts
Riglet
A riglet is executable knowledge packaged with its dependencies, as a Nix module:
- Metadata: When should this riglet be used, is it production-ready or experimental, etc.
- Knowledge: SKILL.md + detailed
references/*.mdfiles documenting processes and recipes - Tools: Nix packages needed to execute those recipes
- Configuration: Settings to adapt tools' behaviour to project context
Rig
A project-level structure that declares which riglets are active:
- Uses
buildRigto compose riglet modules - Builds combined tool environment declaratively
- Exposes riglets' tools and documentation
rigup
A Nix library and CLI tool: http://github.com/YPares/rigup.nix
rigup Nix library
Main functions:
buildRig: evaluates riglet modules and ensures they comply with the riglet schema used by rigup. Returns the rig as an attrset:{ toolRoot = <derivation>; meta = { <riglet> = {...}; }; docAttrs = { <riglet> = <derivation>; }; docRoot = <derivation>; home = <derivation>; shell = <derivation>; }resolveProject: inspects theriglets/folder of a project and itsrigup.tomlto find out which riglets and rigs it defines. It callsbuildRigfor each rig in therigup.tomlgenManifest: generates a markdown+XML manifest file describing the contents of a rig, primarily for AI agent's consumptionmkRiglib: creates a set of utility functions to be used to define riglet Nix modules
Defined in {{repoRoot}}/lib/default.nix.
rigup CLI tool
A Rust app. It provides convenient access to rig outputs, via commands like rigup build and rigup shell. This tool is meant for the user primarily. Agents should not have to call it directly.
Defined in {{repoRoot}}/packages/rigup
Riglet Structure
Riglets are Nix modules with access to riglib helpers
Example Riglet
# First argument: the defining flake's `self`
# Gives access to `self.inputs.*` and `self.riglets.*`
# Use `_:` if you don't need it
self:
# Second argument: module args from evalModules
{ config, pkgs, lib, riglib, ... }: {
# Riglet-specific options (optional)
options.myRiglet = {
myOption = lib.mkOption {
type = lib.types.str;
description = "Example option";
};
};
# Riglet definition
config.riglets.my-riglet = {
# Dependency relationship/Inheritance mechanism: if B imports A, then whenever B is included in a rig, A will automatically be included too
imports = [ self.riglets.base-riglet self.inputs.foo.riglets.bar ... ];
# Tools can be:
# - Nix packages: pkgs.jujutsu, pkgs.git, etc.
# - Script paths: ./scripts/my-script (auto-wrapped as executables)
tools = [
pkgs.tool1
pkgs.tool2
./scripts/helper-script # Becomes executable "helper-script"
];
# Metadata for discovery and context
meta = {
description = "What this riglet provides";
mainDocFile = "SKILL.md"; # Where to start reading the docs (SKILL.md by default)
intent = "cookbook"; # What the agent should expect from this riglet
whenToUse = [
# When the AI Agent should read/use this riglet's knowledge, recipes and tools
"Situation 1" # or
"Situation 2" # or
...
];
keywords = [ "keyword1" "keyword2" ];
status = "experimental"; # Maturity level
version = "x.y.z"; # Semantic version of riglet's interface (configuration + provided methods, procedures, docs...)
disclosure = lib.mkDefault "lazy" # How much to show about riglet in manifest
# mkDefault makes it possible for end users to override this in their rigup.toml
};
# Documentation file(s) (Skills pattern: SKILL.md + references/*.md)
docs = riglib.writeFileTree {
"SKILL.md" = ...; # A main documentation file
references = { # Optional. To add deeper knowledge about more specific topics, less common recipes, etc.
# SKILL.md MUST mention when each reference becomes relevant
"advanced.md" = ...;
"troubleshooting.md" = ...;
};
};
# Files can be defined either as inlined strings or nix file derivations/paths.
# Folders can be defined either as nested attrsets or nix folder derivations/paths,
# so if you have a ready to use folder you can do:
#docs = ./path/to/skill/folder;
# Configuration files (optional) for tools following the
# [XDG Base Directory Specification](https://specifications.freedesktop.org/basedir/latest/)
configFiles = riglib.writeFileTree {
# Built from a Nix attrset
myapp."config.toml" = riglib.toTOML {
setting = "value";
};
# Read from existing file
myapp."stuff.json" = ./path/to/stuff.json;
# Inlined as plain text
myapp."script.sh" = ''
#!/bin/bash
echo hello
'';
};
# EXPERIMENTAL: Prompt commands (slash commands for harnesses like Claude Code)
promptCommands.my-cmd = {
template = "Do something specific with $ARGUMENTS";
description = "What this command does";
useSubAgent = false;
};
};
# EXPERIMENTAL: MCP (Model Context Protocol) servers
mcpServers.some-local-mcp.command = pkgs.my-mcp-server;
mcpServers.some-remote-mcp = {
url = "https://...";
useSSE = true; # false by default
};
}
The full Nix module schema of a riglet is defined in {{repoRoot}}/lib/rigletSchema.nix.
Examples of actual riglets: {{repoRoot}}/riglets.
Metadata
When defining a riglet, the meta section specifies its purpose, maturity, and visibility. See references/metadata-guide.md for comprehensive details on:
- meta.intent - Primary focus (base, sourcebook, toolbox, cookbook, playbook)
- meta.status - Maturity level (stable, experimental, draft, deprecated, example)
- meta.version - Semantic versioning of the riglet's interface
- meta.broken - Temporary non-functional state flag
- meta.disclosure - Visibility control (none, lazy, shallow-toc, deep-toc, eager)
Implementation Utilities
See references/riglib-utilities.md for details on helper functions available via riglib:
- riglib.writeFileTree - Convert nested attrsets to directory trees
- riglib.useScriptFolder - Convert folder of scripts into wrapped tool packages
riglib is defined in {{repoRoot}}/lib/mkRiglib.nix
Experimental Features
WARNING: These features are still experimental and their schema may change.
Prompt Commands
Riglets can define reusable prompt templates (slash commands) for agent harnesses like Claude Code:
promptCommands.analyze = {
template = "Analyze $1 for potential issues";
description = "Perform code analysis";
useSubAgent = false; # Whether to run in a sub-agent
};
Templates use standard Claude command syntax: $ARGUMENTS for all args, or $1, $2, etc. for specific positional arguments.
MCP Servers
Riglets can provide MCP (Model Context Protocol) servers to extend agent capabilities:
mcpServers.my-tools = {
command = pkgs.my-mcp-server; # Package that starts the server
};
WARNING: API still experimental.
Cross-Riglet/Flake Interaction
Advanced patterns for composing riglets together and sharing configuration. See references/advanced-patterns.md for:
- Sharing configuration via
config - Dependencies and inheritance via
imports - Using packages from external flakes
Defining Rigs in Projects
Recommended: Use rigup.toml
Add a rigup.toml file to your project root:
[rigs.default.riglets]
self = ["my-riglet"]
rigup = ["git-setup"]
[rigs.default.config.agent.identity]
name = "Alice"
email = "alice@example.com"
Then use rigup.lib.resolveProject in your flake.nix:
{
inputs.rigup.url = "github:YPares/rigup.nix";
outputs = { self, rigup, ... }@inputs:
# Using the rigup flake directly as a function is equivalent to calling
# `rigup.lib.resolveProject`, as `rigup` defines the __functor attr.
#
# rigup follows the same pattern as the 'blueprint' flake (https://github.com/numtide/blueprint):
# - exposes one main "entrypoint" function, callable through the flake "object" itself
# - inspects user flake's inputs and repository's contents
# - constructs (part of) user flake's outputs
rigup {
inherit inputs;
# A unique name, used in error messages, to make it more explicit where mentioned riglets come from
projectUri = "some-username/some-project-name";
}
}
Advanced: Directly use buildRig for complex config
For config not representable in TOML:
{
inputs.rigup.url = "github:YPares/rigup.nix";
outputs = { self, rigup, nixpkgs, ... }@inputs:
let
system = "x86_64-linux";
pkgs = import nixpkgs { inherit system; };
in
pkgs.lib.recursiveUpdate # merges both recursively, second arg taking precedence
(rigup.lib.resolveProject {
inherit inputs;
projectUri = "...";
})
{
rigs.${system}.custom = rigup.lib.buildRig {
name = "my-custom-rig";
inherit pkgs;
modules = [
# A module from rigup:
rigup.riglets.git-setup
# A module defined directly inline:
{
# Complex Nix expressions
agent.complexOption = lib.mkIf condition value;
}
];
};
};
}
resolveProject outputs
riglets.<riglet>- Auto-discovered riglet modulesrigs.<system>.<rig>- Output ofbuildRigfor each discovered rig:toolRoot- Folder derivation. Tools combined via nixpkgsbuildEnvfunction (bin/, lib/, share/, etc.) and wrapped (when needed) to fix their XDG_CONFIG_HOMEconfigRoot- Folder derivation. The combined config files for the whole rig, with config files for all rig's wrapped tools.meta.<riglet>- Attrset. Per-riglet metadata, as defined by the riglet's moduledocAttrs.<riglet>- Folder derivation. Per-riglet documentation folder derivationsdocRoot- Folder derivation. Combined derivation with docs for all riglets (one subfolder for each)home- Folder derivation. All-in-one directory for the rig: RIG.md manifest + .local/ + docs/ + .config/ foldersshell- Shell derivation (viapkgs.mkShell) exposing ready-to-use RIG_MANIFEST and PATH env varsextend- Nix function. Adds riglets to a pre-existing rig: takes{newName, extraModules}and returns a new rigmanifest- A manifest for this rig, overridable with options to shorten included paths to avoid repeatedly including long explicit paths into the Nix store
resolveProject is defined in {{repoRoot}}/lib/resolveProject.nix.
Using a Rig
The user decides how they and their agent should use the rig: either via its shell, home or entrypoint output derivations.
In any case, the agent's focus should be is the RIG.md manifest file. This file lists all available riglets with:
- Name
- Description
- When to use each riglet
- Keywords for searching
- Documentation paths
Agents should read this file first to understand available capabilities.
buildRig output derivations
buildRig outputs a Nix attrset ("object") that notably contains several "all-in-one" derivations which all allow an AI agent to access the rig's tools and documentation.
Which derivation to use depends on what is the most convenient given the user's setup.
This section lists how and when to use each.
buildRig is defined in {{repoRoot}}/lib/buildRig.nix
shell output
The AI agent runs in a subshell: a $RIG_MANIFEST env var is set that contains the path to the RIG.md manifest the agent should read.
Also, $PATH is already properly set up by the subshell so all tools are readily usable.
# Start a rig as a sub-shell (the user should do that)
rigup shell ".#<rig>" [-c <command>...] # Does `nix develop ".#rigs.<system>.<rig>.shell" [-c <command>...]`
# Read the rig manifest
cat $RIG_MANIFEST
Advantages of using shell:
- No extra setup needed: a single command gets everything ready to use
- No risk of using an incorrect tool or config file if the agent misses a step
- Convenient to use when AI agent runs inside a terminal application (like claude-code)
home output
The AI agent reads from a complete locally-symlinked "home-like" folder.
The RIG.md manifest and an activate.sh script will be added at the root of this folder.
The activate.sh, once sourced, provides the needed PATH.
# Build complete home directory with tools + docs + config as a `.rigup/<rig>` folder at the top-level of the project (the user should do that)
rigup build ".#<rig>" # Does `nix build ".#rigs.<system>.<rig>.home"`
# Read the rig manifest to see what's available
cat .rigup/<rig>/RIG.md
# Source the activation script to use the tools
source .rigup/<rig>/activate.sh && git --version && other-tool ...
# Read documentation (paths shown in RIG.md)
ls .rigup/<rig>/docs/
cat .rigup/<rig>/docs/<riglet>/SKILL.md
Advantages of using home:
- Rig can be rebuilt without having to restart the agent's harness: home folder contents are just symlinks that can be updated, paths remain valid
- Manifest file is right next to doc files: can refer to them via short and simple relative paths
- More convenient to use in contexts where setting up env vars is impractical (e.g. AI agent running inside an IDE, like Cursor)
entrypoint output
The entrypoint output is special in that it does not exist unless some riglet sets it, by defining config.entrypoint.
It is mainly used to provide direct integration with common coding agent harnesses.
Similar to home and shell, entrypoint packages the whole rig as a Nix derivation, but this time as a wrapper shell script that starts the harness with the proper config files and CLI args.
rigup run <flake>#<rig> executes a rig's entrypoint.
Internally it just runs nix run <flake>#rigs.<system>.<rig>.entrypoint.
Claude Code integration is currently available via the claude-code riglet.
See references/harness-integration.md for more details.
Advantages of using entrypoint:
- More direct integration with the harness when such integration exists
More efficient Markdown reading: extract-md-toc
This riglet (agent-rig-system) comes with extract-md-toc. This is the tool that renders the inline table of contents of the rig manifests (for riglets with disclosure = "{shallow,deep}-toc";).
It can also be used to extract a similar ToC out of ANY Markdown file: e.g. extract-md-toc foo.md --max-level 3 will show all headers from # to ### with their line numbers.
It can also read from stdin: extract-md-toc - < foo.md
Defined in {{repoRoot}}/packages/extract-md-toc
Adding Riglets to a Rig
In the project defining the riglets OR in another one importing it as an input flake, either add riglets and their config to the rigs defined in the top-level rigup.toml file, or directly edit the flake.nix if more advanced configuration is needed.
In both cases, the flake should call rigup.lib.resolveProject (or just rigup, which contains a __functor attr which defers to resolveProject) to discover rigs and riglets, and the rigs should be under the rigs.<system>.<rig-name> output.
Creating New Riglets
In some project:
- Create
riglets/my-riglet.nix, orriglets/my-riglet/default.nixfor riglets with multiple supporting files - Add the needed tools, documentation, metadata
- Define options (schema) and config (values) in this module
- Ensure the project has a top-level
flake.nixthat usesrigup.lib.resolveProjectas mentioned above, so all the riglets will be exposed by the flake
If your rig contains riglet-creator, consult it for more detailed information about writing proper riglets.
Design Principles
- Knowledge-first: Docs are the payload, tools are dependencies
- Declarative: Configuration via Nix module options
- Composable: Riglets build on each other
- Reproducible: Nix ensures consistent tool versions
Source
git clone https://github.com/YPares/rigup.nix/blob/main/riglets/agent-rig-system/SKILL.mdView on GitHub Overview
Agent Rig System defines rigs as project-scoped collections of riglets that provide knowledge and tools for AI agents. Each riglet is executable knowledge packaged as a Nix module, with metadata, knowledge, tools, and configuration. A rig exposes riglets via buildRig and uses rigup for management and manifests.
How This Skill Works
Riglets are Nix modules with access to riglib helpers and contain metadata (when to use, production-ready vs experimental), knowledge (SKILL.md and references), tools (Nix packages or scripts), and configuration. The Rig uses buildRig to compose riglets into a coherent tool environment and expose documentation; rigup provides the library functions (buildRig, resolveProject, genManifest, mkRiglib) and a CLI for user-facing actions like rigup build and rigup shell.
When to Use It
- When provisioning an AI agent with project-specific knowledge and tools via riglets.
- When you need to declare and manage which riglets are active in a project using rigup.toml.
- When packaging executable knowledge as Nix modules for reproducible tool environments.
- When generating an agent-facing manifest describing rig contents (markdown+XML).
- When you want automatic dependency handling via riglet imports to compose rigs.
Quick Start
- Step 1: Inspect riglets/ and rigup.toml to understand defined riglets and rigs.
- Step 2: Run 'rigup build' to compose riglets into a unified environment and generate the manifest.
- Step 3: Use 'rigup shell' or consult the generated manifest for agent initialization and usage.
Best Practices
- Define riglets with clear metadata fields: description, mainDocFile, intent, whenToUse, keywords, status, version, and disclosure.
- Use imports to pull in base riglets so tools and knowledge can be shared across riglets.
- Keep tools simple and executable: include Nix packages and script paths (e.g., ./scripts/) that become executables.
- Document depth: place SKILL.md as the main doc and reference additional docs in references/*.md.
- Use genManifest to create agent-friendly manifests and keep rigup.toml up to date with rig versions and disclosures.
Example Use Cases
- A base riglet providing common data-processing tools (pkgs.jujutsu, pkgs.git) and SKILL.md guidance for data workflows.
- A riglet that exposes a script as an executable like 'helper-script' by placing it under ./scripts/.
- A rig where importing another riglet causes automatic inclusion of base capabilities (imports = [ self.riglets.base-riglet ... ]).
- Using rigup build to declaratively compose rigs and generate a toolRoot and manifest for agent consumption.
- Using rigup resolveProject to inspect riglets/ and rigup.toml to determine which rigs to build for a given project.