0
Fork 0
mirror of https://github.com/project-zot/zot.git synced 2025-01-13 22:50:38 -05:00

routes: strip query parameter from request URL

reuqest url also contains query parameter due to this in some scenarios
location header is setting up incorrectly, strip query parameter from
request url to correctly setup location header.

Closes #573 #575

Signed-off-by: Shivam Mishra <shimish2@cisco.com>
This commit is contained in:
Shivam Mishra 2022-06-02 20:54:42 +00:00 committed by Ramkumar Chinchani
parent f52c950d04
commit 620bc7c517
4 changed files with 50 additions and 10 deletions

View file

@ -3,6 +3,8 @@ package constants
const ( const (
ArtifactSpecRoutePrefix = "/oras/artifacts/v1" ArtifactSpecRoutePrefix = "/oras/artifacts/v1"
RoutePrefix = "/v2" RoutePrefix = "/v2"
Blobs = "blobs"
Uploads = "uploads"
DistAPIVersion = "Docker-Distribution-API-Version" DistAPIVersion = "Docker-Distribution-API-Version"
DistContentDigestKey = "Docker-Content-Digest" DistContentDigestKey = "Docker-Content-Digest"
BlobUploadUUID = "Blob-Upload-UUID" BlobUploadUUID = "Blob-Upload-UUID"

View file

@ -2719,6 +2719,10 @@ func TestCrossRepoMount(t *testing.T) {
Post(baseURL + "/v2/zot-c-test/blobs/uploads/") Post(baseURL + "/v2/zot-c-test/blobs/uploads/")
So(err, ShouldBeNil) So(err, ShouldBeNil)
So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted) So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted)
location, err := postResponse.RawResponse.Location()
So(err, ShouldBeNil)
So(location.String(), ShouldStartWith, fmt.Sprintf("%s%s/zot-c-test/%s/%s",
baseURL, constants.RoutePrefix, constants.Blobs, constants.Uploads))
incorrectParams := make(map[string]string) incorrectParams := make(map[string]string)
incorrectParams["mount"] = "sha256:63a795ca90aa6e7dda60941e826810a4cd0a2e73ea02bf458241df2a5c973e29" incorrectParams["mount"] = "sha256:63a795ca90aa6e7dda60941e826810a4cd0a2e73ea02bf458241df2a5c973e29"
@ -2729,6 +2733,8 @@ func TestCrossRepoMount(t *testing.T) {
Post(baseURL + "/v2/zot-y-test/blobs/uploads/") Post(baseURL + "/v2/zot-y-test/blobs/uploads/")
So(err, ShouldBeNil) So(err, ShouldBeNil)
So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted) So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted)
So(test.Location(baseURL, postResponse), ShouldStartWith, fmt.Sprintf("%s%s/zot-y-test/%s/%s",
baseURL, constants.RoutePrefix, constants.Blobs, constants.Uploads))
// Use correct request // Use correct request
// This is correct request but it will return 202 because blob is not present in cache. // This is correct request but it will return 202 because blob is not present in cache.
@ -2738,6 +2744,8 @@ func TestCrossRepoMount(t *testing.T) {
Post(baseURL + "/v2/zot-c-test/blobs/uploads/") Post(baseURL + "/v2/zot-c-test/blobs/uploads/")
So(err, ShouldBeNil) So(err, ShouldBeNil)
So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted) So(postResponse.StatusCode(), ShouldEqual, http.StatusAccepted)
So(test.Location(baseURL, postResponse), ShouldStartWith, fmt.Sprintf("%s%s/zot-c-test/%s/%s",
baseURL, constants.RoutePrefix, constants.Blobs, constants.Uploads))
// Send same request again // Send same request again
postResponse, err = client.R(). postResponse, err = client.R().
@ -2792,6 +2800,8 @@ func TestCrossRepoMount(t *testing.T) {
Post(baseURL + "/v2/zot-mount-test/blobs/uploads/") Post(baseURL + "/v2/zot-mount-test/blobs/uploads/")
So(err, ShouldBeNil) So(err, ShouldBeNil)
So(postResponse.StatusCode(), ShouldEqual, http.StatusCreated) So(postResponse.StatusCode(), ShouldEqual, http.StatusCreated)
So(test.Location(baseURL, postResponse), ShouldEqual, fmt.Sprintf("%s%s/zot-mount-test/%s/%s:%s",
baseURL, constants.RoutePrefix, constants.Blobs, godigest.SHA256, blob))
// Check os.SameFile here // Check os.SameFile here
cachePath := path.Join(ctlr.Config.Storage.RootDirectory, "zot-d-test", "blobs/sha256", dgst.Hex()) cachePath := path.Join(ctlr.Config.Storage.RootDirectory, "zot-d-test", "blobs/sha256", dgst.Hex())
@ -2815,6 +2825,8 @@ func TestCrossRepoMount(t *testing.T) {
Post(baseURL + "/v2/zot-mount1-test/blobs/uploads/") Post(baseURL + "/v2/zot-mount1-test/blobs/uploads/")
So(err, ShouldBeNil) So(err, ShouldBeNil)
So(postResponse.StatusCode(), ShouldEqual, http.StatusCreated) So(postResponse.StatusCode(), ShouldEqual, http.StatusCreated)
So(test.Location(baseURL, postResponse), ShouldEqual, fmt.Sprintf("%s%s/zot-mount1-test/%s/%s:%s",
baseURL, constants.RoutePrefix, constants.Blobs, godigest.SHA256, blob))
linkPath = path.Join(ctlr.Config.Storage.RootDirectory, "zot-mount1-test", "blobs/sha256", dgst.Hex()) linkPath = path.Join(ctlr.Config.Storage.RootDirectory, "zot-mount1-test", "blobs/sha256", dgst.Hex())

View file

@ -17,6 +17,7 @@ import (
"io" "io"
"io/ioutil" "io/ioutil"
"net/http" "net/http"
"net/url"
"path" "path"
"sort" "sort"
"strconv" "strconv"
@ -759,14 +760,14 @@ func (rh *RouteHandler) CreateBlobUpload(response http.ResponseWriter, request *
return return
} }
response.Header().Set("Location", path.Join(request.URL.String(), upload)) response.Header().Set("Location", getBlobUploadSessionLocation(request.URL, upload))
response.Header().Set("Range", "bytes=0-0") response.Header().Set("Range", "bytes=0-0")
response.WriteHeader(http.StatusAccepted) response.WriteHeader(http.StatusAccepted)
return return
} }
response.Header().Set("Location", fmt.Sprintf("/v2/%s/blobs/%s", name, mountDigests[0])) response.Header().Set("Location", getBlobUploadLocation(request.URL, name, mountDigests[0]))
response.WriteHeader(http.StatusCreated) response.WriteHeader(http.StatusCreated)
return return
@ -826,7 +827,7 @@ func (rh *RouteHandler) CreateBlobUpload(response http.ResponseWriter, request *
return return
} }
response.Header().Set("Location", fmt.Sprintf("/v2/%s/blobs/%s", name, digest)) response.Header().Set("Location", getBlobUploadLocation(request.URL, name, digest))
response.Header().Set(constants.BlobUploadUUID, sessionID) response.Header().Set(constants.BlobUploadUUID, sessionID)
response.WriteHeader(http.StatusCreated) response.WriteHeader(http.StatusCreated)
@ -845,7 +846,7 @@ func (rh *RouteHandler) CreateBlobUpload(response http.ResponseWriter, request *
return return
} }
response.Header().Set("Location", path.Join(request.URL.String(), upload)) response.Header().Set("Location", getBlobUploadSessionLocation(request.URL, upload))
response.Header().Set("Range", "bytes=0-0") response.Header().Set("Range", "bytes=0-0")
response.WriteHeader(http.StatusAccepted) response.WriteHeader(http.StatusAccepted)
} }
@ -904,7 +905,7 @@ func (rh *RouteHandler) GetBlobUpload(response http.ResponseWriter, request *htt
return return
} }
response.Header().Set("Location", path.Join(request.URL.String(), sessionID)) response.Header().Set("Location", getBlobUploadSessionLocation(request.URL, sessionID))
response.Header().Set("Range", fmt.Sprintf("bytes=0-%d", size-1)) response.Header().Set("Range", fmt.Sprintf("bytes=0-%d", size-1))
response.WriteHeader(http.StatusNoContent) response.WriteHeader(http.StatusNoContent)
} }
@ -996,7 +997,7 @@ func (rh *RouteHandler) PatchBlobUpload(response http.ResponseWriter, request *h
return return
} }
response.Header().Set("Location", request.URL.String()) response.Header().Set("Location", getBlobUploadSessionLocation(request.URL, sessionID))
response.Header().Set("Range", fmt.Sprintf("bytes=0-%d", clen-1)) response.Header().Set("Range", fmt.Sprintf("bytes=0-%d", clen-1))
response.Header().Set("Content-Length", "0") response.Header().Set("Content-Length", "0")
response.Header().Set(constants.BlobUploadUUID, sessionID) response.Header().Set(constants.BlobUploadUUID, sessionID)
@ -1139,7 +1140,7 @@ finish:
return return
} }
response.Header().Set("Location", fmt.Sprintf("/v2/%s/blobs/%s", name, digest)) response.Header().Set("Location", getBlobUploadLocation(request.URL, name, digest))
response.Header().Set("Content-Length", "0") response.Header().Set("Content-Length", "0")
response.Header().Set(constants.DistContentDigestKey, digest) response.Header().Set(constants.DistContentDigestKey, digest)
response.WriteHeader(http.StatusCreated) response.WriteHeader(http.StatusCreated)
@ -1465,3 +1466,31 @@ func (rh *RouteHandler) GetReferrers(response http.ResponseWriter, request *http
WriteJSON(response, http.StatusOK, rs) WriteJSON(response, http.StatusOK, rs)
} }
// GetBlobUploadSessionLocation returns actual blob location to start/resume uploading blobs.
// e.g. /v2/<name>/blobs/uploads/<session-id>.
func getBlobUploadSessionLocation(url *url.URL, sessionID string) string {
url.RawQuery = ""
if !strings.Contains(url.Path, sessionID) {
url.Path = path.Join(url.Path, sessionID)
}
return url.String()
}
// GetBlobUploadLocation returns actual blob location on registry
// e.g /v2/<name>/blobs/<digest>.
func getBlobUploadLocation(url *url.URL, name, digest string) string {
url.RawQuery = ""
// we are relying on request URL to set location and
// if request URL contains uploads either we are resuming blob upload or starting a new blob upload.
// getBlobUploadLocation will be called only when blob upload is completed and
// location should be set as blob url <v2/<name>/blobs/<digest>>.
if strings.Contains(url.Path, "uploads") {
url.Path = path.Join(constants.RoutePrefix, name, constants.Blobs, digest)
}
return url.String()
}

View file

@ -80,9 +80,6 @@ func Location(baseURL string, resp *resty.Response) string {
} }
path := uloc.Path path := uloc.Path
if query := uloc.RawQuery; query != "" {
path += "?" + query
}
return baseURL + path return baseURL + path
} }