page-title">Go Package Management: Go Modules, go install &amp; get Through the Versions</h1><p class="page-description"></p><table class="properties"><tbody><tr class="property-row property-row-created_by"><th><span class="icon property-icon"><svg aria-hidden="true" role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesCreatedBy"><path d="M8 15.126C11.8623 15.126 15.0615 11.9336 15.0615 8.06445C15.0615 4.20215 11.8623 1.00293 7.99316 1.00293C4.13086 1.00293 0.938477 4.20215 0.938477 8.06445C0.938477 11.9336 4.1377 15.126 8 15.126ZM8 10.4229C6.05176 10.4229 4.54785 11.1133 3.83008 11.9131C2.90039 10.9082 2.33301 9.55469 2.33301 8.06445C2.33301 4.91992 4.84863 2.39746 7.99316 2.39746C11.1377 2.39746 13.6738 4.91992 13.6738 8.06445C13.6738 9.55469 13.1064 10.9082 12.1699 11.9131C11.4521 11.1133 9.94824 10.4229 8 10.4229ZM8 9.30176C9.32617 9.30859 10.3516 8.18066 10.3516 6.71094C10.3516 5.33008 9.31934 4.18164 8 4.18164C6.6875 4.18164 5.6416 5.33008 5.64844 6.71094C5.65527 8.18066 6.68066 9.28809 8 9.30176Z"></path></svg></span>Created by</th><td><span class="user"><img src="Go%20Package%20Management%20Go%20Modules,%20go%20install%20&amp;%20get%201dd6cd51990d80e2ad3ed0d6755c8731/IMG_2295.jpg" class="icon user-icon"/>JiaLin Huang</span></td></tr><tr class="property-row property-row-last_edited_time"><th><span class="icon property-icon"><svg aria-hidden="true" role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesCreatedAt"><path d="M8 15.126C11.8623 15.126 15.0615 11.9336 15.0615 8.06445C15.0615 4.20215 11.8623 1.00293 7.99316 1.00293C4.13086 1.00293 0.938477 4.20215 0.938477 8.06445C0.938477 11.9336 4.1377 15.126 8 15.126ZM8 13.7383C4.85547 13.7383 2.33301 11.209 2.33301 8.06445C2.33301 4.91992 4.84863 2.39746 7.99316 2.39746C11.1377 2.39746 13.6738 4.91992 13.6738 8.06445C13.6738 11.209 11.1445 13.7383 8 13.7383ZM4.54102 8.91211H7.99316C8.30078 8.91211 8.54004 8.67285 8.54004 8.37207V3.8877C8.54004 3.58691 8.30078 3.34766 7.99316 3.34766C7.69238 3.34766 7.45312 3.58691 7.45312 3.8877V7.83203H4.54102C4.2334 7.83203 4.00098 8.06445 4.00098 8.37207C4.00098 8.67285 4.2334 8.91211 4.54102 8.91211Z"></path></svg></span>Last edited</th><td><time>@2025年4月26日 20:48</time></td></tr><tr class="property-row property-row-multi_select"><th><span class="icon property-icon"><svg aria-hidden="true" role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesMultipleSelect"><path d="M1.91602 4.83789C2.44238 4.83789 2.87305 4.40723 2.87305 3.87402C2.87305 3.34766 2.44238 2.91699 1.91602 2.91699C1.38281 2.91699 0.952148 3.34766 0.952148 3.87402C0.952148 4.40723 1.38281 4.83789 1.91602 4.83789ZM5.1084 4.52344H14.3984C14.7607 4.52344 15.0479 4.23633 15.0479 3.87402C15.0479 3.51172 14.7607 3.22461 14.3984 3.22461H5.1084C4.74609 3.22461 4.45898 3.51172 4.45898 3.87402C4.45898 4.23633 4.74609 4.52344 5.1084 4.52344ZM1.91602 9.03516C2.44238 9.03516 2.87305 8.60449 2.87305 8.07129C2.87305 7.54492 2.44238 7.11426 1.91602 7.11426C1.38281 7.11426 0.952148 7.54492 0.952148 8.07129C0.952148 8.60449 1.38281 9.03516 1.91602 9.03516ZM5.1084 8.7207H14.3984C14.7607 8.7207 15.0479 8.43359 15.0479 8.07129C15.0479 7.70898 14.7607 7.42188 14.3984 7.42188H5.1084C4.74609 7.42188 4.45898 7.70898 4.45898 8.07129C4.45898 8.43359 4.74609 8.7207 5.1084 8.7207ZM1.91602 13.2324C2.44238 13.2324 2.87305 12.8018 2.87305 12.2686C2.87305 11.7422 2.44238 11.3115 1.91602 11.3115C1.38281 11.3115 0.952148 11.7422 0.952148 12.2686C0.952148 12.8018 1.38281 13.2324 1.91602 13.2324ZM5.1084 12.918H14.3984C14.7607 12.918 15.0479 12.6309 15.0479 12.2686C15.0479 11.9062 14.7607 11.6191 14.3984 11.6191H5.1084C4.74609 11.6191 4.45898 11.9062 4.45898 12.2686C4.45898 12.6309 4.74609 12.918 5.1084 12.918Z"></path></svg></span>Tags</th><td><span class="selected-value select-value-color-purple">Post</span><span class="selected-value select-value-color-default">golang</span><span class="selected-value select-value-color-green">package management</span></td></tr></tbody></table></header><div class="page-body"><h1 class="">Why</h1><p class="">I used to mix <code>go get</code> and <code>go install</code>, not really understanding the differences, even when being suggested to use one over the other. Based on growing uncertainty and the fact that modules weren&#x27;t there from the beginning, I&#x27;m documenting this for clarity.</p><p class="">
</p><h1 class="">Go Module Era (Go 1.11~)</h1><p class="">We can call the time before 1.11 the &quot;GOPATH era&quot; since everything was placed there without version control.</p><p class="">The confusion comes from using JavaScript with npm, which always had version control. So when Go modules appeared, I wondered why this needed special explanation.</p><p class="">
</p><p class="">If you&#x27;ve only used Go after version 1.11, you probably aren&#x27;t confused because modules are the normal way to control package versions.</p><p class="">
</p><p class="">Module evolution timeline:</p><ol type="1" class="numbered-list" start="1"><li>1.11-12: Required manual setting of <code>GO111MODULE=on</code><ol type="a" class="numbered-list" start="1"><li><code>111</code> in <code>GO111MODULE</code> represents a variable supported after version 1.11</li></ol></li></ol><ol type="1" class="numbered-list" start="2"><li>During Go 1.13-1.15: the default is <code>GO111MODULE=auto</code></li></ol><ol type="1" class="numbered-list" start="3"><li>1.16+: Default setting changed to <code>GO111MODULE=on</code></li></ol><ol type="1" class="numbered-list" start="4"><li>Related commands<ol type="a" class="numbered-list" start="1"><li>go mod init</li></ol><ol type="a" class="numbered-list" start="2"><li>go mod vendor</li></ol><ol type="a" class="numbered-list" start="3"><li>go mod download</li></ol><p class="">
</p></li></ol><p class="">
</p><p class="">
</p><h1 class="">Go Install</h1><ol type="1" class="numbered-list" start="1"><li>Before 1.11:<ol type="a" class="numbered-list" start="1"><li>Installed binaries to <code>$GOPATH/bin</code></li></ol><ol type="a" class="numbered-list" start="2"><li>Also installed packages to <code>$GOPATH/pkg</code></li></ol></li></ol><ol type="1" class="numbered-list" start="2"><li>1.11-15 (transition period with modules):<ol type="a" class="numbered-list" start="1"><li>Still installed binaries to <code>$GOPATH/bin</code> or <code>$GOBIN</code></li></ol><ol type="a" class="numbered-list" start="2"><li>No longer used <code>$GOPATH/pkg</code> for packages - you had to run go install in your project for version dependencies</li></ol></li></ol><ol type="1" class="numbered-list" start="3"><li>1.16+: Supports version specification and became independent from project dependencies:<ol type="a" class="numbered-list" start="1"><li>Still installs binaries to <code>$GOPATH/bin</code> or <code>$GOBIN</code></li></ol><ol type="a" class="numbered-list" start="2"><li>No longer manages project package dependencies (not related to go.mod now)</li></ol><ol type="a" class="numbered-list" start="3"><li>Similar to <code>npm install --global xxx</code></li></ol></li></ol><p class="">
</p><p class="">
</p><h1 class="">Go Get</h1><ol type="1" class="numbered-list" start="1"><li>After 1.16, <code>go install</code> stopped managing packages and dependencies - that job went to <code>go get</code></li></ol><ol type="1" class="numbered-list" start="2"><li>1.11-1.15: Transition period where it downloaded packages and installed binaries</li></ol><ol type="1" class="numbered-list" start="3"><li>Starting 1.16: <mark class="highlight-red"><strong>Not recommended</strong></mark> for installing binaries</li></ol><ol type="1" class="numbered-list" start="4"><li>1.17: <strong>Officially not recommended</strong> for binary installation, shows warnings but doesn&#x27;t block it</li></ol><ol type="1" class="numbered-list" start="5"><li>1.18: The <code>-d</code> flag (<mark class="highlight-red"><strong>download only, no installation</strong></mark>) became the default behavior, completing the transition</li></ol><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Version Migration of Go Install and Go Get</h1><p class="">Originally, these commands had clearly defined separate purposes. Over time, features were added for convenience, which led to overlapping responsibilities and confused users. This issue was finally addressed starting in Go 1.16.</p><p class="">
</p><p class=""><mark class="highlight-red"><strong>Go 1.16</strong></mark> began separating the commands <mark class="highlight-red"><strong>conceptually</strong></mark>. <code>go install</code> could specify versions and didn&#x27;t affect go.mod files.</p><p class=""><a href="https://go.dev/doc/go1.16#modules">https://go.dev/doc/go1.16#modules</a></p><p class="">From Go docs 1.16:</p><blockquote class=""><code>go</code> <code>install</code> now accepts arguments with version suffixes (for example, <code>go installexample.com/cmd@v1.0.0</code>). This causes <code>goinstall</code> to build and install packages in module-aware mode, ignoring the <code>go.mod</code> file in the current directory or any parent directory, if there is one. This is useful for installing executables without affecting the dependencies of the main module.</blockquote><blockquote class=""><code>go</code> <code>install</code>, with or without a version suffix (as described above), is now the recommended way to build and install packages in module mode. <code>go</code> <code>get</code> should be used with the <code>-d</code> flag to adjust the current module’s dependencies without building packages, and use of <code>go</code> <code>get</code> to build and install packages is deprecated. In a future release, the <code>-d</code> flag will always be enabled.</blockquote><p class="">
</p><p class="">Go 1.17 <mark class="highlight-red"><strong>reinforced this separation but still allowed</strong></mark> <code>go get</code> for binary installation with warnings:</p><p class=""><a href="https://go.dev/doc/go1.17#module-deprecation-comments">https://go.dev/doc/go1.17#module-deprecation-comments</a></p><blockquote class=""><code>go</code> <code>get</code> prints a deprecation warning when installing commands outside the main module (without the <code>-d</code> flag). <code>go install cmd@version</code> <mark class="highlight-red"><strong>should be used</strong></mark> instead to install a command at a specific version, using a suffix like <code>@latest</code> or <code>@v1.2.3</code>. <br/>In Go 1.18, the <br/><code>-d</code> flag will always be enabled, and <code>go</code> <code>get</code> will only be used to change dependencies in <code>go.mod</code>.</blockquote><p class="">
</p><p class="">Go 1.18 completed the separation - <code>go get</code> no longer supports binary installation and uses <code>-d</code> by default.</p><p class=""><a href="https://go.dev/doc/go1.18#go-command">https://go.dev/doc/go1.18#go-command</a></p><p class="">
</p><p class="">Official summary:</p><p class=""><a href="https://go.dev/ref/mod#go-get">https://go.dev/ref/mod#go-get</a></p><blockquote class="">Since Go 1.16, <code><a href="https://go.dev/ref/mod#go-install">go install</a></code> is the recommended command for building and installing programs. When used with a version suffix (like <code>@latest</code> or <code>@v1.4.6</code>), <code>go install</code> builds packages in module-aware mode, ignoring the <code>go.mod</code> file in the current directory or any parent directory, if there is one.</blockquote><blockquote class=""><code>go get</code> is more focused on managing requirements in <code>go.mod</code>. The <code>-d</code> flag is deprecated, and in Go 1.18, it will always be enabled.</blockquote><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Evolution of $GOPATH/pkg, bin, src Folders</h1><p class="">The src, pkg, bin folders were core components in the GOPATH era:</p><ol type="1" class="numbered-list" start="1"><li><code>src/</code>: Where you put cloned projects (yours or third-party)<ul class="bulleted-list"><li style="list-style-type:disc">Then: Seemed intrusive but was necessary to resolve import paths</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Now: Less important with modules; <mark class="highlight-red"><strong>I don&#x27;t even have this folder anymore</strong></mark></li></ul></li></ol><ol type="1" class="numbered-list" start="2"><li><code>pkg/</code> <ul class="bulleted-list"><li style="list-style-type:disc">Then: Stored compiled artifacts (<code>*.a</code>)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Now: Changed to <code>pkg/mod/</code> as a cache (<strong>note that it&#x27;s read-only</strong>) for multiple projects; like a repository that can be copied to a project using <code>go mod vendor</code></li></ul></li></ol><ol type="1" class="numbered-list" start="3"><li><code>bin/</code>: Hasn&#x27;t changed - always stored executables/binaries</li></ol><p class=""><strong>GOPATH role change</strong>: From core dependency path to <mark class="highlight-red"><strong>mainly storing cache and binary files</strong></mark></p><p class="">
</p><h1 class="">For Node/npm Users</h1><ol type="1" class="numbered-list" start="1"><li>Think of <code>go install</code> as <code>npm i --global</code></li></ol><ol type="1" class="numbered-list" start="2"><li>Think of <code>go get</code> as <code>npm i --save</code> (updates go.mod/go.sum like package.json/package-lock.json)<ol type="a" class="numbered-list" start="1"><li>npm local installs to <code>node_modules/</code>, similar to <code>go mod vendor</code></li></ol><ol type="a" class="numbered-list" start="2"><li><code>go get</code> installs to <code>$GOPATH/pkg/mod/..</code></li></ol><ol type="a" class="numbered-list" start="3"><li><code>go mod vendor</code> first downloads to <code>$GOPATH/pkg/mod/</code>, then copies to project&#x27;s <code>/vendor/..</code></li></ol></li></ol><p class="">
</p><p class="">
</p><h1 class="">How to Run a Cloned Project?</h1><p class="">When cloning someone&#x27;s project, you usually don&#x27;t need to run <code>go mod tidy</code> if they followed best practices (running <code>go mod tidy</code> before commits).</p><p class=""><strong>To be safe, you can run </strong><code><strong>go mod tidy</strong></code><strong> yourself.</strong></p><p class="">Based on trust, you can directly:</p><ol type="1" class="numbered-list" start="1"><li><code>go build</code>: Downloads and compiles <p class="">if you need an executable For CRUD servers, <code>go run main.go</code> is more common</p></li></ol><ol type="1" class="numbered-list" start="2"><li><code>go mod download</code>: Downloads without compiling <p class="">Useful for debugging in Docker by separating download from build</p><p class="">
</p></li></ol><ol type="1" class="numbered-list" start="3"><li><code>go mod vendor</code>: Needed when: <ol type="a" class="numbered-list" start="1"><li>There&#x27;s a <code>vendor/</code> directory in version control</li></ol><ol type="a" class="numbered-list" start="2"><li>You want npm-like local package control <p class="">Vendor vs <code>node_modules</code>: vendor should be in version control; when cloning, Go still caches in $GOPATH/pkg/mod then copies to /vendor</p><p class="">
</p></li></ol><ol type="a" class="numbered-list" start="3"><li>You need to work offline (must go with <code>go build -mod=vendor</code>)</li></ol><ol type="a" class="numbered-list" start="4"><li>You need to modify dependency code (must go with <code>go build -mod=vendor</code>)</li></ol><ol type="a" class="numbered-list" start="5"><li>In air-gapped env or enterprise, vendor mode remains an important dependency management approach. In vendor mode, Go doesn&#x27;t download dependencies from the network but uses code from the vendor directory</li></ol><p class=""><code>go mod vendor</code> is an active choice, useful for customizing packages</p></li></ol><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Go Install vs Go Build</h1><p class="">The main difference: <code>go install</code> puts executables in <code>$GOPATH/bin</code> or <code>$GOBIN</code> directory.</p><p class=""><code><strong>go build</strong></code><strong> just builds in the current directory.</strong></p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Global Cache vs Project-Local Storage</h1><p class="">When you use <code>go get</code> to download a package used in several different projects, the file structure looks like this:</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-Bash">$GOPATH/pkg/mod/
└── github.com/
    └── gin-gonic/
        └── gin@v1.6.3/          # a version
            ├── LICENSE
            ├── gin.go
            ├── ...else files
        └── gin@v1.7.7/          # b
            ├── LICENSE
            ├── gin.go
            ├── ...else files
        └── gin@v1.9.1/          # c
            ├── LICENSE
            ├── gin.go
            ├── ...else files</code></pre><p class="">go mod vendor: has only one version:</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-Bash">Your-project/
├── go.mod
├── go.sum
├── vendor/
│   ├── github.com/
│   │   └── gin-gonic/
│   │       └── gin/          # only one version (specified in go.mod)
│   │           ├── LICENSE
│   │           ├── gin.go
│   │           └── ...else
│   └── modules.txt           # records all vendored modules</code></pre><p class="">
</p><p class="">
</p><h1 class="">References</h1><p class=""><a href="https://go.dev/doc/go1.16#modules">https://go.dev/doc/go1.16#modules</a></p><p class=""><a href="https://go.dev/doc/go1.17#module-deprecation-comments">https://go.dev/doc/go1.17#module-deprecation-comments</a></p><p class=""><a href="https://go.dev/doc/go1.18#go-command">https://go.dev/doc/go1.18#go-command</a></p><p class="">
</p></div></article><span class="sans" style="font-size:14px;padding-top:2em"></span></body>
~/
about
posts
frontbacknetworkoscloud
readings
css
bookmarks
archives
© 2024 jialin00.com Original content since 2022
And maybe its just slow involvement at first, but try to sort of creep your career in that direction, because if youre not being challenged, if youre not a little bit scared all the time, just a little bit, then youre not gonna improve. - The Myth of the Genius Programmer