Skip to content

Commit ac01f20

Browse files
authored
Merge pull request #2577 from samuelkarp/stop-signal
ctr: make kill optionally use stop-signal
2 parents 655ba65 + 607888c commit ac01f20

8 files changed

+144
-31
lines changed

cmd/ctr/commands/run/run_unix.go

+4-3
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ import (
2727
"github.com/containerd/containerd/cmd/ctr/commands"
2828
"github.com/containerd/containerd/contrib/nvidia"
2929
"github.com/containerd/containerd/oci"
30-
specs "github.com/opencontainers/runtime-spec/specs-go"
30+
"github.com/opencontainers/runtime-spec/specs-go"
3131
"github.com/pkg/errors"
3232
"github.com/urfave/cli"
3333
)
@@ -58,6 +58,7 @@ func NewContainer(ctx gocontext.Context, client *containerd.Client, context *cli
5858
spec containerd.NewContainerOpts
5959
)
6060

61+
cOpts = append(cOpts, containerd.WithContainerLabels(commands.LabelArgs(context.StringSlice("label"))))
6162
if config {
6263
opts = append(opts, oci.WithSpecFromFile(context.String("config")))
6364
} else {
@@ -98,7 +99,8 @@ func NewContainer(ctx gocontext.Context, client *containerd.Client, context *cli
9899
// Even when "readonly" is set, we don't use KindView snapshot here. (#1495)
99100
// We pass writable snapshot to the OCI runtime, and the runtime remounts it as read-only,
100101
// after creating some mount points on demand.
101-
containerd.WithNewSnapshot(id, image))
102+
containerd.WithNewSnapshot(id, image),
103+
containerd.WithImageStopSignal(image, "SIGTERM"))
102104
}
103105
if context.Bool("readonly") {
104106
opts = append(opts, oci.WithRootFSReadonly())
@@ -141,7 +143,6 @@ func NewContainer(ctx gocontext.Context, client *containerd.Client, context *cli
141143
}
142144
}
143145

144-
cOpts = append(cOpts, containerd.WithContainerLabels(commands.LabelArgs(context.StringSlice("label"))))
145146
cOpts = append(cOpts, containerd.WithRuntime(context.String("runtime"), nil))
146147

147148
var s specs.Spec

cmd/ctr/commands/signals.go

-23
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,8 @@ package commands
1818

1919
import (
2020
gocontext "context"
21-
"fmt"
2221
"os"
2322
"os/signal"
24-
"strconv"
25-
"strings"
2623
"syscall"
2724

2825
"github.com/containerd/containerd"
@@ -53,23 +50,3 @@ func StopCatch(sigc chan os.Signal) {
5350
signal.Stop(sigc)
5451
close(sigc)
5552
}
56-
57-
// ParseSignal parses a given string into a syscall.Signal
58-
// it checks that the signal exists in the platform-appropriate signalMap
59-
func ParseSignal(rawSignal string) (syscall.Signal, error) {
60-
s, err := strconv.Atoi(rawSignal)
61-
if err == nil {
62-
sig := syscall.Signal(s)
63-
for _, msig := range signalMap {
64-
if sig == msig {
65-
return sig, nil
66-
}
67-
}
68-
return -1, fmt.Errorf("unknown signal %q", rawSignal)
69-
}
70-
signal, ok := signalMap[strings.TrimPrefix(strings.ToUpper(rawSignal), "SIG")]
71-
if !ok {
72-
return -1, fmt.Errorf("unknown signal %q", rawSignal)
73-
}
74-
return signal, nil
75-
}

cmd/ctr/commands/tasks/kill.go

+15-2
Original file line numberDiff line numberDiff line change
@@ -23,14 +23,16 @@ import (
2323
"github.com/urfave/cli"
2424
)
2525

26+
const defaultSignal = "SIGTERM"
27+
2628
var killCommand = cli.Command{
2729
Name: "kill",
2830
Usage: "signal a container (default: SIGTERM)",
2931
ArgsUsage: "[flags] CONTAINER",
3032
Flags: []cli.Flag{
3133
cli.StringFlag{
3234
Name: "signal, s",
33-
Value: "SIGTERM",
35+
Value: "",
3436
Usage: "signal to send to the container",
3537
},
3638
cli.StringFlag{
@@ -47,7 +49,7 @@ var killCommand = cli.Command{
4749
if id == "" {
4850
return errors.New("container id must be provided")
4951
}
50-
signal, err := commands.ParseSignal(context.String("signal"))
52+
signal, err := containerd.ParseSignal(defaultSignal)
5153
if err != nil {
5254
return err
5355
}
@@ -74,6 +76,17 @@ var killCommand = cli.Command{
7476
if err != nil {
7577
return err
7678
}
79+
if context.String("signal") != "" {
80+
signal, err = containerd.ParseSignal(context.String("signal"))
81+
if err != nil {
82+
return err
83+
}
84+
} else {
85+
signal, err = containerd.GetStopSignal(ctx, container, signal)
86+
if err != nil {
87+
return err
88+
}
89+
}
7790
task, err := container.Task(ctx, nil)
7891
if err != nil {
7992
return err

container_opts.go

+17
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,23 @@ func WithContainerLabels(labels map[string]string) NewContainerOpts {
7676
}
7777
}
7878

79+
// WithImageStopSignal sets a well-known containerd label (StopSignalLabel)
80+
// on the container for storing the stop signal specified in the OCI image
81+
// config
82+
func WithImageStopSignal(image Image, defaultSignal string) NewContainerOpts {
83+
return func(ctx context.Context, _ *Client, c *containers.Container) error {
84+
if c.Labels == nil {
85+
c.Labels = make(map[string]string)
86+
}
87+
stopSignal, err := GetOCIStopSignal(ctx, image, defaultSignal)
88+
if err != nil {
89+
return err
90+
}
91+
c.Labels[StopSignalLabel] = stopSignal
92+
return nil
93+
}
94+
}
95+
7996
// WithSnapshotter sets the provided snapshotter for use by the container
8097
//
8198
// This option must appear before other snapshotter options to have an effect.

cmd/ctr/commands/signal_map_linux.go signal_map_linux.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
limitations under the License.
1515
*/
1616

17-
package commands
17+
package containerd
1818

1919
import (
2020
"syscall"

cmd/ctr/commands/signal_map_unix.go signal_map_unix.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
limitations under the License.
1717
*/
1818

19-
package commands
19+
package containerd
2020

2121
import (
2222
"syscall"

cmd/ctr/commands/signal_map_windows.go signal_map_windows.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
limitations under the License.
1515
*/
1616

17-
package commands
17+
package containerd
1818

1919
import (
2020
"syscall"

signals.go

+105
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
1+
/*
2+
Copyright The containerd Authors.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package containerd
18+
19+
import (
20+
"context"
21+
"encoding/json"
22+
"fmt"
23+
"strconv"
24+
"strings"
25+
"syscall"
26+
27+
"github.com/containerd/containerd/content"
28+
"github.com/containerd/containerd/images"
29+
"github.com/opencontainers/image-spec/specs-go/v1"
30+
)
31+
32+
// StopSignalLabel is a well-known containerd label for storing the stop
33+
// signal specified in the OCI image config
34+
const StopSignalLabel = "io.containerd.image.config.stop-signal"
35+
36+
// GetStopSignal retrieves the container stop signal, specified by the
37+
// well-known containerd label (StopSignalLabel)
38+
func GetStopSignal(ctx context.Context, container Container, defaultSignal syscall.Signal) (syscall.Signal, error) {
39+
labels, err := container.Labels(ctx)
40+
if err != nil {
41+
return -1, err
42+
}
43+
44+
if stopSignal, ok := labels[StopSignalLabel]; ok {
45+
return ParseSignal(stopSignal)
46+
}
47+
48+
return defaultSignal, nil
49+
}
50+
51+
// GetOCIStopSignal retrieves the stop signal specified in the OCI image config
52+
func GetOCIStopSignal(ctx context.Context, image Image, defaultSignal string) (string, error) {
53+
_, err := ParseSignal(defaultSignal)
54+
if err != nil {
55+
return "", err
56+
}
57+
ic, err := image.Config(ctx)
58+
if err != nil {
59+
return "", err
60+
}
61+
var (
62+
ociimage v1.Image
63+
config v1.ImageConfig
64+
)
65+
switch ic.MediaType {
66+
case v1.MediaTypeImageConfig, images.MediaTypeDockerSchema2Config:
67+
p, err := content.ReadBlob(ctx, image.ContentStore(), ic)
68+
if err != nil {
69+
return "", err
70+
}
71+
72+
if err := json.Unmarshal(p, &ociimage); err != nil {
73+
return "", err
74+
}
75+
config = ociimage.Config
76+
default:
77+
return "", fmt.Errorf("unknown image config media type %s", ic.MediaType)
78+
}
79+
80+
if config.StopSignal == "" {
81+
return defaultSignal, nil
82+
}
83+
84+
return config.StopSignal, nil
85+
}
86+
87+
// ParseSignal parses a given string into a syscall.Signal
88+
// it checks that the signal exists in the platform-appropriate signalMap
89+
func ParseSignal(rawSignal string) (syscall.Signal, error) {
90+
s, err := strconv.Atoi(rawSignal)
91+
if err == nil {
92+
sig := syscall.Signal(s)
93+
for _, msig := range signalMap {
94+
if sig == msig {
95+
return sig, nil
96+
}
97+
}
98+
return -1, fmt.Errorf("unknown signal %q", rawSignal)
99+
}
100+
signal, ok := signalMap[strings.TrimPrefix(strings.ToUpper(rawSignal), "SIG")]
101+
if !ok {
102+
return -1, fmt.Errorf("unknown signal %q", rawSignal)
103+
}
104+
return signal, nil
105+
}

0 commit comments

Comments
 (0)