package libpod import ( "context" "fmt" "strings" is "github.com/containers/image/storage" "github.com/pkg/errors" "github.com/projectatomic/buildah" "github.com/projectatomic/libpod/libpod/image" "github.com/sirupsen/logrus" ) // ContainerCommitOptions is a struct used to commit a container to an image // It uses buildah's CommitOptions as a base. Long-term we might wish to // add these to the buildah struct once buildah is more integrated with //libpod type ContainerCommitOptions struct { buildah.CommitOptions Pause bool Author string Message string Changes []string } // Commit commits the changes between a container and its image, creating a new // image func (c *Container) Commit(ctx context.Context, destImage string, options ContainerCommitOptions) (*image.Image, error) { if !c.batched { c.lock.Lock() defer c.lock.Unlock() if err := c.syncContainer(); err != nil { return nil, err } } if c.state.State == ContainerStateRunning && options.Pause { if err := c.runtime.ociRuntime.pauseContainer(c); err != nil { return nil, errors.Wrapf(err, "error pausing container %q", c.ID()) } defer func() { if err := c.runtime.ociRuntime.unpauseContainer(c); err != nil { logrus.Errorf("error unpausing container %q: %v", c.ID(), err) } }() } sc := image.GetSystemContext(options.SignaturePolicyPath, "", false) builderOptions := buildah.ImportOptions{ Container: c.ID(), SignaturePolicyPath: options.SignaturePolicyPath, } commitOptions := buildah.CommitOptions{ SignaturePolicyPath: options.SignaturePolicyPath, ReportWriter: options.ReportWriter, SystemContext: sc, PreferredManifestType: options.PreferredManifestType, } importBuilder, err := buildah.ImportBuilder(ctx, c.runtime.store, builderOptions) if err != nil { return nil, err } if options.Author != "" { importBuilder.SetMaintainer(options.Author) } if options.Message != "" { importBuilder.SetComment(options.Message) } // We need to take meta we find in the current container and // add it to the resulting image. // Entrypoint - always set this first or cmd will get wiped out importBuilder.SetEntrypoint(c.Spec().Process.Args) // Cmd // We cannot differentiate between cmd and entrypoint here // so we assign args to both importBuilder.SetCmd(c.Spec().Process.Args) // Env for _, e := range c.config.Spec.Process.Env { splitEnv := strings.Split(e, "=") importBuilder.SetEnv(splitEnv[0], splitEnv[1]) } // Expose ports for _, p := range c.config.PortMappings { importBuilder.SetPort(string(p.ContainerPort)) } // Labels for k, v := range c.Labels() { importBuilder.SetLabel(k, v) } // No stop signal // User importBuilder.SetUser(c.User()) // Volumes for _, v := range c.config.Spec.Mounts { importBuilder.AddVolume(v.Source) } // Workdir importBuilder.SetWorkDir(c.Spec().Process.Cwd) // Process user changes for _, change := range options.Changes { splitChange := strings.Split(change, "=") switch strings.ToUpper(splitChange[0]) { case "CMD": importBuilder.SetCmd(splitChange[1:]) case "ENTRYPOINT": importBuilder.SetEntrypoint(splitChange[1:]) case "ENV": importBuilder.ClearEnv() importBuilder.SetEnv(splitChange[1], splitChange[2]) case "EXPOSE": importBuilder.ClearPorts() importBuilder.SetPort(splitChange[1]) case "LABEL": importBuilder.ClearLabels() importBuilder.SetLabel(splitChange[1], splitChange[2]) case "STOPSIGNAL": // No Set StopSignal case "USER": importBuilder.SetUser(splitChange[1]) case "VOLUME": importBuilder.ClearVolumes() importBuilder.AddVolume(splitChange[1]) case "WORKDIR": importBuilder.SetWorkDir(splitChange[1]) } } imageRef, err := is.Transport.ParseStoreReference(c.runtime.store, destImage) if err != nil { return nil, err } if err = importBuilder.Commit(ctx, imageRef, commitOptions); err != nil { return nil, err } fmt.Println(importBuilder.Comment()) return c.runtime.imageRuntime.NewFromLocal(imageRef.DockerReference().String()) }