Skip to content
This repository has been archived by the owner on Sep 30, 2022. It is now read-only.

Latest commit

 

History

History
829 lines (621 loc) · 22.7 KB

README.md

File metadata and controls

829 lines (621 loc) · 22.7 KB

** REVIEW REQUIRED **

This guide is a WIP.


Examples

This section demonstrates what it's like to use the go tool with modules. Please follow along and experiment with variations as you do.

TODO: add a note here about ensuring that people are running Go 1.11 or later

TODO: add a note here to ensure people are not in their GOPATH, and some commentary about what it means to be in Go module mode (as opposed to GOPATH mode).... and that every reference to the go command that follows assumes we are in Go module mode.

Hello, world

Let's write an interesting “hello, world” program. Create a directory outside your GOPATH/src tree and change into it:

$ cd /home/gopher
$ mkdir hello
$ cd hello

Then create a file hello.go:

package main

import (
	"fmt"
	"rsc.io/quote"
)

func main() {
	fmt.Println(quote.Hello())
}

Create an empty go.mod file to mark the root of this module, and then build and run your new program:

$ go mod init github.com/you/hello
go: creating new go.mod: module github.com/you/hello
$ go build
go: finding rsc.io/quote v1.5.2
go: downloading rsc.io/quote v1.5.2
go: extracting rsc.io/quote v1.5.2
go: finding rsc.io/sampler v1.3.0
go: finding golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c
go: downloading rsc.io/sampler v1.3.0
go: extracting rsc.io/sampler v1.3.0
go: downloading golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c
go: extracting golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c
$ ./hello
Hello, world.

Notice that there is no explicit go get required. Plain go build will, upon encountering an unknown import, look up the module that contains it and add the latest version of that module as a requirement to the current module.

A side effect of running any go command is to update go.mod if necessary. In this case, the go build wrote a new go.mod:

$ cat go.mod
module github.com/you/hello

go 1.12

require rsc.io/quote v1.5.2

Because the go.mod was written, the next go build will not resolve the import again or print nearly as much:

$ go build
$ ./hello
Hello, world.

Even if rsc.io/quote v1.5.3 or v1.6.0 is released tomorrow, builds in this directory will keep using v1.5.2 until an explicit upgrade (see below).

The go.mod file lists a minimal set of requirements, omitting those implied by the ones already listed. In this case, rsc.io/quote v1.5.2 requires the specific versions of rsc.io/sampler and golang.org/x/text that were reported, so it would be redundant to repeat those in the go.mod file.

It is still possible to find out the full set of modules required by a build, using go list -m:

$ go list -m
github.com/you/hello

At this point you might wonder why our simple “hello world” program uses golang.org/x/text. It turns out that rsc.io/quote depends on rsc.io/sampler, which in turn uses golang.org/x/text for language matching.

$ LANG=fr ./hello
Bonjour le monde.

Upgrading

We've seen that when a new module must be added to a build to resolve a new import, go takes the latest one. Earlier, it needed rsc.io/quote and found that v1.5.2 was the latest. But except when resolving new imports, go uses only versions listed in go.mod files. In our example, rsc.io/quote depended indirectly on specific versions of golang.org/x/text and rsc.io/sampler. It turns out that both of those packages have newer releases, as we can see by adding -u (check for updated packages) to the go list command:

$ go list -m -u
github.com/you/hello

Both of those packages have newer releases, so we might want to upgrade them in our hello program.

Let's upgrade golang.org/x/text first:

$ go get golang.org/x/text
go: finding golang.org/x/text v0.3.0
go: downloading golang.org/x/text v0.3.0
go: extracting golang.org/x/text v0.3.0
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
)

The go get command looks up the latest version of the given modules and adds that version as a requirement for the current module, by updating go.mod. Now future builds will use the newer text module:

$ go list -m
github.com/you/hello

Of course, after an upgrade, it's a good idea to test that everything still works. Our dependencies rsc.io/quote and rsc.io/sampler have not been tested with the newer text module. We can run their tests in the configuration we've created:

$ go test github.com/you/hello rsc.io/quote
?   	github.com/you/hello	[no test files]
ok  	rsc.io/quote	0.003s

In the original go command, the package pattern all meant all packages found in GOPATH. That's almost always too many to be useful. With Go mouldes, we've narrowed the meaning of all to be “all packages in the current module, and the packages they import, recursively.” Version 1.5.2 of the rsc.io/quote module contains a buggy package:

$ go test rsc.io/quote/...
ok  	rsc.io/quote	(cached)
--- FAIL: Test (0.00s)
    buggy_test.go:10: buggy!
FAIL
FAIL	rsc.io/quote/buggy	0.002s

Until something in our module imports buggy, however, it's irrelevant to us, so it's not included in all. In any event, the upgraded x/text seems to work. At this point we'd probably commit go.mod.

Another option is to upgrade all modules needed by the build, using go get -u:

$ go get -u
go: finding rsc.io/sampler v1.99.99
go: downloading rsc.io/sampler v1.99.99
go: extracting rsc.io/sampler v1.99.99
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
	rsc.io/sampler v1.99.99 // indirect
)

Here, go get -u has kept the upgraded text module and also upgraded rsc.io/sampler to its latest version, v1.99.99.

Let's run our tests:

$ go test github.com/you/hello rsc.io/quote
?   	github.com/you/hello	[no test files]
--- FAIL: TestHello (0.00s)
    quote_test.go:19: Hello() = "99 bottles of beer on the wall, 99 bottles of beer, ...", want "Hello, world."
FAIL
FAIL	rsc.io/quote	0.002s

It appears that something is wrong with rsc.io/sampler v1.99.99. Sure enough:

$ go build
$ ./hello
99 bottles of beer on the wall, 99 bottles of beer, ...

The go get -u behavior of taking the latest of every dependency is exactly what go get does when packages being downloaded aren't in GOPATH.

The important difference is that go modules do not behave this way by default. Also you can undo it by downgrading.

Downgrading

To downgrade a package, use go list -t to show the available tagged versions:

$ go list -m -versions rsc.io/sampler
rsc.io/sampler v1.0.0 v1.2.0 v1.2.1 v1.3.0 v1.3.1 v1.99.99

Then use go get to ask for a specific version, like maybe v1.3.1:

$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
	rsc.io/sampler v1.99.99 // indirect
)
$ go get rsc.io/sampler@v1.3.1
go: finding rsc.io/sampler v1.3.1
go: downloading rsc.io/sampler v1.3.1
go: extracting rsc.io/sampler v1.3.1
$ go list -m
github.com/you/hello
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
	rsc.io/sampler v1.3.1 // indirect
)
$ go test github.com/you/hello rsc.io/quote
?   	github.com/you/hello	[no test files]
ok  	rsc.io/quote	0.003s

Downgrading one package may require downgrading others. For example:

$ go get rsc.io/sampler@v1.2.0
go: finding rsc.io/sampler v1.2.0
go: finding rsc.io/quote v1.5.1
go: finding rsc.io/quote v1.5.0
go: finding rsc.io/quote v1.4.0
go: finding rsc.io/sampler v1.0.0
go: downloading rsc.io/sampler v1.2.0
go: extracting rsc.io/sampler v1.2.0
$ go list -m
github.com/you/hello
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.4.0
	rsc.io/sampler v1.2.0 // indirect
)

In this case, rsc.io/quote v1.5.0 was the first to require rsc.io/sampler v1.3.0; earlier versions only needed v1.0.0 (or later). The downgrade selected rsc.io/quote v1.4.0, the last version compatible with v1.2.0.

It is also possible to remove a dependency entirely, an extreme form of downgrade, by specifying none as the version.

$ go mod edit -droprequire rsc.io/sampler
$ go list -m
github.com/you/hello
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.4.0
)
$ go test github.com/you/hello rsc.io/quote
go: downloading rsc.io/quote v1.4.0
go: extracting rsc.io/quote v1.4.0
go: downloading rsc.io/sampler v1.0.0
go: extracting rsc.io/sampler v1.0.0
?   	github.com/you/hello	[no test files]
ok  	rsc.io/quote	0.002s

Let's go back to the state where everything is the latest version, including rsc.io/sampler v1.99.99:

$ go get -u
$ go list -m
github.com/you/hello

Excluding

Having identified that v1.99.99 isn't okay to use in our hello world program, we may want to record that fact, to avoid future problems. We can do that by adding an exclude directive to go.mod. Future operations behave as if that module does not exist:

$ go mod edit -exclude=rsc.io/sampler@v1.99.99
$ echo "** TODO: REMOVE THIS HACK; see https://github.com/golang/go/issues/26454 **"
** TODO: REMOVE THIS HACK; see https://github.com/golang/go/issues/26454 **
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
	rsc.io/sampler v1.99.99 // indirect
)

exclude rsc.io/sampler v1.99.99
$ go mod edit -require rsc.io/sampler@v1.3.1
$ go list -m -versions rsc.io/sampler
rsc.io/sampler v1.0.0 v1.2.0 v1.2.1 v1.3.0 v1.3.1 v1.99.99
$ go list -m
github.com/you/hello
$ cat go.mod
module github.com/you/hello

go 1.12

require (
	golang.org/x/text v0.3.0 // indirect
	rsc.io/quote v1.5.2
	rsc.io/sampler v1.3.1 // indirect
)

exclude rsc.io/sampler v1.99.99
$ go test github.com/you/hello rsc.io/quote
?   	github.com/you/hello	[no test files]
ok  	rsc.io/quote	(cached)
$ quoteVersion=$(go list -m -f "{{.Version}}" rsc.io/quote)

Exclusions only apply to builds of the current module. If the current module were required by a larger build, the exclusions would not apply. For example, an exclusion in rsc.io/quote's go.mod will not apply to our “hello, world” build. This policy balances giving the authors of the current module almost arbitrary control over their own build, without also subjecting them to almost arbitrary control exerted by the modules they depend on.

At this point, the right next step is to contact the author of rsc.io/sampler and report the problem in v1.99.99, so it can be fixed in v1.99.100. Unfortunately, the author has a blog post that depends on not fixing the bug.

Replacing

If you do identify a problem in a dependency, you need a way to replace it with a fixed copy temporarily. Suppose we want to change something about the behavior of rsc.io/quote. Perhaps we want to work around the problem in rsc.io/sampler, or perhaps we want to do something else. The first step is to check out the quote module, using an ordinary git command:

$ git clone -q https://github.com/rsc/quote ../quote

Then edit ../quote/quote.go to change something about func Hello. For example, I'm going to change its return value from sampler.Hello() to sampler.Glass(), a more interesting greeting.

$ cd ../quote
$ git checkout -q -b my_quote $quoteVersion
$ echo "<edit quote.go>"
<edit quote.go>

Having changed the fork, we can make our build use it in place of the real one by adding a replacement directive to go.mod:

replace "rsc.io/quote" v1.5.2 => "../quote"

Then we can build our program using it:

$ cd ../hello
$ go mod edit -replace=rsc.io/quote=../quote
$ go list -m
github.com/you/hello
$ go build
$ ./hello
I can eat glass and it doesn't hurt me.

You can also name a different module as a replacement. For example, you can fork github.com/rsc/quote and then push your change to your fork.

$ cd ../quote
$ git remote add go-modules-by-example https://github.com/go-modules-by-example/quote-fork
$ git commit -a -m 'my fork'
[my_quote 1ab8a1a] my fork
 1 file changed, 1 insertion(+), 1 deletion(-)
$ git push -q go-modules-by-example
$ git tag v0.0.0-myfork
$ git push -q go-modules-by-example v0.0.0-myfork

Then you can use that as the replacement:

$ cd ../hello
$ go mod edit -replace=rsc.io/quote=github.com/go-modules-by-example/quote-fork@v0.0.0-myfork
$ go list -m
go: finding github.com/go-modules-by-example/quote-fork v0.0.0-myfork
github.com/you/hello
$ go build
go: downloading github.com/go-modules-by-example/quote-fork v0.0.0-myfork
go: extracting github.com/go-modules-by-example/quote-fork v0.0.0-myfork
$ LANG=fr ./hello
Je peux manger du verre, ça ne me fait pas mal.

Backwards Compatibility

Even if you want to use vgo for your project, you probably don't want to require all your users to have vgo. Instead, you can create a vendor directory that allows go command users to produce nearly the same builds (building inside GOPATH, of course):

$ go mod vendor
$ mkdir -p $GOPATH/src/github.com/you
$ cp -a . $GOPATH/src/github.com/you/hello
$ go build -o vhello github.com/you/hello
$ LANG=es ./vhello
Puedo comer vidrio, no me hace daño.

I said the builds are “nearly the same,” because the import paths seen by the toolchain and recorded in the final binary are different. The vendored builds see vendor directories:

$ go tool nm hello | grep sampler.hello
  591d48 D rsc.io/sampler.hello
$ go tool nm vhello | grep sampler.hello
  591d48 D rsc.io/sampler.hello

Except for this difference, the builds should produce the same binaries. In order to provide for a graceful transition, vgo-based builds ignore vendor directories entirely, as will module-aware go command builds.

What's Next?

Please try vgo. Start tagging versions in your repositories. Create and check in go.mod files. File issues at golang.org/issue, and please include “x/vgo:” at the start of the title. More posts tomorrow. Thanks, and have fun!

Version details

go version go1.12 linux/amd64