2015-12-02 22:41:49 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"text/tabwriter"
|
|
|
|
|
|
|
|
"github.com/codegangsta/cli"
|
2015-12-09 23:37:04 +00:00
|
|
|
"github.com/docker/containerd/api/grpc/types"
|
|
|
|
netcontext "golang.org/x/net/context"
|
|
|
|
"google.golang.org/grpc"
|
2015-12-02 22:41:49 +00:00
|
|
|
)
|
|
|
|
|
2015-12-09 23:37:04 +00:00
|
|
|
// TODO: parse flags and pass opts
|
|
|
|
func getClient() types.APIClient {
|
|
|
|
conn, err := grpc.Dial("localhost:8888", grpc.WithInsecure())
|
|
|
|
if err != nil {
|
|
|
|
fatal(err.Error(), 1)
|
|
|
|
}
|
|
|
|
return types.NewAPIClient(conn)
|
|
|
|
}
|
|
|
|
|
2015-12-02 22:41:49 +00:00
|
|
|
var ContainersCommand = cli.Command{
|
|
|
|
Name: "containers",
|
|
|
|
Usage: "interact with running containers",
|
|
|
|
Subcommands: []cli.Command{
|
|
|
|
StartCommand,
|
|
|
|
ListCommand,
|
|
|
|
KillCommand,
|
2015-12-10 19:03:36 +00:00
|
|
|
ExecCommand,
|
2015-12-02 22:41:49 +00:00
|
|
|
},
|
|
|
|
Action: listContainers,
|
|
|
|
}
|
|
|
|
|
|
|
|
var ListCommand = cli.Command{
|
|
|
|
Name: "list",
|
|
|
|
Usage: "list all running containers",
|
|
|
|
Action: listContainers,
|
|
|
|
}
|
|
|
|
|
|
|
|
func listContainers(context *cli.Context) {
|
2015-12-10 19:03:36 +00:00
|
|
|
c := getClient()
|
|
|
|
resp, err := c.State(netcontext.Background(), &types.StateRequest{})
|
2015-12-02 22:41:49 +00:00
|
|
|
if err != nil {
|
|
|
|
fatal(err.Error(), 1)
|
|
|
|
}
|
|
|
|
w := tabwriter.NewWriter(os.Stdout, 20, 1, 3, ' ', 0)
|
2015-12-09 23:53:23 +00:00
|
|
|
fmt.Fprint(w, "ID\tPATH\tSTATUS\tPID1\n")
|
2015-12-09 23:37:04 +00:00
|
|
|
for _, c := range resp.Containers {
|
2015-12-10 01:03:53 +00:00
|
|
|
fmt.Fprintf(w, "%s\t%s\t%s\t%d\n", c.Id, c.BundlePath, c.Status, c.Processes[0].Pid)
|
2015-12-02 22:41:49 +00:00
|
|
|
}
|
|
|
|
if err := w.Flush(); err != nil {
|
2015-12-10 01:03:53 +00:00
|
|
|
fatal(err.Error(), 1)
|
2015-12-02 22:41:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var StartCommand = cli.Command{
|
|
|
|
Name: "start",
|
|
|
|
Usage: "start a container",
|
|
|
|
Flags: []cli.Flag{
|
2015-12-08 21:31:20 +00:00
|
|
|
cli.StringFlag{
|
|
|
|
Name: "checkpoint,c",
|
|
|
|
Value: "",
|
|
|
|
Usage: "checkpoint to start the container from",
|
|
|
|
},
|
2015-12-02 22:41:49 +00:00
|
|
|
},
|
|
|
|
Action: func(context *cli.Context) {
|
2015-12-10 01:03:53 +00:00
|
|
|
var (
|
|
|
|
id = context.Args().Get(0)
|
|
|
|
path = context.Args().Get(1)
|
|
|
|
)
|
2015-12-02 22:41:49 +00:00
|
|
|
if path == "" {
|
|
|
|
fatal("bundle path cannot be empty", 1)
|
|
|
|
}
|
|
|
|
if id == "" {
|
|
|
|
fatal("container id cannot be empty", 1)
|
|
|
|
}
|
2015-12-09 23:37:04 +00:00
|
|
|
c := getClient()
|
|
|
|
if _, err := c.CreateContainer(netcontext.Background(), &types.CreateContainerRequest{
|
|
|
|
Id: id,
|
|
|
|
BundlePath: path,
|
2015-12-08 21:31:20 +00:00
|
|
|
Checkpoint: context.String("checkpoint"),
|
|
|
|
}); err != nil {
|
2015-12-02 22:41:49 +00:00
|
|
|
fatal(err.Error(), 1)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var KillCommand = cli.Command{
|
|
|
|
Name: "kill",
|
|
|
|
Usage: "send a signal to a container or it's processes",
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
cli.IntFlag{
|
|
|
|
Name: "pid,p",
|
|
|
|
Usage: "pid of the process to signal within the container",
|
|
|
|
},
|
|
|
|
cli.IntFlag{
|
|
|
|
Name: "signal,s",
|
|
|
|
Value: 15,
|
|
|
|
Usage: "signal to send to the container",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Action: func(context *cli.Context) {
|
|
|
|
id := context.Args().First()
|
|
|
|
if id == "" {
|
|
|
|
fatal("container id cannot be empty", 1)
|
|
|
|
}
|
2015-12-09 23:37:04 +00:00
|
|
|
c := getClient()
|
|
|
|
if _, err := c.Signal(netcontext.Background(), &types.SignalRequest{
|
|
|
|
Id: id,
|
|
|
|
Pid: uint32(context.Int("pid")),
|
|
|
|
Signal: uint32(context.Int("signal")),
|
|
|
|
}); err != nil {
|
2015-12-02 22:41:49 +00:00
|
|
|
fatal(err.Error(), 1)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2015-12-10 19:03:36 +00:00
|
|
|
|
|
|
|
var ExecCommand = cli.Command{
|
|
|
|
Name: "exec",
|
|
|
|
Usage: "exec another process in an existing container",
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
cli.StringFlag{
|
|
|
|
Name: "id",
|
|
|
|
Usage: "container id to add the process to",
|
|
|
|
},
|
|
|
|
cli.StringFlag{
|
|
|
|
Name: "cwd",
|
|
|
|
Usage: "current working directory for the process",
|
|
|
|
},
|
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "tty,t",
|
|
|
|
Usage: "create a terminal for the process",
|
|
|
|
},
|
|
|
|
cli.StringSliceFlag{
|
|
|
|
Name: "env,e",
|
|
|
|
Value: &cli.StringSlice{},
|
|
|
|
Usage: "environment variables for the process",
|
|
|
|
},
|
|
|
|
cli.IntFlag{
|
|
|
|
Name: "uid,u",
|
|
|
|
Usage: "user id of the user for the process",
|
|
|
|
},
|
|
|
|
cli.IntFlag{
|
|
|
|
Name: "gid,g",
|
|
|
|
Usage: "group id of the user for the process",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Action: func(context *cli.Context) {
|
|
|
|
p := &types.AddProcessRequest{
|
|
|
|
Args: context.Args(),
|
|
|
|
Cwd: context.String("cwd"),
|
|
|
|
Terminal: context.Bool("tty"),
|
|
|
|
Id: context.String("id"),
|
|
|
|
Env: context.StringSlice("env"),
|
|
|
|
User: &types.User{
|
|
|
|
Uid: uint32(context.Int("uid")),
|
|
|
|
Gid: uint32(context.Int("gid")),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
c := getClient()
|
|
|
|
if _, err := c.AddProcess(netcontext.Background(), p); err != nil {
|
|
|
|
fatal(err.Error(), 1)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|