github.com/containerd/Containerd@v1.4.13/cmd/containerd-stress/exec_worker.go (about)

     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 main
    18  
    19  import (
    20  	"context"
    21  	"fmt"
    22  	"strings"
    23  	"syscall"
    24  	"time"
    25  
    26  	"github.com/containerd/containerd"
    27  	"github.com/containerd/containerd/cio"
    28  	"github.com/containerd/containerd/oci"
    29  	specs "github.com/opencontainers/runtime-spec/specs-go"
    30  	"github.com/sirupsen/logrus"
    31  )
    32  
    33  type execWorker struct {
    34  	worker
    35  }
    36  
    37  func (w *execWorker) exec(ctx, tctx context.Context) {
    38  	defer func() {
    39  		w.wg.Done()
    40  		logrus.Infof("worker %d finished", w.id)
    41  	}()
    42  	id := fmt.Sprintf("exec-container-%d", w.id)
    43  	c, err := w.client.NewContainer(ctx, id,
    44  		containerd.WithNewSnapshot(id, w.image),
    45  		containerd.WithNewSpec(oci.WithImageConfig(w.image), oci.WithUsername("games"), oci.WithProcessArgs("sleep", "30d")),
    46  	)
    47  	if err != nil {
    48  		logrus.WithError(err).Error("create exec container")
    49  		return
    50  	}
    51  	defer c.Delete(ctx, containerd.WithSnapshotCleanup)
    52  
    53  	task, err := c.NewTask(ctx, cio.NullIO)
    54  	if err != nil {
    55  		logrus.WithError(err).Error("create exec container's task")
    56  		return
    57  	}
    58  	defer task.Delete(ctx, containerd.WithProcessKill)
    59  
    60  	statusC, err := task.Wait(ctx)
    61  	if err != nil {
    62  		logrus.WithError(err).Error("wait exec container's task")
    63  		return
    64  	}
    65  	spec, err := c.Spec(ctx)
    66  	if err != nil {
    67  		logrus.WithError(err).Error("failed to get spec")
    68  		return
    69  	}
    70  
    71  	pspec := spec.Process
    72  	pspec.Args = []string{"true"}
    73  
    74  	for {
    75  		select {
    76  		case <-tctx.Done():
    77  			if err := task.Kill(ctx, syscall.SIGKILL); err != nil {
    78  				logrus.WithError(err).Error("kill exec container's task")
    79  			}
    80  			<-statusC
    81  			return
    82  		default:
    83  		}
    84  
    85  		w.count++
    86  		id := w.getID()
    87  		logrus.Debugf("starting exec %s", id)
    88  		start := time.Now()
    89  
    90  		if err := w.runExec(ctx, task, id, pspec); err != nil {
    91  			if err != context.DeadlineExceeded ||
    92  				!strings.Contains(err.Error(), context.DeadlineExceeded.Error()) {
    93  				w.failures++
    94  				logrus.WithError(err).Errorf("running exec %s", id)
    95  				errCounter.WithValues(err.Error()).Inc()
    96  			}
    97  			continue
    98  		}
    99  		// only log times are success so we don't scew the results from failures that go really fast
   100  		execTimer.WithValues(w.commit).UpdateSince(start)
   101  	}
   102  }
   103  
   104  func (w *execWorker) runExec(ctx context.Context, task containerd.Task, id string, spec *specs.Process) (err error) {
   105  	process, err := task.Exec(ctx, id, spec, cio.NullIO)
   106  	if err != nil {
   107  		return err
   108  	}
   109  	defer func() {
   110  		if _, derr := process.Delete(ctx, containerd.WithProcessKill); err == nil {
   111  			err = derr
   112  		}
   113  	}()
   114  	statusC, err := process.Wait(ctx)
   115  	if err != nil {
   116  		return err
   117  	}
   118  	if err := process.Start(ctx); err != nil {
   119  		return err
   120  	}
   121  	status := <-statusC
   122  	_, _, err = status.Result()
   123  	if err != nil {
   124  		if err == context.DeadlineExceeded || err == context.Canceled {
   125  			return nil
   126  		}
   127  		w.failures++
   128  		errCounter.WithValues(err.Error()).Inc()
   129  	}
   130  	return nil
   131  }