2018-03-20 01:33:56 -04:00
|
|
|
package registry
|
|
|
|
|
|
|
|
import (
|
2018-09-25 12:27:46 -04:00
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2018-03-20 01:33:56 -04:00
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/docker/distribution"
|
|
|
|
"github.com/docker/distribution/manifest/manifestlist"
|
|
|
|
"github.com/docker/distribution/manifest/schema1"
|
|
|
|
"github.com/docker/distribution/manifest/schema2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Manifest returns the manifest for a specific repository:tag.
|
|
|
|
func (r *Registry) Manifest(repository, ref string) (distribution.Manifest, error) {
|
|
|
|
uri := r.url("/v2/%s/manifests/%s", repository, ref)
|
|
|
|
r.Logf("registry.manifests uri=%s repository=%s ref=%s", uri, repository, ref)
|
|
|
|
|
|
|
|
req, err := http.NewRequest("GET", uri, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-09-25 12:27:46 -04:00
|
|
|
req.Header.Add("Accept", fmt.Sprintf("%s;q=0.9", schema2.MediaTypeManifest))
|
2018-03-20 01:33:56 -04:00
|
|
|
|
|
|
|
resp, err := r.Client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
r.Logf("registry.manifests resp.Status=%s, body=%s", resp.Status, body)
|
|
|
|
|
|
|
|
m, _, err := distribution.UnmarshalManifest(resp.Header.Get("Content-Type"), body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ManifestList gets the registry v2 manifest list.
|
|
|
|
func (r *Registry) ManifestList(repository, ref string) (manifestlist.ManifestList, error) {
|
|
|
|
uri := r.url("/v2/%s/manifests/%s", repository, ref)
|
|
|
|
r.Logf("registry.manifests uri=%s repository=%s ref=%s", uri, repository, ref)
|
|
|
|
|
|
|
|
var m manifestlist.ManifestList
|
|
|
|
if _, err := r.getJSON(uri, &m, true); err != nil {
|
|
|
|
r.Logf("registry.manifests response=%v", m)
|
|
|
|
return m, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ManifestV2 gets the registry v2 manifest.
|
|
|
|
func (r *Registry) ManifestV2(repository, ref string) (schema2.Manifest, error) {
|
|
|
|
uri := r.url("/v2/%s/manifests/%s", repository, ref)
|
|
|
|
r.Logf("registry.manifests uri=%s repository=%s ref=%s", uri, repository, ref)
|
|
|
|
|
|
|
|
var m schema2.Manifest
|
|
|
|
if _, err := r.getJSON(uri, &m, true); err != nil {
|
|
|
|
r.Logf("registry.manifests response=%v", m)
|
|
|
|
return m, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ManifestV1 gets the registry v1 manifest.
|
|
|
|
func (r *Registry) ManifestV1(repository, ref string) (schema1.SignedManifest, error) {
|
|
|
|
uri := r.url("/v2/%s/manifests/%s", repository, ref)
|
|
|
|
r.Logf("registry.manifests uri=%s repository=%s ref=%s", uri, repository, ref)
|
|
|
|
|
|
|
|
var m schema1.SignedManifest
|
|
|
|
if _, err := r.getJSON(uri, &m, false); err != nil {
|
|
|
|
r.Logf("registry.manifests response=%v", m)
|
|
|
|
return m, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m, nil
|
|
|
|
}
|
2018-09-25 12:27:46 -04:00
|
|
|
|
|
|
|
// PutManifest calls a PUT for the specific manifest for an image.
|
|
|
|
func (r *Registry) PutManifest(repository, ref string, manifest distribution.Manifest) error {
|
|
|
|
url := r.url("/v2/%s/manifests/%s", repository, ref)
|
|
|
|
r.Logf("registry.manifest.put url=%s repository=%s reference=%s", url, repository, ref)
|
|
|
|
|
|
|
|
b, err := json.Marshal(manifest)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("PUT", url, bytes.NewBuffer(b))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Set("Content-Type", schema2.MediaTypeManifest)
|
|
|
|
resp, err := r.Client.Do(req)
|
|
|
|
if resp != nil {
|
|
|
|
defer resp.Body.Close()
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|