Skip to content

Commit 1e168f0

Browse files
committed
docker: mimic docker upstream registry authentication
Signed-off-by: Antonio Murdaca <[email protected]>
1 parent b5c6de3 commit 1e168f0

File tree

4 files changed

+115
-100
lines changed

4 files changed

+115
-100
lines changed

docker/docker_client.go

Lines changed: 42 additions & 88 deletions
Original file line numberDiff line numberDiff line change
@@ -45,14 +45,14 @@ var ErrV1NotSupported = errors.New("can't talk to a V1 docker registry")
4545

4646
// dockerClient is configuration for dealing with a single Docker registry.
4747
type dockerClient struct {
48-
ctx *types.SystemContext
49-
registry string
50-
username string
51-
password string
52-
wwwAuthenticate string // Cache of a value set by ping() if scheme is not empty
53-
scheme string // Cache of a value returned by a successful ping() if not empty
54-
client *http.Client
55-
signatureBase signatureStorageBase
48+
ctx *types.SystemContext
49+
registry string
50+
username string
51+
password string
52+
scheme string // Cache of a value returned by a successful ping() if not empty
53+
client *http.Client
54+
signatureBase signatureStorageBase
55+
challenges []challenge
5656
}
5757

5858
// this is cloned from docker/go-connections because upstream docker has changed
@@ -187,45 +187,55 @@ func newDockerClient(ctx *types.SystemContext, ref dockerReference, write bool)
187187
}, nil
188188
}
189189

190+
type authScope struct {
191+
remoteName string
192+
actions string
193+
}
194+
195+
type requestOptions struct {
196+
scope authScope
197+
headers map[string][]string
198+
stream io.Reader
199+
}
200+
190201
// makeRequest creates and executes a http.Request with the specified parameters, adding authentication and TLS options for the Docker client.
191202
// url is NOT an absolute URL, but a path relative to the /v2/ top-level API path. The host name and schema is taken from the client or autodetected.
192-
func (c *dockerClient) makeRequest(method, url string, headers map[string][]string, stream io.Reader) (*http.Response, error) {
203+
func (c *dockerClient) makeRequest(method, url string, opts requestOptions) (*http.Response, error) {
193204
if c.scheme == "" {
194205
pr, err := c.ping()
195206
if err != nil {
196207
return nil, err
197208
}
198-
c.wwwAuthenticate = pr.WWWAuthenticate
199209
c.scheme = pr.scheme
200210
}
201211

202212
url = fmt.Sprintf(baseURL, c.scheme, c.registry) + url
203-
return c.makeRequestToResolvedURL(method, url, headers, stream, -1, true)
213+
return c.makeRequestToResolvedURL(method, url, opts, -1, true)
204214
}
205215

206216
// makeRequestToResolvedURL creates and executes a http.Request with the specified parameters, adding authentication and TLS options for the Docker client.
207217
// streamLen, if not -1, specifies the length of the data expected on stream.
208218
// makeRequest should generally be preferred.
209219
// TODO(runcom): too many arguments here, use a struct
210-
func (c *dockerClient) makeRequestToResolvedURL(method, url string, headers map[string][]string, stream io.Reader, streamLen int64, sendAuth bool) (*http.Response, error) {
211-
req, err := http.NewRequest(method, url, stream)
220+
func (c *dockerClient) makeRequestToResolvedURL(method, url string, opts requestOptions, streamLen int64, sendAuth bool) (*http.Response, error) {
221+
req, err := http.NewRequest(method, url, opts.stream)
212222
if err != nil {
213223
return nil, err
214224
}
215225
if streamLen != -1 { // Do not blindly overwrite if streamLen == -1, http.NewRequest above can figure out the length of bytes.Reader and similar objects without us having to compute it.
216226
req.ContentLength = streamLen
217227
}
218228
req.Header.Set("Docker-Distribution-API-Version", "registry/2.0")
219-
for n, h := range headers {
229+
for n, h := range opts.headers {
220230
for _, hh := range h {
221231
req.Header.Add(n, hh)
222232
}
223233
}
224234
if c.ctx != nil && c.ctx.DockerRegistryUserAgent != "" {
225235
req.Header.Add("User-Agent", c.ctx.DockerRegistryUserAgent)
226236
}
227-
if c.wwwAuthenticate != "" && sendAuth {
228-
if err := c.setupRequestAuth(req); err != nil {
237+
if sendAuth {
238+
if err := c.setupRequestAuth(req, opts.scope.remoteName, opts.scope.actions); err != nil {
229239
return nil, err
230240
}
231241
}
@@ -237,87 +247,31 @@ func (c *dockerClient) makeRequestToResolvedURL(method, url string, headers map[
237247
return res, nil
238248
}
239249

240-
func (c *dockerClient) setupRequestAuth(req *http.Request) error {
241-
tokens := strings.SplitN(strings.TrimSpace(c.wwwAuthenticate), " ", 2)
242-
if len(tokens) != 2 {
243-
return errors.Errorf("expected 2 tokens in WWW-Authenticate: %d, %s", len(tokens), c.wwwAuthenticate)
250+
func (c *dockerClient) setupRequestAuth(req *http.Request, remoteName, actions string) error {
251+
if len(c.challenges) == 0 {
252+
return nil
244253
}
245-
switch tokens[0] {
246-
case "Basic":
254+
// assume just one...
255+
challenge := c.challenges[0]
256+
switch challenge.Scheme {
257+
case "basic":
247258
req.SetBasicAuth(c.username, c.password)
248259
return nil
249-
case "Bearer":
250-
// FIXME? This gets a new token for every API request;
251-
// we may be easily able to reuse a previous token, e.g.
252-
// for OpenShift the token only identifies the user and does not vary
253-
// across operations. Should we just try the request first, and
254-
// only get a new token on failure?
255-
// OTOH what to do with the single-use body stream in that case?
256-
257-
// Try performing the request, expecting it to fail.
258-
testReq := *req
259-
// Do not use the body stream, or we couldn't reuse it for the "real" call later.
260-
testReq.Body = nil
261-
testReq.ContentLength = 0
262-
res, err := c.client.Do(&testReq)
263-
if err != nil {
264-
return err
265-
}
266-
chs := parseAuthHeader(res.Header)
267-
// We could end up in this "if" statement if the /v2/ call (during ping)
268-
// returned 401 with a valid WWW-Authenticate=Bearer header.
269-
// That doesn't **always** mean, however, that the specific API request
270-
// (different from /v2/) actually needs to be authorized.
271-
// One example of this _weird_ scenario happens with GCR.io docker
272-
// registries.
273-
if res.StatusCode != http.StatusUnauthorized || chs == nil || len(chs) == 0 {
274-
// With gcr.io, the /v2/ call returns a 401 with a valid WWW-Authenticate=Bearer
275-
// header but the repository could be _public_ (no authorization is needed).
276-
// Hence, the registry response contains no challenges and the status
277-
// code is not 401.
278-
// We just skip this case as it's not standard on docker/distribution
279-
// registries (https://github.com/docker/distribution/blob/master/docs/spec/api.md#api-version-check)
280-
if res.StatusCode != http.StatusUnauthorized {
281-
return nil
282-
}
283-
// gcr.io private repositories pull instead requires us to send user:pass pair in
284-
// order to retrieve a token and setup the correct Bearer token.
285-
// try again one last time with Basic Auth
286-
testReq2 := *req
287-
// Do not use the body stream, or we couldn't reuse it for the "real" call later.
288-
testReq2.Body = nil
289-
testReq2.ContentLength = 0
290-
testReq2.SetBasicAuth(c.username, c.password)
291-
res, err := c.client.Do(&testReq2)
292-
if err != nil {
293-
return err
294-
}
295-
chs = parseAuthHeader(res.Header)
296-
if res.StatusCode != http.StatusUnauthorized || chs == nil || len(chs) == 0 {
297-
// no need for bearer? wtf?
298-
return nil
299-
}
300-
}
301-
// Arbitrarily use the first challenge, there is no reason to expect more than one.
302-
challenge := chs[0]
303-
if challenge.Scheme != "bearer" { // Another artifact of trying to handle WWW-Authenticate before it actually happens.
304-
return errors.Errorf("Unimplemented: WWW-Authenticate Bearer replaced by %#v", challenge.Scheme)
305-
}
260+
case "bearer":
306261
realm, ok := challenge.Parameters["realm"]
307262
if !ok {
308263
return errors.Errorf("missing realm in bearer auth challenge")
309264
}
310265
service, _ := challenge.Parameters["service"] // Will be "" if not present
311-
scope, _ := challenge.Parameters["scope"] // Will be "" if not present
266+
scope := fmt.Sprintf("repository:%s:%s", remoteName, actions)
312267
token, err := c.getBearerToken(realm, service, scope)
313268
if err != nil {
314269
return err
315270
}
316271
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", token))
317272
return nil
318273
}
319-
return errors.Errorf("no handler for %s authentication", tokens[0])
320-
// support docker bearer with authconfig's Auth string? see docker2aci
274+
return errors.Errorf("no handler for %s authentication", challenge.Scheme)
321275
}
322276

323277
func (c *dockerClient) getBearerToken(realm, service, scope string) (string, error) {
@@ -428,15 +382,14 @@ func getAuth(ctx *types.SystemContext, registry string) (string, string, error)
428382
}
429383

430384
type pingResponse struct {
431-
WWWAuthenticate string
432-
APIVersion string
433-
scheme string
385+
APIVersion string
386+
scheme string
434387
}
435388

436389
func (c *dockerClient) ping() (*pingResponse, error) {
437390
ping := func(scheme string) (*pingResponse, error) {
438391
url := fmt.Sprintf(baseURL, scheme, c.registry)
439-
resp, err := c.makeRequestToResolvedURL("GET", url, nil, nil, -1, true)
392+
resp, err := c.makeRequestToResolvedURL("GET", url, requestOptions{}, -1, true)
440393
logrus.Debugf("Ping %s err %#v", url, err)
441394
if err != nil {
442395
return nil, err
@@ -446,8 +399,9 @@ func (c *dockerClient) ping() (*pingResponse, error) {
446399
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusUnauthorized {
447400
return nil, errors.Errorf("error pinging repository, response code %d", resp.StatusCode)
448401
}
402+
c.challenges = parseAuthHeader(resp.Header)
403+
449404
pr := &pingResponse{}
450-
pr.WWWAuthenticate = resp.Header.Get("WWW-Authenticate")
451405
pr.APIVersion = resp.Header.Get("Docker-Distribution-Api-Version")
452406
pr.scheme = scheme
453407
return pr, nil
@@ -464,7 +418,7 @@ func (c *dockerClient) ping() (*pingResponse, error) {
464418
// best effort to understand if we're talking to a V1 registry
465419
pingV1 := func(scheme string) bool {
466420
url := fmt.Sprintf(baseURLV1, scheme, c.registry)
467-
resp, err := c.makeRequestToResolvedURL("GET", url, nil, nil, -1, true)
421+
resp, err := c.makeRequestToResolvedURL("GET", url, requestOptions{}, -1, true)
468422
logrus.Debugf("Ping %s err %#v", url, err)
469423
if err != nil {
470424
return false

docker/docker_image.go

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,13 @@ func (i *Image) SourceRefFullName() string {
4040
// GetRepositoryTags list all tags available in the repository. Note that this has no connection with the tag(s) used for this specific image, if any.
4141
func (i *Image) GetRepositoryTags() ([]string, error) {
4242
url := fmt.Sprintf(tagsURL, i.src.ref.ref.RemoteName())
43-
res, err := i.src.c.makeRequest("GET", url, nil, nil)
43+
opts := requestOptions{
44+
scope: authScope{
45+
remoteName: i.src.ref.ref.RemoteName(),
46+
actions: "pull",
47+
},
48+
}
49+
res, err := i.src.c.makeRequest("GET", url, opts)
4450
if err != nil {
4551
return nil, err
4652
}

docker/docker_image_dest.go

Lines changed: 35 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,13 @@ func (d *dockerImageDestination) PutBlob(stream io.Reader, inputInfo types.BlobI
9191
checkURL := fmt.Sprintf(blobsURL, d.ref.ref.RemoteName(), inputInfo.Digest.String())
9292

9393
logrus.Debugf("Checking %s", checkURL)
94-
res, err := d.c.makeRequest("HEAD", checkURL, nil, nil)
94+
opts := requestOptions{
95+
scope: authScope{
96+
remoteName: d.ref.ref.RemoteName(),
97+
actions: "push",
98+
},
99+
}
100+
res, err := d.c.makeRequest("HEAD", checkURL, opts)
95101
if err != nil {
96102
return types.BlobInfo{}, err
97103
}
@@ -114,7 +120,13 @@ func (d *dockerImageDestination) PutBlob(stream io.Reader, inputInfo types.BlobI
114120
// FIXME? Chunked upload, progress reporting, etc.
115121
uploadURL := fmt.Sprintf(blobUploadURL, d.ref.ref.RemoteName())
116122
logrus.Debugf("Uploading %s", uploadURL)
117-
res, err := d.c.makeRequest("POST", uploadURL, nil, nil)
123+
opts := requestOptions{
124+
scope: authScope{
125+
remoteName: d.ref.ref.RemoteName(),
126+
actions: "push",
127+
},
128+
}
129+
res, err := d.c.makeRequest("POST", uploadURL, opts)
118130
if err != nil {
119131
return types.BlobInfo{}, err
120132
}
@@ -131,7 +143,9 @@ func (d *dockerImageDestination) PutBlob(stream io.Reader, inputInfo types.BlobI
131143
digester := digest.Canonical.Digester()
132144
sizeCounter := &sizeCounter{}
133145
tee := io.TeeReader(stream, io.MultiWriter(digester.Hash(), sizeCounter))
134-
res, err = d.c.makeRequestToResolvedURL("PATCH", uploadLocation.String(), map[string][]string{"Content-Type": {"application/octet-stream"}}, tee, inputInfo.Size, true)
146+
opts.headers = map[string][]string{"Content-Type": {"application/octet-stream"}}
147+
opts.stream = tee
148+
res, err = d.c.makeRequestToResolvedURL("PATCH", uploadLocation.String(), opts, inputInfo.Size, true)
135149
if err != nil {
136150
logrus.Debugf("Error uploading layer chunked, response %#v", *res)
137151
return types.BlobInfo{}, err
@@ -150,7 +164,8 @@ func (d *dockerImageDestination) PutBlob(stream io.Reader, inputInfo types.BlobI
150164
// TODO: check inputInfo.Digest == computedDigest https://github.com/containers/image/pull/70#discussion_r77646717
151165
locationQuery.Set("digest", computedDigest.String())
152166
uploadLocation.RawQuery = locationQuery.Encode()
153-
res, err = d.c.makeRequestToResolvedURL("PUT", uploadLocation.String(), map[string][]string{"Content-Type": {"application/octet-stream"}}, nil, -1, true)
167+
opts.stream = nil
168+
res, err = d.c.makeRequestToResolvedURL("PUT", uploadLocation.String(), opts, -1, true)
154169
if err != nil {
155170
return types.BlobInfo{}, err
156171
}
@@ -171,7 +186,13 @@ func (d *dockerImageDestination) HasBlob(info types.BlobInfo) (bool, int64, erro
171186
checkURL := fmt.Sprintf(blobsURL, d.ref.ref.RemoteName(), info.Digest.String())
172187

173188
logrus.Debugf("Checking %s", checkURL)
174-
res, err := d.c.makeRequest("HEAD", checkURL, nil, nil)
189+
opts := requestOptions{
190+
scope: authScope{
191+
remoteName: d.ref.ref.RemoteName(),
192+
actions: "push",
193+
},
194+
}
195+
res, err := d.c.makeRequest("HEAD", checkURL, opts)
175196
if err != nil {
176197
return false, -1, err
177198
}
@@ -215,7 +236,15 @@ func (d *dockerImageDestination) PutManifest(m []byte) error {
215236
if mimeType != "" {
216237
headers["Content-Type"] = []string{mimeType}
217238
}
218-
res, err := d.c.makeRequest("PUT", url, headers, bytes.NewReader(m))
239+
opts := requestOptions{
240+
scope: authScope{
241+
remoteName: d.ref.ref.RemoteName(),
242+
actions: "push",
243+
},
244+
headers: headers,
245+
stream: bytes.NewReader(m),
246+
}
247+
res, err := d.c.makeRequest("PUT", url, opts)
219248
if err != nil {
220249
return err
221250
}

docker/docker_image_src.go

Lines changed: 31 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,14 @@ func (s *dockerImageSource) fetchManifest(tagOrDigest string) ([]byte, string, e
8383
url := fmt.Sprintf(manifestURL, s.ref.ref.RemoteName(), tagOrDigest)
8484
headers := make(map[string][]string)
8585
headers["Accept"] = s.requestedManifestMIMETypes
86-
res, err := s.c.makeRequest("GET", url, headers, nil)
86+
opts := requestOptions{
87+
scope: authScope{
88+
remoteName: s.ref.ref.RemoteName(),
89+
actions: "pull",
90+
},
91+
headers: headers,
92+
}
93+
res, err := s.c.makeRequest("GET", url, opts)
8794
if err != nil {
8895
return nil, "", err
8996
}
@@ -135,9 +142,15 @@ func (s *dockerImageSource) getExternalBlob(urls []string) (io.ReadCloser, int64
135142
var (
136143
resp *http.Response
137144
err error
145+
opts = requestOptions{
146+
scope: authScope{
147+
remoteName: s.ref.ref.RemoteName(),
148+
actions: "pull",
149+
},
150+
}
138151
)
139152
for _, url := range urls {
140-
resp, err = s.c.makeRequestToResolvedURL("GET", url, nil, nil, -1, false)
153+
resp, err = s.c.makeRequestToResolvedURL("GET", url, opts, -1, false)
141154
if err == nil {
142155
if resp.StatusCode != http.StatusOK {
143156
err = errors.Errorf("error fetching external blob from %q: %d", url, resp.StatusCode)
@@ -168,7 +181,13 @@ func (s *dockerImageSource) GetBlob(info types.BlobInfo) (io.ReadCloser, int64,
168181

169182
url := fmt.Sprintf(blobsURL, s.ref.ref.RemoteName(), info.Digest.String())
170183
logrus.Debugf("Downloading %s", url)
171-
res, err := s.c.makeRequest("GET", url, nil, nil)
184+
opts := requestOptions{
185+
scope: authScope{
186+
remoteName: s.ref.ref.RemoteName(),
187+
actions: "pull",
188+
},
189+
}
190+
res, err := s.c.makeRequest("GET", url, opts)
172191
if err != nil {
173192
return nil, 0, err
174193
}
@@ -265,7 +284,14 @@ func deleteImage(ctx *types.SystemContext, ref dockerReference) error {
265284
return err
266285
}
267286
getURL := fmt.Sprintf(manifestURL, ref.ref.RemoteName(), reference)
268-
get, err := c.makeRequest("GET", getURL, headers, nil)
287+
opts := requestOptions{
288+
scope: authScope{
289+
remoteName: ref.ref.RemoteName(),
290+
actions: "pull",
291+
},
292+
headers: headers,
293+
}
294+
get, err := c.makeRequest("GET", getURL, opts)
269295
if err != nil {
270296
return err
271297
}
@@ -287,7 +313,7 @@ func deleteImage(ctx *types.SystemContext, ref dockerReference) error {
287313

288314
// When retrieving the digest from a registry >= 2.3 use the following header:
289315
// "Accept": "application/vnd.docker.distribution.manifest.v2+json"
290-
delete, err := c.makeRequest("DELETE", deleteURL, headers, nil)
316+
delete, err := c.makeRequest("DELETE", deleteURL, opts)
291317
if err != nil {
292318
return err
293319
}

0 commit comments

Comments
 (0)