mirror of
https://github.com/moby/moby.git
synced 2026-01-11 18:51:37 +00:00
93 lines
2.3 KiB
Go
93 lines
2.3 KiB
Go
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/go-archive/compression"
|
|
"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: compression.None,
|
|
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
|
|
}
|