mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Extract squash and tagging from the Dockerfile builder.
Remove pathCache and replace it with syncmap Cleanup NewBuilder Create an api/server/backend/build Extract BuildTagger Signed-off-by: Daniel Nephin <dnephin@docker.com>
This commit is contained in:
parent
f07811f766
commit
0296797f0f
15 changed files with 337 additions and 259 deletions
70
api/server/backend/build/backend.go
Normal file
70
api/server/backend/build/backend.go
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
package build
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/docker/distribution/reference"
|
||||||
|
"github.com/docker/docker/api/types/backend"
|
||||||
|
"github.com/docker/docker/builder"
|
||||||
|
"github.com/docker/docker/builder/dockerfile"
|
||||||
|
"github.com/docker/docker/image"
|
||||||
|
"github.com/docker/docker/pkg/stringid"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"golang.org/x/net/context"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ImageComponent provides an interface for working with images
|
||||||
|
type ImageComponent interface {
|
||||||
|
SquashImage(from string, to string) (string, error)
|
||||||
|
TagImageWithReference(image.ID, reference.Named) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// Backend provides build functionality to the API router
|
||||||
|
type Backend struct {
|
||||||
|
manager *dockerfile.BuildManager
|
||||||
|
imageComponent ImageComponent
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewBackend creates a new build backend from components
|
||||||
|
func NewBackend(components ImageComponent, builderBackend builder.Backend) *Backend {
|
||||||
|
manager := dockerfile.NewBuildManager(builderBackend)
|
||||||
|
return &Backend{imageComponent: components, manager: manager}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build builds an image from a Source
|
||||||
|
func (b *Backend) Build(ctx context.Context, config backend.BuildConfig) (string, error) {
|
||||||
|
options := config.Options
|
||||||
|
tagger, err := NewTagger(b.imageComponent, config.ProgressWriter.StdoutFormatter, options.Tags)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
build, err := b.manager.Build(ctx, config)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
var imageID = build.ImageID
|
||||||
|
if options.Squash {
|
||||||
|
if imageID, err = squashBuild(build, b.imageComponent); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stdout := config.ProgressWriter.StdoutFormatter
|
||||||
|
fmt.Fprintf(stdout, "Successfully built %s\n", stringid.TruncateID(imageID))
|
||||||
|
err = tagger.TagImages(image.ID(imageID))
|
||||||
|
return imageID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func squashBuild(build *builder.Result, imageComponent ImageComponent) (string, error) {
|
||||||
|
var fromID string
|
||||||
|
if build.FromImage != nil {
|
||||||
|
fromID = build.FromImage.ImageID()
|
||||||
|
}
|
||||||
|
imageID, err := imageComponent.SquashImage(build.ImageID, fromID)
|
||||||
|
if err != nil {
|
||||||
|
return "", errors.Wrap(err, "error squashing image")
|
||||||
|
}
|
||||||
|
return imageID, nil
|
||||||
|
}
|
77
api/server/backend/build/tag.go
Normal file
77
api/server/backend/build/tag.go
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
package build
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/docker/distribution/reference"
|
||||||
|
"github.com/docker/docker/image"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Tagger is responsible for tagging an image created by a builder
|
||||||
|
type Tagger struct {
|
||||||
|
imageComponent ImageComponent
|
||||||
|
stdout io.Writer
|
||||||
|
repoAndTags []reference.Named
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewTagger returns a new Tagger for tagging the images of a build.
|
||||||
|
// If any of the names are invalid tags an error is returned.
|
||||||
|
func NewTagger(backend ImageComponent, stdout io.Writer, names []string) (*Tagger, error) {
|
||||||
|
reposAndTags, err := sanitizeRepoAndTags(names)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Tagger{
|
||||||
|
imageComponent: backend,
|
||||||
|
stdout: stdout,
|
||||||
|
repoAndTags: reposAndTags,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// TagImages creates image tags for the imageID
|
||||||
|
func (bt *Tagger) TagImages(imageID image.ID) error {
|
||||||
|
for _, rt := range bt.repoAndTags {
|
||||||
|
if err := bt.imageComponent.TagImageWithReference(imageID, rt); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Fprintf(bt.stdout, "Successfully tagged %s\n", reference.FamiliarString(rt))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// sanitizeRepoAndTags parses the raw "t" parameter received from the client
|
||||||
|
// to a slice of repoAndTag.
|
||||||
|
// It also validates each repoName and tag.
|
||||||
|
func sanitizeRepoAndTags(names []string) ([]reference.Named, error) {
|
||||||
|
var (
|
||||||
|
repoAndTags []reference.Named
|
||||||
|
// This map is used for deduplicating the "-t" parameter.
|
||||||
|
uniqNames = make(map[string]struct{})
|
||||||
|
)
|
||||||
|
for _, repo := range names {
|
||||||
|
if repo == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
ref, err := reference.ParseNormalizedNamed(repo)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, isCanonical := ref.(reference.Canonical); isCanonical {
|
||||||
|
return nil, errors.New("build tag cannot contain a digest")
|
||||||
|
}
|
||||||
|
|
||||||
|
ref = reference.TagNameOnly(ref)
|
||||||
|
|
||||||
|
nameWithTag := ref.String()
|
||||||
|
|
||||||
|
if _, exists := uniqNames[nameWithTag]; !exists {
|
||||||
|
uniqNames[nameWithTag] = struct{}{}
|
||||||
|
repoAndTags = append(repoAndTags, ref)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return repoAndTags, nil
|
||||||
|
}
|
|
@ -1,20 +1,17 @@
|
||||||
package build
|
package build
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
|
||||||
|
|
||||||
"github.com/docker/docker/api/types"
|
|
||||||
"github.com/docker/docker/api/types/backend"
|
"github.com/docker/docker/api/types/backend"
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Backend abstracts an image builder whose only purpose is to build an image referenced by an imageID.
|
// Backend abstracts an image builder whose only purpose is to build an image referenced by an imageID.
|
||||||
type Backend interface {
|
type Backend interface {
|
||||||
// BuildFromContext builds a Docker image referenced by an imageID string.
|
// Build a Docker image returning the id of the image
|
||||||
//
|
|
||||||
// Note: Tagging an image should not be done by a Builder, it should instead be done
|
|
||||||
// by the caller.
|
|
||||||
//
|
|
||||||
// TODO: make this return a reference instead of string
|
// TODO: make this return a reference instead of string
|
||||||
BuildFromContext(ctx context.Context, src io.ReadCloser, buildOptions *types.ImageBuildOptions, pg backend.ProgressWriter) (string, error)
|
Build(context.Context, backend.BuildConfig) (string, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type experimentalProvider interface {
|
||||||
|
HasExperimental() bool
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,14 +5,13 @@ import "github.com/docker/docker/api/server/router"
|
||||||
// buildRouter is a router to talk with the build controller
|
// buildRouter is a router to talk with the build controller
|
||||||
type buildRouter struct {
|
type buildRouter struct {
|
||||||
backend Backend
|
backend Backend
|
||||||
|
daemon experimentalProvider
|
||||||
routes []router.Route
|
routes []router.Route
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewRouter initializes a new build router
|
// NewRouter initializes a new build router
|
||||||
func NewRouter(b Backend) router.Router {
|
func NewRouter(b Backend, d experimentalProvider) router.Router {
|
||||||
r := &buildRouter{
|
r := &buildRouter{backend: b, daemon: d}
|
||||||
backend: b,
|
|
||||||
}
|
|
||||||
r.initRoutes()
|
r.initRoutes()
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/Sirupsen/logrus"
|
"github.com/Sirupsen/logrus"
|
||||||
|
apierrors "github.com/docker/docker/api/errors"
|
||||||
"github.com/docker/docker/api/server/httputils"
|
"github.com/docker/docker/api/server/httputils"
|
||||||
"github.com/docker/docker/api/types"
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/api/types/backend"
|
"github.com/docker/docker/api/types/backend"
|
||||||
|
@ -22,6 +23,7 @@ import (
|
||||||
"github.com/docker/docker/pkg/progress"
|
"github.com/docker/docker/pkg/progress"
|
||||||
"github.com/docker/docker/pkg/streamformatter"
|
"github.com/docker/docker/pkg/streamformatter"
|
||||||
units "github.com/docker/go-units"
|
units "github.com/docker/go-units"
|
||||||
|
"github.com/pkg/errors"
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -87,9 +89,6 @@ func newImageBuildOptions(ctx context.Context, r *http.Request) (*types.ImageBui
|
||||||
options.Ulimits = buildUlimits
|
options.Ulimits = buildUlimits
|
||||||
}
|
}
|
||||||
|
|
||||||
var buildArgs = map[string]*string{}
|
|
||||||
buildArgsJSON := r.FormValue("buildargs")
|
|
||||||
|
|
||||||
// Note that there are two ways a --build-arg might appear in the
|
// Note that there are two ways a --build-arg might appear in the
|
||||||
// json of the query param:
|
// json of the query param:
|
||||||
// "foo":"bar"
|
// "foo":"bar"
|
||||||
|
@ -102,25 +101,27 @@ func newImageBuildOptions(ctx context.Context, r *http.Request) (*types.ImageBui
|
||||||
// the fact they mentioned it, we need to pass that along to the builder
|
// the fact they mentioned it, we need to pass that along to the builder
|
||||||
// so that it can print a warning about "foo" being unused if there is
|
// so that it can print a warning about "foo" being unused if there is
|
||||||
// no "ARG foo" in the Dockerfile.
|
// no "ARG foo" in the Dockerfile.
|
||||||
|
buildArgsJSON := r.FormValue("buildargs")
|
||||||
if buildArgsJSON != "" {
|
if buildArgsJSON != "" {
|
||||||
|
var buildArgs = map[string]*string{}
|
||||||
if err := json.Unmarshal([]byte(buildArgsJSON), &buildArgs); err != nil {
|
if err := json.Unmarshal([]byte(buildArgsJSON), &buildArgs); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
options.BuildArgs = buildArgs
|
options.BuildArgs = buildArgs
|
||||||
}
|
}
|
||||||
|
|
||||||
var labels = map[string]string{}
|
|
||||||
labelsJSON := r.FormValue("labels")
|
labelsJSON := r.FormValue("labels")
|
||||||
if labelsJSON != "" {
|
if labelsJSON != "" {
|
||||||
|
var labels = map[string]string{}
|
||||||
if err := json.Unmarshal([]byte(labelsJSON), &labels); err != nil {
|
if err := json.Unmarshal([]byte(labelsJSON), &labels); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
options.Labels = labels
|
options.Labels = labels
|
||||||
}
|
}
|
||||||
|
|
||||||
var cacheFrom = []string{}
|
|
||||||
cacheFromJSON := r.FormValue("cachefrom")
|
cacheFromJSON := r.FormValue("cachefrom")
|
||||||
if cacheFromJSON != "" {
|
if cacheFromJSON != "" {
|
||||||
|
var cacheFrom = []string{}
|
||||||
if err := json.Unmarshal([]byte(cacheFromJSON), &cacheFrom); err != nil {
|
if err := json.Unmarshal([]byte(cacheFromJSON), &cacheFrom); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -130,33 +131,8 @@ func newImageBuildOptions(ctx context.Context, r *http.Request) (*types.ImageBui
|
||||||
return options, nil
|
return options, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type syncWriter struct {
|
|
||||||
w io.Writer
|
|
||||||
mu sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *syncWriter) Write(b []byte) (count int, err error) {
|
|
||||||
s.mu.Lock()
|
|
||||||
count, err = s.w.Write(b)
|
|
||||||
s.mu.Unlock()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (br *buildRouter) postBuild(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
func (br *buildRouter) postBuild(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
|
||||||
var (
|
var notVerboseBuffer = bytes.NewBuffer(nil)
|
||||||
authConfigs = map[string]types.AuthConfig{}
|
|
||||||
authConfigsEncoded = r.Header.Get("X-Registry-Config")
|
|
||||||
notVerboseBuffer = bytes.NewBuffer(nil)
|
|
||||||
)
|
|
||||||
|
|
||||||
if authConfigsEncoded != "" {
|
|
||||||
authConfigsJSON := base64.NewDecoder(base64.URLEncoding, strings.NewReader(authConfigsEncoded))
|
|
||||||
if err := json.NewDecoder(authConfigsJSON).Decode(&authConfigs); err != nil {
|
|
||||||
// for a pull it is not an error if no auth was given
|
|
||||||
// to increase compatibility with the existing api it is defaulting
|
|
||||||
// to be empty.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "application/json")
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
@ -183,7 +159,12 @@ func (br *buildRouter) postBuild(ctx context.Context, w http.ResponseWriter, r *
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errf(err)
|
return errf(err)
|
||||||
}
|
}
|
||||||
buildOptions.AuthConfigs = authConfigs
|
buildOptions.AuthConfigs = getAuthConfigs(r.Header)
|
||||||
|
|
||||||
|
if buildOptions.Squash && !br.daemon.HasExperimental() {
|
||||||
|
return apierrors.NewBadRequestError(
|
||||||
|
errors.New("squash is only supported with experimental mode"))
|
||||||
|
}
|
||||||
|
|
||||||
// Currently, only used if context is from a remote url.
|
// Currently, only used if context is from a remote url.
|
||||||
// Look at code in DetectContextFromRemoteURL for more information.
|
// Look at code in DetectContextFromRemoteURL for more information.
|
||||||
|
@ -199,18 +180,12 @@ func (br *buildRouter) postBuild(ctx context.Context, w http.ResponseWriter, r *
|
||||||
if buildOptions.SuppressOutput {
|
if buildOptions.SuppressOutput {
|
||||||
out = notVerboseBuffer
|
out = notVerboseBuffer
|
||||||
}
|
}
|
||||||
out = &syncWriter{w: out}
|
|
||||||
stdout := &streamformatter.StdoutFormatter{Writer: out, StreamFormatter: sf}
|
|
||||||
stderr := &streamformatter.StderrFormatter{Writer: out, StreamFormatter: sf}
|
|
||||||
|
|
||||||
pg := backend.ProgressWriter{
|
imgID, err := br.backend.Build(ctx, backend.BuildConfig{
|
||||||
Output: out,
|
Source: r.Body,
|
||||||
StdoutFormatter: stdout,
|
Options: buildOptions,
|
||||||
StderrFormatter: stderr,
|
ProgressWriter: buildProgressWriter(out, sf, createProgressReader),
|
||||||
ProgressReaderFunc: createProgressReader,
|
})
|
||||||
}
|
|
||||||
|
|
||||||
imgID, err := br.backend.BuildFromContext(ctx, r.Body, buildOptions, pg)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errf(err)
|
return errf(err)
|
||||||
}
|
}
|
||||||
|
@ -219,8 +194,47 @@ func (br *buildRouter) postBuild(ctx context.Context, w http.ResponseWriter, r *
|
||||||
// should be just the image ID and we'll print that to stdout.
|
// should be just the image ID and we'll print that to stdout.
|
||||||
if buildOptions.SuppressOutput {
|
if buildOptions.SuppressOutput {
|
||||||
stdout := &streamformatter.StdoutFormatter{Writer: output, StreamFormatter: sf}
|
stdout := &streamformatter.StdoutFormatter{Writer: output, StreamFormatter: sf}
|
||||||
fmt.Fprintf(stdout, "%s\n", string(imgID))
|
fmt.Fprintln(stdout, imgID)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func getAuthConfigs(header http.Header) map[string]types.AuthConfig {
|
||||||
|
authConfigs := map[string]types.AuthConfig{}
|
||||||
|
authConfigsEncoded := header.Get("X-Registry-Config")
|
||||||
|
|
||||||
|
if authConfigsEncoded == "" {
|
||||||
|
return authConfigs
|
||||||
|
}
|
||||||
|
|
||||||
|
authConfigsJSON := base64.NewDecoder(base64.URLEncoding, strings.NewReader(authConfigsEncoded))
|
||||||
|
// Pulling an image does not error when no auth is provided so to remain
|
||||||
|
// consistent with the existing api decode errors are ignored
|
||||||
|
json.NewDecoder(authConfigsJSON).Decode(&authConfigs)
|
||||||
|
return authConfigs
|
||||||
|
}
|
||||||
|
|
||||||
|
type syncWriter struct {
|
||||||
|
w io.Writer
|
||||||
|
mu sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *syncWriter) Write(b []byte) (count int, err error) {
|
||||||
|
s.mu.Lock()
|
||||||
|
count, err = s.w.Write(b)
|
||||||
|
s.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func buildProgressWriter(out io.Writer, sf *streamformatter.StreamFormatter, createProgressReader func(io.ReadCloser) io.ReadCloser) backend.ProgressWriter {
|
||||||
|
out = &syncWriter{w: out}
|
||||||
|
stdout := &streamformatter.StdoutFormatter{Writer: out, StreamFormatter: sf}
|
||||||
|
stderr := &streamformatter.StderrFormatter{Writer: out, StreamFormatter: sf}
|
||||||
|
|
||||||
|
return backend.ProgressWriter{
|
||||||
|
Output: out,
|
||||||
|
StdoutFormatter: stdout,
|
||||||
|
StderrFormatter: stderr,
|
||||||
|
ProgressReaderFunc: createProgressReader,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -6,7 +6,6 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/docker/docker/api/types"
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/pkg/streamformatter"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// ContainerAttachConfig holds the streams to use when connecting to a container to view logs.
|
// ContainerAttachConfig holds the streams to use when connecting to a container to view logs.
|
||||||
|
@ -99,11 +98,3 @@ type ContainerCommitConfig struct {
|
||||||
types.ContainerCommitConfig
|
types.ContainerCommitConfig
|
||||||
Changes []string
|
Changes []string
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProgressWriter is a data object to transport progress streams to the client
|
|
||||||
type ProgressWriter struct {
|
|
||||||
Output io.Writer
|
|
||||||
StdoutFormatter *streamformatter.StdoutFormatter
|
|
||||||
StderrFormatter *streamformatter.StderrFormatter
|
|
||||||
ProgressReaderFunc func(io.ReadCloser) io.ReadCloser
|
|
||||||
}
|
|
||||||
|
|
23
api/types/backend/build.go
Normal file
23
api/types/backend/build.go
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
package backend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/docker/docker/pkg/streamformatter"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ProgressWriter is a data object to transport progress streams to the client
|
||||||
|
type ProgressWriter struct {
|
||||||
|
Output io.Writer
|
||||||
|
StdoutFormatter *streamformatter.StdoutFormatter
|
||||||
|
StderrFormatter *streamformatter.StderrFormatter
|
||||||
|
ProgressReaderFunc func(io.ReadCloser) io.ReadCloser
|
||||||
|
}
|
||||||
|
|
||||||
|
// BuildConfig is the configuration used by a BuildManager to start a build
|
||||||
|
type BuildConfig struct {
|
||||||
|
Source io.ReadCloser
|
||||||
|
ProgressWriter ProgressWriter
|
||||||
|
Options *types.ImageBuildOptions
|
||||||
|
}
|
|
@ -8,11 +8,9 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/docker/distribution/reference"
|
|
||||||
"github.com/docker/docker/api/types"
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/api/types/backend"
|
"github.com/docker/docker/api/types/backend"
|
||||||
"github.com/docker/docker/api/types/container"
|
"github.com/docker/docker/api/types/container"
|
||||||
"github.com/docker/docker/image"
|
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -40,8 +38,6 @@ type Backend interface {
|
||||||
|
|
||||||
// GetImageOnBuild looks up a Docker image referenced by `name`.
|
// GetImageOnBuild looks up a Docker image referenced by `name`.
|
||||||
GetImageOnBuild(name string) (Image, error)
|
GetImageOnBuild(name string) (Image, error)
|
||||||
// TagImageWithReference tags an image with newTag
|
|
||||||
TagImageWithReference(image.ID, reference.Named) error
|
|
||||||
// PullOnBuild tells Docker to pull image referenced by `name`.
|
// PullOnBuild tells Docker to pull image referenced by `name`.
|
||||||
PullOnBuild(ctx context.Context, name string, authConfigs map[string]types.AuthConfig, output io.Writer) (Image, error)
|
PullOnBuild(ctx context.Context, name string, authConfigs map[string]types.AuthConfig, output io.Writer) (Image, error)
|
||||||
// ContainerAttachRaw attaches to container.
|
// ContainerAttachRaw attaches to container.
|
||||||
|
@ -69,12 +65,6 @@ type Backend interface {
|
||||||
// TODO: use containerd/fs.changestream instead as a source
|
// TODO: use containerd/fs.changestream instead as a source
|
||||||
CopyOnBuild(containerID string, destPath string, srcRoot string, srcPath string, decompress bool) error
|
CopyOnBuild(containerID string, destPath string, srcRoot string, srcPath string, decompress bool) error
|
||||||
|
|
||||||
// HasExperimental checks if the backend supports experimental features
|
|
||||||
HasExperimental() bool
|
|
||||||
|
|
||||||
// SquashImage squashes the fs layers from the provided image down to the specified `to` image
|
|
||||||
SquashImage(from string, to string) (string, error)
|
|
||||||
|
|
||||||
// MountImage returns mounted path with rootfs of an image.
|
// MountImage returns mounted path with rootfs of an image.
|
||||||
MountImage(name string) (string, func() error, error)
|
MountImage(name string) (string, func() error, error)
|
||||||
}
|
}
|
||||||
|
@ -85,6 +75,12 @@ type Image interface {
|
||||||
RunConfig() *container.Config
|
RunConfig() *container.Config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Result is the output produced by a Builder
|
||||||
|
type Result struct {
|
||||||
|
ImageID string
|
||||||
|
FromImage Image
|
||||||
|
}
|
||||||
|
|
||||||
// ImageCacheBuilder represents a generator for stateful image cache.
|
// ImageCacheBuilder represents a generator for stateful image cache.
|
||||||
type ImageCacheBuilder interface {
|
type ImageCacheBuilder interface {
|
||||||
// MakeImageCache creates a stateful image cache.
|
// MakeImageCache creates a stateful image cache.
|
||||||
|
|
|
@ -5,12 +5,9 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/Sirupsen/logrus"
|
"github.com/Sirupsen/logrus"
|
||||||
"github.com/docker/distribution/reference"
|
|
||||||
apierrors "github.com/docker/docker/api/errors"
|
|
||||||
"github.com/docker/docker/api/types"
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/api/types/backend"
|
"github.com/docker/docker/api/types/backend"
|
||||||
"github.com/docker/docker/api/types/container"
|
"github.com/docker/docker/api/types/container"
|
||||||
|
@ -18,10 +15,10 @@ import (
|
||||||
"github.com/docker/docker/builder/dockerfile/command"
|
"github.com/docker/docker/builder/dockerfile/command"
|
||||||
"github.com/docker/docker/builder/dockerfile/parser"
|
"github.com/docker/docker/builder/dockerfile/parser"
|
||||||
"github.com/docker/docker/builder/remotecontext"
|
"github.com/docker/docker/builder/remotecontext"
|
||||||
"github.com/docker/docker/image"
|
|
||||||
"github.com/docker/docker/pkg/stringid"
|
"github.com/docker/docker/pkg/stringid"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
|
"golang.org/x/sync/syncmap"
|
||||||
)
|
)
|
||||||
|
|
||||||
var validCommitCommands = map[string]bool{
|
var validCommitCommands = map[string]bool{
|
||||||
|
@ -39,6 +36,52 @@ var validCommitCommands = map[string]bool{
|
||||||
|
|
||||||
var defaultLogConfig = container.LogConfig{Type: "none"}
|
var defaultLogConfig = container.LogConfig{Type: "none"}
|
||||||
|
|
||||||
|
// BuildManager is shared across all Builder objects
|
||||||
|
type BuildManager struct {
|
||||||
|
backend builder.Backend
|
||||||
|
pathCache pathCache // TODO: make this persistent
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewBuildManager creates a BuildManager
|
||||||
|
func NewBuildManager(b builder.Backend) *BuildManager {
|
||||||
|
return &BuildManager{backend: b, pathCache: &syncmap.Map{}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build starts a new build from a BuildConfig
|
||||||
|
func (bm *BuildManager) Build(ctx context.Context, config backend.BuildConfig) (*builder.Result, error) {
|
||||||
|
if config.Options.Dockerfile == "" {
|
||||||
|
config.Options.Dockerfile = builder.DefaultDockerfileName
|
||||||
|
}
|
||||||
|
|
||||||
|
source, dockerfile, err := remotecontext.Detect(config)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if source != nil {
|
||||||
|
defer func() {
|
||||||
|
if err := source.Close(); err != nil {
|
||||||
|
logrus.Debugf("[BUILDER] failed to remove temporary context: %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
builderOptions := builderOptions{
|
||||||
|
Options: config.Options,
|
||||||
|
ProgressWriter: config.ProgressWriter,
|
||||||
|
Backend: bm.backend,
|
||||||
|
PathCache: bm.pathCache,
|
||||||
|
}
|
||||||
|
return newBuilder(ctx, builderOptions).build(source, dockerfile)
|
||||||
|
}
|
||||||
|
|
||||||
|
// builderOptions are the dependencies required by the builder
|
||||||
|
type builderOptions struct {
|
||||||
|
Options *types.ImageBuildOptions
|
||||||
|
Backend builder.Backend
|
||||||
|
ProgressWriter backend.ProgressWriter
|
||||||
|
PathCache pathCache
|
||||||
|
}
|
||||||
|
|
||||||
// Builder is a Dockerfile builder
|
// Builder is a Dockerfile builder
|
||||||
// It implements the builder.Backend interface.
|
// It implements the builder.Backend interface.
|
||||||
type Builder struct {
|
type Builder struct {
|
||||||
|
@ -68,65 +111,25 @@ type Builder struct {
|
||||||
from builder.Image
|
from builder.Image
|
||||||
}
|
}
|
||||||
|
|
||||||
// BuildManager implements builder.Backend and is shared across all Builder objects.
|
// newBuilder creates a new Dockerfile builder from an optional dockerfile and a Options.
|
||||||
type BuildManager struct {
|
func newBuilder(clientCtx context.Context, options builderOptions) *Builder {
|
||||||
backend builder.Backend
|
config := options.Options
|
||||||
pathCache *pathCache // TODO: make this persistent
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewBuildManager creates a BuildManager.
|
|
||||||
func NewBuildManager(b builder.Backend) (bm *BuildManager) {
|
|
||||||
return &BuildManager{backend: b, pathCache: &pathCache{}}
|
|
||||||
}
|
|
||||||
|
|
||||||
// BuildFromContext builds a new image from a given context.
|
|
||||||
func (bm *BuildManager) BuildFromContext(ctx context.Context, src io.ReadCloser, buildOptions *types.ImageBuildOptions, pg backend.ProgressWriter) (string, error) {
|
|
||||||
if buildOptions.Squash && !bm.backend.HasExperimental() {
|
|
||||||
return "", apierrors.NewBadRequestError(errors.New("squash is only supported with experimental mode"))
|
|
||||||
}
|
|
||||||
if buildOptions.Dockerfile == "" {
|
|
||||||
buildOptions.Dockerfile = builder.DefaultDockerfileName
|
|
||||||
}
|
|
||||||
|
|
||||||
source, dockerfile, err := remotecontext.Detect(ctx, buildOptions.RemoteContext, buildOptions.Dockerfile, src, pg.ProgressReaderFunc)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
if source != nil {
|
|
||||||
defer func() {
|
|
||||||
if err := source.Close(); err != nil {
|
|
||||||
logrus.Debugf("[BUILDER] failed to remove temporary context: %v", err)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
b, err := NewBuilder(ctx, buildOptions, bm.backend, source)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
b.imageContexts.cache = bm.pathCache
|
|
||||||
return b.build(dockerfile, pg.StdoutFormatter, pg.StderrFormatter, pg.Output)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewBuilder creates a new Dockerfile builder from an optional dockerfile and a Config.
|
|
||||||
// If dockerfile is nil, the Dockerfile specified by Config.DockerfileName,
|
|
||||||
// will be read from the Context passed to Build().
|
|
||||||
func NewBuilder(clientCtx context.Context, config *types.ImageBuildOptions, backend builder.Backend, source builder.Source) (b *Builder, err error) {
|
|
||||||
if config == nil {
|
if config == nil {
|
||||||
config = new(types.ImageBuildOptions)
|
config = new(types.ImageBuildOptions)
|
||||||
}
|
}
|
||||||
b = &Builder{
|
b := &Builder{
|
||||||
clientCtx: clientCtx,
|
clientCtx: clientCtx,
|
||||||
options: config,
|
options: config,
|
||||||
Stdout: os.Stdout,
|
Stdout: options.ProgressWriter.StdoutFormatter,
|
||||||
Stderr: os.Stderr,
|
Stderr: options.ProgressWriter.StderrFormatter,
|
||||||
docker: backend,
|
Output: options.ProgressWriter.Output,
|
||||||
source: source,
|
docker: options.Backend,
|
||||||
runConfig: new(container.Config),
|
runConfig: new(container.Config),
|
||||||
tmpContainers: map[string]struct{}{},
|
tmpContainers: map[string]struct{}{},
|
||||||
buildArgs: newBuildArgs(config.BuildArgs),
|
buildArgs: newBuildArgs(config.BuildArgs),
|
||||||
}
|
}
|
||||||
b.imageContexts = &imageContexts{b: b}
|
b.imageContexts = &imageContexts{b: b, cache: options.PathCache}
|
||||||
return b, nil
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) resetImageCache() {
|
func (b *Builder) resetImageCache() {
|
||||||
|
@ -137,83 +140,31 @@ func (b *Builder) resetImageCache() {
|
||||||
b.cacheBusted = false
|
b.cacheBusted = false
|
||||||
}
|
}
|
||||||
|
|
||||||
// sanitizeRepoAndTags parses the raw "t" parameter received from the client
|
// Build runs the Dockerfile builder by parsing the Dockerfile and executing
|
||||||
// to a slice of repoAndTag.
|
// the instructions from the file.
|
||||||
// It also validates each repoName and tag.
|
func (b *Builder) build(source builder.Source, dockerfile *parser.Result) (*builder.Result, error) {
|
||||||
func sanitizeRepoAndTags(names []string) ([]reference.Named, error) {
|
|
||||||
var (
|
|
||||||
repoAndTags []reference.Named
|
|
||||||
// This map is used for deduplicating the "-t" parameter.
|
|
||||||
uniqNames = make(map[string]struct{})
|
|
||||||
)
|
|
||||||
for _, repo := range names {
|
|
||||||
if repo == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
ref, err := reference.ParseNormalizedNamed(repo)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if _, isCanonical := ref.(reference.Canonical); isCanonical {
|
|
||||||
return nil, errors.New("build tag cannot contain a digest")
|
|
||||||
}
|
|
||||||
|
|
||||||
ref = reference.TagNameOnly(ref)
|
|
||||||
|
|
||||||
nameWithTag := ref.String()
|
|
||||||
|
|
||||||
if _, exists := uniqNames[nameWithTag]; !exists {
|
|
||||||
uniqNames[nameWithTag] = struct{}{}
|
|
||||||
repoAndTags = append(repoAndTags, ref)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return repoAndTags, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// build runs the Dockerfile builder from a context and a docker object that allows to make calls
|
|
||||||
// to Docker.
|
|
||||||
func (b *Builder) build(dockerfile *parser.Result, stdout io.Writer, stderr io.Writer, out io.Writer) (string, error) {
|
|
||||||
defer b.imageContexts.unmount()
|
defer b.imageContexts.unmount()
|
||||||
|
|
||||||
b.Stdout = stdout
|
// TODO: Remove source field from Builder
|
||||||
b.Stderr = stderr
|
b.source = source
|
||||||
b.Output = out
|
|
||||||
|
|
||||||
repoAndTags, err := sanitizeRepoAndTags(b.options.Tags)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
addNodesForLabelOption(dockerfile.AST, b.options.Labels)
|
addNodesForLabelOption(dockerfile.AST, b.options.Labels)
|
||||||
|
|
||||||
if err := checkDispatchDockerfile(dockerfile.AST); err != nil {
|
if err := checkDispatchDockerfile(dockerfile.AST); err != nil {
|
||||||
return "", err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
imageID, err := b.dispatchDockerfileWithCancellation(dockerfile)
|
imageID, err := b.dispatchDockerfileWithCancellation(dockerfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
b.warnOnUnusedBuildArgs()
|
b.warnOnUnusedBuildArgs()
|
||||||
|
|
||||||
if imageID == "" {
|
if imageID == "" {
|
||||||
return "", errors.New("No image was generated. Is your Dockerfile empty?")
|
return nil, errors.New("No image was generated. Is your Dockerfile empty?")
|
||||||
}
|
}
|
||||||
|
return &builder.Result{ImageID: imageID, FromImage: b.from}, nil
|
||||||
if b.options.Squash {
|
|
||||||
if imageID, err = b.squashBuild(imageID); err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Fprintf(b.Stdout, "Successfully built %s\n", stringid.TruncateID(imageID))
|
|
||||||
if err := b.tagImages(imageID, repoAndTags); err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return imageID, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) dispatchDockerfileWithCancellation(dockerfile *parser.Result) (string, error) {
|
func (b *Builder) dispatchDockerfileWithCancellation(dockerfile *parser.Result) (string, error) {
|
||||||
|
@ -258,19 +209,6 @@ func (b *Builder) dispatchDockerfileWithCancellation(dockerfile *parser.Result)
|
||||||
return imageID, nil
|
return imageID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) squashBuild(imageID string) (string, error) {
|
|
||||||
var fromID string
|
|
||||||
var err error
|
|
||||||
if b.from != nil {
|
|
||||||
fromID = b.from.ImageID()
|
|
||||||
}
|
|
||||||
imageID, err = b.docker.SquashImage(imageID, fromID)
|
|
||||||
if err != nil {
|
|
||||||
return "", errors.Wrap(err, "error squashing image")
|
|
||||||
}
|
|
||||||
return imageID, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func addNodesForLabelOption(dockerfile *parser.Node, labels map[string]string) {
|
func addNodesForLabelOption(dockerfile *parser.Node, labels map[string]string) {
|
||||||
if len(labels) == 0 {
|
if len(labels) == 0 {
|
||||||
return
|
return
|
||||||
|
@ -289,17 +227,6 @@ func (b *Builder) warnOnUnusedBuildArgs() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) tagImages(id string, repoAndTags []reference.Named) error {
|
|
||||||
imageID := image.ID(id)
|
|
||||||
for _, rt := range repoAndTags {
|
|
||||||
if err := b.docker.TagImageWithReference(imageID, rt); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
fmt.Fprintf(b.Stdout, "Successfully tagged %s\n", reference.FamiliarString(rt))
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// hasFromImage returns true if the builder has processed a `FROM <image>` line
|
// hasFromImage returns true if the builder has processed a `FROM <image>` line
|
||||||
// TODO: move to DispatchState
|
// TODO: move to DispatchState
|
||||||
func (b *Builder) hasFromImage() bool {
|
func (b *Builder) hasFromImage() bool {
|
||||||
|
@ -316,10 +243,7 @@ func (b *Builder) hasFromImage() bool {
|
||||||
//
|
//
|
||||||
// TODO: Remove?
|
// TODO: Remove?
|
||||||
func BuildFromConfig(config *container.Config, changes []string) (*container.Config, error) {
|
func BuildFromConfig(config *container.Config, changes []string) (*container.Config, error) {
|
||||||
b, err := NewBuilder(context.Background(), nil, nil, nil)
|
b := newBuilder(context.Background(), builderOptions{})
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
result, err := parser.Parse(bytes.NewBufferString(strings.Join(changes, "\n")))
|
result, err := parser.Parse(bytes.NewBufferString(strings.Join(changes, "\n")))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -595,7 +595,7 @@ func healthcheck(req dispatchRequest) error {
|
||||||
// to /usr/sbin/nginx. Uses the default shell if not in JSON format.
|
// to /usr/sbin/nginx. Uses the default shell if not in JSON format.
|
||||||
//
|
//
|
||||||
// Handles command processing similar to CMD and RUN, only req.runConfig.Entrypoint
|
// Handles command processing similar to CMD and RUN, only req.runConfig.Entrypoint
|
||||||
// is initialized at NewBuilder time instead of through argument parsing.
|
// is initialized at newBuilder time instead of through argument parsing.
|
||||||
//
|
//
|
||||||
func entrypoint(req dispatchRequest) error {
|
func entrypoint(req dispatchRequest) error {
|
||||||
if err := req.flags.Parse(); err != nil {
|
if err := req.flags.Parse(); err != nil {
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
//
|
//
|
||||||
// It incorporates a dispatch table based on the parser.Node values (see the
|
// It incorporates a dispatch table based on the parser.Node values (see the
|
||||||
// parser package for more information) that are yielded from the parser itself.
|
// parser package for more information) that are yielded from the parser itself.
|
||||||
// Calling NewBuilder with the BuildOpts struct can be used to customize the
|
// Calling newBuilder with the BuildOpts struct can be used to customize the
|
||||||
// experience for execution purposes only. Parsing is controlled in the parser
|
// experience for execution purposes only. Parsing is controlled in the parser
|
||||||
// package, and this division of responsibility should be respected.
|
// package, and this division of responsibility should be respected.
|
||||||
//
|
//
|
||||||
|
|
|
@ -3,7 +3,6 @@ package dockerfile
|
||||||
import (
|
import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
|
||||||
|
|
||||||
"github.com/Sirupsen/logrus"
|
"github.com/Sirupsen/logrus"
|
||||||
"github.com/docker/docker/api/types/container"
|
"github.com/docker/docker/api/types/container"
|
||||||
|
@ -12,13 +11,18 @@ import (
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type pathCache interface {
|
||||||
|
Load(key interface{}) (value interface{}, ok bool)
|
||||||
|
Store(key, value interface{})
|
||||||
|
}
|
||||||
|
|
||||||
// imageContexts is a helper for stacking up built image rootfs and reusing
|
// imageContexts is a helper for stacking up built image rootfs and reusing
|
||||||
// them as contexts
|
// them as contexts
|
||||||
type imageContexts struct {
|
type imageContexts struct {
|
||||||
b *Builder
|
b *Builder
|
||||||
list []*imageMount
|
list []*imageMount
|
||||||
byName map[string]*imageMount
|
byName map[string]*imageMount
|
||||||
cache *pathCache
|
cache pathCache
|
||||||
currentName string
|
currentName string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,14 +108,14 @@ func (ic *imageContexts) getCache(id, path string) (interface{}, bool) {
|
||||||
if id == "" {
|
if id == "" {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
return ic.cache.get(id + path)
|
return ic.cache.Load(id + path)
|
||||||
}
|
}
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ic *imageContexts) setCache(id, path string, v interface{}) {
|
func (ic *imageContexts) setCache(id, path string, v interface{}) {
|
||||||
if ic.cache != nil {
|
if ic.cache != nil {
|
||||||
ic.cache.set(id+path, v)
|
ic.cache.Store(id+path, v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -160,28 +164,3 @@ func (im *imageMount) ImageID() string {
|
||||||
func (im *imageMount) RunConfig() *container.Config {
|
func (im *imageMount) RunConfig() *container.Config {
|
||||||
return im.runConfig
|
return im.runConfig
|
||||||
}
|
}
|
||||||
|
|
||||||
type pathCache struct {
|
|
||||||
mu sync.Mutex
|
|
||||||
items map[string]interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *pathCache) set(k string, v interface{}) {
|
|
||||||
c.mu.Lock()
|
|
||||||
if c.items == nil {
|
|
||||||
c.items = make(map[string]interface{})
|
|
||||||
}
|
|
||||||
c.items[k] = v
|
|
||||||
c.mu.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *pathCache) get(k string) (interface{}, bool) {
|
|
||||||
c.mu.Lock()
|
|
||||||
if c.items == nil {
|
|
||||||
c.mu.Unlock()
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
v, ok := c.items[k]
|
|
||||||
c.mu.Unlock()
|
|
||||||
return v, ok
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
package dockerfile
|
package dockerfile
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/docker/docker/api/types/backend"
|
||||||
"github.com/docker/docker/api/types/container"
|
"github.com/docker/docker/api/types/container"
|
||||||
"github.com/docker/docker/builder"
|
"github.com/docker/docker/builder"
|
||||||
"github.com/docker/docker/builder/remotecontext"
|
"github.com/docker/docker/builder/remotecontext"
|
||||||
|
@ -69,7 +70,11 @@ func readAndCheckDockerfile(t *testing.T, testName, contextDir, dockerfilePath,
|
||||||
dockerfilePath = builder.DefaultDockerfileName
|
dockerfilePath = builder.DefaultDockerfileName
|
||||||
}
|
}
|
||||||
|
|
||||||
_, _, err = remotecontext.Detect(context.Background(), "", dockerfilePath, tarStream, nil)
|
config := backend.BuildConfig{
|
||||||
|
Options: &types.ImageBuildOptions{Dockerfile: dockerfilePath},
|
||||||
|
Source: tarStream,
|
||||||
|
}
|
||||||
|
_, _, err = remotecontext.Detect(config)
|
||||||
assert.EqualError(t, err, expectedError)
|
assert.EqualError(t, err, expectedError)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,6 @@ package remotecontext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"context"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
|
@ -10,6 +9,7 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/Sirupsen/logrus"
|
"github.com/Sirupsen/logrus"
|
||||||
|
"github.com/docker/docker/api/types/backend"
|
||||||
"github.com/docker/docker/builder"
|
"github.com/docker/docker/builder"
|
||||||
"github.com/docker/docker/builder/dockerfile/parser"
|
"github.com/docker/docker/builder/dockerfile/parser"
|
||||||
"github.com/docker/docker/builder/dockerignore"
|
"github.com/docker/docker/builder/dockerignore"
|
||||||
|
@ -23,14 +23,17 @@ import (
|
||||||
// Detect returns a context and dockerfile from remote location or local
|
// Detect returns a context and dockerfile from remote location or local
|
||||||
// archive. progressReader is only used if remoteURL is actually a URL
|
// archive. progressReader is only used if remoteURL is actually a URL
|
||||||
// (not empty, and not a Git endpoint).
|
// (not empty, and not a Git endpoint).
|
||||||
func Detect(ctx context.Context, remoteURL string, dockerfilePath string, r io.ReadCloser, progressReader func(in io.ReadCloser) io.ReadCloser) (remote builder.Source, dockerfile *parser.Result, err error) {
|
func Detect(config backend.BuildConfig) (remote builder.Source, dockerfile *parser.Result, err error) {
|
||||||
|
remoteURL := config.Options.RemoteContext
|
||||||
|
dockerfilePath := config.Options.Dockerfile
|
||||||
|
|
||||||
switch {
|
switch {
|
||||||
case remoteURL == "":
|
case remoteURL == "":
|
||||||
remote, dockerfile, err = newArchiveRemote(r, dockerfilePath)
|
remote, dockerfile, err = newArchiveRemote(config.Source, dockerfilePath)
|
||||||
case urlutil.IsGitURL(remoteURL):
|
case urlutil.IsGitURL(remoteURL):
|
||||||
remote, dockerfile, err = newGitRemote(remoteURL, dockerfilePath)
|
remote, dockerfile, err = newGitRemote(remoteURL, dockerfilePath)
|
||||||
case urlutil.IsURL(remoteURL):
|
case urlutil.IsURL(remoteURL):
|
||||||
remote, dockerfile, err = newURLRemote(remoteURL, dockerfilePath, progressReader)
|
remote, dockerfile, err = newURLRemote(remoteURL, dockerfilePath, config.ProgressWriter.ProgressReaderFunc)
|
||||||
default:
|
default:
|
||||||
err = fmt.Errorf("remoteURL (%s) could not be recognized as URL", remoteURL)
|
err = fmt.Errorf("remoteURL (%s) could not be recognized as URL", remoteURL)
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,6 +14,7 @@ import (
|
||||||
"github.com/docker/distribution/uuid"
|
"github.com/docker/distribution/uuid"
|
||||||
"github.com/docker/docker/api"
|
"github.com/docker/docker/api"
|
||||||
apiserver "github.com/docker/docker/api/server"
|
apiserver "github.com/docker/docker/api/server"
|
||||||
|
buildbackend "github.com/docker/docker/api/server/backend/build"
|
||||||
"github.com/docker/docker/api/server/middleware"
|
"github.com/docker/docker/api/server/middleware"
|
||||||
"github.com/docker/docker/api/server/router"
|
"github.com/docker/docker/api/server/router"
|
||||||
"github.com/docker/docker/api/server/router/build"
|
"github.com/docker/docker/api/server/router/build"
|
||||||
|
@ -25,7 +26,6 @@ import (
|
||||||
swarmrouter "github.com/docker/docker/api/server/router/swarm"
|
swarmrouter "github.com/docker/docker/api/server/router/swarm"
|
||||||
systemrouter "github.com/docker/docker/api/server/router/system"
|
systemrouter "github.com/docker/docker/api/server/router/system"
|
||||||
"github.com/docker/docker/api/server/router/volume"
|
"github.com/docker/docker/api/server/router/volume"
|
||||||
"github.com/docker/docker/builder/dockerfile"
|
|
||||||
cliconfig "github.com/docker/docker/cli/config"
|
cliconfig "github.com/docker/docker/cli/config"
|
||||||
"github.com/docker/docker/cli/debug"
|
"github.com/docker/docker/cli/debug"
|
||||||
cliflags "github.com/docker/docker/cli/flags"
|
cliflags "github.com/docker/docker/cli/flags"
|
||||||
|
@ -484,7 +484,7 @@ func initRouter(s *apiserver.Server, d *daemon.Daemon, c *cluster.Cluster) {
|
||||||
image.NewRouter(d, decoder),
|
image.NewRouter(d, decoder),
|
||||||
systemrouter.NewRouter(d, c),
|
systemrouter.NewRouter(d, c),
|
||||||
volume.NewRouter(d),
|
volume.NewRouter(d),
|
||||||
build.NewRouter(dockerfile.NewBuildManager(d)),
|
build.NewRouter(buildbackend.NewBackend(d, d), d),
|
||||||
swarmrouter.NewRouter(c),
|
swarmrouter.NewRouter(c),
|
||||||
pluginrouter.NewRouter(d.PluginManager()),
|
pluginrouter.NewRouter(d.PluginManager()),
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue