Skip to content

Commit 5445572

Browse files
committed
Add oci-hook command to containerd
This allows many different commands to be used as OCI hooks. It allows these commands to template out different args and env vars so that normal commands can accept the OCI spec State payload over stdin. Signed-off-by: Michael Crosby <[email protected]>
1 parent a15e7a0 commit 5445572

File tree

3 files changed

+192
-0
lines changed

3 files changed

+192
-0
lines changed

cmd/containerd/command/main.go

+1
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,7 @@ func App() *cli.App {
8989
app.Commands = []cli.Command{
9090
configCommand,
9191
publishCommand,
92+
ociHook,
9293
}
9394
app.Action = func(context *cli.Context) error {
9495
var (

cmd/containerd/command/oci-hook.go

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
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 command
18+
19+
import (
20+
"bytes"
21+
"encoding/json"
22+
"io"
23+
"os"
24+
"path/filepath"
25+
"syscall"
26+
"text/template"
27+
28+
specs "github.com/opencontainers/runtime-spec/specs-go"
29+
"github.com/urfave/cli"
30+
)
31+
32+
var ociHook = cli.Command{
33+
Name: "oci-hook",
34+
Usage: "provides a base for OCI runtime hooks that allow arguements to be templated.",
35+
Action: func(context *cli.Context) error {
36+
state, err := loadHookState(os.Stdin)
37+
if err != nil {
38+
return err
39+
}
40+
var (
41+
ctx = newTemplateContext(state)
42+
args = []string(context.Args())
43+
env = os.Environ()
44+
)
45+
if err := newList(&args).render(ctx); err != nil {
46+
return err
47+
}
48+
if err := newList(&env).render(ctx); err != nil {
49+
return err
50+
}
51+
return syscall.Exec(args[0], args, env)
52+
},
53+
}
54+
55+
func loadHookState(r io.Reader) (*specs.State, error) {
56+
var s specs.State
57+
if err := json.NewDecoder(r).Decode(&s); err != nil {
58+
return nil, err
59+
}
60+
return &s, nil
61+
}
62+
63+
func newTemplateContext(state *specs.State) *templateContext {
64+
t := &templateContext{
65+
state: state,
66+
}
67+
t.funcs = template.FuncMap{
68+
"id": t.id,
69+
"bundle": t.bundle,
70+
"rootfs": t.rootfs,
71+
"pid": t.pid,
72+
"annotation": t.annotation,
73+
}
74+
return t
75+
}
76+
77+
type templateContext struct {
78+
state *specs.State
79+
funcs template.FuncMap
80+
}
81+
82+
func (t *templateContext) id() string {
83+
return t.state.ID
84+
}
85+
86+
func (t *templateContext) bundle() string {
87+
return t.state.Bundle
88+
}
89+
90+
func (t *templateContext) rootfs() string {
91+
return filepath.Join(t.state.Bundle, "rootfs")
92+
}
93+
94+
func (t *templateContext) pid() int {
95+
return t.state.Pid
96+
}
97+
98+
func (t *templateContext) annotation(k string) string {
99+
return t.state.Annotations[k]
100+
}
101+
102+
func render(ctx *templateContext, source string, out io.Writer) error {
103+
t, err := template.New("oci-hook").Funcs(ctx.funcs).Parse(source)
104+
if err != nil {
105+
return err
106+
}
107+
return t.Execute(out, ctx)
108+
}
109+
110+
func newList(l *[]string) *templateList {
111+
return &templateList{
112+
l: l,
113+
}
114+
}
115+
116+
type templateList struct {
117+
l *[]string
118+
}
119+
120+
func (l *templateList) render(ctx *templateContext) error {
121+
buf := bytes.NewBuffer(nil)
122+
for i, s := range *l.l {
123+
buf.Reset()
124+
if err := render(ctx, s, buf); err != nil {
125+
return err
126+
}
127+
(*l.l)[i] = buf.String()
128+
}
129+
buf.Reset()
130+
return nil
131+
}

container_test.go

+60
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import (
2222
"io"
2323
"io/ioutil"
2424
"os"
25+
"os/exec"
2526
"runtime"
2627
"strings"
2728
"syscall"
@@ -34,6 +35,7 @@ import (
3435
"github.com/containerd/containerd/oci"
3536
_ "github.com/containerd/containerd/runtime"
3637
"github.com/containerd/typeurl"
38+
specs "github.com/opencontainers/runtime-spec/specs-go"
3739

3840
"github.com/containerd/containerd/errdefs"
3941
"github.com/containerd/containerd/windows/hcsshimtypes"
@@ -1469,3 +1471,61 @@ func TestContainerLabels(t *testing.T) {
14691471
t.Fatalf("expected label \"test\" to be \"no\"")
14701472
}
14711473
}
1474+
1475+
func TestContainerHook(t *testing.T) {
1476+
t.Parallel()
1477+
1478+
client, err := newClient(t, address)
1479+
if err != nil {
1480+
t.Fatal(err)
1481+
}
1482+
defer client.Close()
1483+
1484+
var (
1485+
image Image
1486+
ctx, cancel = testContext()
1487+
id = t.Name()
1488+
)
1489+
defer cancel()
1490+
1491+
image, err = client.GetImage(ctx, testImage)
1492+
if err != nil {
1493+
t.Fatal(err)
1494+
}
1495+
hook := func(_ context.Context, _ oci.Client, _ *containers.Container, s *specs.Spec) error {
1496+
if s.Hooks == nil {
1497+
s.Hooks = &specs.Hooks{}
1498+
}
1499+
path, err := exec.LookPath("containerd")
1500+
if err != nil {
1501+
return err
1502+
}
1503+
psPath, err := exec.LookPath("ps")
1504+
if err != nil {
1505+
return err
1506+
}
1507+
s.Hooks.Prestart = []specs.Hook{
1508+
{
1509+
Path: path,
1510+
Args: []string{
1511+
"containerd",
1512+
"oci-hook", "--",
1513+
psPath, "--pid", "{{pid}}",
1514+
},
1515+
Env: os.Environ(),
1516+
},
1517+
}
1518+
return nil
1519+
}
1520+
container, err := client.NewContainer(ctx, id, WithNewSpec(oci.WithImageConfig(image), hook), WithNewSnapshot(id, image))
1521+
if err != nil {
1522+
t.Fatal(err)
1523+
}
1524+
defer container.Delete(ctx, WithSnapshotCleanup)
1525+
1526+
task, err := container.NewTask(ctx, empty())
1527+
if err != nil {
1528+
t.Fatal(err)
1529+
}
1530+
defer task.Delete(ctx, WithProcessKill)
1531+
}

0 commit comments

Comments
 (0)