/workspace-go-libp2p

workspace for go-libp2p contributors

Primary LanguageShell

go-libp2p workspace

go-libp2p is a p2p networking stack based on the principle of modularity. Its modules are scattered across a number of Git repos. We have recently migrated from gx to gomod as the primary dependency and build management tool.

To make developing with gomod easy, this meta-repository aggregates all go-libp2p repos under a single roof via git submodules, and provides a management script (./workspace.sh) for automating common workflows.

⚠️ Work in progress

This setup is currently at pilot stage and heavily WIP. Feedback and contributions are welcome. Some wishlist items include:

  • go mod <> IPFS integration, to consume and publish content-addressed dependencies easily. Stebalien/ipgo is one possible direction.
  • Automating the "bubbling" and update workflow, via scripts or tooling such as renovate.
  • Shared libp2p/IPFS workspace.
  • Nightly master builds.

👉 Prerequisite: Git pre-commit hook

This pre-commit hook will alert you when you are trying to commit go.mod file with local replace directives. This is useful pattern for developing, but has no place on remote.

It is best to install it as global githook. Instructions here.

Usage

Getting started

This will initialise the submodules by cloning repos into their relevant subdirectories.

$ git clone --recursive <this repo>

Switch to local module resolution

Interlinks all repos for local development through replace directives.

$ ./workspace.sh local

Switch to remote module resolution

Removes the replace directives added by local.

$ ./workspace.sh remote

Stash local changes and update all repos from origin/master

$ ./workspace.sh remote    # to reset go.mod files to original
$ ./workspace.sh refresh

Background: Why is this necessary?

Gomod is great for mono-repo projects that depend on 3rd party dependencies, but there are some challenges when working with an intervowen set of modularised packages:

  • Go tools resolve module versions from remotes.
  • Developers want to make changesets across a number of local repos and have them visible by interdependent modules.

Aggregating modules under one roof, and using go mod replace directives to interlink them locally via replace directives, provides a neat GOPATH-like development experience.

Detour: GOPATH vs go modules

The GOPATH monolith is irrelevant in the Go modules universe. Now you can check out modules anywhere in the filesystem.

Running go get from inside a Go module will download packages into the global go mod cache ($GOPATH/pkg/mod), where they are indexed by module path and version.

All go tools (go build, go test, etc.) are now module-friendly. They automatically add require directives go go.mod for new imports in code, download packages from remotes, and more. It's like magic.

NOTE: Running go get in a tree without a go.mod will still place that package under your GOPATH. Running go commands from within your GOPATH will behave like before (i.e. no module-based builds) even if the package has a go.mod, unless you explicitly set GO111MODULE=on.

Fun fact: Go mod makes bold assertions about immutability, and this transpires even to file permissions under the module cache. A quick ls -l therein shows that go strips away write permissions from downloaded modules, even for the owner.

Recommended reading

Familiarise yourself with Go modules. In particular, pay attention to how minimal version selection works.

Reacquaint yourself with go commands. Most of the commands you know and have come to love (go get, go build, go test) now deal with modules transparently. Suggested read: go command manual, and pay special attention to the sections dedicated to modules behaviour.