273 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			273 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
   Copyright The containerd Authors.
 | 
						|
 | 
						|
   Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
   you may not use this file except in compliance with the License.
 | 
						|
   You may obtain a copy of the License at
 | 
						|
 | 
						|
       http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 | 
						|
   Unless required by applicable law or agreed to in writing, software
 | 
						|
   distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
   See the License for the specific language governing permissions and
 | 
						|
   limitations under the License.
 | 
						|
*/
 | 
						|
 | 
						|
package run
 | 
						|
 | 
						|
import (
 | 
						|
	gocontext "context"
 | 
						|
	"encoding/csv"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/containerd/console"
 | 
						|
	"github.com/containerd/containerd"
 | 
						|
	"github.com/containerd/containerd/cio"
 | 
						|
	"github.com/containerd/containerd/cmd/ctr/commands"
 | 
						|
	"github.com/containerd/containerd/cmd/ctr/commands/tasks"
 | 
						|
	"github.com/containerd/containerd/containers"
 | 
						|
	clabels "github.com/containerd/containerd/labels"
 | 
						|
	"github.com/containerd/containerd/oci"
 | 
						|
	gocni "github.com/containerd/go-cni"
 | 
						|
	specs "github.com/opencontainers/runtime-spec/specs-go"
 | 
						|
	"github.com/sirupsen/logrus"
 | 
						|
	"github.com/urfave/cli"
 | 
						|
)
 | 
						|
 | 
						|
func withMounts(context *cli.Context) oci.SpecOpts {
 | 
						|
	return func(ctx gocontext.Context, client oci.Client, container *containers.Container, s *specs.Spec) error {
 | 
						|
		mounts := make([]specs.Mount, 0)
 | 
						|
		for _, mount := range context.StringSlice("mount") {
 | 
						|
			m, err := parseMountFlag(mount)
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
			mounts = append(mounts, m)
 | 
						|
		}
 | 
						|
		return oci.WithMounts(mounts)(ctx, client, container, s)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// parseMountFlag parses a mount string in the form "type=foo,source=/path,destination=/target,options=rbind:rw"
 | 
						|
func parseMountFlag(m string) (specs.Mount, error) {
 | 
						|
	mount := specs.Mount{}
 | 
						|
	r := csv.NewReader(strings.NewReader(m))
 | 
						|
 | 
						|
	fields, err := r.Read()
 | 
						|
	if err != nil {
 | 
						|
		return mount, err
 | 
						|
	}
 | 
						|
 | 
						|
	for _, field := range fields {
 | 
						|
		key, val, ok := strings.Cut(field, "=")
 | 
						|
		if !ok {
 | 
						|
			return mount, fmt.Errorf("invalid mount specification: expected key=val")
 | 
						|
		}
 | 
						|
 | 
						|
		switch key {
 | 
						|
		case "type":
 | 
						|
			mount.Type = val
 | 
						|
		case "source", "src":
 | 
						|
			mount.Source = val
 | 
						|
		case "destination", "dst":
 | 
						|
			mount.Destination = val
 | 
						|
		case "options":
 | 
						|
			mount.Options = strings.Split(val, ":")
 | 
						|
		default:
 | 
						|
			return mount, fmt.Errorf("mount option %q not supported", key)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return mount, nil
 | 
						|
}
 | 
						|
 | 
						|
// Command runs a container
 | 
						|
var Command = cli.Command{
 | 
						|
	Name:           "run",
 | 
						|
	Usage:          "run a container",
 | 
						|
	ArgsUsage:      "[flags] Image|RootFS ID [COMMAND] [ARG...]",
 | 
						|
	SkipArgReorder: true,
 | 
						|
	Flags: append([]cli.Flag{
 | 
						|
		cli.BoolFlag{
 | 
						|
			Name:  "rm",
 | 
						|
			Usage: "remove the container after running, cannot be used with --detach",
 | 
						|
		},
 | 
						|
		cli.BoolFlag{
 | 
						|
			Name:  "null-io",
 | 
						|
			Usage: "send all IO to /dev/null",
 | 
						|
		},
 | 
						|
		cli.StringFlag{
 | 
						|
			Name:  "log-uri",
 | 
						|
			Usage: "log uri",
 | 
						|
		},
 | 
						|
		cli.BoolFlag{
 | 
						|
			Name:  "detach,d",
 | 
						|
			Usage: "detach from the task after it has started execution, cannot be used with --rm",
 | 
						|
		},
 | 
						|
		cli.StringFlag{
 | 
						|
			Name:  "fifo-dir",
 | 
						|
			Usage: "directory used for storing IO FIFOs",
 | 
						|
		},
 | 
						|
		cli.StringFlag{
 | 
						|
			Name:  "cgroup",
 | 
						|
			Usage: "cgroup path (To disable use of cgroup, set to \"\" explicitly)",
 | 
						|
		},
 | 
						|
		cli.StringFlag{
 | 
						|
			Name:  "platform",
 | 
						|
			Usage: "run image for specific platform",
 | 
						|
		},
 | 
						|
		cli.BoolFlag{
 | 
						|
			Name:  "cni",
 | 
						|
			Usage: "enable cni networking for the container",
 | 
						|
		},
 | 
						|
	}, append(platformRunFlags,
 | 
						|
		append(append(commands.SnapshotterFlags, []cli.Flag{commands.SnapshotterLabels}...),
 | 
						|
			commands.ContainerFlags...)...)...),
 | 
						|
	Action: func(context *cli.Context) error {
 | 
						|
		var (
 | 
						|
			err error
 | 
						|
			id  string
 | 
						|
			ref string
 | 
						|
 | 
						|
			rm        = context.Bool("rm")
 | 
						|
			tty       = context.Bool("tty")
 | 
						|
			detach    = context.Bool("detach")
 | 
						|
			config    = context.IsSet("config")
 | 
						|
			enableCNI = context.Bool("cni")
 | 
						|
		)
 | 
						|
 | 
						|
		if config {
 | 
						|
			id = context.Args().First()
 | 
						|
			if context.NArg() > 1 {
 | 
						|
				return errors.New("with spec config file, only container id should be provided")
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			id = context.Args().Get(1)
 | 
						|
			ref = context.Args().First()
 | 
						|
 | 
						|
			if ref == "" {
 | 
						|
				return errors.New("image ref must be provided")
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if id == "" {
 | 
						|
			return errors.New("container id must be provided")
 | 
						|
		}
 | 
						|
		if rm && detach {
 | 
						|
			return errors.New("flags --detach and --rm cannot be specified together")
 | 
						|
		}
 | 
						|
 | 
						|
		client, ctx, cancel, err := commands.NewClient(context)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		defer cancel()
 | 
						|
		container, err := NewContainer(ctx, client, context)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if rm && !detach {
 | 
						|
			defer container.Delete(ctx, containerd.WithSnapshotCleanup)
 | 
						|
		}
 | 
						|
		var con console.Console
 | 
						|
		if tty {
 | 
						|
			con = console.Current()
 | 
						|
			defer con.Reset()
 | 
						|
			if err := con.SetRaw(); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
		var network gocni.CNI
 | 
						|
		if enableCNI {
 | 
						|
			if network, err = gocni.New(gocni.WithDefaultConf); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		opts := getNewTaskOpts(context)
 | 
						|
		ioOpts := []cio.Opt{cio.WithFIFODir(context.String("fifo-dir"))}
 | 
						|
		task, err := tasks.NewTask(ctx, client, container, context.String("checkpoint"), con, context.Bool("null-io"), context.String("log-uri"), ioOpts, opts...)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		var statusC <-chan containerd.ExitStatus
 | 
						|
		if !detach {
 | 
						|
			defer func() {
 | 
						|
				if enableCNI {
 | 
						|
					if err := network.Remove(ctx, commands.FullID(ctx, container), ""); err != nil {
 | 
						|
						logrus.WithError(err).Error("network review")
 | 
						|
					}
 | 
						|
				}
 | 
						|
				task.Delete(ctx)
 | 
						|
			}()
 | 
						|
 | 
						|
			if statusC, err = task.Wait(ctx); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if context.IsSet("pid-file") {
 | 
						|
			if err := commands.WritePidFile(context.String("pid-file"), int(task.Pid())); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if enableCNI {
 | 
						|
			netNsPath, err := getNetNSPath(ctx, task)
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
 | 
						|
			if _, err := network.Setup(ctx, commands.FullID(ctx, container), netNsPath); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if err := task.Start(ctx); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if detach {
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		if tty {
 | 
						|
			if err := tasks.HandleConsoleResize(ctx, task, con); err != nil {
 | 
						|
				logrus.WithError(err).Error("console resize")
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			sigc := commands.ForwardAllSignals(ctx, task)
 | 
						|
			defer commands.StopCatch(sigc)
 | 
						|
		}
 | 
						|
		status := <-statusC
 | 
						|
		code, _, err := status.Result()
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if _, err := task.Delete(ctx); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if code != 0 {
 | 
						|
			return cli.NewExitError("", int(code))
 | 
						|
		}
 | 
						|
		return nil
 | 
						|
	},
 | 
						|
}
 | 
						|
 | 
						|
// buildLabel builds the labels from command line labels and the image labels
 | 
						|
func buildLabels(cmdLabels, imageLabels map[string]string) map[string]string {
 | 
						|
	labels := make(map[string]string)
 | 
						|
	for k, v := range imageLabels {
 | 
						|
		if err := clabels.Validate(k, v); err == nil {
 | 
						|
			labels[k] = v
 | 
						|
		} else {
 | 
						|
			// In case the image label is invalid, we output a warning and skip adding it to the
 | 
						|
			// container.
 | 
						|
			logrus.WithError(err).Warnf("unable to add image label with key %s to the container", k)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	// labels from the command line will override image and the initial image config labels
 | 
						|
	for k, v := range cmdLabels {
 | 
						|
		labels[k] = v
 | 
						|
	}
 | 
						|
	return labels
 | 
						|
}
 |