Skip to content

Commit

Permalink
add retry attempts in logger to wait for task start
Browse files Browse the repository at this point in the history
Signed-off-by: Mrudul Harwani <[email protected]>
  • Loading branch information
mharwani committed Aug 1, 2023
1 parent 808dd0c commit 0050ac5
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 11 deletions.
7 changes: 7 additions & 0 deletions cmd/nerdctl/container_logs_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ package main

import (
"fmt"
"runtime"
"strings"
"testing"
"time"
Expand Down Expand Up @@ -160,6 +161,9 @@ func TestLogsWithRunningContainer(t *testing.T) {
}

func TestLogsWithoutNewlineOrEOF(t *testing.T) {
if runtime.GOOS != "linux" {
t.Skip("FIXME: test does not work on Windows yet because containerd doesn't send an exit event appropriately after task exit on Windows")
}
t.Parallel()
base := testutil.NewBase(t)
containerName := testutil.Identifier(t)
Expand All @@ -172,6 +176,9 @@ func TestLogsWithoutNewlineOrEOF(t *testing.T) {
}

func TestLogsAfterRestartingContainer(t *testing.T) {
if runtime.GOOS != "linux" {
t.Skip("FIXME: test does not work on Windows yet. Restarting a container fails with: failed to create shim task: hcs::CreateComputeSystem <id>: The requested operation for attach namespace failed.: unknown")
}
t.Parallel()
base := testutil.NewBase(t)
containerName := testutil.Identifier(t)
Expand Down
47 changes: 36 additions & 11 deletions pkg/logging/logging.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ import (
"path/filepath"
"sort"
"sync"
"time"

"github.com/containerd/containerd"
"github.com/containerd/containerd/errdefs"
Expand Down Expand Up @@ -164,21 +165,39 @@ func getContainerWait(ctx context.Context, hostAddress string, config *logging.C
if err != nil {
return nil, err
}

task, err := con.Task(ctx, nil)
if err != nil {
if err == nil {
return task.Wait(ctx)
}
if !errdefs.IsNotFound(err) {
return nil, err
}
return task.Wait(ctx)

// If task was not found, it's possible that the container runtime is still being created.
// Retry every 100ms.
ticker := time.NewTicker(100 * time.Millisecond)
for {
select {
case <-ctx.Done():
return nil, fmt.Errorf("timed out waiting for container task to start")
case <-ticker.C:
task, err = con.Task(ctx, nil)
if err != nil {
if errdefs.IsNotFound(err) {
continue
}
return nil, err
}
return task.Wait(ctx)
}
}
}

func loggingProcessAdapter(ctx context.Context, driver Driver, dataStore, hostAddress string, config *logging.Config) error {
if err := driver.PreProcess(dataStore, config); err != nil {
return err
}
exitCh, err := getContainerWait(ctx, hostAddress, config)
if err != nil {
return err
}

// initialize goroutines to copy stdout and stderr streams to a closable pipe
stdoutR, stdoutW := io.Pipe()
Expand Down Expand Up @@ -220,9 +239,15 @@ func loggingProcessAdapter(ctx context.Context, driver Driver, dataStore, hostAd
}()
go func() {
// close stdout and stderr upon container exit
defer stdoutW.Close()
defer stderrW.Close()

exitCh, err := getContainerWait(ctx, hostAddress, config)
if err != nil {
logrus.Errorf("failed to get container task wait channel: %v", err)
return
}
<-exitCh
stdoutW.Close()
stderrW.Close()
}()
wg.Wait()
return driver.PostProcess()
Expand All @@ -247,8 +272,8 @@ func loggerFunc(dataStore string) (logging.LoggerFunc, error) {
return err
}

lockFile := getLockPath(dataStore, config.Namespace, config.ID)
f, err := os.Create(lockFile)
loggerLock := getLockPath(dataStore, config.Namespace, config.ID)
f, err := os.Create(loggerLock)
if err != nil {
return err
}
Expand All @@ -257,7 +282,7 @@ func loggerFunc(dataStore string) (logging.LoggerFunc, error) {
// the logger will obtain an exclusive lock on a file until the container is
// stopped and the driver has finished processing all output,
// so that waiting log viewers can be signalled when the process is complete.
return lockutil.WithDirLock(lockFile, func() error {
return lockutil.WithDirLock(loggerLock, func() error {
if err := ready(); err != nil {
return err
}
Expand Down

0 comments on commit 0050ac5

Please sign in to comment.