Skip to content

Commit 1ba4aa0

Browse files
authored
Merge pull request #2528 from crosbymichael/shim-debug
Add shim log pipe for log forwarding to the daemon
2 parents 4fb9230 + 6ba4ddf commit 1ba4aa0

File tree

9 files changed

+178
-19
lines changed

9 files changed

+178
-19
lines changed

cmd/containerd-shim-runc-v1/main.go

+1-7
Original file line numberDiff line numberDiff line change
@@ -19,16 +19,10 @@
1919
package main
2020

2121
import (
22-
"fmt"
23-
"os"
24-
2522
"github.com/containerd/containerd/runtime/v2/runc"
2623
"github.com/containerd/containerd/runtime/v2/shim"
2724
)
2825

2926
func main() {
30-
if err := shim.Run(runc.New); err != nil {
31-
fmt.Fprintf(os.Stderr, "containerd-shim-runc-v1: %s\n", err)
32-
os.Exit(1)
33-
}
27+
shim.Run("io.containerd.runc.v1", runc.New)
3428
}

runtime/v2/README.md

+7
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,13 @@ If the shim collects Out of Memory events, it SHOULD also publish a `runtime.Tas
157157

158158
If a shim does not or cannot implement an rpc call, it MUST return a `github.com/containerd/containerd/errdefs.ErrNotImplemented` error.
159159

160+
#### Debugging and Shim Logs
161+
162+
A fifo on unix or named pipe on Windows will be provided to the shim.
163+
It can be located inside the `cwd` of the shim named "log".
164+
The shims can use the existing `github.com/containerd/containerd/log` package to log debug messages.
165+
Messages will automatically be output in the containerd's daemon logs with the correct fields and runtime set.
166+
160167
#### ttrpc
161168

162169
[ttrpc](https://github.com/containerd/ttrpc) is the only currently supported protocol for shims.

runtime/v2/binary.go

+29-2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,8 @@ package v2
1919
import (
2020
"bytes"
2121
"context"
22+
"io"
23+
"os"
2224
"strings"
2325

2426
eventstypes "github.com/containerd/containerd/api/events"
@@ -49,11 +51,36 @@ type binary struct {
4951
rtTasks *runtime.TaskList
5052
}
5153

52-
func (b *binary) Start(ctx context.Context) (*shim, error) {
53-
cmd, err := client.Command(ctx, b.runtime, b.containerdAddress, b.bundle.Path, "-id", b.bundle.ID, "start")
54+
func (b *binary) Start(ctx context.Context) (_ *shim, err error) {
55+
cmd, err := client.Command(
56+
ctx,
57+
b.runtime,
58+
b.containerdAddress,
59+
b.bundle.Path,
60+
"-id", b.bundle.ID,
61+
"start",
62+
)
5463
if err != nil {
5564
return nil, err
5665
}
66+
f, err := openShimLog(ctx, b.bundle)
67+
if err != nil {
68+
return nil, errors.Wrap(err, "open shim log pipe")
69+
}
70+
defer func() {
71+
if err != nil {
72+
f.Close()
73+
}
74+
}()
75+
// open the log pipe and block until the writer is ready
76+
// this helps with syncronization of the shim
77+
// copy the shim's logs to containerd's output
78+
go func() {
79+
defer f.Close()
80+
if _, err := io.Copy(os.Stderr, f); err != nil {
81+
log.G(ctx).WithError(err).Error("copy shim log")
82+
}
83+
}()
5784
out, err := cmd.CombinedOutput()
5885
if err != nil {
5986
return nil, errors.Wrapf(err, "%s", out)

runtime/v2/shim.go

+21
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,9 @@ package v2
1818

1919
import (
2020
"context"
21+
"io"
2122
"io/ioutil"
23+
"os"
2224
"path/filepath"
2325
"time"
2426

@@ -54,6 +56,25 @@ func loadShim(ctx context.Context, bundle *Bundle, events *exchange.Exchange, rt
5456
if err != nil {
5557
return nil, err
5658
}
59+
f, err := openShimLog(ctx, bundle)
60+
if err != nil {
61+
return nil, errors.Wrap(err, "open shim log pipe")
62+
}
63+
defer func() {
64+
if err != nil {
65+
f.Close()
66+
}
67+
}()
68+
// open the log pipe and block until the writer is ready
69+
// this helps with syncronization of the shim
70+
// copy the shim's logs to containerd's output
71+
go func() {
72+
defer f.Close()
73+
if _, err := io.Copy(os.Stderr, f); err != nil {
74+
log.G(ctx).WithError(err).Error("copy shim log")
75+
}
76+
}()
77+
5778
client := ttrpc.NewClient(conn)
5879
client.OnClose(func() { conn.Close() })
5980
s := &shim{

runtime/v2/shim/shim.go

+30-10
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ import (
2727
"time"
2828

2929
"github.com/containerd/containerd/events"
30+
"github.com/containerd/containerd/log"
3031
"github.com/containerd/containerd/namespaces"
3132
shimapi "github.com/containerd/containerd/runtime/v2/task"
3233
"github.com/containerd/ttrpc"
@@ -82,24 +83,38 @@ func setRuntime() {
8283
debug.FreeOSMemory()
8384
}
8485
}()
85-
if debugFlag {
86-
f, err := os.OpenFile("shim.log", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
87-
if err != nil {
88-
fmt.Fprintf(os.Stderr, "open shim log %s", err)
89-
os.Exit(1)
90-
}
91-
logrus.SetLevel(logrus.DebugLevel)
92-
logrus.SetOutput(f)
93-
}
9486
if os.Getenv("GOMAXPROCS") == "" {
9587
// If GOMAXPROCS hasn't been set, we default to a value of 2 to reduce
9688
// the number of Go stacks present in the shim.
9789
runtime.GOMAXPROCS(2)
9890
}
9991
}
10092

93+
func setLogger(ctx context.Context, id string) error {
94+
logrus.SetFormatter(&logrus.TextFormatter{
95+
TimestampFormat: log.RFC3339NanoFixed,
96+
FullTimestamp: true,
97+
})
98+
if debugFlag {
99+
logrus.SetLevel(logrus.DebugLevel)
100+
}
101+
f, err := openLog(ctx, id)
102+
if err != nil {
103+
return err
104+
}
105+
logrus.SetOutput(f)
106+
return nil
107+
}
108+
101109
// Run initializes and runs a shim server
102-
func Run(initFunc Init) error {
110+
func Run(id string, initFunc Init) {
111+
if err := run(id, initFunc); err != nil {
112+
fmt.Fprintf(os.Stderr, "%s: %s\n", id, err)
113+
os.Exit(1)
114+
}
115+
}
116+
117+
func run(id string, initFunc Init) error {
103118
parseFlags()
104119
setRuntime()
105120

@@ -118,6 +133,8 @@ func Run(initFunc Init) error {
118133
return fmt.Errorf("shim namespace cannot be empty")
119134
}
120135
ctx := namespaces.WithNamespace(context.Background(), namespaceFlag)
136+
ctx = log.WithLogger(ctx, log.G(ctx).WithField("runtime", id))
137+
121138
service, err := initFunc(ctx, idFlag, publisher)
122139
if err != nil {
123140
return err
@@ -151,6 +168,9 @@ func Run(initFunc Init) error {
151168
}
152169
return nil
153170
default:
171+
if err := setLogger(ctx, idFlag); err != nil {
172+
return err
173+
}
154174
client := NewShimClient(ctx, service, signals)
155175
return client.Serve()
156176
}

runtime/v2/shim/shim_unix.go

+6
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ package shim
2121
import (
2222
"bytes"
2323
"context"
24+
"io"
2425
"net"
2526
"os"
2627
"os/exec"
@@ -29,6 +30,7 @@ import (
2930

3031
"github.com/containerd/containerd/events"
3132
"github.com/containerd/containerd/namespaces"
33+
"github.com/containerd/fifo"
3234
"github.com/containerd/typeurl"
3335
"github.com/pkg/errors"
3436
"github.com/sirupsen/logrus"
@@ -84,6 +86,10 @@ func handleSignals(logger *logrus.Entry, signals chan os.Signal) error {
8486
}
8587
}
8688

89+
func openLog(ctx context.Context, _ string) (io.Writer, error) {
90+
return fifo.OpenFifo(context.Background(), "log", unix.O_WRONLY, 0700)
91+
}
92+
8793
func (l *remoteEventsPublisher) Publish(ctx context.Context, topic string, event events.Event) error {
8894
ns, _ := namespaces.Namespace(ctx)
8995
encoded, err := typeurl.MarshalAny(event)

runtime/v2/shim/shim_windows.go

+10
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,8 @@ package shim
2121
import (
2222
"bytes"
2323
"context"
24+
"fmt"
25+
"io"
2426
"net"
2527
"os"
2628
"os/exec"
@@ -79,6 +81,14 @@ func handleSignals(logger *logrus.Entry, signals chan os.Signal) error {
7981
}
8082
}
8183

84+
func openLog(ctx context.Context, id string) (io.Writer, error) {
85+
ns, err := namespaces.NamespaceRequired(ctx)
86+
if err != nil {
87+
return nil, err
88+
}
89+
return winio.DialPipe(fmt.Sprintf("\\\\.\\pipe\\containerd-shim-%s-%s-log", ns, id), nil)
90+
}
91+
8292
func (l *remoteEventsPublisher) Publish(ctx context.Context, topic string, event events.Event) error {
8393
ns, _ := namespaces.Namespace(ctx)
8494
encoded, err := typeurl.MarshalAny(event)

runtime/v2/shim_unix.go

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
// +build !windows
2+
3+
/*
4+
Copyright The containerd Authors.
5+
6+
Licensed under the Apache License, Version 2.0 (the "License");
7+
you may not use this file except in compliance with the License.
8+
You may obtain a copy of the License at
9+
10+
http://www.apache.org/licenses/LICENSE-2.0
11+
12+
Unless required by applicable law or agreed to in writing, software
13+
distributed under the License is distributed on an "AS IS" BASIS,
14+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
See the License for the specific language governing permissions and
16+
limitations under the License.
17+
*/
18+
19+
package v2
20+
21+
import (
22+
"context"
23+
"io"
24+
"path/filepath"
25+
26+
"github.com/containerd/fifo"
27+
"golang.org/x/sys/unix"
28+
)
29+
30+
func openShimLog(ctx context.Context, bundle *Bundle) (io.ReadCloser, error) {
31+
return fifo.OpenFifo(ctx, filepath.Join(bundle.Path, "log"), unix.O_RDONLY|unix.O_CREAT|unix.O_NONBLOCK, 0700)
32+
}

runtime/v2/shim_windows.go

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
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 v2
18+
19+
import (
20+
"context"
21+
"fmt"
22+
"io"
23+
24+
winio "github.com/Microsoft/go-winio"
25+
"github.com/containerd/containerd/namespaces"
26+
)
27+
28+
func openShimLog(ctx context.Context, bundle *Bundle) (io.ReadCloser, error) {
29+
ns, err := namespaces.NamespaceRequired(ctx)
30+
if err != nil {
31+
return nil, err
32+
}
33+
l, err := winio.ListenPipe(fmt.Sprintf("\\\\.\\pipe\\containerd-shim-%s-%s-log", ns, bundle.ID), nil)
34+
if err != nil {
35+
return nil, err
36+
}
37+
c, err := l.Accept()
38+
if err != nil {
39+
l.Close()
40+
}
41+
return c, nil
42+
}

0 commit comments

Comments
 (0)