@@ -24,8 +24,10 @@ import (
24
24
"net/http"
25
25
"net/url"
26
26
"strings"
27
+ "sync"
27
28
28
29
authchallenge "github.com/docker/distribution/registry/client/auth/challenge"
30
+
29
31
"github.com/google/go-containerregistry/internal/redact"
30
32
"github.com/google/go-containerregistry/pkg/authn"
31
33
"github.com/google/go-containerregistry/pkg/logs"
@@ -98,6 +100,7 @@ func fromChallenge(reg name.Registry, auth authn.Authenticator, t http.RoundTrip
98
100
}
99
101
100
102
type bearerTransport struct {
103
+ mx sync.RWMutex
101
104
// Wrapped by bearerTransport.
102
105
inner http.RoundTripper
103
106
// Basic credentials that we exchange for bearer tokens.
@@ -139,7 +142,10 @@ func (bt *bearerTransport) RoundTrip(in *http.Request) (*http.Response, error) {
139
142
// the registry with which we are interacting.
140
143
// In case of redirect http.Client can use an empty Host, check URL too.
141
144
if matchesHost (bt .registry .RegistryStr (), in , bt .scheme ) {
142
- hdr := fmt .Sprintf ("Bearer %s" , bt .bearer .RegistryToken )
145
+ bt .mx .RLock ()
146
+ localToken := bt .bearer .RegistryToken
147
+ bt .mx .RUnlock ()
148
+ hdr := fmt .Sprintf ("Bearer %s" , localToken )
143
149
in .Header .Set ("Authorization" , hdr )
144
150
}
145
151
return bt .inner .RoundTrip (in )
@@ -156,11 +162,12 @@ func (bt *bearerTransport) RoundTrip(in *http.Request) (*http.Response, error) {
156
162
res .Body .Close ()
157
163
158
164
newScopes := []string {}
165
+ bt .mx .Lock ()
166
+ got := stringSet (bt .scopes )
159
167
for _ , wac := range challenges {
160
168
// TODO(jonjohnsonjr): Should we also update "realm" or "service"?
161
169
if want , ok := wac .Parameters ["scope" ]; ok {
162
170
// Add any scopes that we don't already request.
163
- got := stringSet (bt .scopes )
164
171
if _ , ok := got [want ]; ! ok {
165
172
newScopes = append (newScopes , want )
166
173
}
@@ -172,6 +179,7 @@ func (bt *bearerTransport) RoundTrip(in *http.Request) (*http.Response, error) {
172
179
// otherwise the registry might just ignore it :/
173
180
newScopes = append (newScopes , bt .scopes ... )
174
181
bt .scopes = newScopes
182
+ bt .mx .Unlock ()
175
183
176
184
// TODO(jonjohnsonjr): Teach transport.Error about "error" and "error_description" from challenge.
177
185
@@ -196,7 +204,9 @@ func (bt *bearerTransport) refresh(ctx context.Context) error {
196
204
}
197
205
198
206
if auth .RegistryToken != "" {
207
+ bt .mx .Lock ()
199
208
bt .bearer .RegistryToken = auth .RegistryToken
209
+ bt .mx .Unlock ()
200
210
return nil
201
211
}
202
212
@@ -212,7 +222,9 @@ func (bt *bearerTransport) refresh(ctx context.Context) error {
212
222
213
223
// Find a token to turn into a Bearer authenticator
214
224
if response .Token != "" {
225
+ bt .mx .Lock ()
215
226
bt .bearer .RegistryToken = response .Token
227
+ bt .mx .Unlock ()
216
228
}
217
229
218
230
// If we obtained a refresh token from the oauth flow, use that for refresh() now.
@@ -306,7 +318,9 @@ func (bt *bearerTransport) refreshOauth(ctx context.Context) ([]byte, error) {
306
318
}
307
319
308
320
v := url.Values {}
321
+ bt .mx .RLock ()
309
322
v .Set ("scope" , strings .Join (bt .scopes , " " ))
323
+ bt .mx .RUnlock ()
310
324
if bt .service != "" {
311
325
v .Set ("service" , bt .service )
312
326
}
@@ -362,7 +376,9 @@ func (bt *bearerTransport) refreshBasic(ctx context.Context) ([]byte, error) {
362
376
client := http.Client {Transport : b }
363
377
364
378
v := u .Query ()
379
+ bt .mx .RLock ()
365
380
v ["scope" ] = bt .scopes
381
+ bt .mx .RUnlock ()
366
382
v .Set ("service" , bt .service )
367
383
u .RawQuery = v .Encode ()
368
384
0 commit comments