|
@@ -4,102 +4,49 @@ import (
|
|
|
"context"
|
|
|
"encoding/json"
|
|
|
"io"
|
|
|
- "net/http"
|
|
|
- "net/url"
|
|
|
- "strings"
|
|
|
"time"
|
|
|
|
|
|
"github.com/containerd/containerd/platforms"
|
|
|
"github.com/docker/distribution/reference"
|
|
|
"github.com/docker/docker/api/types/container"
|
|
|
"github.com/docker/docker/builder/dockerfile"
|
|
|
- "github.com/docker/docker/builder/remotecontext"
|
|
|
"github.com/docker/docker/dockerversion"
|
|
|
"github.com/docker/docker/errdefs"
|
|
|
"github.com/docker/docker/image"
|
|
|
"github.com/docker/docker/layer"
|
|
|
"github.com/docker/docker/pkg/archive"
|
|
|
- "github.com/docker/docker/pkg/progress"
|
|
|
- "github.com/docker/docker/pkg/streamformatter"
|
|
|
"github.com/docker/docker/pkg/system"
|
|
|
specs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
- "github.com/pkg/errors"
|
|
|
)
|
|
|
|
|
|
-// ImportImage imports an image, getting the archived layer data either from
|
|
|
-// inConfig (if src is "-"), or from a URI specified in src. Progress output is
|
|
|
-// written to outStream. Repository and tag names can optionally be given in
|
|
|
-// the repo and tag arguments, respectively.
|
|
|
-func (i *ImageService) ImportImage(ctx context.Context, src string, repository string, platform *specs.Platform, tag string, msg string, inConfig io.ReadCloser, outStream io.Writer, changes []string) error {
|
|
|
- var (
|
|
|
- rc io.ReadCloser
|
|
|
- resp *http.Response
|
|
|
- newRef reference.Named
|
|
|
- )
|
|
|
-
|
|
|
- if repository != "" {
|
|
|
- var err error
|
|
|
- newRef, err = reference.ParseNormalizedNamed(repository)
|
|
|
- if err != nil {
|
|
|
- return errdefs.InvalidParameter(err)
|
|
|
- }
|
|
|
- if _, isCanonical := newRef.(reference.Canonical); isCanonical {
|
|
|
- return errdefs.InvalidParameter(errors.New("cannot import digest reference"))
|
|
|
- }
|
|
|
-
|
|
|
- if tag != "" {
|
|
|
- newRef, err = reference.WithTag(newRef, tag)
|
|
|
- if err != nil {
|
|
|
- return errdefs.InvalidParameter(err)
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- // Normalize platform - default to the operating system and architecture if not supplied.
|
|
|
+// ImportImage imports an image, getting the archived layer data from layerReader.
|
|
|
+// Uncompressed layer archive is passed to the layerStore and handled by the
|
|
|
+// underlying graph driver.
|
|
|
+// Image is tagged with the given reference.
|
|
|
+// If the platform is nil, the default host platform is used.
|
|
|
+// Message is used as the image's history comment.
|
|
|
+// Image configuration is derived from the dockerfile instructions in changes.
|
|
|
+func (i *ImageService) ImportImage(ctx context.Context, newRef reference.Named, platform *specs.Platform, msg string, layerReader io.Reader, changes []string) (image.ID, error) {
|
|
|
if platform == nil {
|
|
|
- p := platforms.DefaultSpec()
|
|
|
- platform = &p
|
|
|
+ def := platforms.DefaultSpec()
|
|
|
+ platform = &def
|
|
|
}
|
|
|
if !system.IsOSSupported(platform.OS) {
|
|
|
- return errdefs.InvalidParameter(system.ErrNotSupportedOperatingSystem)
|
|
|
+ return "", errdefs.InvalidParameter(system.ErrNotSupportedOperatingSystem)
|
|
|
}
|
|
|
+
|
|
|
config, err := dockerfile.BuildFromConfig(ctx, &container.Config{}, changes, platform.OS)
|
|
|
if err != nil {
|
|
|
- return err
|
|
|
- }
|
|
|
- if src == "-" {
|
|
|
- rc = inConfig
|
|
|
- } else {
|
|
|
- inConfig.Close()
|
|
|
- if len(strings.Split(src, "://")) == 1 {
|
|
|
- src = "http://" + src
|
|
|
- }
|
|
|
- u, err := url.Parse(src)
|
|
|
- if err != nil {
|
|
|
- return errdefs.InvalidParameter(err)
|
|
|
- }
|
|
|
-
|
|
|
- resp, err = remotecontext.GetWithStatusError(u.String())
|
|
|
- if err != nil {
|
|
|
- return err
|
|
|
- }
|
|
|
- outStream.Write(streamformatter.FormatStatus("", "Downloading from %s", u))
|
|
|
- progressOutput := streamformatter.NewJSONProgressOutput(outStream, true)
|
|
|
- rc = progress.NewProgressReader(resp.Body, progressOutput, resp.ContentLength, "", "Importing")
|
|
|
- }
|
|
|
-
|
|
|
- defer rc.Close()
|
|
|
- if len(msg) == 0 {
|
|
|
- msg = "Imported from " + src
|
|
|
+ return "", errdefs.InvalidParameter(err)
|
|
|
}
|
|
|
|
|
|
- inflatedLayerData, err := archive.DecompressStream(rc)
|
|
|
+ inflatedLayerData, err := archive.DecompressStream(layerReader)
|
|
|
if err != nil {
|
|
|
- return err
|
|
|
+ return "", err
|
|
|
}
|
|
|
l, err := i.layerStore.Register(inflatedLayerData, "")
|
|
|
if err != nil {
|
|
|
- return err
|
|
|
+ return "", err
|
|
|
}
|
|
|
defer layer.ReleaseAndLog(i.layerStore, l)
|
|
|
|
|
@@ -124,22 +71,20 @@ func (i *ImageService) ImportImage(ctx context.Context, src string, repository s
|
|
|
}},
|
|
|
})
|
|
|
if err != nil {
|
|
|
- return err
|
|
|
+ return "", err
|
|
|
}
|
|
|
|
|
|
id, err := i.imageStore.Create(imgConfig)
|
|
|
if err != nil {
|
|
|
- return err
|
|
|
+ return "", err
|
|
|
}
|
|
|
|
|
|
- // FIXME: connect with commit code and call refstore directly
|
|
|
if newRef != nil {
|
|
|
if err := i.TagImageWithReference(id, newRef); err != nil {
|
|
|
- return err
|
|
|
+ return "", err
|
|
|
}
|
|
|
}
|
|
|
|
|
|
i.LogImageEvent(id.String(), id.String(), "import")
|
|
|
- outStream.Write(streamformatter.FormatStatus("", id.String()))
|
|
|
- return nil
|
|
|
+ return id, nil
|
|
|
}
|