open-nomad/client/driver/rkt.go

466 lines
14 KiB
Go
Raw Normal View History

package driver
import (
2015-10-07 22:24:16 +00:00
"bytes"
"encoding/json"
"fmt"
"log"
"net"
2016-10-07 19:37:52 +00:00
"os"
2015-10-07 22:24:16 +00:00
"os/exec"
2015-10-09 19:14:56 +00:00
"path/filepath"
2015-10-07 22:24:16 +00:00
"regexp"
"runtime"
"strings"
"syscall"
"time"
"github.com/hashicorp/go-plugin"
"github.com/hashicorp/go-version"
2015-10-09 19:14:56 +00:00
"github.com/hashicorp/nomad/client/allocdir"
2015-10-07 22:24:16 +00:00
"github.com/hashicorp/nomad/client/config"
"github.com/hashicorp/nomad/client/driver/executor"
dstructs "github.com/hashicorp/nomad/client/driver/structs"
2015-11-05 21:46:02 +00:00
"github.com/hashicorp/nomad/client/fingerprint"
cstructs "github.com/hashicorp/nomad/client/structs"
"github.com/hashicorp/nomad/helper/discover"
"github.com/hashicorp/nomad/helper/fields"
2015-10-07 22:24:16 +00:00
"github.com/hashicorp/nomad/nomad/structs"
"github.com/mitchellh/mapstructure"
)
var (
2016-02-02 23:39:45 +00:00
reRktVersion = regexp.MustCompile(`rkt [vV]ersion[:]? (\d[.\d]+)`)
reAppcVersion = regexp.MustCompile(`appc [vV]ersion[:]? (\d[.\d]+)`)
)
2015-12-22 05:15:37 +00:00
const (
2015-12-22 18:11:22 +00:00
// minRktVersion is the earliest supported version of rkt. rkt added support
// for CPU and memory isolators in 0.14.0. We cannot support an earlier
// version to maintain an uniform interface across all drivers
minRktVersion = "0.14.0"
// The key populated in the Node Attributes to indicate the presence of the
// Rkt driver
rktDriverAttr = "driver.rkt"
2016-10-13 23:18:18 +00:00
// rktVolumesConfigOption is the key for enabling the use of custom
// bind volumes.
rktVolumesConfigOption = "rkt.volumes.enabled"
2015-12-22 05:15:37 +00:00
)
// RktDriver is a driver for running images via Rkt
// We attempt to chose sane defaults for now, with more configuration available
// planned in the future
type RktDriver struct {
2015-10-07 22:24:16 +00:00
DriverContext
2015-11-05 21:46:02 +00:00
fingerprint.StaticFingerprinter
}
2015-11-14 04:22:49 +00:00
type RktDriverConfig struct {
ImageName string `mapstructure:"image"`
Command string `mapstructure:"command"`
Args []string `mapstructure:"args"`
TrustPrefix string `mapstructure:"trust_prefix"`
DNSServers []string `mapstructure:"dns_servers"` // DNS Server for containers
DNSSearchDomains []string `mapstructure:"dns_search_domains"` // DNS Search domains for containers
Debug bool `mapstructure:"debug"` // Enable debug option for rkt command
}
// rktHandle is returned from Start/Open as a handle to the PID
type rktHandle struct {
pluginClient *plugin.Client
executorPid int
executor executor.Executor
allocDir *allocdir.AllocDir
logger *log.Logger
killTimeout time.Duration
maxKillTimeout time.Duration
waitCh chan *dstructs.WaitResult
doneCh chan struct{}
}
// rktPID is a struct to map the pid running the process to the vm image on
// disk
type rktPID struct {
PluginConfig *PluginReattachConfig
AllocDir *allocdir.AllocDir
ExecutorPid int
KillTimeout time.Duration
MaxKillTimeout time.Duration
}
// NewRktDriver is used to create a new exec driver
func NewRktDriver(ctx *DriverContext) Driver {
2015-11-05 21:46:02 +00:00
return &RktDriver{DriverContext: *ctx}
}
// Validate is used to validate the driver configuration
func (d *RktDriver) Validate(config map[string]interface{}) error {
fd := &fields.FieldData{
Raw: config,
Schema: map[string]*fields.FieldSchema{
"image": &fields.FieldSchema{
Type: fields.TypeString,
Required: true,
},
"command": &fields.FieldSchema{
Type: fields.TypeString,
},
"args": &fields.FieldSchema{
Type: fields.TypeArray,
},
"trust_prefix": &fields.FieldSchema{
Type: fields.TypeString,
},
"dns_servers": &fields.FieldSchema{
Type: fields.TypeArray,
},
"dns_search_domains": &fields.FieldSchema{
Type: fields.TypeArray,
},
2016-08-08 10:30:47 +00:00
"debug": &fields.FieldSchema{
Type: fields.TypeBool,
2016-08-08 10:30:47 +00:00
},
},
}
if err := fd.Validate(); err != nil {
return err
}
return nil
}
func (d *RktDriver) Fingerprint(cfg *config.Config, node *structs.Node) (bool, error) {
// Get the current status so that we can log any debug messages only if the
// state changes
_, currentlyEnabled := node.Attributes[rktDriverAttr]
2015-10-07 22:24:16 +00:00
// Only enable if we are root when running on non-windows systems.
if runtime.GOOS != "windows" && syscall.Geteuid() != 0 {
if currentlyEnabled {
d.logger.Printf("[DEBUG] driver.rkt: must run as root user, disabling")
}
delete(node.Attributes, rktDriverAttr)
2015-10-07 22:24:16 +00:00
return false, nil
}
outBytes, err := exec.Command("rkt", "version").Output()
if err != nil {
delete(node.Attributes, rktDriverAttr)
2015-10-07 22:24:16 +00:00
return false, nil
}
out := strings.TrimSpace(string(outBytes))
rktMatches := reRktVersion.FindStringSubmatch(out)
2015-10-21 01:08:46 +00:00
appcMatches := reAppcVersion.FindStringSubmatch(out)
2015-10-07 22:24:16 +00:00
if len(rktMatches) != 2 || len(appcMatches) != 2 {
delete(node.Attributes, rktDriverAttr)
2015-10-07 22:24:16 +00:00
return false, fmt.Errorf("Unable to parse Rkt version string: %#v", rktMatches)
}
node.Attributes[rktDriverAttr] = "1"
2015-10-21 01:08:46 +00:00
node.Attributes["driver.rkt.version"] = rktMatches[1]
2015-10-07 22:24:16 +00:00
node.Attributes["driver.rkt.appc.version"] = appcMatches[1]
2015-12-22 05:15:37 +00:00
minVersion, _ := version.NewVersion(minRktVersion)
currentVersion, _ := version.NewVersion(node.Attributes["driver.rkt.version"])
if currentVersion.LessThan(minVersion) {
// Do not allow rkt < 0.14.0
d.logger.Printf("[WARN] driver.rkt: please upgrade rkt to a version >= %s", minVersion)
node.Attributes[rktDriverAttr] = "0"
}
2015-10-07 22:24:16 +00:00
return true, nil
}
// Run an existing Rkt image.
func (d *RktDriver) Start(ctx *ExecContext, task *structs.Task) (DriverHandle, error) {
2015-11-14 04:22:49 +00:00
var driverConfig RktDriverConfig
if err := mapstructure.WeakDecode(task.Config, &driverConfig); err != nil {
return nil, err
}
// ACI image
img := driverConfig.ImageName
2015-10-09 19:14:56 +00:00
// Get the tasks local directory.
taskName := d.DriverContext.taskName
taskDir, ok := ctx.AllocDir.TaskDirs[taskName]
if !ok {
return nil, fmt.Errorf("Could not find task directory for task: %v", d.DriverContext.taskName)
}
2015-12-22 18:23:29 +00:00
// Build the command.
var cmdArgs []string
2016-08-08 10:30:47 +00:00
// Add debug option to rkt command.
debug := driverConfig.Debug
2016-08-08 10:30:47 +00:00
2015-10-07 22:24:16 +00:00
// Add the given trust prefix
2016-08-05 17:55:20 +00:00
trustPrefix := driverConfig.TrustPrefix
insecure := false
2016-08-05 17:55:20 +00:00
if trustPrefix != "" {
var outBuf, errBuf bytes.Buffer
cmd := exec.Command("rkt", "trust", "--skip-fingerprint-review=true", fmt.Sprintf("--prefix=%s", trustPrefix), fmt.Sprintf("--debug=%t", debug))
cmd.Stdout = &outBuf
cmd.Stderr = &errBuf
if err := cmd.Run(); err != nil {
return nil, fmt.Errorf("Error running rkt trust: %s\n\nOutput: %s\n\nError: %s",
err, outBuf.String(), errBuf.String())
}
2015-12-21 10:10:37 +00:00
d.logger.Printf("[DEBUG] driver.rkt: added trust prefix: %q", trustPrefix)
2015-12-22 18:23:29 +00:00
} else {
// Disble signature verification if the trust command was not run.
insecure = true
}
cmdArgs = append(cmdArgs, "run")
2016-10-13 23:18:18 +00:00
// Mount /alloc
cmdArgs = append(cmdArgs, fmt.Sprintf("--volume=%salloc,kind=host,source=%s", task.Name, ctx.AllocDir.SharedDir))
cmdArgs = append(cmdArgs, fmt.Sprintf("--mount=volume=%salloc,target=%s", task.Name, allocdir.SharedAllocContainerPath))
// Mount /local
cmdArgs = append(cmdArgs, fmt.Sprintf("--volume=%slocal,kind=host,source=%s", task.Name, filepath.Join(taskDir, allocdir.TaskLocal)))
cmdArgs = append(cmdArgs, fmt.Sprintf("--mount=volume=%slocal,target=%s", task.Name, allocdir.TaskLocalContainerPath))
// Mount /secrets
cmdArgs = append(cmdArgs, fmt.Sprintf("--volume=%ssecrets,kind=host,source=%s", task.Name, filepath.Join(taskDir, allocdir.TaskSecrets)))
cmdArgs = append(cmdArgs, fmt.Sprintf("--mount=volume=%ssecrets,target=/%s", task.Name, allocdir.TaskSecretsContainerPath))
cmdArgs = append(cmdArgs, img)
2016-10-13 23:18:18 +00:00
if insecure {
cmdArgs = append(cmdArgs, "--insecure-options=all")
2015-10-07 22:24:16 +00:00
}
cmdArgs = append(cmdArgs, fmt.Sprintf("--debug=%t", debug))
2015-10-07 22:24:16 +00:00
2016-05-15 16:41:34 +00:00
// Inject environment variables
2016-10-13 23:18:18 +00:00
d.taskEnv.SetAllocDir(allocdir.SharedAllocContainerPath)
d.taskEnv.SetTaskLocalDir(allocdir.TaskLocalContainerPath)
d.taskEnv.SetTaskLocalDir(allocdir.TaskSecretsContainerPath)
d.taskEnv.Build()
2016-01-11 17:58:26 +00:00
for k, v := range d.taskEnv.EnvMap() {
2015-12-21 10:10:37 +00:00
cmdArgs = append(cmdArgs, fmt.Sprintf("--set-env=%v=%v", k, v))
2015-10-07 22:24:16 +00:00
}
2016-05-15 16:41:34 +00:00
// Check if the user has overridden the exec command.
2016-08-05 17:55:20 +00:00
if driverConfig.Command != "" {
cmdArgs = append(cmdArgs, fmt.Sprintf("--exec=%v", driverConfig.Command))
2015-10-07 22:24:16 +00:00
}
// Add memory isolator
2016-08-14 05:27:42 +00:00
cmdArgs = append(cmdArgs, fmt.Sprintf("--memory=%vM", int64(task.Resources.MemoryMB)))
// Add CPU isolator
2015-12-21 10:10:37 +00:00
cmdArgs = append(cmdArgs, fmt.Sprintf("--cpu=%vm", int64(task.Resources.CPU)))
// Add DNS servers
for _, ip := range driverConfig.DNSServers {
if err := net.ParseIP(ip); err == nil {
msg := fmt.Errorf("invalid ip address for container dns server %q", ip)
d.logger.Printf("[DEBUG] driver.rkt: %v", msg)
return nil, msg
} else {
cmdArgs = append(cmdArgs, fmt.Sprintf("--dns=%s", ip))
}
}
// set DNS search domains
for _, domain := range driverConfig.DNSSearchDomains {
cmdArgs = append(cmdArgs, fmt.Sprintf("--dns-search=%s", domain))
}
// Add user passed arguments.
if len(driverConfig.Args) != 0 {
2016-01-11 17:58:26 +00:00
parsed := d.taskEnv.ParseAndReplace(driverConfig.Args)
// Need to start arguments with "--"
if len(parsed) > 0 {
2015-12-21 10:10:37 +00:00
cmdArgs = append(cmdArgs, "--")
}
for _, arg := range parsed {
2015-12-21 10:10:37 +00:00
cmdArgs = append(cmdArgs, fmt.Sprintf("%v", arg))
}
2015-10-07 22:24:16 +00:00
}
// Set the host environment variables.
filter := strings.Split(d.config.ReadDefault("env.blacklist", config.DefaultEnvBlacklist), ",")
d.taskEnv.AppendHostEnvvars(filter)
bin, err := discover.NomadExecutable()
2015-10-09 19:14:56 +00:00
if err != nil {
return nil, fmt.Errorf("unable to find the nomad binary: %v", err)
2015-10-09 19:14:56 +00:00
}
pluginLogFile := filepath.Join(taskDir, fmt.Sprintf("%s-executor.out", task.Name))
pluginConfig := &plugin.ClientConfig{
Cmd: exec.Command(bin, "executor", pluginLogFile),
2015-10-09 19:14:56 +00:00
}
2016-03-16 03:21:52 +00:00
execIntf, pluginClient, err := createExecutor(pluginConfig, d.config.LogOutput, d.config)
if err != nil {
return nil, err
}
executorCtx := &executor.ExecutorContext{
2016-03-24 22:39:10 +00:00
TaskEnv: d.taskEnv,
Driver: "rkt",
AllocDir: ctx.AllocDir,
AllocID: ctx.AllocID,
Task: task,
}
2016-10-12 18:35:29 +00:00
if err := execIntf.SetContext(executorCtx); err != nil {
pluginClient.Kill()
return nil, fmt.Errorf("failed to set executor context: %v", err)
}
2015-10-09 19:14:56 +00:00
2016-03-16 02:22:40 +00:00
absPath, err := GetAbsolutePath("rkt")
if err != nil {
return nil, err
}
2016-10-12 18:35:29 +00:00
execCmd := &executor.ExecCommand{
Cmd: absPath,
Args: cmdArgs,
User: task.User,
2016-10-12 18:35:29 +00:00
}
ps, err := execIntf.LaunchCmd(execCmd)
if err != nil {
pluginClient.Kill()
2016-03-19 19:18:10 +00:00
return nil, err
2015-10-07 22:24:16 +00:00
}
d.logger.Printf("[DEBUG] driver.rkt: started ACI %q with: %v", img, cmdArgs)
maxKill := d.DriverContext.config.MaxKillTimeout
2015-10-07 22:24:16 +00:00
h := &rktHandle{
pluginClient: pluginClient,
2016-03-16 03:21:52 +00:00
executor: execIntf,
executorPid: ps.Pid,
allocDir: ctx.AllocDir,
logger: d.logger,
killTimeout: GetKillTimeout(task.KillTimeout, maxKill),
maxKillTimeout: maxKill,
doneCh: make(chan struct{}),
waitCh: make(chan *dstructs.WaitResult, 1),
2015-10-07 22:24:16 +00:00
}
if err := h.executor.SyncServices(consulContext(d.config, "")); err != nil {
2016-03-24 21:53:53 +00:00
h.logger.Printf("[ERR] driver.rkt: error registering services for task: %q: %v", task.Name, err)
2016-03-23 20:19:45 +00:00
}
2015-10-07 22:24:16 +00:00
go h.run()
return h, nil
}
func (d *RktDriver) Open(ctx *ExecContext, handleID string) (DriverHandle, error) {
2015-10-07 22:24:16 +00:00
// Parse the handle
pidBytes := []byte(strings.TrimPrefix(handleID, "Rkt:"))
id := &rktPID{}
if err := json.Unmarshal(pidBytes, id); err != nil {
2015-10-07 22:24:16 +00:00
return nil, fmt.Errorf("failed to parse Rkt handle '%s': %v", handleID, err)
}
pluginConfig := &plugin.ClientConfig{
Reattach: id.PluginConfig.PluginConfig(),
}
2016-03-29 23:27:31 +00:00
exec, pluginClient, err := createExecutor(pluginConfig, d.config.LogOutput, d.config)
if err != nil {
d.logger.Println("[ERROR] driver.rkt: error connecting to plugin so destroying plugin pid and user pid")
if e := destroyPlugin(id.PluginConfig.Pid, id.ExecutorPid); e != nil {
d.logger.Printf("[ERROR] driver.rkt: error destroying plugin and executor pid: %v", e)
}
return nil, fmt.Errorf("error connecting to plugin: %v", err)
2015-10-07 22:24:16 +00:00
}
2016-03-30 05:05:02 +00:00
ver, _ := exec.Version()
d.logger.Printf("[DEBUG] driver.rkt: version of executor: %v", ver.Version)
2015-10-07 22:24:16 +00:00
// Return a driver handle
h := &rktHandle{
pluginClient: pluginClient,
executorPid: id.ExecutorPid,
allocDir: id.AllocDir,
2016-03-29 23:27:31 +00:00
executor: exec,
logger: d.logger,
killTimeout: id.KillTimeout,
maxKillTimeout: id.MaxKillTimeout,
doneCh: make(chan struct{}),
waitCh: make(chan *dstructs.WaitResult, 1),
2015-10-07 22:24:16 +00:00
}
if err := h.executor.SyncServices(consulContext(d.config, "")); err != nil {
2016-03-24 22:39:10 +00:00
h.logger.Printf("[ERR] driver.rkt: error registering services: %v", err)
}
2015-10-07 22:24:16 +00:00
go h.run()
return h, nil
}
func (h *rktHandle) ID() string {
2015-10-07 22:24:16 +00:00
// Return a handle to the PID
pid := &rktPID{
PluginConfig: NewPluginReattachConfig(h.pluginClient.ReattachConfig()),
KillTimeout: h.killTimeout,
MaxKillTimeout: h.maxKillTimeout,
ExecutorPid: h.executorPid,
AllocDir: h.allocDir,
2015-10-07 22:24:16 +00:00
}
data, err := json.Marshal(pid)
if err != nil {
h.logger.Printf("[ERR] driver.rkt: failed to marshal rkt PID to JSON: %s", err)
}
return fmt.Sprintf("Rkt:%s", string(data))
}
func (h *rktHandle) WaitCh() chan *dstructs.WaitResult {
2015-10-07 22:24:16 +00:00
return h.waitCh
}
func (h *rktHandle) Update(task *structs.Task) error {
// Store the updated kill timeout.
h.killTimeout = GetKillTimeout(task.KillTimeout, h.maxKillTimeout)
h.executor.UpdateTask(task)
2015-10-07 22:24:16 +00:00
// Update is not possible
return nil
}
2016-10-07 19:37:52 +00:00
func (h *rktHandle) Signal(s os.Signal) error {
2016-10-07 23:49:00 +00:00
return fmt.Errorf("Rkt does not support signals")
2016-10-07 19:37:52 +00:00
}
// Kill is used to terminate the task. We send an Interrupt
// and then provide a 5 second grace period before doing a Kill.
func (h *rktHandle) Kill() error {
h.executor.ShutDown()
2015-10-07 22:24:16 +00:00
select {
case <-h.doneCh:
return nil
case <-time.After(h.killTimeout):
return h.executor.Exit()
2015-10-07 22:24:16 +00:00
}
}
2016-04-28 23:06:01 +00:00
func (h *rktHandle) Stats() (*cstructs.TaskResourceUsage, error) {
return nil, fmt.Errorf("stats not implemented for rkt")
}
func (h *rktHandle) run() {
ps, err := h.executor.Wait()
2015-10-07 22:24:16 +00:00
close(h.doneCh)
if ps.ExitCode == 0 && err != nil {
if e := killProcess(h.executorPid); e != nil {
h.logger.Printf("[ERROR] driver.rkt: error killing user process: %v", e)
}
if e := h.allocDir.UnmountAll(); e != nil {
h.logger.Printf("[ERROR] driver.rkt: unmounting dev,proc and alloc dirs failed: %v", e)
}
}
h.waitCh <- dstructs.NewWaitResult(ps.ExitCode, 0, err)
2015-10-07 22:24:16 +00:00
close(h.waitCh)
2016-03-23 20:19:45 +00:00
// Remove services
if err := h.executor.DeregisterServices(); err != nil {
h.logger.Printf("[ERR] driver.rkt: failed to deregister services: %v", err)
}
if err := h.executor.Exit(); err != nil {
h.logger.Printf("[ERR] driver.rkt: error killing executor: %v", err)
}
h.pluginClient.Kill()
}