Skip to content

Commit 8362d9a

Browse files
committed
switch shim log Windows client/server direction
Switches the client/server direction of the shim-log pipe on Windows so that the shim is the listener. This allows the containerd client to reconnect as needed to the log streams. Signed-off-by: Justin Terry (VM) <[email protected]>
1 parent 2c85ae2 commit 8362d9a

File tree

2 files changed

+95
-10
lines changed

2 files changed

+95
-10
lines changed

runtime/v2/shim/shim_windows.go

+41-1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import (
2626
"net"
2727
"os"
2828
"os/exec"
29+
"sync"
2930

3031
winio "github.com/Microsoft/go-winio"
3132
"github.com/containerd/containerd/events"
@@ -81,12 +82,51 @@ func handleSignals(logger *logrus.Entry, signals chan os.Signal) error {
8182
}
8283
}
8384

85+
type deferredShimWriteLogger struct {
86+
ctx context.Context
87+
88+
wg sync.WaitGroup
89+
90+
c net.Conn
91+
conerr error
92+
}
93+
94+
func (dswl *deferredShimWriteLogger) Write(p []byte) (int, error) {
95+
dswl.wg.Wait()
96+
if dswl.c == nil {
97+
return 0, dswl.conerr
98+
}
99+
return dswl.c.Write(p)
100+
}
101+
102+
// openLog on Windows acts as the server of the log pipe. This allows the
103+
// containerd daemon to independently restart and reconnect to the logs.
84104
func openLog(ctx context.Context, id string) (io.Writer, error) {
85105
ns, err := namespaces.NamespaceRequired(ctx)
86106
if err != nil {
87107
return nil, err
88108
}
89-
return winio.DialPipe(fmt.Sprintf("\\\\.\\pipe\\containerd-shim-%s-%s-log", ns, id), nil)
109+
l, err := winio.ListenPipe(fmt.Sprintf("\\\\.\\pipe\\containerd-shim-%s-%s-log", ns, id), nil)
110+
if err != nil {
111+
return nil, err
112+
}
113+
dswl := &deferredShimWriteLogger{
114+
ctx: ctx,
115+
}
116+
// TODO: JTERRY75 - this will not work with restarts. Only the first
117+
// connection will work and all +1 connections will return 'use of closed
118+
// network connection'. Make this reconnect aware.
119+
dswl.wg.Add(1)
120+
go func() {
121+
c, conerr := l.Accept()
122+
if conerr != nil {
123+
l.Close()
124+
dswl.conerr = conerr
125+
}
126+
dswl.c = c
127+
dswl.wg.Done()
128+
}()
129+
return dswl, nil
90130
}
91131

92132
func (l *remoteEventsPublisher) Publish(ctx context.Context, topic string, event events.Event) error {

runtime/v2/shim_windows.go

+54-9
Original file line numberDiff line numberDiff line change
@@ -20,23 +20,68 @@ import (
2020
"context"
2121
"fmt"
2222
"io"
23+
"net"
24+
"sync"
25+
"time"
2326

24-
winio "github.com/Microsoft/go-winio"
2527
"github.com/containerd/containerd/namespaces"
28+
client "github.com/containerd/containerd/runtime/v2/shim"
29+
"github.com/pkg/errors"
2630
)
2731

32+
type deferredPipeConnection struct {
33+
ctx context.Context
34+
35+
wg sync.WaitGroup
36+
once sync.Once
37+
38+
c net.Conn
39+
conerr error
40+
}
41+
42+
func (dpc *deferredPipeConnection) Read(p []byte) (n int, err error) {
43+
if dpc.c == nil {
44+
dpc.wg.Wait()
45+
if dpc.c == nil {
46+
return 0, dpc.conerr
47+
}
48+
}
49+
return dpc.c.Read(p)
50+
}
51+
func (dpc *deferredPipeConnection) Close() error {
52+
var err error
53+
dpc.once.Do(func() {
54+
dpc.wg.Wait()
55+
if dpc.c != nil {
56+
err = dpc.c.Close()
57+
} else if dpc.conerr != nil {
58+
err = dpc.conerr
59+
}
60+
})
61+
return err
62+
}
63+
64+
// openShimLog on Windows acts as the client of the log pipe. In this way the
65+
// containerd daemon can reconnect to the shim log stream if it is restarted.
2866
func openShimLog(ctx context.Context, bundle *Bundle) (io.ReadCloser, error) {
2967
ns, err := namespaces.NamespaceRequired(ctx)
3068
if err != nil {
3169
return nil, err
3270
}
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()
71+
dpc := &deferredPipeConnection{
72+
ctx: ctx,
4073
}
41-
return c, nil
74+
dpc.wg.Add(1)
75+
go func() {
76+
c, conerr := client.AnonDialer(
77+
fmt.Sprintf("\\\\.\\pipe\\containerd-shim-%s-%s-log", ns, bundle.ID),
78+
time.Second*10,
79+
)
80+
if conerr != nil {
81+
dpc.conerr = errors.Wrap(err, "failed to connect to shim log")
82+
}
83+
dpc.c = c
84+
dpc.wg.Done()
85+
}()
86+
return dpc, nil
4287
}

0 commit comments

Comments
 (0)