| /* |
| 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 events |
| |
| import ( |
| "context" |
| "fmt" |
| |
| api "github.com/containerd/containerd/api/services/events/v1" |
| apittrpc "github.com/containerd/containerd/api/services/ttrpc/events/v1" |
| "github.com/containerd/containerd/api/types" |
| "github.com/containerd/errdefs/pkg/errgrpc" |
| "github.com/containerd/plugin" |
| "github.com/containerd/plugin/registry" |
| "github.com/containerd/ttrpc" |
| "github.com/containerd/typeurl/v2" |
| "google.golang.org/grpc" |
| |
| "github.com/containerd/containerd/v2/core/events" |
| "github.com/containerd/containerd/v2/core/events/exchange" |
| "github.com/containerd/containerd/v2/pkg/protobuf" |
| ptypes "github.com/containerd/containerd/v2/pkg/protobuf/types" |
| "github.com/containerd/containerd/v2/plugins" |
| ) |
| |
| var empty = &ptypes.Empty{} |
| |
| func init() { |
| registry.Register(&plugin.Registration{ |
| Type: plugins.GRPCPlugin, |
| ID: "events", |
| Requires: []plugin.Type{ |
| plugins.EventPlugin, |
| }, |
| InitFn: func(ic *plugin.InitContext) (interface{}, error) { |
| ep, err := ic.GetByID(plugins.EventPlugin, "exchange") |
| if err != nil { |
| return nil, err |
| } |
| return NewService(ep.(*exchange.Exchange)), nil |
| }, |
| }) |
| } |
| |
| type service struct { |
| ttService *ttrpcService |
| events *exchange.Exchange |
| api.UnimplementedEventsServer |
| } |
| |
| // NewService returns the GRPC events server |
| func NewService(events *exchange.Exchange) api.EventsServer { |
| return &service{ |
| ttService: &ttrpcService{ |
| events: events, |
| }, |
| events: events, |
| } |
| } |
| |
| func (s *service) Register(server *grpc.Server) error { |
| api.RegisterEventsServer(server, s) |
| return nil |
| } |
| |
| func (s *service) RegisterTTRPC(server *ttrpc.Server) error { |
| apittrpc.RegisterEventsService(server, s.ttService) |
| return nil |
| } |
| |
| func (s *service) Publish(ctx context.Context, r *api.PublishRequest) (*ptypes.Empty, error) { |
| if err := s.events.Publish(ctx, r.Topic, r.Event); err != nil { |
| return nil, errgrpc.ToGRPC(err) |
| } |
| |
| return empty, nil |
| } |
| |
| func (s *service) Forward(ctx context.Context, r *api.ForwardRequest) (*ptypes.Empty, error) { |
| if err := s.events.Forward(ctx, fromProto(r.Envelope)); err != nil { |
| return nil, errgrpc.ToGRPC(err) |
| } |
| |
| return empty, nil |
| } |
| |
| func (s *service) Subscribe(req *api.SubscribeRequest, srv api.Events_SubscribeServer) error { |
| ctx, cancel := context.WithCancel(srv.Context()) |
| defer cancel() |
| |
| eventq, errq := s.events.Subscribe(ctx, req.Filters...) |
| for { |
| select { |
| case ev := <-eventq: |
| if err := srv.Send(toProto(ev)); err != nil { |
| return fmt.Errorf("failed sending event to subscriber: %w", err) |
| } |
| case err := <-errq: |
| if err != nil { |
| return fmt.Errorf("subscription error: %w", err) |
| } |
| |
| return nil |
| } |
| } |
| } |
| |
| func toProto(env *events.Envelope) *types.Envelope { |
| return &types.Envelope{ |
| Timestamp: protobuf.ToTimestamp(env.Timestamp), |
| Namespace: env.Namespace, |
| Topic: env.Topic, |
| Event: typeurl.MarshalProto(env.Event), |
| } |
| } |
| |
| func fromProto(env *types.Envelope) *events.Envelope { |
| return &events.Envelope{ |
| Timestamp: protobuf.FromTimestamp(env.Timestamp), |
| Namespace: env.Namespace, |
| Topic: env.Topic, |
| Event: env.Event, |
| } |
| } |