1
2
3
4
5
6 package auth
7
8 import (
9 "cmd/go/internal/base"
10 "cmd/go/internal/cfg"
11 "fmt"
12 "log"
13 "net/http"
14 "os"
15 "path/filepath"
16 "slices"
17 "strings"
18 "sync"
19 )
20
21 var (
22 credentialCache sync.Map
23 authOnce sync.Once
24 )
25
26
27
28
29
30
31 func AddCredentials(client *http.Client, req *http.Request, res *http.Response, url string) bool {
32 if req.URL.Scheme != "https" {
33 panic("GOAUTH called without https")
34 }
35 if cfg.GOAUTH == "off" {
36 return false
37 }
38
39 authOnce.Do(func() {
40 runGoAuth(client, res, "")
41 })
42 if url != "" {
43
44 runGoAuth(client, res, url)
45 }
46 return loadCredential(req, req.URL.String())
47 }
48
49
50
51
52 func runGoAuth(client *http.Client, res *http.Response, url string) {
53 var cmdErrs []error
54 goAuthCmds := strings.Split(cfg.GOAUTH, ";")
55
56
57 slices.Reverse(goAuthCmds)
58 for _, command := range goAuthCmds {
59 command = strings.TrimSpace(command)
60 words := strings.Fields(command)
61 if len(words) == 0 {
62 base.Fatalf("go: GOAUTH encountered an empty command (GOAUTH=%s)", cfg.GOAUTH)
63 }
64 switch words[0] {
65 case "off":
66 if len(goAuthCmds) != 1 {
67 base.Fatalf("go: GOAUTH=off cannot be combined with other authentication commands (GOAUTH=%s)", cfg.GOAUTH)
68 }
69 return
70 case "netrc":
71 lines, err := readNetrc()
72 if err != nil {
73 cmdErrs = append(cmdErrs, fmt.Errorf("GOAUTH=%s: %v", command, err))
74 continue
75 }
76
77
78
79
80 for i := len(lines) - 1; i >= 0; i-- {
81 l := lines[i]
82 r := http.Request{Header: make(http.Header)}
83 r.SetBasicAuth(l.login, l.password)
84 storeCredential(l.machine, r.Header)
85 }
86 case "git":
87 if len(words) != 2 {
88 base.Fatalf("go: GOAUTH=git dir method requires an absolute path to the git working directory")
89 }
90 dir := words[1]
91 if !filepath.IsAbs(dir) {
92 base.Fatalf("go: GOAUTH=git dir method requires an absolute path to the git working directory, dir is not absolute")
93 }
94 fs, err := os.Stat(dir)
95 if err != nil {
96 base.Fatalf("go: GOAUTH=git encountered an error; cannot stat %s: %v", dir, err)
97 }
98 if !fs.IsDir() {
99 base.Fatalf("go: GOAUTH=git dir method requires an absolute path to the git working directory, dir is not a directory")
100 }
101
102 if url == "" {
103
104
105 continue
106 }
107 prefix, header, err := runGitAuth(client, dir, url)
108 if err != nil {
109
110
111 cmdErrs = append(cmdErrs, fmt.Errorf("GOAUTH=%s: %v", command, err))
112 } else {
113 storeCredential(prefix, header)
114 }
115 default:
116 credentials, err := runAuthCommand(command, url, res)
117 if err != nil {
118
119
120 cmdErrs = append(cmdErrs, fmt.Errorf("GOAUTH=%s: %v", command, err))
121 continue
122 }
123 for prefix := range credentials {
124 storeCredential(prefix, credentials[prefix])
125 }
126 }
127 }
128
129
130 if cfg.BuildX && url != "" {
131 req := &http.Request{Header: make(http.Header)}
132 if ok := loadCredential(req, url); !ok && len(cmdErrs) > 0 {
133 log.Printf("GOAUTH encountered errors for %s:", url)
134 for _, err := range cmdErrs {
135 log.Printf(" %v", err)
136 }
137 }
138 }
139 }
140
141
142
143 func loadCredential(req *http.Request, url string) bool {
144 currentPrefix := strings.TrimPrefix(url, "https://")
145
146 for {
147 headers, ok := credentialCache.Load(currentPrefix)
148 if !ok {
149 currentPrefix, _, ok = strings.Cut(currentPrefix, "/")
150 if !ok {
151 return false
152 }
153 continue
154 }
155 for key, values := range headers.(http.Header) {
156 for _, value := range values {
157 req.Header.Add(key, value)
158 }
159 }
160 return true
161 }
162 }
163
164
165
166 func storeCredential(prefix string, header http.Header) {
167
168 prefix = strings.TrimPrefix(prefix, "https://")
169 if len(header) == 0 {
170 credentialCache.Delete(prefix)
171 } else {
172 credentialCache.Store(prefix, header)
173 }
174 }
175
View as plain text