Merge pull request #5641 from baude/v2commit

podmanv2 commit
This commit is contained in:
OpenShift Merge Robot
2020-03-29 13:22:25 +02:00
committed by GitHub
5 changed files with 183 additions and 0 deletions

View File

@ -0,0 +1,79 @@
package containers
import (
"context"
"fmt"
"io/ioutil"
"os"
"strings"
"github.com/containers/libpod/cmd/podmanV2/registry"
"github.com/containers/libpod/pkg/domain/entities"
"github.com/pkg/errors"
"github.com/spf13/cobra"
)
var (
commitDescription = `Create an image from a container's changes. Optionally tag the image created, set the author with the --author flag, set the commit message with the --message flag, and make changes to the instructions with the --change flag.`
commitCommand = &cobra.Command{
Use: "commit [flags] CONTAINER [IMAGE]",
Short: "Create new image based on the changed container",
Long: commitDescription,
RunE: commit,
PreRunE: preRunE,
Args: cobra.MinimumNArgs(1),
Example: `podman commit -q --message "committing container to image" reverent_golick image-committed
podman commit -q --author "firstName lastName" reverent_golick image-committed
podman commit -q --pause=false containerID image-committed
podman commit containerID`,
}
// ChangeCmds is the list of valid Changes commands to passed to the Commit call
ChangeCmds = []string{"CMD", "ENTRYPOINT", "ENV", "EXPOSE", "LABEL", "ONBUILD", "STOPSIGNAL", "USER", "VOLUME", "WORKDIR"}
)
var (
commitOptions = entities.CommitOptions{
ImageName: "",
}
iidFile string
)
func init() {
registry.Commands = append(registry.Commands, registry.CliCommand{
Mode: []entities.EngineMode{entities.ABIMode, entities.TunnelMode},
Command: commitCommand,
})
flags := commitCommand.Flags()
flags.StringArrayVarP(&commitOptions.Changes, "change", "c", []string{}, "Apply the following possible instructions to the created image (default []): "+strings.Join(ChangeCmds, " | "))
flags.StringVarP(&commitOptions.Format, "format", "f", "oci", "`Format` of the image manifest and metadata")
flags.StringVarP(&iidFile, "iidfile", "", "", "`file` to write the image ID to")
flags.StringVarP(&commitOptions.Message, "message", "m", "", "Set commit message for imported image")
flags.StringVarP(&commitOptions.Author, "author", "a", "", "Set the author for the image committed")
flags.BoolVarP(&commitOptions.Pause, "pause", "p", false, "Pause container during commit")
flags.BoolVarP(&commitOptions.Quiet, "quiet", "q", false, "Suppress output")
flags.BoolVar(&commitOptions.IncludeVolumes, "include-volumes", false, "Include container volumes as image volumes")
}
func commit(cmd *cobra.Command, args []string) error {
container := args[0]
if len(args) > 1 {
commitOptions.ImageName = args[1]
}
if !commitOptions.Quiet {
commitOptions.Writer = os.Stderr
}
response, err := registry.ContainerEngine().ContainerCommit(context.Background(), container, commitOptions)
if err != nil {
return err
}
if len(iidFile) > 0 {
if err = ioutil.WriteFile(iidFile, []byte(response.Id), 0644); err != nil {
return errors.Wrapf(err, "failed to write image ID to file %q", iidFile)
}
}
fmt.Println(response.Id)
return nil
}

View File

@ -1,6 +1,7 @@
package entities package entities
import ( import (
"io"
"time" "time"
"github.com/containers/libpod/libpod/define" "github.com/containers/libpod/libpod/define"
@ -106,3 +107,19 @@ type ContainerInspectOptions struct {
type ContainerInspectReport struct { type ContainerInspectReport struct {
*define.InspectContainerData *define.InspectContainerData
} }
type CommitOptions struct {
Author string
Changes []string
Format string
ImageName string
IncludeVolumes bool
Message string
Pause bool
Quiet bool
Writer io.Writer
}
type CommitReport struct {
Id string
}

View File

@ -5,6 +5,7 @@ import (
) )
type ContainerEngine interface { type ContainerEngine interface {
ContainerCommit(ctx context.Context, nameOrId string, options CommitOptions) (*CommitReport, error)
ContainerExists(ctx context.Context, nameOrId string) (*BoolReport, error) ContainerExists(ctx context.Context, nameOrId string) (*BoolReport, error)
ContainerInspect(ctx context.Context, namesOrIds []string, options ContainerInspectOptions) ([]*ContainerInspectReport, error) ContainerInspect(ctx context.Context, namesOrIds []string, options ContainerInspectOptions) ([]*ContainerInspectReport, error)
ContainerKill(ctx context.Context, namesOrIds []string, options KillOptions) ([]*KillReport, error) ContainerKill(ctx context.Context, namesOrIds []string, options KillOptions) ([]*KillReport, error)

View File

@ -7,8 +7,11 @@ import (
"io/ioutil" "io/ioutil"
"strings" "strings"
"github.com/containers/buildah"
"github.com/containers/image/v5/manifest"
"github.com/containers/libpod/libpod" "github.com/containers/libpod/libpod"
"github.com/containers/libpod/libpod/define" "github.com/containers/libpod/libpod/define"
"github.com/containers/libpod/libpod/image"
"github.com/containers/libpod/pkg/adapter/shortcuts" "github.com/containers/libpod/pkg/adapter/shortcuts"
"github.com/containers/libpod/pkg/domain/entities" "github.com/containers/libpod/pkg/domain/entities"
"github.com/containers/libpod/pkg/signal" "github.com/containers/libpod/pkg/signal"
@ -277,3 +280,48 @@ func (ic *ContainerEngine) ContainerTop(ctx context.Context, options entities.To
report.Value, err = container.Top(options.Descriptors) report.Value, err = container.Top(options.Descriptors)
return report, err return report, err
} }
func (ic *ContainerEngine) ContainerCommit(ctx context.Context, nameOrId string, options entities.CommitOptions) (*entities.CommitReport, error) {
var (
mimeType string
)
ctr, err := ic.Libpod.LookupContainer(nameOrId)
if err != nil {
return nil, err
}
rtc, err := ic.Libpod.GetConfig()
if err != nil {
return nil, err
}
switch options.Format {
case "oci":
mimeType = buildah.OCIv1ImageManifest
if len(options.Message) > 0 {
return nil, errors.Errorf("messages are only compatible with the docker image format (-f docker)")
}
case "docker":
mimeType = manifest.DockerV2Schema2MediaType
default:
return nil, errors.Errorf("unrecognized image format %q", options.Format)
}
sc := image.GetSystemContext(rtc.Engine.SignaturePolicyPath, "", false)
coptions := buildah.CommitOptions{
SignaturePolicyPath: rtc.Engine.SignaturePolicyPath,
ReportWriter: options.Writer,
SystemContext: sc,
PreferredManifestType: mimeType,
}
opts := libpod.ContainerCommitOptions{
CommitOptions: coptions,
Pause: options.Pause,
IncludeVolumes: options.IncludeVolumes,
Message: options.Message,
Changes: options.Changes,
Author: options.Author,
}
newImage, err := ctr.Commit(ctx, options.ImageName, opts)
if err != nil {
return nil, err
}
return &entities.CommitReport{Id: newImage.ID()}, nil
}

View File

@ -3,6 +3,8 @@ package tunnel
import ( import (
"context" "context"
"github.com/containers/image/v5/docker/reference"
"github.com/containers/libpod/pkg/bindings/containers" "github.com/containers/libpod/pkg/bindings/containers"
"github.com/containers/libpod/pkg/domain/entities" "github.com/containers/libpod/pkg/domain/entities"
"github.com/pkg/errors" "github.com/pkg/errors"
@ -172,3 +174,39 @@ func (ic *ContainerEngine) ContainerTop(ctx context.Context, options entities.To
} }
return &entities.StringSliceReport{Value: topOutput}, nil return &entities.StringSliceReport{Value: topOutput}, nil
} }
func (ic *ContainerEngine) ContainerCommit(ctx context.Context, nameOrId string, options entities.CommitOptions) (*entities.CommitReport, error) {
var (
repo string
tag string = "latest"
)
if len(options.ImageName) > 0 {
ref, err := reference.Parse(options.ImageName)
if err != nil {
return nil, err
}
if t, ok := ref.(reference.Tagged); ok {
tag = t.Tag()
}
if r, ok := ref.(reference.Named); ok {
repo = r.Name()
}
if len(repo) < 1 {
return nil, errors.Errorf("invalid image name %q", options.ImageName)
}
}
commitOpts := containers.CommitOptions{
Author: &options.Author,
Changes: options.Changes,
Comment: &options.Message,
Format: &options.Format,
Pause: &options.Pause,
Repo: &repo,
Tag: &tag,
}
response, err := containers.Commit(ic.ClientCxt, nameOrId, commitOpts)
if err != nil {
return nil, err
}
return &entities.CommitReport{Id: response.ID}, nil
}