git clone https://github.com/YPares/rigup.nix
T=$(mktemp -d) && git clone --depth=1 https://github.com/YPares/rigup.nix "$T" && mkdir -p ~/.claude/skills && cp -r "$T/riglets/agent-rig-system" ~/.claude/skills/ypares-rigup-nix-agent-rig-system && rm -rf "$T"
riglets/agent-rig-system/SKILL.mdAgent 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
files documenting processes and recipesreferences/*.md - 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
to compose riglet modulesbuildRig - 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:
: evaluates riglet modules and ensures they comply with the riglet schema used by rigup. Returns the rig as an attrset:buildRig{ toolRoot = <derivation>; meta = { <riglet> = {...}; }; docAttrs = { <riglet> = <derivation>; }; docRoot = <derivation>; home = <derivation>; shell = <derivation>; }
: inspects theresolveProject
folder of a project and itsriglets/
to find out which riglets and rigs it defines. It callsrigup.toml
for each rig in thebuildRigrigup.toml
: generates a markdown+XML manifest file describing the contents of a rig, primarily for AI agent's consumptiongenManifest
: creates a set of utility functions to be used to define riglet Nix modulesmkRiglib
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
resolveProject
- Auto-discovered riglet modulesriglets.<riglet>
- Output ofrigs.<system>.<rig>
for each discovered rig:buildRig
- Folder derivation. Tools combined via nixpkgstoolRoot
function (bin/, lib/, share/, etc.) and wrapped (when needed) to fix their XDG_CONFIG_HOMEbuildEnv
- Folder derivation. The combined config files for the whole rig, with config files for all rig's wrapped tools.configRoot
- Attrset. Per-riglet metadata, as defined by the riglet's modulemeta.<riglet>
- Folder derivation. Per-riglet documentation folder derivationsdocAttrs.<riglet>
- Folder derivation. Combined derivation with docs for all riglets (one subfolder for each)docRoot
- Folder derivation. All-in-one directory for the rig: RIG.md manifest + .local/ + docs/ + .config/ foldershome
- Shell derivation (viashell
) exposing ready-to-use RIG_MANIFEST and PATH env varspkgs.mkShell
- Nix function. Adds riglets to a pre-existing rig: takesextend
and returns a new rig{newName, extraModules}
- A manifest for this rig, overridable with options to shorten included paths to avoid repeatedly including long explicit paths into the Nix storemanifest
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
buildRigbuildRig 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
shellThe 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
homeThe 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
entrypointThe
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
extract-md-tocThis 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
, orriglets/my-riglet.nix
for riglets with multiple supporting filesriglets/my-riglet/default.nix - Add the needed tools, documentation, metadata
- Define options (schema) and config (values) in this module
- Ensure the project has a top-level
that usesflake.nix
as mentioned above, so all the riglets will be exposed by the flakerigup.lib.resolveProject
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