Skip to content

Commit

Permalink
implement kill tests
Browse files Browse the repository at this point in the history
Signed-off-by: Liang Chenye <[email protected]>
  • Loading branch information
liangchenye committed Feb 26, 2018
1 parent 4902e9c commit 55d7e14
Show file tree
Hide file tree
Showing 4 changed files with 161 additions and 1 deletion.
75 changes: 75 additions & 0 deletions validation/kill.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
package main

import (
"fmt"
"os"
"time"

"github.com/mndrix/tap-go"
rspecs "github.com/opencontainers/runtime-spec/specs-go"
"github.com/opencontainers/runtime-tools/generate"
"github.com/opencontainers/runtime-tools/specerror"
"github.com/opencontainers/runtime-tools/validation/util"
uuid "github.com/satori/go.uuid"
)

func main() {
t := tap.New()
t.Header(0)
bundleDir, err := util.PrepareBundle()
if err != nil {
util.Fatal(err)
}
defer os.RemoveAll(bundleDir)

stoppedConfig := util.GetDefaultGenerator()
stoppedConfig.SetProcessArgs([]string{"true"})
runningConfig := util.GetDefaultGenerator()
runningConfig.SetProcessArgs([]string{"sleep", "30"})
containerID := uuid.NewV4().String()

cases := []struct {
config *generate.Generator
id string
action util.LifecycleAction
errExpected bool
err error
}{
// Note: the nil config test case should run first since we are re-using the bundle
// kill without id
{nil, "", util.LifecycleActionNone, false, specerror.NewError(specerror.KillWithoutIDGenError, fmt.Errorf("`kill` operation MUST generate an error if it is not provided the container ID"), rspecs.Version)},
// kill a non exist container
{nil, containerID, util.LifecycleActionNone, false, specerror.NewError(specerror.KillNonCreateRunGenError, fmt.Errorf("attempting to send a signal to a container that is neither `created` nor `running` MUST generate an error"), rspecs.Version)},
// kill a created
{stoppedConfig, containerID, util.LifecycleActionCreate | util.LifecycleActionDelete, true, specerror.NewError(specerror.KillSignalImplement, fmt.Errorf("`kill` operation MUST send the specified signal to the container process"), rspecs.Version)},
// kill a stopped
{stoppedConfig, containerID, util.LifecycleActionCreate | util.LifecycleActionStart | util.LifecycleActionDelete, false, specerror.NewError(specerror.KillSignalImplement, fmt.Errorf("`kill` operation MUST send the specified signal to the container process"), rspecs.Version)},
// kill a running
{runningConfig, containerID, util.LifecycleActionCreate | util.LifecycleActionStart | util.LifecycleActionDelete, true, specerror.NewError(specerror.KillSignalImplement, fmt.Errorf("`kill` operation MUST send the specified signal to the container process"), rspecs.Version)},
}

for _, c := range cases {
config := util.LifecycleConfig{
Config: c.config,
BundleDir: bundleDir,
Actions: c.action,
PreCreate: func(r *util.Runtime) error {
r.SetID(c.id)
return nil
},
PreDelete: func(r *util.Runtime) error {
// waiting the 'stoppedConfig' testcase to stop
// the 'runningConfig' testcase sleeps 30 seconds, so 10 seconds are enough for this case
util.WaitingForStatus(*r, util.LifecycleStatusCreated|util.LifecycleStatusStopped, time.Second*10, time.Second*1)
// KILL MUST be supported and KILL cannot be trapped
err := r.Kill("KILL")
util.WaitingForStatus(*r, util.LifecycleStatusStopped, time.Second*10, time.Second*1)
return err
},
}
err := util.RuntimeLifecycleValidate(config)
util.SpecErrorOK(t, (err == nil) == c.errExpected, c.err, err)
}

t.AutoPlan()
}
59 changes: 59 additions & 0 deletions validation/killsig.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
package main

import (
"fmt"
"os"
"path/filepath"
"time"

"github.com/mndrix/tap-go"
rspecs "github.com/opencontainers/runtime-spec/specs-go"
"github.com/opencontainers/runtime-tools/specerror"
"github.com/opencontainers/runtime-tools/validation/util"
uuid "github.com/satori/go.uuid"
)

var signals = []string{
"TERM",
}

func main() {
t := tap.New()
t.Header(0)
bundleDir, err := util.PrepareBundle()
if err != nil {
util.Fatal(err)
}
defer os.RemoveAll(bundleDir)

containerID := uuid.NewV4().String()
sigConfig := util.GetDefaultGenerator()
rootDir := filepath.Join(bundleDir, sigConfig.Spec().Root.Path)
for _, signal := range signals {
sigConfig.SetProcessArgs([]string{"sh", "-c", fmt.Sprintf("trap 'touch /%s' %s; sleep 10 & wait $!", signal, signal)})
config := util.LifecycleConfig{
Config: sigConfig,
BundleDir: bundleDir,
Actions: util.LifecycleActionCreate | util.LifecycleActionStart | util.LifecycleActionDelete,
PreCreate: func(r *util.Runtime) error {
r.SetID(containerID)
return nil
},
PreDelete: func(r *util.Runtime) error {
util.WaitingForStatus(*r, util.LifecycleStatusRunning, time.Second*5, time.Second*1)
err := r.Kill(signal)
// wait before the container been deleted
util.WaitingForStatus(*r, util.LifecycleStatusStopped, time.Second*5, time.Second*1)
return err
},
}
err := util.RuntimeLifecycleValidate(config)
if err != nil {
util.SpecErrorOK(t, false, specerror.NewError(specerror.KillSignalImplement, fmt.Errorf("`kill` operation MUST send the specified signal to the container process"), rspecs.Version), err)
} else {
_, err = os.Stat(filepath.Join(rootDir, signal))
util.SpecErrorOK(t, err == nil, specerror.NewError(specerror.KillSignalImplement, fmt.Errorf("`kill` operation MUST send the specified signal to the container process"), rspecs.Version), err)
}
}
t.AutoPlan()
}
22 changes: 22 additions & 0 deletions validation/util/container.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@ type Runtime struct {
stderr *os.File
}

// DefaultSignal represents the default signal sends to a container
const DefaultSignal = "TERM"

// NewRuntime create a runtime by command and the bundle directory
func NewRuntime(runtimeCommand string, bundleDir string) (Runtime, error) {
var r Runtime
Expand Down Expand Up @@ -158,6 +161,25 @@ func (r *Runtime) State() (rspecs.State, error) {
return state, err
}

// Kill a container
func (r *Runtime) Kill(sig string) (err error) {
var args []string
args = append(args, "kill")
if r.ID != "" {
args = append(args, r.ID)
}
if sig != "" {
// TODO: runc does not support this
// args = append(args, "--signal", sig)
args = append(args, sig)
} else {
args = append(args, DefaultSignal)
}

cmd := exec.Command(r.RuntimeCommand, args...)
return execWithStderrFallbackToStdout(cmd)
}

// Delete a container
func (r *Runtime) Delete() (err error) {
var args []string
Expand Down
6 changes: 5 additions & 1 deletion validation/util/test.go
Original file line number Diff line number Diff line change
Expand Up @@ -327,8 +327,12 @@ func RuntimeLifecycleValidate(config LifecycleConfig) error {
if _, err := r.State(); err != nil {
return
}
if err := WaitingForStatus(r, LifecycleStatusCreated|LifecycleStatusStopped, time.Second*10, time.Second*1); err != nil {
err := WaitingForStatus(r, LifecycleStatusCreated|LifecycleStatusStopped, time.Second*10, time.Second*1)
if err == nil {
r.Delete()
} else {
os.Stderr.WriteString("failed to delete the container\n")
os.Stderr.WriteString(err.Error())
}
}()
}
Expand Down

0 comments on commit 55d7e14

Please sign in to comment.