Skip to main content

Reporoot workspace manager

Project description


title: Home layout: home nav_order: 1 permalink: /

Software lives in multiple repos. Even a monorepo has upstream dependencies, vendored libraries, etc., where the source of truth is elsewhere. The code you work with almost always spans repos you own, repos you depend on (or forked), and maybe even repos you just want around as a reference.

The value of a monorepo is the workspace — all your code in one directory tree, so every tool that touches the filesystem works across all of it. Reporoot gives you the workspace without merging repos. A project .repos file declares which repos belong together; reporoot activate wires them into ecosystem workspace mechanisms so cross-repo imports resolve locally. Repos stay sovereign: normal clones, normal branches, normal git.

reporoot/
├── github/
│   ├── myorg/
│   │   ├── server/          # your code
│   │   ├── web/             # your code
│   │   └── protocol/        # shared types, used by both
│   └── socketio/
│       └── engine.io/       # your fork with reconnection fixes
├── projects/
│   └── web-app/
│       ├── web-app.repos    # which repos, what roles
│       └── web-app.lock.repos
├── package.json             # generated: npm workspaces
├── go.work                  # generated: Go workspace
└── web-app.code-workspace   # generated: VS Code workspace

One reporoot activate web-app generates the ecosystem workspace files, and import { Thing } from '@myorg/protocol' just works — resolved locally, no file:../../ paths.

Why not just...

...use a monorepo? You'd need everyone to buy in, and you still have external deps, forks, and reference code outside the repo. The coordination problem exists either way.

...use git submodules? Submodules take ownership: detached HEAD by default, can't adopt existing clones, the parent controls the relationship. For repos you don't control, this is backwards.

...clone repos into a flat directory? Works for one person who set it up. Fails for: reproducing on a new machine, onboarding someone, remembering why a repo was cloned six months later.

Reporoot is the layer in between — structure and reproducibility without giving up repo independence.

Install

pipx install reporoot

Quickstart

Starting fresh:

mkdir ~/reporoot && cd ~/reporoot
reporoot fetch myorg/web-app    # clones project + all its repos

Adopting existing repos:

cd ~/reporoot
reporoot activate web-app       # wires existing repos into workspace

reporoot activate reads the project's .repos file and runs integration hooks — generating npm workspaces, go.work, uv workspaces, gita config, and a VS Code workspace — so cross-repo imports resolve locally without path hacks.

Three layers

1. The directory tree

Repos live under one root at {registry}/{owner}/{repo}/. This is just a directory convention — no tooling required. But every tool benefits: grep finds results across repos, editors navigate the full tree, agents see all the code.

2. Ecosystem wiring

reporoot activate generates per-ecosystem workspace files from the active project's repos:

Ecosystem Generated file What it enables
Node (npm) package.json with workspaces import { x } from '@myorg/shared' resolves locally
Go go.work import "myorg/shared" resolves locally
Python (uv) pyproject.toml with [tool.uv.workspace] editable installs across repos
gita .gita/ config gita ll, gita super pull, role-based groups
VS Code {project}.code-workspace single-root workspace, non-project repos hidden

Each integration auto-detects relevant repos (has package.json? include in npm workspaces) and skips gracefully if the tool isn't installed.

3. Reproducibility

A .repos file declares which repos belong to a project. reporoot lock snapshots every repo's HEAD into a .lock.repos file — the multi-repo equivalent of a monorepo commit hash.

# On a new machine — one command to reproduce the full workspace
reporoot fetch myorg/web-app

sha256sum web-app.lock.repos gives a single fingerprint for the entire project state.

Projects

Projects are named views over subsets of repos, with roles that signal how freely code should be changed:

# projects/web-app/web-app.repos
repositories:
  github/myorg/server:
    type: git
    url: https://github.com/myorg/server.git
    version: main
    role: primary              # your code — change freely
  github/myorg/protocol:
    type: git
    url: https://github.com/myorg/protocol.git
    version: main
    role: primary
  github/socketio/engine.io:
    type: git
    url: https://github.com/myorg/engine.io.git
    version: main
    role: fork                 # your fork — changes ideally go upstream

Repos can appear in multiple projects with different roles. Switching projects is fast — repos are already on disk, only the ecosystem wiring changes:

reporoot activate mobile-app
# Regenerates package.json, go.work, etc. for mobile-app's repos

Commands

Command What it does
reporoot Show active project and help
reporoot activate {project} Set active project, run integration hooks
reporoot deactivate Remove derived files, clear active project
reporoot add {url|path} Clone a repo and register it in the active project
reporoot remove {path} Remove a repo from the active project, re-run hooks
reporoot fetch {source} Clone a project and all its repos
reporoot resolve Print the workspace root path
reporoot lock Snapshot repo versions for the active project
reporoot lock-all Snapshot repo versions for all projects
reporoot check Run convention enforcement checks

Docs

  • Conventions — full design: directory layout, projects, roles, workflows, adjacent tools
  • Integrations — how each integration works, generated file formats, configuration

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

reporoot-0.3.1.tar.gz (42.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

reporoot-0.3.1-py3-none-any.whl (33.9 kB view details)

Uploaded Python 3

File details

Details for the file reporoot-0.3.1.tar.gz.

File metadata

  • Download URL: reporoot-0.3.1.tar.gz
  • Upload date:
  • Size: 42.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for reporoot-0.3.1.tar.gz
Algorithm Hash digest
SHA256 00d55a751132362ef8291176777f06cc05de6acc7f06e1fb49006876ef2bf4ed
MD5 76fee02745a0475d55cd353bc9b48649
BLAKE2b-256 d51f071f09a3ccab9dc219a70979ec9e8aee2720cbba1bb8e41b5128cd50ae1d

See more details on using hashes here.

Provenance

The following attestation bundles were made for reporoot-0.3.1.tar.gz:

Publisher: publish.yml on cwalv/reporoot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file reporoot-0.3.1-py3-none-any.whl.

File metadata

  • Download URL: reporoot-0.3.1-py3-none-any.whl
  • Upload date:
  • Size: 33.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for reporoot-0.3.1-py3-none-any.whl
Algorithm Hash digest
SHA256 05f2e6eee27c16961f4b71643fa751a8a8ea3d21213bdc8b67a8f10dbac9b18e
MD5 49aec0821a821601e19c36b0ba8f9dca
BLAKE2b-256 9c6bc115dc0dd120ecd5ef8eb175b23a66c13cff9aa4d4292f53202b82de98f9

See more details on using hashes here.

Provenance

The following attestation bundles were made for reporoot-0.3.1-py3-none-any.whl:

Publisher: publish.yml on cwalv/reporoot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page