Initial implementation of Manifest HTTP API
Push, pull and delete of manifest files in the registry have been implemented
on top of the storage services. Basic workflows, including reporting of missing
manifests are tested, including various proposed response codes. Common testing
functionality has been collected into shared methods. A test suite may be
emerging but it might better to capture more edge cases (such as resumable
upload, range requests, etc.) before we commit to a full approach.
To support clearer test cases and simpler handler methods, an application aware
urlBuilder has been added. We may want to export the functionality for use in
the client, which could allow us to abstract away from gorilla/mux.
A few error codes have been added to fill in error conditions missing from the
proposal. Some use cases have identified some problems with the approach to
error reporting that requires more work to reconcile. To resolve this, the
mapping of Go errors into error types needs to pulled out of the handlers and
into the application. We also need to move to type-based errors, with rich
information, rather than value-based errors. ErrorHandlers will probably
replace the http.Handlers to make this work correctly.
Unrelated to the above, the "length" parameter has been migrated to "size" for
completing layer uploads. This change should have gone out before but these
diffs ending up being coupled with the parameter name change due to updates to
the layer unit tests.
2014-11-26 20:16:58 +00:00
|
|
|
package registry
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
|
|
|
|
"github.com/docker/docker-registry/digest"
|
|
|
|
"github.com/docker/docker-registry/storage"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
)
|
|
|
|
|
|
|
|
type urlBuilder struct {
|
|
|
|
url *url.URL // url root (ie http://localhost/)
|
|
|
|
router *mux.Router
|
|
|
|
}
|
|
|
|
|
|
|
|
func newURLBuilder(root *url.URL) *urlBuilder {
|
|
|
|
return &urlBuilder{
|
|
|
|
url: root,
|
|
|
|
router: v2APIRouter(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newURLBuilderFromRequest(r *http.Request) *urlBuilder {
|
|
|
|
u := &url.URL{
|
|
|
|
Scheme: r.URL.Scheme,
|
|
|
|
Host: r.Host,
|
|
|
|
}
|
|
|
|
|
|
|
|
return newURLBuilder(u)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newURLBuilderFromString(root string) (*urlBuilder, error) {
|
|
|
|
u, err := url.Parse(root)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return newURLBuilder(u), nil
|
|
|
|
}
|
|
|
|
|
2014-12-11 06:33:36 +00:00
|
|
|
func (ub *urlBuilder) buildBaseURL() (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameBase)
|
|
|
|
|
|
|
|
baseURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return baseURL.String(), nil
|
|
|
|
}
|
|
|
|
|
2014-12-09 23:36:26 +00:00
|
|
|
func (ub *urlBuilder) buildTagsURL(name string) (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameTags)
|
|
|
|
|
|
|
|
tagsURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL("name", name)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return tagsURL.String(), nil
|
|
|
|
}
|
|
|
|
|
Initial implementation of Manifest HTTP API
Push, pull and delete of manifest files in the registry have been implemented
on top of the storage services. Basic workflows, including reporting of missing
manifests are tested, including various proposed response codes. Common testing
functionality has been collected into shared methods. A test suite may be
emerging but it might better to capture more edge cases (such as resumable
upload, range requests, etc.) before we commit to a full approach.
To support clearer test cases and simpler handler methods, an application aware
urlBuilder has been added. We may want to export the functionality for use in
the client, which could allow us to abstract away from gorilla/mux.
A few error codes have been added to fill in error conditions missing from the
proposal. Some use cases have identified some problems with the approach to
error reporting that requires more work to reconcile. To resolve this, the
mapping of Go errors into error types needs to pulled out of the handlers and
into the application. We also need to move to type-based errors, with rich
information, rather than value-based errors. ErrorHandlers will probably
replace the http.Handlers to make this work correctly.
Unrelated to the above, the "length" parameter has been migrated to "size" for
completing layer uploads. This change should have gone out before but these
diffs ending up being coupled with the parameter name change due to updates to
the layer unit tests.
2014-11-26 20:16:58 +00:00
|
|
|
func (ub *urlBuilder) forManifest(m *storage.Manifest) (string, error) {
|
|
|
|
return ub.buildManifestURL(m.Name, m.Tag)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) buildManifestURL(name, tag string) (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameImageManifest)
|
|
|
|
|
|
|
|
manifestURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL("name", name, "tag", tag)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return manifestURL.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) forLayer(l storage.Layer) (string, error) {
|
|
|
|
return ub.buildLayerURL(l.Name(), l.Digest())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) buildLayerURL(name string, dgst digest.Digest) (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameBlob)
|
|
|
|
|
|
|
|
layerURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL("name", name, "digest", dgst.String())
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return layerURL.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) buildLayerUploadURL(name string) (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameBlobUpload)
|
|
|
|
|
|
|
|
uploadURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL("name", name)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return uploadURL.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) forLayerUpload(layerUpload storage.LayerUpload) (string, error) {
|
|
|
|
return ub.buildLayerUploadResumeURL(layerUpload.Name(), layerUpload.UUID())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ub *urlBuilder) buildLayerUploadResumeURL(name, uuid string, values ...url.Values) (string, error) {
|
|
|
|
route := clonedRoute(ub.router, routeNameBlobUploadResume)
|
|
|
|
|
|
|
|
uploadURL, err := route.
|
|
|
|
Schemes(ub.url.Scheme).
|
|
|
|
Host(ub.url.Host).
|
|
|
|
URL("name", name, "uuid", uuid)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return appendValuesURL(uploadURL, values...).String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// appendValuesURL appends the parameters to the url.
|
|
|
|
func appendValuesURL(u *url.URL, values ...url.Values) *url.URL {
|
|
|
|
merged := u.Query()
|
|
|
|
|
|
|
|
for _, v := range values {
|
|
|
|
for k, vv := range v {
|
|
|
|
merged[k] = append(merged[k], vv...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
u.RawQuery = merged.Encode()
|
|
|
|
return u
|
|
|
|
}
|
|
|
|
|
|
|
|
// appendValues appends the parameters to the url. Panics if the string is not
|
|
|
|
// a url.
|
|
|
|
func appendValues(u string, values ...url.Values) string {
|
|
|
|
up, err := url.Parse(u)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err) // should never happen
|
|
|
|
}
|
|
|
|
|
|
|
|
return appendValuesURL(up, values...).String()
|
|
|
|
}
|
|
|
|
|
|
|
|
// clondedRoute returns a clone of the named route from the router.
|
|
|
|
func clonedRoute(router *mux.Router, name string) *mux.Route {
|
|
|
|
route := new(mux.Route)
|
|
|
|
*route = *router.GetRoute(name) // clone the route
|
|
|
|
return route
|
|
|
|
}
|