Understanding and using the vendor folder
With the release of Go 1.5, there is a new way the go tool can discover go
packages. This method is off by default and the surrounding tools, such as
goimports, do not understand that folder layout. However in Go 1.6 this method
will be on for everyone and other tools will have support for it as well. This
new package discovery method is the vendor folder.
Before we look into the solution and semantics of the vendor folder, let’s explore the problem that prompted it.
The prompting problem
Go programs are often comprised of packages from many different sources.
Each one of these sources are pulled in from the GOPATH or from the standard
library. However, only their project was subject their own source control.
Projects that cared about not breaking when their dependent packages changed
or went away did one of the following:
- Copied the dependent packages into the project source tree and rewrote imports that referenced it.
- Copied the dependent packages into the project source tree and modified
the GOPATHvariable to include a project specific sub-tree.
- Wrote the repository revision down in a file, then updated the existing
GOPATHpackages to be that revision.
Although different projects did slight variations of this, these were the major trends present in all.
There was one large problem and several smaller ones. The largest problem was that each of these were different. The individual problems included;
- Many people did not like modifying the import paths or being required to include dependent packages in their repository.
- Modifying GOPATHimplied using a barego buildwould not be sufficient. Wrappers for thegocommand emerged, each slightly different.
- Modifying packages in the normal GOPATHrequired each project to have a uniqueGOPATH.
A solution to these problems
With Go 1.5 a new method to discover go packages was released.
Nothing has been changed or added
to the go language or the go compilers. Packages must still reside in GOPATH.
However, if a package or a parent folder of a package contains folder named
vendor it will be searched for dependencies using the vendor folder as an
import path root. While vendor folders can be nested, in most cases it is
not advised or needed.
Any package in the vendor folder will be found before the standard library.
To enable this in Go 1.5 set the environment variable GO15VENDOREXPERIMENT=1.
In Go 1.6 this will be on by default without an environment variable.
An example
This simple package lives in $GOPATH/src/github.com/kardianos/spider.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | $ cat main.go package main import ( "bytes" "flag" "io" "io/ioutil" "log" "net/http" "net/url" "os" "path" "path/filepath" "strings" "sync" "time" "github.com/andybalholm/cascadia" "github.com/tdewolff/parse/css" "golang.org/x/net/html" ) ... | 
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 | $ tree
.
├── css_test.go
├── main.go
└── vendor
    ├── github.com
    │   ├── andybalholm
    │   │   └── cascadia
    │   │       ├── LICENSE
    │   │       ├── parser.go
    │   │       ├── README.md
    │   │       └── selector.go
    │   └── tdewolff
    │       ├── buffer
    │       │   ├── buffer.go
    │       │   ├── lexer.go
    │       │   ├── LICENSE.md
    │       │   ├── reader.go
    │       │   ├── README.md
    │       │   ├── shifter.go
    │       │   └── writer.go
    │       └── parse
    │           ├── common.go
    │           ├── css
    │           │   ├── hash.go
    │           │   ├── lex.go
    │           │   ├── parse.go
    │           │   ├── README.md
    │           │   └── util.go
    │           ├── LICENSE.md
    │           ├── README.md
    │           └── util.go
    ├── golang.org
    │   └── x
    │       └── net
    │           └── html
    │               ├── atom
    │               │   ├── atom.go
    │               │   ├── gen.go
    │               │   └── table.go
    │               ├── const.go
    │               ├── doc.go
    │               ├── doctype.go
    │               ├── entity.go
    │               ├── escape.go
    │               ├── foreign.go
    │               ├── node.go
    │               ├── parse.go
    │               ├── render.go
    │               └── token.go
    └── vendor.json | 
How to use the vendor folder
Advice on how to use the vendor folder is varied. Some will shutter at the thought of including dependencies in the project repository. Others hold it is unthinkable to not include the dependencies in the project repository. Some will just want to include a manifest or lock file and fetch the dependencies before building.
Regardless of what you choose to do with it, the vendor folder enables you to do more.
Two general guidelines:
- If you vendor a single package, you probably want to vendor all your dependencies.
- You probably want a flat vendor structure (you probably don’t want nested vendor folders).
Tools that use this.
There are many tools that use the vendor folder today. There are even more
tools that have support for the vendor folder in a feature branch, so the future
is bright for the vendor folder.
I am the author of govendor and
vendor-spec. The primary goal
for govendor was to prove and use a common vendor specification file.
The secondary goals were to make a tool that worked at the package level
and could provide quick insight into the status of vendor packages.
It also has always ensured the dependencies are “flattened” and only contains
a single vendor folder. Today
govendor list and govendor list -v are some of my favorite commands.
If you are coming from godep, you just need to run govendor migrate
to start using the vendor folder today.
Example govendor commands
Here are some examples of govendor commands on the github.com/kardianos/spider repo.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 | $ govendor list
v github.com/andybalholm/cascadia
v github.com/tdewolff/buffer
v github.com/tdewolff/parse
v github.com/tdewolff/parse/css
v golang.org/x/net/html
v golang.org/x/net/html/atom
p github.com/kardianos/spider
$ govendor list -no-status +v
github.com/andybalholm/cascadia
github.com/tdewolff/buffer
github.com/tdewolff/parse
github.com/tdewolff/parse/css
golang.org/x/net/html
golang.org/x/net/html/atom
$ govendor list -v
v github.com/andybalholm/cascadia ::/vendor/github.com/andybalholm/cascadia
  └── github.com/kardianos/spider
v github.com/tdewolff/buffer ::/vendor/github.com/tdewolff/buffer
  └── github.com/kardianos/spider/vendor/github.com/tdewolff/parse/css
v github.com/tdewolff/parse ::/vendor/github.com/tdewolff/parse
  └── github.com/kardianos/spider/vendor/github.com/tdewolff/parse/css
v github.com/tdewolff/parse/css ::/vendor/github.com/tdewolff/parse/css
  └── github.com/kardianos/spider
v golang.org/x/net/html ::/vendor/golang.org/x/net/html
  ├── github.com/kardianos/spider
  └── github.com/kardianos/spider/vendor/github.com/andybalholm/cascadia
v golang.org/x/net/html/atom ::/vendor/golang.org/x/net/html/atom
  └── github.com/kardianos/spider/vendor/golang.org/x/net/html
p github.com/kardianos/spider
$ govendor remove +v
$ govendor list
p github.com/kardianos/spider
e github.com/andybalholm/cascadia
e github.com/tdewolff/buffer
e github.com/tdewolff/parse
e github.com/tdewolff/parse/css
e golang.org/x/net/html
e golang.org/x/net/html/atom
$ tree
.
├── css_test.go
├── main.go
└── vendor
    └── vendor.json
$ govendor add +e
$ govendor list
v github.com/andybalholm/cascadia
v github.com/tdewolff/buffer
v github.com/tdewolff/parse
v github.com/tdewolff/parse/css
v golang.org/x/net/html
v golang.org/x/net/html/atom
p github.com/kardianos/spider
$ govendor update -n golang.org/x/net/html/...
Copy "/home/daniel/src/golang.org/x/net/html" -> "/home/daniel/src/github.com/kardianos/spider/vendor/golang.org/x/net/html"
	Ignore "escape_test.go"
	Ignore "node_test.go"
	Ignore "parse_test.go"
	Ignore "entity_test.go"
	Ignore "render_test.go"
	Ignore "token_test.go"
	Ignore "example_test.go"
Copy "/home/daniel/src/golang.org/x/net/html/atom" -> "/home/daniel/src/github.com/kardianos/spider/vendor/golang.org/x/net/html/atom"
	Ignore "atom_test.go"
	Ignore "table_test.go" | 
As a side note, the test files are ignored in this repo because the “vendor/vendor.json” file
has a field called "ignore": "test". Multiple tags can be ignored. I also often
ignore files with appengine and appenginevm build tags.
