From fe2b3e382d691b946dc2aacb7fa236d52a2c8c44 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Fri, 21 Apr 2017 12:20:05 -0700 Subject: [PATCH 1/9] Restart tasks on upgrade with script checks and old executors --- client/alloc_runner.go | 11 ++++++++-- client/task_runner.go | 50 ++++++++++++++++++++++++++++++++++-------- 2 files changed, 50 insertions(+), 11 deletions(-) diff --git a/client/alloc_runner.go b/client/alloc_runner.go index 8ca6f671eae..7da40c67814 100644 --- a/client/alloc_runner.go +++ b/client/alloc_runner.go @@ -186,13 +186,20 @@ func (r *AllocRunner) RestoreState() error { continue } - if err := tr.RestoreState(); err != nil { - r.logger.Printf("[ERR] client: failed to restore state for alloc %s task '%s': %v", r.alloc.ID, name, err) + if restartReason, err := tr.RestoreState(); err != nil { + r.logger.Printf("[ERR] client: failed to restore state for alloc %s task %q: %v", r.alloc.ID, name, err) mErr.Errors = append(mErr.Errors, err) } else if !r.alloc.TerminalStatus() { // Only start if the alloc isn't in a terminal status. go tr.Run() + + // Restart task runner if RestoreState gave a reason + if restartReason != "" { + r.logger.Printf("[INFO] client: restarting alloc %s task %q due to upgrade: %s", r.alloc.ID, name, restartReason) + tr.Restart("upgrade", restartReason) + } } + } return mErr.ErrorOrNil() diff --git a/client/task_runner.go b/client/task_runner.go index c578ca5ea0b..a9f1d744292 100644 --- a/client/task_runner.go +++ b/client/task_runner.go @@ -16,6 +16,7 @@ import ( "github.com/golang/snappy" "github.com/hashicorp/consul-template/signals" "github.com/hashicorp/go-multierror" + version "github.com/hashicorp/go-version" "github.com/hashicorp/nomad/client/allocdir" "github.com/hashicorp/nomad/client/config" "github.com/hashicorp/nomad/client/driver" @@ -234,17 +235,20 @@ func (r *TaskRunner) stateFilePath() string { return path } -// RestoreState is used to restore our state -func (r *TaskRunner) RestoreState() error { +// RestoreState is used to restore our state. If a non-empty string is returned +// the task is restarted with the string as the reason. This is useful for +// backwards incompatible upgrades that need to restart tasks with a new +// executor. +func (r *TaskRunner) RestoreState() (string, error) { // Load the snapshot var snap taskRunnerState if err := restoreState(r.stateFilePath(), &snap); err != nil { - return err + return "", err } // Restore fields if snap.Task == nil { - return fmt.Errorf("task runner snapshot includes nil Task") + return "", fmt.Errorf("task runner snapshot includes nil Task") } else { r.task = snap.Task } @@ -255,7 +259,7 @@ func (r *TaskRunner) RestoreState() error { r.setCreatedResources(snap.CreatedResources) if err := r.setTaskEnv(); err != nil { - return fmt.Errorf("client: failed to create task environment for task %q in allocation %q: %v", + return "", fmt.Errorf("client: failed to create task environment for task %q in allocation %q: %v", r.task.Name, r.alloc.ID, err) } @@ -265,7 +269,7 @@ func (r *TaskRunner) RestoreState() error { data, err := ioutil.ReadFile(tokenPath) if err != nil { if !os.IsNotExist(err) { - return fmt.Errorf("failed to read token for task %q in alloc %q: %v", r.task.Name, r.alloc.ID, err) + return "", fmt.Errorf("failed to read token for task %q in alloc %q: %v", r.task.Name, r.alloc.ID, err) } // Token file doesn't exist @@ -276,10 +280,11 @@ func (r *TaskRunner) RestoreState() error { } // Restore the driver + restartReason := "" if snap.HandleID != "" { d, err := r.createDriver() if err != nil { - return err + return "", err } ctx := driver.NewExecContext(r.taskDir) @@ -289,7 +294,11 @@ func (r *TaskRunner) RestoreState() error { if err != nil { r.logger.Printf("[ERR] client: failed to open handle to task %q for alloc %q: %v", r.task.Name, r.alloc.ID, err) - return nil + return "", nil + } + + if pre06ScriptCheck(snap.Version, r.task.Services) { + restartReason = "upgrading pre-0.6 script checks" } if err := r.registerServices(d, handle); err != nil { @@ -308,7 +317,30 @@ func (r *TaskRunner) RestoreState() error { r.running = true r.runningLock.Unlock() } - return nil + return restartReason, nil +} + +var ver06 = version.Must(version.NewVersion("0.6.0dev")) + +// pre06ScriptCheck returns true if version is prior to 0.6.0dev. +func pre06ScriptCheck(ver string, services []*structs.Service) bool { + v, err := version.NewVersion(ver) + if err != nil { + // Treat it as old + return true + } + if !v.LessThan(ver06) { + // >= 0.6.0dev + return false + } + for _, service := range services { + for _, check := range service.Checks { + if check.Type == "script" { + return true + } + } + } + return false } // SaveState is used to snapshot our state From ba1c2e7936d8f6d9b6945d9823e9b251f8baec85 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Fri, 21 Apr 2017 12:20:48 -0700 Subject: [PATCH 2/9] Upgrade go-version to one compatible with Nomad versioning --- .../hashicorp/go-version/.travis.yml | 11 -- .../github.com/hashicorp/go-version/README.md | 2 +- .../hashicorp/go-version/constraint.go | 24 +++- .../hashicorp/go-version/version.go | 115 ++++++++++++++---- vendor/vendor.json | 4 +- 5 files changed, 120 insertions(+), 36 deletions(-) delete mode 100644 vendor/github.com/hashicorp/go-version/.travis.yml diff --git a/vendor/github.com/hashicorp/go-version/.travis.yml b/vendor/github.com/hashicorp/go-version/.travis.yml deleted file mode 100644 index 9f30eecd73f..00000000000 --- a/vendor/github.com/hashicorp/go-version/.travis.yml +++ /dev/null @@ -1,11 +0,0 @@ -language: go - -go: - - 1.0 - - 1.1 - - 1.2 - - 1.3 - - 1.4 - -script: - - go test diff --git a/vendor/github.com/hashicorp/go-version/README.md b/vendor/github.com/hashicorp/go-version/README.md index 1d50070f131..6f3a15ce772 100644 --- a/vendor/github.com/hashicorp/go-version/README.md +++ b/vendor/github.com/hashicorp/go-version/README.md @@ -1,5 +1,5 @@ # Versioning Library for Go -[![Build Status](https://travis-ci.org/hashicorp/go-version.svg?branch=master)](https://travis-ci.org/hashicorp/go-version) +[![Build Status](https://travis-ci.org/hashicorp/go-version.svg?branch=master)](https://travis-ci.org/hashicorp/go-version) go-version is a library for parsing versions and version constraints, and verifying versions against a set of constraints. go-version diff --git a/vendor/github.com/hashicorp/go-version/constraint.go b/vendor/github.com/hashicorp/go-version/constraint.go index 091cfab389f..8c73df0602b 100644 --- a/vendor/github.com/hashicorp/go-version/constraint.go +++ b/vendor/github.com/hashicorp/go-version/constraint.go @@ -37,7 +37,7 @@ func init() { } ops := make([]string, 0, len(constraintOperators)) - for k, _ := range constraintOperators { + for k := range constraintOperators { ops = append(ops, regexp.QuoteMeta(k)) } @@ -142,15 +142,37 @@ func constraintLessThanEqual(v, c *Version) bool { } func constraintPessimistic(v, c *Version) bool { + // If the version being checked is naturally less than the constraint, then there + // is no way for the version to be valid against the constraint if v.LessThan(c) { return false } + // We'll use this more than once, so grab the length now so it's a little cleaner + // to write the later checks + cs := len(c.segments) + // If the version being checked has less specificity than the constraint, then there + // is no way for the version to be valid against the constraint + if cs > len(v.segments) { + return false + } + + // Check the segments in the constraint against those in the version. If the version + // being checked, at any point, does not have the same values in each index of the + // constraints segments, then it cannot be valid against the constraint. for i := 0; i < c.si-1; i++ { if v.segments[i] != c.segments[i] { return false } } + // Check the last part of the segment in the constraint. If the version segment at + // this index is less than the constraints segment at this index, then it cannot + // be valid against the constraint + if c.segments[cs-1] > v.segments[cs-1] { + return false + } + + // If nothing has rejected the version by now, it's valid return true } diff --git a/vendor/github.com/hashicorp/go-version/version.go b/vendor/github.com/hashicorp/go-version/version.go index c7147ff7bf5..dfe509caa0c 100644 --- a/vendor/github.com/hashicorp/go-version/version.go +++ b/vendor/github.com/hashicorp/go-version/version.go @@ -14,8 +14,8 @@ var versionRegexp *regexp.Regexp // The raw regular expression string used for testing the validity // of a version. -const VersionRegexpRaw string = `v?([0-9]+(\.[0-9]+){0,2})` + - `(-([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*))?` + +const VersionRegexpRaw string = `v?([0-9]+(\.[0-9]+)*?)` + + `(-?([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*))?` + `(\+([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*))?` + `?` @@ -23,7 +23,7 @@ const VersionRegexpRaw string = `v?([0-9]+(\.[0-9]+){0,2})` + type Version struct { metadata string pre string - segments []int + segments []int64 si int } @@ -38,20 +38,23 @@ func NewVersion(v string) (*Version, error) { if matches == nil { return nil, fmt.Errorf("Malformed version: %s", v) } - segmentsStr := strings.Split(matches[1], ".") - segments := make([]int, len(segmentsStr), 3) + segments := make([]int64, len(segmentsStr)) si := 0 for i, str := range segmentsStr { - val, err := strconv.ParseInt(str, 10, 32) + val, err := strconv.ParseInt(str, 10, 64) if err != nil { return nil, fmt.Errorf( "Error parsing version: %s", err) } - segments[i] = int(val) - si += 1 + segments[i] = int64(val) + si++ } + + // Even though we could support more than three segments, if we + // got less than three, pad it with 0s. This is to cover the basic + // default usecase of semver, which is MAJOR.MINOR.PATCH at the minimum for i := len(segments); i < 3; i++ { segments = append(segments, 0) } @@ -86,8 +89,8 @@ func (v *Version) Compare(other *Version) int { return 0 } - segmentsSelf := v.Segments() - segmentsOther := other.Segments() + segmentsSelf := v.Segments64() + segmentsOther := other.Segments64() // If the segments are the same, we must compare on prerelease info if reflect.DeepEqual(segmentsSelf, segmentsOther) { @@ -106,21 +109,56 @@ func (v *Version) Compare(other *Version) int { return comparePrereleases(preSelf, preOther) } + // Get the highest specificity (hS), or if they're equal, just use segmentSelf length + lenSelf := len(segmentsSelf) + lenOther := len(segmentsOther) + hS := lenSelf + if lenSelf < lenOther { + hS = lenOther + } // Compare the segments - for i := 0; i < len(segmentsSelf); i++ { + // Because a constraint could have more/less specificity than the version it's + // checking, we need to account for a lopsided or jagged comparison + for i := 0; i < hS; i++ { + if i > lenSelf-1 { + // This means Self had the lower specificity + // Check to see if the remaining segments in Other are all zeros + if !allZero(segmentsOther[i:]) { + // if not, it means that Other has to be greater than Self + return -1 + } + break + } else if i > lenOther-1 { + // this means Other had the lower specificity + // Check to see if the remaining segments in Self are all zeros - + if !allZero(segmentsSelf[i:]) { + //if not, it means that Self has to be greater than Other + return 1 + } + break + } lhs := segmentsSelf[i] rhs := segmentsOther[i] - if lhs == rhs { continue } else if lhs < rhs { return -1 - } else { - return 1 } + // Otherwis, rhs was > lhs, they're not equal + return 1 } - panic("should not be reached") + // if we got this far, they're equal + return 0 +} + +func allZero(segs []int64) bool { + for _, s := range segs { + if s != 0 { + return false + } + } + return true } func comparePart(preSelf string, preOther string) int { @@ -128,24 +166,38 @@ func comparePart(preSelf string, preOther string) int { return 0 } + selfNumeric := true + _, err := strconv.ParseInt(preSelf, 10, 64) + if err != nil { + selfNumeric = false + } + + otherNumeric := true + _, err = strconv.ParseInt(preOther, 10, 64) + if err != nil { + otherNumeric = false + } + // if a part is empty, we use the other to decide if preSelf == "" { - _, notIsNumeric := strconv.ParseInt(preOther, 10, 64) - if notIsNumeric == nil { + if otherNumeric { return -1 } return 1 } if preOther == "" { - _, notIsNumeric := strconv.ParseInt(preSelf, 10, 64) - if notIsNumeric == nil { + if selfNumeric { return 1 } return -1 } - if preSelf > preOther { + if selfNumeric && !otherNumeric { + return -1 + } else if !selfNumeric && otherNumeric { + return 1 + } else if preSelf > preOther { return 1 } @@ -226,12 +278,25 @@ func (v *Version) Prerelease() string { return v.pre } -// Segments returns the numeric segments of the version as a slice. +// Segments returns the numeric segments of the version as a slice of ints. // // This excludes any metadata or pre-release information. For example, // for a version "1.2.3-beta", segments will return a slice of // 1, 2, 3. func (v *Version) Segments() []int { + segmentSlice := make([]int, len(v.segments)) + for i, v := range v.segments { + segmentSlice[i] = int(v) + } + return segmentSlice +} + +// Segments64 returns the numeric segments of the version as a slice of int64s. +// +// This excludes any metadata or pre-release information. For example, +// for a version "1.2.3-beta", segments will return a slice of +// 1, 2, 3. +func (v *Version) Segments64() []int64 { return v.segments } @@ -239,7 +304,13 @@ func (v *Version) Segments() []int { // and metadata information. func (v *Version) String() string { var buf bytes.Buffer - fmt.Fprintf(&buf, "%d.%d.%d", v.segments[0], v.segments[1], v.segments[2]) + fmtParts := make([]string, len(v.segments)) + for i, s := range v.segments { + // We can ignore err here since we've pre-parsed the values in segments + str := strconv.FormatInt(s, 10) + fmtParts[i] = str + } + fmt.Fprintf(&buf, strings.Join(fmtParts, ".")) if v.pre != "" { fmt.Fprintf(&buf, "-%s", v.pre) } diff --git a/vendor/vendor.json b/vendor/vendor.json index ec1b1787731..1c4d1fc6431 100644 --- a/vendor/vendor.json +++ b/vendor/vendor.json @@ -743,8 +743,10 @@ "revision": "42a2b573b664dbf281bd48c3cc12c086b17a39ba" }, { + "checksumSHA1": "tUGxc7rfX0cmhOOUDhMuAZ9rWsA=", "path": "github.com/hashicorp/go-version", - "revision": "2e7f5ea8e27bb3fdf9baa0881d16757ac4637332" + "revision": "03c5bf6be031b6dd45afec16b1cf94fc8938bc77", + "revisionTime": "2017-02-02T08:07:59Z" }, { "checksumSHA1": "d9PxF1XQGLMJZRct2R8qVM/eYlE=", From 90f5e232a5f6a07c3d52a37b97ddd6df17e33170 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Fri, 21 Apr 2017 16:20:37 -0700 Subject: [PATCH 3/9] Switch java/exec to use Exec in Executor --- client/alloc_runner.go | 1 - client/driver/exec.go | 7 ++++- client/driver/executor/executor.go | 46 ++++++++++++++++++++++++++++-- client/driver/executor_plugin.go | 36 +++++++++++++++++++++++ client/driver/java.go | 7 ++++- client/task_runner.go | 8 ++++-- 6 files changed, 98 insertions(+), 7 deletions(-) diff --git a/client/alloc_runner.go b/client/alloc_runner.go index 7da40c67814..35a58e1cd05 100644 --- a/client/alloc_runner.go +++ b/client/alloc_runner.go @@ -195,7 +195,6 @@ func (r *AllocRunner) RestoreState() error { // Restart task runner if RestoreState gave a reason if restartReason != "" { - r.logger.Printf("[INFO] client: restarting alloc %s task %q due to upgrade: %s", r.alloc.ID, name, restartReason) tr.Restart("upgrade", restartReason) } } diff --git a/client/driver/exec.go b/client/driver/exec.go index bc6ee3aee25..fdc3bbc0186 100644 --- a/client/driver/exec.go +++ b/client/driver/exec.go @@ -259,7 +259,12 @@ func (h *execHandle) Update(task *structs.Task) error { } func (h *execHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte, int, error) { - return execChroot(ctx, h.taskDir.Dir, cmd, args) + deadline, ok := ctx.Deadline() + if !ok { + // No deadline set on context; default to 1 minute + deadline = time.Now().Add(time.Minute) + } + return h.executor.Exec(deadline, cmd, args) } func (h *execHandle) Signal(s os.Signal) error { diff --git a/client/driver/executor/executor.go b/client/driver/executor/executor.go index 90797fbefa2..295dc89283a 100644 --- a/client/driver/executor/executor.go +++ b/client/driver/executor/executor.go @@ -1,6 +1,7 @@ package executor import ( + "context" "fmt" "io/ioutil" "log" @@ -15,6 +16,7 @@ import ( "syscall" "time" + "github.com/armon/circbuf" "github.com/hashicorp/go-multierror" "github.com/mitchellh/go-ps" "github.com/shirou/gopsutil/process" @@ -57,6 +59,7 @@ type Executor interface { Version() (*ExecutorVersion, error) Stats() (*cstructs.TaskResourceUsage, error) Signal(s os.Signal) error + Exec(deadline time.Time, cmd string, args []string) ([]byte, int, error) } // ExecutorContext holds context to configure the command user @@ -203,8 +206,8 @@ func (e *UniversalExecutor) SetContext(ctx *ExecutorContext) error { return nil } -// LaunchCmd launches a process and returns it's state. It also configures an -// applies isolation on certain platforms. +// LaunchCmd launches the main process and returns its state. It also +// configures an applies isolation on certain platforms. func (e *UniversalExecutor) LaunchCmd(command *ExecCommand) (*ProcessState, error) { e.logger.Printf("[DEBUG] executor: launching command %v %v", command.Cmd, strings.Join(command.Args, " ")) @@ -283,6 +286,45 @@ func (e *UniversalExecutor) LaunchCmd(command *ExecCommand) (*ProcessState, erro return &ProcessState{Pid: e.cmd.Process.Pid, ExitCode: -1, IsolationConfig: ic, Time: time.Now()}, nil } +// Exec a command inside a container for exec and java drivers. +func (e *UniversalExecutor) Exec(deadline time.Time, name string, args []string) ([]byte, int, error) { + ctx, cancel := context.WithDeadline(context.Background(), deadline) + defer cancel() + + name = e.ctx.TaskEnv.ReplaceEnv(name) + cmd := exec.CommandContext(ctx, name, e.ctx.TaskEnv.ParseAndReplace(args)...) + + // Copy runtime environment from the main command + cmd.SysProcAttr = e.cmd.SysProcAttr + cmd.Dir = e.cmd.Dir + cmd.Env = e.ctx.TaskEnv.EnvList() + + // Capture output + buf, _ := circbuf.NewBuffer(int64(dstructs.CheckBufSize)) + cmd.Stdout = buf + cmd.Stderr = buf + + if err := cmd.Run(); err != nil { + exitErr, ok := err.(*exec.ExitError) + if !ok { + // Non-exit error, return it and let the caller treat + // it as a critical failure + return nil, 0, err + } + + // Some kind of error happened; default to critical + exitCode := 2 + if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { + exitCode = status.ExitStatus() + } + + // Don't return the exitError as the caller only needs the + // output and code. + return buf.Bytes(), exitCode, nil + } + return buf.Bytes(), 0, nil +} + // configureLoggers sets up the standard out/error file rotators func (e *UniversalExecutor) configureLoggers() error { e.rotatorLock.Lock() diff --git a/client/driver/executor_plugin.go b/client/driver/executor_plugin.go index 7c4074feffc..01e63343d28 100644 --- a/client/driver/executor_plugin.go +++ b/client/driver/executor_plugin.go @@ -6,6 +6,7 @@ import ( "net/rpc" "os" "syscall" + "time" "github.com/hashicorp/go-plugin" "github.com/hashicorp/nomad/client/driver/executor" @@ -33,6 +34,17 @@ type LaunchCmdArgs struct { Cmd *executor.ExecCommand } +type ExecCmdArgs struct { + Deadline time.Time + Name string + Args []string +} + +type ExecCmdReturn struct { + Output []byte + Code int +} + func (e *ExecutorRPC) LaunchCmd(cmd *executor.ExecCommand) (*executor.ProcessState, error) { var ps *executor.ProcessState err := e.client.Call("Plugin.LaunchCmd", LaunchCmdArgs{Cmd: cmd}, &ps) @@ -91,6 +103,20 @@ func (e *ExecutorRPC) Signal(s os.Signal) error { return e.client.Call("Plugin.Signal", &s, new(interface{})) } +func (e *ExecutorRPC) Exec(deadline time.Time, name string, args []string) ([]byte, int, error) { + req := ExecCmdArgs{ + Deadline: deadline, + Name: name, + Args: args, + } + var resp *ExecCmdReturn + err := e.client.Call("Plugin.Exec", req, &resp) + if resp == nil { + return nil, 0, err + } + return resp.Output, resp.Code, err +} + type ExecutorRPCServer struct { Impl executor.Executor logger *log.Logger @@ -165,6 +191,16 @@ func (e *ExecutorRPCServer) Signal(args os.Signal, resp *interface{}) error { return e.Impl.Signal(args) } +func (e *ExecutorRPCServer) Exec(args ExecCmdArgs, result *ExecCmdReturn) error { + out, code, err := e.Impl.Exec(args.Deadline, args.Name, args.Args) + ret := &ExecCmdReturn{ + Output: out, + Code: code, + } + *result = *ret + return err +} + type ExecutorPlugin struct { logger *log.Logger Impl *ExecutorRPCServer diff --git a/client/driver/java.go b/client/driver/java.go index c215e6882d5..5801d2e83aa 100644 --- a/client/driver/java.go +++ b/client/driver/java.go @@ -390,7 +390,12 @@ func (h *javaHandle) Update(task *structs.Task) error { } func (h *javaHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte, int, error) { - return execChroot(ctx, h.taskDir, cmd, args) + deadline, ok := ctx.Deadline() + if !ok { + // No deadline set on context; default to 1 minute + deadline = time.Now().Add(time.Minute) + } + return h.executor.Exec(deadline, cmd, args) } func (h *javaHandle) Signal(s os.Signal) error { diff --git a/client/task_runner.go b/client/task_runner.go index a9f1d744292..b88bc6a7d20 100644 --- a/client/task_runner.go +++ b/client/task_runner.go @@ -297,7 +297,7 @@ func (r *TaskRunner) RestoreState() (string, error) { return "", nil } - if pre06ScriptCheck(snap.Version, r.task.Services) { + if pre06ScriptCheck(snap.Version, r.task.Driver, r.task.Services) { restartReason = "upgrading pre-0.6 script checks" } @@ -323,7 +323,11 @@ func (r *TaskRunner) RestoreState() (string, error) { var ver06 = version.Must(version.NewVersion("0.6.0dev")) // pre06ScriptCheck returns true if version is prior to 0.6.0dev. -func pre06ScriptCheck(ver string, services []*structs.Service) bool { +func pre06ScriptCheck(ver, driver string, services []*structs.Service) bool { + if driver != "exec" && driver != "java" { + // Only exec and java are affected + return false + } v, err := version.NewVersion(ver) if err != nil { // Treat it as old From 29f222d4615117e9b26d1ce91e91a9e144fb5162 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Fri, 21 Apr 2017 16:50:20 -0700 Subject: [PATCH 4/9] Change raw_exec to use simplified exec wrapper --- client/driver/raw_exec.go | 10 +++++++++- client/driver/rkt.go | 9 ++++++++- client/driver/utils.go | 18 +++++++++--------- 3 files changed, 26 insertions(+), 11 deletions(-) diff --git a/client/driver/raw_exec.go b/client/driver/raw_exec.go index 5e602f47bb1..b799a32041d 100644 --- a/client/driver/raw_exec.go +++ b/client/driver/raw_exec.go @@ -10,7 +10,9 @@ import ( "time" "github.com/hashicorp/go-plugin" + "github.com/hashicorp/nomad/client/allocdir" "github.com/hashicorp/nomad/client/config" + "github.com/hashicorp/nomad/client/driver/env" "github.com/hashicorp/nomad/client/driver/executor" dstructs "github.com/hashicorp/nomad/client/driver/structs" "github.com/hashicorp/nomad/client/fingerprint" @@ -48,6 +50,8 @@ type rawExecHandle struct { logger *log.Logger waitCh chan *dstructs.WaitResult doneCh chan struct{} + taskEnv *env.TaskEnvironment + taskDir *allocdir.TaskDir } // NewRawExecDriver is used to create a new raw exec driver @@ -165,6 +169,8 @@ func (d *RawExecDriver) Start(ctx *ExecContext, task *structs.Task) (DriverHandl logger: d.logger, doneCh: make(chan struct{}), waitCh: make(chan *dstructs.WaitResult, 1), + taskEnv: d.taskEnv, + taskDir: ctx.TaskDir, } go h.run() return h, nil @@ -212,6 +218,8 @@ func (d *RawExecDriver) Open(ctx *ExecContext, handleID string) (DriverHandle, e version: id.Version, doneCh: make(chan struct{}), waitCh: make(chan *dstructs.WaitResult, 1), + taskEnv: d.taskEnv, + taskDir: ctx.TaskDir, } go h.run() return h, nil @@ -247,7 +255,7 @@ func (h *rawExecHandle) Update(task *structs.Task) error { } func (h *rawExecHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte, int, error) { - return execChroot(ctx, "", cmd, args) + return execScript(ctx, h.taskDir.Dir, h.taskEnv, cmd, args) } func (h *rawExecHandle) Signal(s os.Signal) error { diff --git a/client/driver/rkt.go b/client/driver/rkt.go index a8cff3c0249..930d6672894 100644 --- a/client/driver/rkt.go +++ b/client/driver/rkt.go @@ -22,6 +22,7 @@ import ( "github.com/hashicorp/go-version" "github.com/hashicorp/nomad/client/allocdir" "github.com/hashicorp/nomad/client/config" + "github.com/hashicorp/nomad/client/driver/env" "github.com/hashicorp/nomad/client/driver/executor" dstructs "github.com/hashicorp/nomad/client/driver/structs" cstructs "github.com/hashicorp/nomad/client/structs" @@ -87,6 +88,8 @@ type RktDriverConfig struct { // rktHandle is returned from Start/Open as a handle to the PID type rktHandle struct { uuid string + env *env.TaskEnvironment + taskDir *allocdir.TaskDir pluginClient *plugin.Client executorPid int executor executor.Executor @@ -474,6 +477,8 @@ func (d *RktDriver) Start(ctx *ExecContext, task *structs.Task) (DriverHandle, e maxKill := d.DriverContext.config.MaxKillTimeout h := &rktHandle{ uuid: uuid, + env: d.taskEnv, + taskDir: ctx.TaskDir, pluginClient: pluginClient, executor: execIntf, executorPid: ps.Pid, @@ -514,6 +519,8 @@ func (d *RktDriver) Open(ctx *ExecContext, handleID string) (DriverHandle, error // Return a driver handle h := &rktHandle{ uuid: id.UUID, + env: d.taskEnv, + taskDir: ctx.TaskDir, pluginClient: pluginClient, executorPid: id.ExecutorPid, executor: exec, @@ -566,7 +573,7 @@ func (h *rktHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte enterArgs[1] = h.uuid enterArgs[2] = cmd copy(enterArgs[3:], args) - return execChroot(ctx, "", rktCmd, enterArgs) + return execScript(ctx, h.taskDir.Dir, h.env, rktCmd, enterArgs) } func (h *rktHandle) Signal(s os.Signal) error { diff --git a/client/driver/utils.go b/client/driver/utils.go index def668548ff..07864bae411 100644 --- a/client/driver/utils.go +++ b/client/driver/utils.go @@ -16,6 +16,7 @@ import ( "github.com/hashicorp/go-multierror" "github.com/hashicorp/go-plugin" "github.com/hashicorp/nomad/client/config" + "github.com/hashicorp/nomad/client/driver/env" "github.com/hashicorp/nomad/client/driver/executor" cstructs "github.com/hashicorp/nomad/client/driver/structs" "github.com/hashicorp/nomad/helper/discover" @@ -182,18 +183,17 @@ func getExecutorUser(task *structs.Task) string { return task.User } -// execChroot executes cmd with args inside chroot if set and returns the -// output, exit code, and error. If chroot is an empty string the command is -// executed on the host. -func execChroot(ctx context.Context, chroot, name string, args []string) ([]byte, int, error) { - buf, _ := circbuf.NewBuffer(int64(cstructs.CheckBufSize)) +// execScript executes cmd with args and returns the output, exit code, and +// error. Output is truncated to client/driver/structs.CheckBufSize +func execScript(ctx context.Context, dir string, env *env.TaskEnvironment, name string, args []string) ([]byte, int, error) { + name = env.ReplaceEnv(name) + args = env.ParseAndReplace(args) cmd := exec.CommandContext(ctx, name, args...) - cmd.Dir = "/" + cmd.Dir = dir + cmd.Env = env.EnvList() + buf, _ := circbuf.NewBuffer(int64(cstructs.CheckBufSize)) cmd.Stdout = buf cmd.Stderr = buf - if chroot != "" { - setChroot(cmd, chroot) - } if err := cmd.Run(); err != nil { exitErr, ok := err.(*exec.ExitError) if !ok { From 8f518c4c1f2d14feeb7e0d858bf884d04d94d5e8 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Tue, 25 Apr 2017 11:13:06 -0700 Subject: [PATCH 5/9] Test env+cgroups for exec driver checks --- client/driver/exec_test.go | 53 +++++++++++++++++++++++++++++++++----- 1 file changed, 47 insertions(+), 6 deletions(-) diff --git a/client/driver/exec_test.go b/client/driver/exec_test.go index 6a580f24352..5760b175163 100644 --- a/client/driver/exec_test.go +++ b/client/driver/exec_test.go @@ -283,7 +283,8 @@ func TestExecDriverUser(t *testing.T) { } } -// TestExecDriver_HandlerExec ensures the exec driver's handle properly executes commands inside the chroot. +// TestExecDriver_HandlerExec ensures the exec driver's handle properly +// executes commands inside the container. func TestExecDriver_HandlerExec(t *testing.T) { ctestutils.ExecCompatible(t) task := &structs.Task{ @@ -315,20 +316,60 @@ func TestExecDriver_HandlerExec(t *testing.T) { t.Fatalf("missing handle") } - // Exec a command that should work - out, code, err := handle.Exec(context.TODO(), "/usr/bin/stat", []string{"/alloc"}) + // Exec a command that should work and dump the environment + out, code, err := handle.Exec(context.Background(), "/bin/sh", []string{"-c", "env | grep NOMAD"}) if err != nil { t.Fatalf("error exec'ing stat: %v", err) } if code != 0 { t.Fatalf("expected `stat /alloc` to succeed but exit code was: %d", code) } - if expected := 100; len(out) < expected { - t.Fatalf("expected at least %d bytes of output but found %d:\n%s", expected, len(out), out) + + // Assert exec'd commands are run in a task-like environment + scriptEnv := make(map[string]string) + for _, line := range strings.Split(string(out), "\n") { + if line == "" { + continue + } + parts := strings.SplitN(string(line), "=", 2) + if len(parts) != 2 { + t.Fatalf("Invalid env var: %q", line) + } + scriptEnv[parts[0]] = parts[1] + } + if v, ok := scriptEnv["NOMAD_SECRETS_DIR"]; !ok || v != "/secrets" { + t.Errorf("Expected NOMAD_SECRETS_DIR=/secrets but found=%t value=%q", ok, v) + } + if v, ok := scriptEnv["NOMAD_ALLOC_ID"]; !ok || v != ctx.DriverCtx.allocID { + t.Errorf("Expected NOMAD_SECRETS_DIR=%q but found=%t value=%q", ok, v) + } + + // Assert cgroup membership + out, code, err = handle.Exec(context.Background(), "/bin/cat", []string{"/proc/self/cgroup"}) + if err != nil { + t.Fatalf("error exec'ing cat /proc/self/cgroup: %v", err) + } + if code != 0 { + t.Fatalf("expected `cat /proc/self/cgroup` to succeed but exit code was: %d", code) + } + found := false + for _, line := range strings.Split(string(out), "\n") { + // Every cgroup entry should be /nomad/$ALLOC_ID + if line == "" { + continue + } + if !strings.Contains(line, ":/nomad/") { + t.Errorf("Not a member of the alloc's cgroup: expected=...:/nomad/... -- found=%q", line) + continue + } + found = true + } + if !found { + t.Errorf("exec'd command isn't in the task's cgroup") } // Exec a command that should fail - out, code, err = handle.Exec(context.TODO(), "/usr/bin/stat", []string{"lkjhdsaflkjshowaisxmcvnlia"}) + out, code, err = handle.Exec(context.Background(), "/usr/bin/stat", []string{"lkjhdsaflkjshowaisxmcvnlia"}) if err != nil { t.Fatalf("error exec'ing stat: %v", err) } From 2cc492c95e06a74b8564146c230d26dc024977f2 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Tue, 25 Apr 2017 11:41:03 -0700 Subject: [PATCH 6/9] Test pre-0.6 script check upgrade path --- client/alloc_runner_test.go | 79 +++++++++++++++++++++++++++++++++++++ client/task_runner.go | 10 ++++- client/task_runner_test.go | 2 +- 3 files changed, 88 insertions(+), 3 deletions(-) diff --git a/client/alloc_runner_test.go b/client/alloc_runner_test.go index f1bc828b807..99b03f65989 100644 --- a/client/alloc_runner_test.go +++ b/client/alloc_runner_test.go @@ -5,6 +5,7 @@ import ( "io/ioutil" "os" "path/filepath" + "strings" "testing" "text/template" "time" @@ -476,6 +477,84 @@ func TestAllocRunner_SaveRestoreState_TerminalAlloc(t *testing.T) { }) } +// TestAllocRunner_SaveRestoreState_Upgrade asserts that pre-0.6 exec tasks are +// restarted on upgrade. +func TestAllocRunner_SaveRestoreState_Upgrade(t *testing.T) { + alloc := mock.Alloc() + task := alloc.Job.TaskGroups[0].Tasks[0] + task.Driver = "mock_driver" + task.Config = map[string]interface{}{ + "exit_code": "0", + "run_for": "10s", + } + + upd, ar := testAllocRunnerFromAlloc(alloc, false) + // Hack in old version to cause an upgrade on RestoreState + origConfig := ar.config.Copy() + ar.config.Version = "0.5.6" + go ar.Run() + + // Snapshot state + testutil.WaitForResult(func() (bool, error) { + return len(ar.tasks) == 1, nil + }, func(err error) { + t.Fatalf("task never started: %v", err) + }) + + err := ar.SaveState() + if err != nil { + t.Fatalf("err: %v", err) + } + + // Create a new alloc runner + l2 := prefixedTestLogger("----- ar2: ") + ar2 := NewAllocRunner(l2, origConfig, upd.Update, + &structs.Allocation{ID: ar.alloc.ID}, ar.vaultClient, + ar.consulClient) + err = ar2.RestoreState() + if err != nil { + t.Fatalf("err: %v", err) + } + go ar2.Run() + + testutil.WaitForResult(func() (bool, error) { + if len(ar2.tasks) != 1 { + return false, fmt.Errorf("Incorrect number of tasks") + } + + if upd.Count < 3 { + return false, nil + } + + for _, ev := range ar2.alloc.TaskStates["web"].Events { + if strings.HasSuffix(ev.RestartReason, pre06ScriptCheckReason) { + return true, nil + } + } + return false, fmt.Errorf("no restart with proper reason found") + }, func(err error) { + t.Fatalf("err: %v\nAllocs: %#v\nWeb State: %#v", err, upd.Allocs, ar2.alloc.TaskStates["web"]) + }) + + // Destroy and wait + ar2.Destroy() + start := time.Now() + + testutil.WaitForResult(func() (bool, error) { + alloc := ar2.Alloc() + if alloc.ClientStatus != structs.AllocClientStatusComplete { + return false, fmt.Errorf("Bad client status; got %v; want %v", alloc.ClientStatus, structs.AllocClientStatusComplete) + } + return true, nil + }, func(err error) { + t.Fatalf("err: %v %#v %#v", err, upd.Allocs[0], ar.alloc.TaskStates) + }) + + if time.Since(start) > time.Duration(testutil.TestMultiplier()*5)*time.Second { + t.Fatalf("took too long to terminate") + } +} + // Ensure pre-#2132 state files containing the Context struct are properly // migrated to the new format. // diff --git a/client/task_runner.go b/client/task_runner.go index b88bc6a7d20..f3715f338bf 100644 --- a/client/task_runner.go +++ b/client/task_runner.go @@ -298,7 +298,7 @@ func (r *TaskRunner) RestoreState() (string, error) { } if pre06ScriptCheck(snap.Version, r.task.Driver, r.task.Services) { - restartReason = "upgrading pre-0.6 script checks" + restartReason = pre06ScriptCheckReason } if err := r.registerServices(d, handle); err != nil { @@ -320,11 +320,16 @@ func (r *TaskRunner) RestoreState() (string, error) { return restartReason, nil } +// ver06 is used for checking for pre-0.6 script checks var ver06 = version.Must(version.NewVersion("0.6.0dev")) +// pre06ScriptCheckReason is the restart reason given when a pre-0.6 script +// check is found on an exec/java task. +const pre06ScriptCheckReason = "upgrading pre-0.6 script checks" + // pre06ScriptCheck returns true if version is prior to 0.6.0dev. func pre06ScriptCheck(ver, driver string, services []*structs.Service) bool { - if driver != "exec" && driver != "java" { + if driver != "exec" && driver != "java" && driver != "mock_driver" { // Only exec and java are affected return false } @@ -352,6 +357,7 @@ func (r *TaskRunner) SaveState() error { r.persistLock.Lock() defer r.persistLock.Unlock() + r.logger.Printf("[XXX] task_runner: %q", r.config.Version) snap := taskRunnerState{ Task: r.task, Version: r.config.Version, diff --git a/client/task_runner_test.go b/client/task_runner_test.go index ede8cb1647c..b373d6db874 100644 --- a/client/task_runner_test.go +++ b/client/task_runner_test.go @@ -369,7 +369,7 @@ func TestTaskRunner_SaveRestoreState(t *testing.T) { tr2 := NewTaskRunner(ctx.tr.logger, ctx.tr.config, ctx.upd.Update, ctx.tr.taskDir, ctx.tr.alloc, task2, ctx.tr.vaultClient, ctx.tr.consul) tr2.restartTracker = noRestartsTracker() - if err := tr2.RestoreState(); err != nil { + if _, err := tr2.RestoreState(); err != nil { t.Fatalf("err: %v", err) } go tr2.Run() From 426c6964047444eae7e1e9ffac36f5fd87182425 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Thu, 4 May 2017 16:21:40 -0700 Subject: [PATCH 7/9] Reuse ExecScript implementation --- client/driver/executor/executor.go | 16 ++++++++----- client/driver/raw_exec.go | 2 +- client/driver/rkt.go | 2 +- client/driver/rkt_test.go | 2 +- client/driver/utils.go | 36 ------------------------------ 5 files changed, 14 insertions(+), 44 deletions(-) diff --git a/client/driver/executor/executor.go b/client/driver/executor/executor.go index 295dc89283a..9cf3ad2b5a7 100644 --- a/client/driver/executor/executor.go +++ b/client/driver/executor/executor.go @@ -290,14 +290,20 @@ func (e *UniversalExecutor) LaunchCmd(command *ExecCommand) (*ProcessState, erro func (e *UniversalExecutor) Exec(deadline time.Time, name string, args []string) ([]byte, int, error) { ctx, cancel := context.WithDeadline(context.Background(), deadline) defer cancel() + return ExecScript(ctx, e.cmd.Dir, e.ctx.TaskEnv, e.cmd.SysProcAttr, name, args) +} - name = e.ctx.TaskEnv.ReplaceEnv(name) - cmd := exec.CommandContext(ctx, name, e.ctx.TaskEnv.ParseAndReplace(args)...) +// ExecScript executes cmd with args and returns the output, exit code, and +// error. Output is truncated to client/driver/structs.CheckBufSize +func ExecScript(ctx context.Context, dir string, env *env.TaskEnvironment, attrs *syscall.SysProcAttr, + name string, args []string) ([]byte, int, error) { + name = env.ReplaceEnv(name) + cmd := exec.CommandContext(ctx, name, env.ParseAndReplace(args)...) // Copy runtime environment from the main command - cmd.SysProcAttr = e.cmd.SysProcAttr - cmd.Dir = e.cmd.Dir - cmd.Env = e.ctx.TaskEnv.EnvList() + cmd.SysProcAttr = attrs + cmd.Dir = dir + cmd.Env = env.EnvList() // Capture output buf, _ := circbuf.NewBuffer(int64(dstructs.CheckBufSize)) diff --git a/client/driver/raw_exec.go b/client/driver/raw_exec.go index b799a32041d..558289fddae 100644 --- a/client/driver/raw_exec.go +++ b/client/driver/raw_exec.go @@ -255,7 +255,7 @@ func (h *rawExecHandle) Update(task *structs.Task) error { } func (h *rawExecHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte, int, error) { - return execScript(ctx, h.taskDir.Dir, h.taskEnv, cmd, args) + return executor.ExecScript(ctx, h.taskDir.Dir, h.taskEnv, nil, cmd, args) } func (h *rawExecHandle) Signal(s os.Signal) error { diff --git a/client/driver/rkt.go b/client/driver/rkt.go index 930d6672894..0c5e350efa0 100644 --- a/client/driver/rkt.go +++ b/client/driver/rkt.go @@ -573,7 +573,7 @@ func (h *rktHandle) Exec(ctx context.Context, cmd string, args []string) ([]byte enterArgs[1] = h.uuid enterArgs[2] = cmd copy(enterArgs[3:], args) - return execScript(ctx, h.taskDir.Dir, h.env, rktCmd, enterArgs) + return executor.ExecScript(ctx, h.taskDir.Dir, h.env, nil, rktCmd, enterArgs) } func (h *rktHandle) Signal(s os.Signal) error { diff --git a/client/driver/rkt_test.go b/client/driver/rkt_test.go index b61ba04af79..3dae05cf860 100644 --- a/client/driver/rkt_test.go +++ b/client/driver/rkt_test.go @@ -22,7 +22,7 @@ import ( func TestRktVersionRegex(t *testing.T) { if os.Getenv("NOMAD_TEST_RKT") == "" { - t.Skip("skipping rkt tests") + t.Skip("NOMAD_TEST_RKT unset, skipping") } input_rkt := "rkt version 0.8.1" diff --git a/client/driver/utils.go b/client/driver/utils.go index 07864bae411..fb61e1f6f16 100644 --- a/client/driver/utils.go +++ b/client/driver/utils.go @@ -1,7 +1,6 @@ package driver import ( - "context" "encoding/json" "fmt" "io" @@ -9,14 +8,11 @@ import ( "os/exec" "path/filepath" "strings" - "syscall" "time" - "github.com/armon/circbuf" "github.com/hashicorp/go-multierror" "github.com/hashicorp/go-plugin" "github.com/hashicorp/nomad/client/config" - "github.com/hashicorp/nomad/client/driver/env" "github.com/hashicorp/nomad/client/driver/executor" cstructs "github.com/hashicorp/nomad/client/driver/structs" "github.com/hashicorp/nomad/helper/discover" @@ -182,35 +178,3 @@ func getExecutorUser(task *structs.Task) string { } return task.User } - -// execScript executes cmd with args and returns the output, exit code, and -// error. Output is truncated to client/driver/structs.CheckBufSize -func execScript(ctx context.Context, dir string, env *env.TaskEnvironment, name string, args []string) ([]byte, int, error) { - name = env.ReplaceEnv(name) - args = env.ParseAndReplace(args) - cmd := exec.CommandContext(ctx, name, args...) - cmd.Dir = dir - cmd.Env = env.EnvList() - buf, _ := circbuf.NewBuffer(int64(cstructs.CheckBufSize)) - cmd.Stdout = buf - cmd.Stderr = buf - if err := cmd.Run(); err != nil { - exitErr, ok := err.(*exec.ExitError) - if !ok { - // Non-exit error, return it and let the caller treat - // it as a critical failure - return nil, 0, err - } - - // Some kind of error happened; default to critical - exitCode := 2 - if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { - exitCode = status.ExitStatus() - } - - // Don't return the exitError as the caller only needs the - // output and code. - return buf.Bytes(), exitCode, nil - } - return buf.Bytes(), 0, nil -} From 90123574141c72a3ae1a6bf60e940101a75b82e7 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Thu, 4 May 2017 16:27:04 -0700 Subject: [PATCH 8/9] Oops, remove dev logging --- client/task_runner.go | 1 - 1 file changed, 1 deletion(-) diff --git a/client/task_runner.go b/client/task_runner.go index f3715f338bf..c68ff7287a6 100644 --- a/client/task_runner.go +++ b/client/task_runner.go @@ -357,7 +357,6 @@ func (r *TaskRunner) SaveState() error { r.persistLock.Lock() defer r.persistLock.Unlock() - r.logger.Printf("[XXX] task_runner: %q", r.config.Version) snap := taskRunnerState{ Task: r.task, Version: r.config.Version, From a39eef0f113ca415f6af6b38891ed6871afd5ba5 Mon Sep 17 00:00:00 2001 From: Michael Schurter Date: Thu, 4 May 2017 16:49:00 -0700 Subject: [PATCH 9/9] Test pre06ScriptCheck --- client/task_runner.go | 3 ++- client/task_runner_test.go | 46 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 48 insertions(+), 1 deletion(-) diff --git a/client/task_runner.go b/client/task_runner.go index c68ff7287a6..5eecdb8f8a9 100644 --- a/client/task_runner.go +++ b/client/task_runner.go @@ -327,7 +327,8 @@ var ver06 = version.Must(version.NewVersion("0.6.0dev")) // check is found on an exec/java task. const pre06ScriptCheckReason = "upgrading pre-0.6 script checks" -// pre06ScriptCheck returns true if version is prior to 0.6.0dev. +// pre06ScriptCheck returns true if version is prior to 0.6.0dev, has a script +// check, and uses exec or java drivers. func pre06ScriptCheck(ver, driver string, services []*structs.Service) bool { if driver != "exec" && driver != "java" && driver != "mock_driver" { // Only exec and java are affected diff --git a/client/task_runner_test.go b/client/task_runner_test.go index b373d6db874..cd818fde8dd 100644 --- a/client/task_runner_test.go +++ b/client/task_runner_test.go @@ -1520,3 +1520,49 @@ func TestTaskRunner_CleanupFail(t *testing.T) { t.Fatalf("expected %#v but found: %#v", expected, ctx.tr.createdResources.Resources) } } + +func TestTaskRunner_Pre06ScriptCheck(t *testing.T) { + run := func(ver, driver, checkType string, exp bool) (string, func(t *testing.T)) { + name := fmt.Sprintf("%s %s %s returns %t", ver, driver, checkType, exp) + return name, func(t *testing.T) { + services := []*structs.Service{ + { + Checks: []*structs.ServiceCheck{ + { + Type: checkType, + }, + }, + }, + } + if act := pre06ScriptCheck(ver, driver, services); act != exp { + t.Errorf("expected %t received %t", exp, act) + } + } + } + t.Run(run("0.5.6", "exec", "script", true)) + t.Run(run("0.5.6", "java", "script", true)) + t.Run(run("0.5.6", "mock_driver", "script", true)) + t.Run(run("0.5.9", "exec", "script", true)) + t.Run(run("0.5.9", "java", "script", true)) + t.Run(run("0.5.9", "mock_driver", "script", true)) + + t.Run(run("0.6.0dev", "exec", "script", false)) + t.Run(run("0.6.0dev", "java", "script", false)) + t.Run(run("0.6.0dev", "mock_driver", "script", false)) + t.Run(run("0.6.0", "exec", "script", false)) + t.Run(run("0.6.0", "java", "script", false)) + t.Run(run("0.6.0", "mock_driver", "script", false)) + t.Run(run("1.0.0", "exec", "script", false)) + t.Run(run("1.0.0", "java", "script", false)) + t.Run(run("1.0.0", "mock_driver", "script", false)) + + t.Run(run("0.5.6", "rkt", "script", false)) + t.Run(run("0.5.6", "docker", "script", false)) + t.Run(run("0.5.6", "qemu", "script", false)) + t.Run(run("0.5.6", "raw_exec", "script", false)) + t.Run(run("0.5.6", "invalid", "script", false)) + + t.Run(run("0.5.6", "exec", "tcp", false)) + t.Run(run("0.5.6", "java", "tcp", false)) + t.Run(run("0.5.6", "mock_driver", "tcp", false)) +}