Go modules, private repos and gopath
Asked Answered
V

5

28

We are converting our internal codebase from the dep dependency manager to go modules (vgo or built in with go1.11.2). Imagine we have code like this:

$GOPATH/src/mycompany/myprogram/main.go:

package main

import (
        "fmt"
        lib "mycompany/mylib" )

func main() {
        fmt.Println("2+3=", lib.add(2, 3)) 
}

$GOPATH/src/mycompany/myprogram/go.mod:

module mycompany/myprogram

(it doesn't have any dependencies; our real-world code does).

$GOPATH/src/mycompany/mylib/lib.go:

package mylib

func Add(x int, y int) int {
        return x + y
}

I didn't module-ize this code; it doesn't seem to matter whether I do or don't.

These are trivial examples but our internal code follows a similar structure as this worked historically.

Since these directories are on the Gopath, export GO111MODULE=auto still builds as before and this works fine (modules not used because we are on the gopath). However, when I set export GO111MODULE=on I immediately get the error:

build mycompany/myprogram: cannot find module for path mycompany/mylib

So I did some research and I would like to validate my understanding. First let me say our old approach worked, but I am more interested in changing to use go modules as it appears to be where the go project itself is headed. So.

  1. It seems the intention of the golang authors was that "dotless" paths belong to the standard repository only; that is there should be a binding between domain name and project. We don't use go get on our internal project, unsurprisingly. Here is the source specifically:

    Dotless paths in general are reserved for the standard library; go get has (to my knowledge) never worked with them, but go get is also the main entry point for working with versioned modules.

    Can anyone with more knowledge of golang than me confirm this?

  2. My key assumption is that once go decides to use modules, all dependencies must be modules and the gopath becomes somewhat irrelevant, except as a cache (for downloaded modules). Is this correct?

  3. If this is true, we need to use a private gitlab (in our case) repository on the path. There's an open issue on handling this that I'm aware of so we can implement this if necessary. I'm more interested in the consequences, specifically for iterating in the private repositories. Previously we could develop these libraries locally before committing any changes; now it seems we have a choice:

    1. Accept this remote dependency and iterate. I was hoping to avoid needing to push and pull remotely like this. There are workarounds to needing an internet connection if strictly necessary.
    2. Merge everything into one big git repository.

If it matters, I'm using go version go1.11.2 linux/amd64 and my colleagues are using darwin/amd64. If it helps, my golang is exactly as installed by Fedora's repositories.

So, tl;dr, my question is: are go modules all-or-nothing, in that any dependency must be resolved using the module system (go get, it seems) and the gopath has become redundant? Or is there something about my setup that might trigger this to fail? Is there some way to indicate a dependency should be resolved explicitly from the gopath?

Updates since asking the question:

  1. I can move myprogram out of the gopath. The same issue occurs (mylib has been left in the gopath).
  2. I can run, or not run, go mod init mycompany/mylib in the mylib directory; it makes no difference at all.
  3. I came across Russ Cox's blog post on vgo. My concerns about offline development that I tried not to dive into too far are resolved by $GOPROXY.
Volumetric answered 28/11, 2018 at 9:39 Comment(8)
It look like you never ran go mod init mycompany in $GOPATH/src/mycompany, therefore there is no module yet. // The comment about dotless paths you quoted is by bcmills, one of the Go maintainers (or author, even?). You can take that as authorative. // Modules are not "all-or-nothing". You can depend on non-modules just fine.Keek
@Keek I left that out of my question, but I did (otherwise it wouldn't build as a module, no?). I'll edit that into the question. So if this is true my question becomes: if I can depend on non-modules, why doesn't it work? :PVolumetric
Ah, I see now. You have to add a replace statement in mycompany/myprogram to make this work.Keek
@Keek this sounds like exactly what I need. I added replace mycompany/mylib => ../mylib/ to my example, didn't work. So, domain name issue. So I renamed all mycompany references to mycompany.com... and now the build (of the example in my Q) hangs until the internet times out and then gives me the usual build mycompany.com/myprogram: cannot find module for path mycompany.com/mylib :(Volumetric
So I guess we need to combine the two approaches; build from private but remote repositories with url-like imports and then we can replace locally to speed up dev.Volumetric
@Volumetric I am in the same boat and get this error. Were you able to find any solution to your problem?Mumble
@TarunTyagi not that was satisfying with gitlab. We could only use top level repositories (gitlab.com/org/repo) and not gitlab.com/org/subproj/repo) because of some crazy gitlab feature. We also had to configure ssh instead of https because there's no token support on gitlab. We've since migrated to github, thankfully.Volumetric
Go vanity urls is something I am also investigating right now as a way to provide the required redirection. It still requires the remote end to support the go modules protocol to discover the correct repo format, and some kind of token support for http auth, but it can at least allow you to redirect go.yourcompany.com/repo -> gitlab.com/org/subproj/repo.git (.git required because gitlab doesn't provide support in subprojects for go modules).Volumetric
D
29

I use a workaround with GITHUB_TOKEN to solve this.

  1. Generate GITHUB_TOKEN here https://github.com/settings/tokens
  2. export GITHUB_TOKEN=xxx
  3. git config --global url."https://${GITHUB_TOKEN}:[email protected]/mycompany".insteadOf "https://github.com/mycompany"

Additionally, use GOPRIVATE. Upstream docs: Private modules

Din answered 28/11, 2018 at 11:29 Comment(4)
Thanks. We use gitlab, but a similar workaround exists for this as well and we'll be using it for sure.Volumetric
This is no longer sufficient, you need to also populate GOPRIVATE now.Kubis
What to do if your are working with a private BitBucket repository?Kubis
I did not set GOPRIVATE and this heps me.Olnay
N
12

I wrote up a solution for this on Medium: Go Modules with Private Git Repositories.

The way we handle it is basically the same as the answer above from Alex Pliutau, and the blog goes into some more detail with examples for how to set up your git config with tokens from GitHub/GitLab/BitBucket.

The relevant bit for GitLab:

git config --global \
  url."https://oauth2:${personal_access_token}@privategitlab.com".insteadOf \
  "https://privategitlab.com"

#or 

git config --global \
  url."https://${user}:${personal_access_token}@privategitlab.com".insteadOf \
  "https://privategitlab.com"

I hope it's helpful.

Navigation answered 24/5, 2019 at 21:24 Comment(3)
Is there a way to use http instead of https. We have a legacy git server behind a firewall where everything is http and the change above does not work when invoking the go command (which is trying to fetch the repo via https on port 443)Eyeless
This is a common solution, but it fails with https://repos.example.com/my-library?go-get=1 request made by go tool before reaching cloning phase.Chinchy
You should be more specific and state that your answer, where it mentions BitBucket, is purposely for BitBucket server not BitBucket cloud. Luckily you mention the .netrc solution in your article and this happens to work in go @1.13 onwards by also specifying the GOPRIVATE](golang.org/doc/go1.13#modules) environment variable.Kubis
I
12

I use ls-remote git command to help resolve private repo tags and go get after it.

$ go env GO111MODULE=on
$ go env GOPRIVATE=yourprivaterepo.com
$ git ls-remote -q https://yourprivaterepo.com/yourproject.git
$ go get
Iridis answered 8/1, 2020 at 6:45 Comment(2)
You can use go env -w GOPRIVATE for private Github repositories as well: go env -w GOPRIVATE=github.com/username/*Kaffir
if nothing helps: https://mcmap.net/q/503708/-go-build-can-39-t-find-a-revisionOtocyst
P
0

I try to fix this issue and then solved with this... you need to set up 3 ENV on your local machine

GONOPROXY=privategitlab.company.com
GONOSUMDB=privategitlab.company.com
GOPRIVATE=privategitlab.company.com

this will solved your issue

Pasha answered 1/12, 2023 at 2:35 Comment(0)
I
0

At first you should set GOPRIVATE

go env -w GOPRIVATE=privategitlab.com

then you have two methods:

first method:

git config --global url."https://{userName}:{access_token}@privategitlab.com".insteadOf "http://privategitlab.com"

second method:

update or create ~/.netrc file and put below line and save it

machine privategitlab.com login {username} password {access_token}

I prefer the second method because if you have group and subgroup in git, first method don't work

Inventory answered 12/2, 2024 at 7:11 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.