Uses the new github.com/containerd/errdefs/pkg module which is intended to hold less stable utility functions separately from the stable github.com/containerd/errdefs error types. Includes temporary update to hcsshim until a release is cut there Signed-off-by: Derek McGowan <derek@mcg.dev>
		
			
				
	
	
		
			262 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
   Copyright The containerd Authors.
 | 
						|
 | 
						|
   Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
   you may not use this file except in compliance with the License.
 | 
						|
   You may obtain a copy of the License at
 | 
						|
 | 
						|
       http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 | 
						|
   Unless required by applicable law or agreed to in writing, software
 | 
						|
   distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
   See the License for the specific language governing permissions and
 | 
						|
   limitations under the License.
 | 
						|
*/
 | 
						|
 | 
						|
package sandbox
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"google.golang.org/grpc"
 | 
						|
	"google.golang.org/protobuf/types/known/anypb"
 | 
						|
 | 
						|
	eventtypes "github.com/containerd/containerd/api/events"
 | 
						|
	api "github.com/containerd/containerd/api/services/sandbox/v1"
 | 
						|
	"github.com/containerd/errdefs"
 | 
						|
	"github.com/containerd/errdefs/pkg/errgrpc"
 | 
						|
	"github.com/containerd/log"
 | 
						|
	"github.com/containerd/plugin"
 | 
						|
	"github.com/containerd/plugin/registry"
 | 
						|
 | 
						|
	"github.com/containerd/containerd/v2/core/events"
 | 
						|
	"github.com/containerd/containerd/v2/core/sandbox"
 | 
						|
	"github.com/containerd/containerd/v2/pkg/protobuf"
 | 
						|
	"github.com/containerd/containerd/v2/plugins"
 | 
						|
)
 | 
						|
 | 
						|
func init() {
 | 
						|
	registry.Register(&plugin.Registration{
 | 
						|
		Type: plugins.GRPCPlugin,
 | 
						|
		ID:   "sandbox-controllers",
 | 
						|
		Requires: []plugin.Type{
 | 
						|
			plugins.PodSandboxPlugin,
 | 
						|
			plugins.SandboxControllerPlugin,
 | 
						|
			plugins.EventPlugin,
 | 
						|
		},
 | 
						|
		InitFn: func(ic *plugin.InitContext) (interface{}, error) {
 | 
						|
			sc := make(map[string]sandbox.Controller)
 | 
						|
 | 
						|
			sandboxers, err := ic.GetByType(plugins.PodSandboxPlugin)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			for name, p := range sandboxers {
 | 
						|
				sc[name] = p.(sandbox.Controller)
 | 
						|
			}
 | 
						|
 | 
						|
			sandboxersV2, err := ic.GetByType(plugins.SandboxControllerPlugin)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			for name, p := range sandboxersV2 {
 | 
						|
				sc[name] = p.(sandbox.Controller)
 | 
						|
			}
 | 
						|
 | 
						|
			ep, err := ic.GetSingle(plugins.EventPlugin)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
 | 
						|
			return &controllerService{
 | 
						|
				sc:        sc,
 | 
						|
				publisher: ep.(events.Publisher),
 | 
						|
			}, nil
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
type controllerService struct {
 | 
						|
	sc        map[string]sandbox.Controller
 | 
						|
	publisher events.Publisher
 | 
						|
	api.UnimplementedControllerServer
 | 
						|
}
 | 
						|
 | 
						|
var _ api.ControllerServer = (*controllerService)(nil)
 | 
						|
 | 
						|
func (s *controllerService) Register(server *grpc.Server) error {
 | 
						|
	api.RegisterControllerServer(server, s)
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) getController(name string) (sandbox.Controller, error) {
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("%w: sandbox controller name can not be empty", errdefs.ErrInvalidArgument)
 | 
						|
	}
 | 
						|
	if ctrl, ok := s.sc[name]; ok {
 | 
						|
		return ctrl, nil
 | 
						|
	}
 | 
						|
	return nil, fmt.Errorf("%w: failed to get sandbox controller by %s", errdefs.ErrNotFound, name)
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Create(ctx context.Context, req *api.ControllerCreateRequest) (*api.ControllerCreateResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("create sandbox")
 | 
						|
	// TODO: Rootfs
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	var sb sandbox.Sandbox
 | 
						|
	if req.Sandbox != nil {
 | 
						|
		sb = sandbox.FromProto(req.Sandbox)
 | 
						|
	} else {
 | 
						|
		sb = sandbox.Sandbox{ID: req.GetSandboxID()}
 | 
						|
	}
 | 
						|
	err = ctrl.Create(ctx, sb, sandbox.WithOptions(req.GetOptions()))
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerCreateResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	if err := s.publisher.Publish(ctx, "sandboxes/create", &eventtypes.SandboxCreate{
 | 
						|
		SandboxID: req.GetSandboxID(),
 | 
						|
	}); err != nil {
 | 
						|
		return &api.ControllerCreateResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	return &api.ControllerCreateResponse{
 | 
						|
		SandboxID: req.GetSandboxID(),
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Start(ctx context.Context, req *api.ControllerStartRequest) (*api.ControllerStartResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("start sandbox")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	inst, err := ctrl.Start(ctx, req.GetSandboxID())
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerStartResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	if err := s.publisher.Publish(ctx, "sandboxes/start", &eventtypes.SandboxStart{
 | 
						|
		SandboxID: req.GetSandboxID(),
 | 
						|
	}); err != nil {
 | 
						|
		return &api.ControllerStartResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	return &api.ControllerStartResponse{
 | 
						|
		SandboxID: inst.SandboxID,
 | 
						|
		Pid:       inst.Pid,
 | 
						|
		CreatedAt: protobuf.ToTimestamp(inst.CreatedAt),
 | 
						|
		Labels:    inst.Labels,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Stop(ctx context.Context, req *api.ControllerStopRequest) (*api.ControllerStopResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("delete sandbox")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	return &api.ControllerStopResponse{}, errgrpc.ToGRPC(ctrl.Stop(ctx, req.GetSandboxID(), sandbox.WithTimeout(time.Duration(req.TimeoutSecs)*time.Second)))
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Wait(ctx context.Context, req *api.ControllerWaitRequest) (*api.ControllerWaitResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("wait sandbox")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	exitStatus, err := ctrl.Wait(ctx, req.GetSandboxID())
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerWaitResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	if err := s.publisher.Publish(ctx, "sandboxes/exit", &eventtypes.SandboxExit{
 | 
						|
		SandboxID:  req.GetSandboxID(),
 | 
						|
		ExitStatus: exitStatus.ExitStatus,
 | 
						|
		ExitedAt:   protobuf.ToTimestamp(exitStatus.ExitedAt),
 | 
						|
	}); err != nil {
 | 
						|
		return &api.ControllerWaitResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
 | 
						|
	return &api.ControllerWaitResponse{
 | 
						|
		ExitStatus: exitStatus.ExitStatus,
 | 
						|
		ExitedAt:   protobuf.ToTimestamp(exitStatus.ExitedAt),
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Status(ctx context.Context, req *api.ControllerStatusRequest) (*api.ControllerStatusResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("sandbox status")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	cstatus, err := ctrl.Status(ctx, req.GetSandboxID(), req.GetVerbose())
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerStatusResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	extra := &anypb.Any{}
 | 
						|
	if cstatus.Extra != nil {
 | 
						|
		extra = &anypb.Any{
 | 
						|
			TypeUrl: cstatus.Extra.GetTypeUrl(),
 | 
						|
			Value:   cstatus.Extra.GetValue(),
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return &api.ControllerStatusResponse{
 | 
						|
		SandboxID: cstatus.SandboxID,
 | 
						|
		Pid:       cstatus.Pid,
 | 
						|
		State:     cstatus.State,
 | 
						|
		Info:      cstatus.Info,
 | 
						|
		CreatedAt: protobuf.ToTimestamp(cstatus.CreatedAt),
 | 
						|
		ExitedAt:  protobuf.ToTimestamp(cstatus.ExitedAt),
 | 
						|
		Extra:     extra,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Shutdown(ctx context.Context, req *api.ControllerShutdownRequest) (*api.ControllerShutdownResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("shutdown sandbox")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	return &api.ControllerShutdownResponse{}, errgrpc.ToGRPC(ctrl.Shutdown(ctx, req.GetSandboxID()))
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Metrics(ctx context.Context, req *api.ControllerMetricsRequest) (*api.ControllerMetricsResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("sandbox metrics")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	metrics, err := ctrl.Metrics(ctx, req.GetSandboxID())
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerMetricsResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	return &api.ControllerMetricsResponse{
 | 
						|
		Metrics: metrics,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func (s *controllerService) Update(
 | 
						|
	ctx context.Context,
 | 
						|
	req *api.ControllerUpdateRequest) (*api.ControllerUpdateResponse, error) {
 | 
						|
	log.G(ctx).WithField("req", req).Debug("sandbox update resource")
 | 
						|
	ctrl, err := s.getController(req.Sandboxer)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	if req.Sandbox == nil {
 | 
						|
		return nil, fmt.Errorf("sandbox can not be nil")
 | 
						|
	}
 | 
						|
	err = ctrl.Update(ctx, req.SandboxID, sandbox.FromProto(req.Sandbox), req.Fields...)
 | 
						|
	if err != nil {
 | 
						|
		return &api.ControllerUpdateResponse{}, errgrpc.ToGRPC(err)
 | 
						|
	}
 | 
						|
	return &api.ControllerUpdateResponse{}, nil
 | 
						|
}
 |