| package daemon |
| |
| import ( |
| "context" |
| "errors" |
| "fmt" |
| "io" |
| |
| "github.com/containerd/log" |
| "github.com/moby/go-archive" |
| "github.com/moby/go-archive/chrootarchive" |
| "github.com/moby/moby/api/types/events" |
| "github.com/moby/moby/v2/daemon/container" |
| "github.com/moby/moby/v2/errdefs" |
| ) |
| |
| // ContainerExport writes the contents of the container to the given |
| // writer. An error is returned if the container cannot be found. |
| func (daemon *Daemon) ContainerExport(ctx context.Context, name string, out io.Writer) error { |
| ctr, err := daemon.GetContainer(name) |
| if err != nil { |
| return err |
| } |
| |
| if isWindows && ctr.ImagePlatform.OS == "windows" { |
| return errors.New("the daemon on this operating system does not support exporting Windows containers") |
| } |
| |
| if ctr.State.IsDead() { |
| err := fmt.Errorf("You cannot export container %s which is Dead", ctr.ID) |
| return errdefs.Conflict(err) |
| } |
| |
| if ctr.State.IsRemovalInProgress() { |
| err := fmt.Errorf("You cannot export container %s which is being removed", ctr.ID) |
| return errdefs.Conflict(err) |
| } |
| |
| err = daemon.containerExport(ctx, ctr, out) |
| if err != nil { |
| return fmt.Errorf("Error exporting container %s: %v", name, err) |
| } |
| |
| return nil |
| } |
| |
| func (daemon *Daemon) containerExport(ctx context.Context, ctr *container.Container, out io.Writer) error { |
| rwl := ctr.RWLayer |
| if rwl == nil { |
| return fmt.Errorf("container %s has no rootfs", ctr.ID) |
| } |
| |
| if err := ctx.Err(); err != nil { |
| return err |
| } |
| |
| basefs, err := rwl.Mount(ctr.GetMountLabel()) |
| if err != nil { |
| return err |
| } |
| defer func() { |
| if err := rwl.Unmount(); err != nil { |
| log.G(ctx).WithFields(log.Fields{"error": err, "container": ctr.ID}).Warn("Failed to unmount container RWLayer after export") |
| } |
| }() |
| |
| archv, err := chrootarchive.Tar(basefs, &archive.TarOptions{ |
| Compression: archive.Uncompressed, |
| IDMap: daemon.idMapping, |
| }, basefs) |
| if err != nil { |
| return err |
| } |
| |
| ctx, cancel := context.WithCancel(ctx) |
| defer cancel() |
| context.AfterFunc(ctx, func() { |
| _ = archv.Close() |
| }) |
| |
| // Stream the entire contents of the container (basically a volatile snapshot) |
| if _, err := io.Copy(out, archv); err != nil { |
| if err := ctx.Err(); err != nil { |
| return errdefs.Cancelled(err) |
| } |
| return err |
| } |
| |
| daemon.LogContainerEvent(ctr, events.ActionExport) |
| return nil |
| } |