-
Notifications
You must be signed in to change notification settings - Fork 4.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Pullthrough broken registry server #13283
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,10 @@ | ||
package server | ||
|
||
import ( | ||
"fmt" | ||
"io" | ||
"net/http" | ||
"os" | ||
"sync" | ||
"time" | ||
|
||
|
@@ -26,6 +28,7 @@ var _ distribution.BlobStore = &pullthroughBlobStore{} | |
// the image stream and looks for those that have the layer. | ||
func (pbs *pullthroughBlobStore) Stat(ctx context.Context, dgst digest.Digest) (distribution.Descriptor, error) { | ||
context.GetLogger(ctx).Debugf("(*pullthroughBlobStore).Stat: starting with dgst=%s", dgst.String()) | ||
|
||
// check the local store for the blob | ||
desc, err := pbs.BlobStore.Stat(ctx, dgst) | ||
switch { | ||
|
@@ -108,6 +111,35 @@ func setResponseHeaders(w http.ResponseWriter, length int64, mediaType string, d | |
w.Header().Set("Etag", digest.String()) | ||
} | ||
|
||
// serveRemoteContent tries to use http.ServeContent for remote content. | ||
func serveRemoteContent(rw http.ResponseWriter, req *http.Request, desc distribution.Descriptor, remoteReader io.ReadSeeker) (bool, error) { | ||
// Set the appropriate content serving headers. | ||
setResponseHeaders(rw, desc.Size, desc.MediaType, desc.Digest) | ||
|
||
// Fallback to Copy if request wasn't given. | ||
if req == nil { | ||
return false, nil | ||
} | ||
|
||
// Check whether remoteReader is seekable. The remoteReader' Seek method must work: ServeContent uses | ||
// a seek to the end of the content to determine its size. | ||
if _, err := remoteReader.Seek(0, os.SEEK_END); err != nil { | ||
// The remoteReader isn't seekable. It means that the remote response under the hood of remoteReader | ||
// doesn't contain any Content-Range or Content-Length headers. In this case we need to rollback to | ||
// simple Copy. | ||
return false, nil | ||
} | ||
|
||
// Move pointer back to begin. | ||
if _, err := remoteReader.Seek(0, os.SEEK_SET); err != nil { | ||
return false, err | ||
} | ||
|
||
http.ServeContent(rw, req, desc.Digest.String(), time.Time{}, remoteReader) | ||
|
||
return true, nil | ||
} | ||
|
||
// inflight tracks currently downloading blobs | ||
var inflight = make(map[digest.Digest]struct{}) | ||
|
||
|
@@ -129,12 +161,16 @@ func (pbs *pullthroughBlobStore) copyContent(store BlobGetterService, ctx contex | |
|
||
rw, ok := writer.(http.ResponseWriter) | ||
if ok { | ||
setResponseHeaders(rw, desc.Size, desc.MediaType, dgst) | ||
// serve range requests | ||
if req != nil { | ||
http.ServeContent(rw, req, desc.Digest.String(), time.Time{}, remoteReader) | ||
contentHandled, err := serveRemoteContent(rw, req, desc, remoteReader) | ||
if err != nil { | ||
return distribution.Descriptor{}, err | ||
} | ||
|
||
if contentHandled { | ||
return desc, nil | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This should set There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Only when ServeContent doesn't run. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @smarterclayton ServeContent will replace anyway. |
||
|
||
rw.Header().Set("Content-Length", fmt.Sprintf("%d", desc.Size)) | ||
} | ||
|
||
if _, err = io.CopyN(writer, remoteReader, desc.Size); err != nil { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems fairly fragile - how do we actually verify this doesn't regress in the future?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@smarterclayton This test is about the acknowledgment that the
remoteReader
is reallyio.ReadSeeker
. Acording documentation thehttp.ServeContent
uses a seek to the end of the content to determine its size.Basically, the problem hidden deeply inside the docker's client library. They return an object with
io.ReadSeeker
interface which in fact isn't seekable. I can understand them, because if they don't know the final size of the content, it's impossible to move through it.So the regression can't happen until the
http.ServeContent
acceptsio.ReadSeeker
as argument. If it stops doing that, then it will regress in golang./cc @mfojtik @dmage