2019-03-14 15:56:06 +01:00
package ingress
2019-02-21 23:08:05 +01:00
import (
"context"
2019-04-01 15:30:07 +02:00
"errors"
2019-02-21 23:08:05 +01:00
"fmt"
"math"
"os"
"sort"
"strconv"
"strings"
"time"
2019-08-03 03:58:23 +02:00
"github.com/cenkalti/backoff/v3"
"github.com/containous/traefik/v2/pkg/config/dynamic"
"github.com/containous/traefik/v2/pkg/job"
"github.com/containous/traefik/v2/pkg/log"
2019-10-14 11:18:04 +02:00
"github.com/containous/traefik/v2/pkg/provider"
2019-08-03 03:58:23 +02:00
"github.com/containous/traefik/v2/pkg/safe"
"github.com/containous/traefik/v2/pkg/tls"
2019-08-30 06:16:04 -04:00
"github.com/containous/traefik/v2/pkg/types"
2019-10-25 15:46:05 +02:00
"github.com/mitchellh/hashstructure"
2019-02-21 23:08:05 +01:00
corev1 "k8s.io/api/core/v1"
"k8s.io/api/extensions/v1beta1"
"k8s.io/apimachinery/pkg/labels"
"k8s.io/apimachinery/pkg/util/intstr"
)
const (
annotationKubernetesIngressClass = "kubernetes.io/ingress.class"
traefikDefaultIngressClass = "traefik"
)
// Provider holds configurations of the provider.
type Provider struct {
2019-07-01 11:30:05 +02:00
Endpoint string ` description:"Kubernetes server endpoint (required for external cluster client)." json:"endpoint,omitempty" toml:"endpoint,omitempty" yaml:"endpoint,omitempty" `
Token string ` description:"Kubernetes bearer token (not needed for in-cluster client)." json:"token,omitempty" toml:"token,omitempty" yaml:"token,omitempty" `
CertAuthFilePath string ` description:"Kubernetes certificate authority file path (not needed for in-cluster client)." json:"certAuthFilePath,omitempty" toml:"certAuthFilePath,omitempty" yaml:"certAuthFilePath,omitempty" `
DisablePassHostHeaders bool ` description:"Kubernetes disable PassHost Headers." json:"disablePassHostHeaders,omitempty" toml:"disablePassHostHeaders,omitempty" yaml:"disablePassHostHeaders,omitempty" export:"true" `
Namespaces [ ] string ` description:"Kubernetes namespaces." json:"namespaces,omitempty" toml:"namespaces,omitempty" yaml:"namespaces,omitempty" export:"true" `
LabelSelector string ` description:"Kubernetes Ingress label selector to use." json:"labelSelector,omitempty" toml:"labelSelector,omitempty" yaml:"labelSelector,omitempty" export:"true" `
IngressClass string ` description:"Value of kubernetes.io/ingress.class annotation to watch for." json:"ingressClass,omitempty" toml:"ingressClass,omitempty" yaml:"ingressClass,omitempty" export:"true" `
IngressEndpoint * EndpointIngress ` description:"Kubernetes Ingress Endpoint." json:"ingressEndpoint,omitempty" toml:"ingressEndpoint,omitempty" yaml:"ingressEndpoint,omitempty" `
2019-08-30 06:16:04 -04:00
ThrottleDuration types . Duration ` description:"Ingress refresh throttle duration" json:"throttleDuration,omitempty" toml:"throttleDuration,omitempty" yaml:"throttleDuration,omitempty" `
2019-02-21 23:08:05 +01:00
lastConfiguration safe . Safe
}
2019-03-18 10:10:04 +01:00
// EndpointIngress holds the endpoint information for the Kubernetes provider
type EndpointIngress struct {
2019-07-01 11:30:05 +02:00
IP string ` description:"IP used for Kubernetes Ingress endpoints." json:"ip,omitempty" toml:"ip,omitempty" yaml:"ip,omitempty" `
Hostname string ` description:"Hostname used for Kubernetes Ingress endpoints." json:"hostname,omitempty" toml:"hostname,omitempty" yaml:"hostname,omitempty" `
PublishedService string ` description:"Published Kubernetes Service to copy status from." json:"publishedService,omitempty" toml:"publishedService,omitempty" yaml:"publishedService,omitempty" `
2019-03-18 10:10:04 +01:00
}
2019-03-14 15:56:06 +01:00
func ( p * Provider ) newK8sClient ( ctx context . Context , ingressLabelSelector string ) ( * clientWrapper , error ) {
2019-02-21 23:08:05 +01:00
ingLabelSel , err := labels . Parse ( ingressLabelSelector )
if err != nil {
return nil , fmt . Errorf ( "invalid ingress label selector: %q" , ingressLabelSelector )
}
2019-03-14 15:56:06 +01:00
logger := log . FromContext ( ctx )
logger . Infof ( "ingress label selector is: %q" , ingLabelSel )
2019-02-21 23:08:05 +01:00
withEndpoint := ""
if p . Endpoint != "" {
withEndpoint = fmt . Sprintf ( " with endpoint %v" , p . Endpoint )
}
2019-03-14 15:56:06 +01:00
var cl * clientWrapper
2019-03-11 14:54:05 +01:00
switch {
case os . Getenv ( "KUBERNETES_SERVICE_HOST" ) != "" && os . Getenv ( "KUBERNETES_SERVICE_PORT" ) != "" :
2019-03-14 15:56:06 +01:00
logger . Infof ( "Creating in-cluster Provider client%s" , withEndpoint )
2019-02-21 23:08:05 +01:00
cl , err = newInClusterClient ( p . Endpoint )
2019-03-11 14:54:05 +01:00
case os . Getenv ( "KUBECONFIG" ) != "" :
2019-03-14 15:56:06 +01:00
logger . Infof ( "Creating cluster-external Provider client from KUBECONFIG %s" , os . Getenv ( "KUBECONFIG" ) )
2019-03-11 14:54:05 +01:00
cl , err = newExternalClusterClientFromFile ( os . Getenv ( "KUBECONFIG" ) )
default :
2019-03-14 15:56:06 +01:00
logger . Infof ( "Creating cluster-external Provider client%s" , withEndpoint )
2019-02-21 23:08:05 +01:00
cl , err = newExternalClusterClient ( p . Endpoint , p . Token , p . CertAuthFilePath )
}
if err == nil {
cl . ingressLabelSelector = ingLabelSel
}
return cl , err
}
// Init the provider.
func ( p * Provider ) Init ( ) error {
2019-03-27 15:02:06 +01:00
return nil
2019-02-21 23:08:05 +01:00
}
// Provide allows the k8s provider to provide configurations to traefik
// using the given configuration channel.
2019-07-10 09:26:04 +02:00
func ( p * Provider ) Provide ( configurationChan chan <- dynamic . Message , pool * safe . Pool ) error {
2019-03-14 15:56:06 +01:00
ctxLog := log . With ( context . Background ( ) , log . Str ( log . ProviderName , "kubernetes" ) )
2019-02-21 23:08:05 +01:00
logger := log . FromContext ( ctxLog )
logger . Debugf ( "Using Ingress label selector: %q" , p . LabelSelector )
k8sClient , err := p . newK8sClient ( ctxLog , p . LabelSelector )
if err != nil {
return err
}
pool . Go ( func ( stop chan bool ) {
operation := func ( ) error {
stopWatch := make ( chan struct { } , 1 )
defer close ( stopWatch )
2019-03-14 15:56:06 +01:00
2019-02-21 23:08:05 +01:00
eventsChan , err := k8sClient . WatchAll ( p . Namespaces , stopWatch )
if err != nil {
logger . Errorf ( "Error watching kubernetes events: %v" , err )
timer := time . NewTimer ( 1 * time . Second )
select {
case <- timer . C :
return err
case <- stop :
return nil
}
}
2019-03-14 15:56:06 +01:00
2019-08-30 06:16:04 -04:00
throttleDuration := time . Duration ( p . ThrottleDuration )
2019-08-31 14:10:04 +02:00
throttledChan := throttleEvents ( ctxLog , throttleDuration , stop , eventsChan )
if throttledChan != nil {
eventsChan = throttledChan
}
2019-08-30 06:16:04 -04:00
2019-02-21 23:08:05 +01:00
for {
select {
case <- stop :
return nil
2019-08-31 14:10:04 +02:00
case event := <- eventsChan :
2019-08-30 06:16:04 -04:00
// Note that event is the *first* event that came in during this
// throttling interval -- if we're hitting our throttle, we may have
// dropped events. This is fine, because we don't treat different
// event types differently. But if we do in the future, we'll need to
// track more information about the dropped events.
2019-02-21 23:08:05 +01:00
conf := p . loadConfigurationFromIngresses ( ctxLog , k8sClient )
2019-10-25 15:46:05 +02:00
confHash , err := hashstructure . Hash ( conf , nil )
switch {
case err != nil :
logger . Error ( "Unable to hash the configuration" )
case p . lastConfiguration . Get ( ) == confHash :
2019-02-21 23:08:05 +01:00
logger . Debugf ( "Skipping Kubernetes event kind %T" , event )
2019-10-25 15:46:05 +02:00
default :
p . lastConfiguration . Set ( confHash )
2019-07-10 09:26:04 +02:00
configurationChan <- dynamic . Message {
2019-02-21 23:08:05 +01:00
ProviderName : "kubernetes" ,
Configuration : conf ,
}
}
2019-08-30 06:16:04 -04:00
// If we're throttling, we sleep here for the throttle duration to
// enforce that we don't refresh faster than our throttle. time.Sleep
// returns immediately if p.ThrottleDuration is 0 (no throttle).
time . Sleep ( throttleDuration )
2019-02-21 23:08:05 +01:00
}
}
}
notify := func ( err error , time time . Duration ) {
logger . Errorf ( "Provider connection error: %s; retrying in %s" , err , time )
}
err := backoff . RetryNotify ( safe . OperationWithRecover ( operation ) , job . NewBackOff ( backoff . NewExponentialBackOff ( ) ) , notify )
if err != nil {
logger . Errorf ( "Cannot connect to Provider: %s" , err )
}
} )
return nil
}
func checkStringQuoteValidity ( value string ) error {
_ , err := strconv . Unquote ( ` " ` + value + ` " ` )
return err
}
2019-07-10 09:26:04 +02:00
func loadService ( client Client , namespace string , backend v1beta1 . IngressBackend ) ( * dynamic . Service , error ) {
2019-02-21 23:08:05 +01:00
service , exists , err := client . GetService ( namespace , backend . ServiceName )
if err != nil {
return nil , err
}
if ! exists {
return nil , errors . New ( "service not found" )
}
2019-07-10 09:26:04 +02:00
var servers [ ] dynamic . Server
2019-02-21 23:08:05 +01:00
var portName string
var portSpec corev1 . ServicePort
var match bool
for _ , p := range service . Spec . Ports {
if ( backend . ServicePort . Type == intstr . Int && backend . ServicePort . IntVal == p . Port ) ||
( backend . ServicePort . Type == intstr . String && backend . ServicePort . StrVal == p . Name ) {
portName = p . Name
portSpec = p
match = true
break
}
}
if ! match {
return nil , errors . New ( "service port not found" )
}
if service . Spec . Type == corev1 . ServiceTypeExternalName {
2019-10-18 11:12:05 +02:00
protocol := "http"
if portSpec . Port == 443 || strings . HasPrefix ( portSpec . Name , "https" ) {
protocol = "https"
}
2019-07-10 09:26:04 +02:00
servers = append ( servers , dynamic . Server {
2019-10-18 11:12:05 +02:00
URL : fmt . Sprintf ( "%s://%s:%d" , protocol , service . Spec . ExternalName , portSpec . Port ) ,
2019-02-21 23:08:05 +01:00
} )
} else {
endpoints , endpointsExists , endpointsErr := client . GetEndpoints ( namespace , backend . ServiceName )
if endpointsErr != nil {
return nil , endpointsErr
}
if ! endpointsExists {
return nil , errors . New ( "endpoints not found" )
}
if len ( endpoints . Subsets ) == 0 {
return nil , errors . New ( "subset not found" )
}
var port int32
for _ , subset := range endpoints . Subsets {
for _ , p := range subset . Ports {
if portName == p . Name {
port = p . Port
break
}
}
if port == 0 {
return nil , errors . New ( "cannot define a port" )
}
protocol := "http"
2019-08-14 12:58:04 -04:00
if portSpec . Port == 443 || strings . HasPrefix ( portName , "https" ) {
2019-02-21 23:08:05 +01:00
protocol = "https"
}
for _ , addr := range subset . Addresses {
2019-07-10 09:26:04 +02:00
servers = append ( servers , dynamic . Server {
2019-06-05 22:18:06 +02:00
URL : fmt . Sprintf ( "%s://%s:%d" , protocol , addr . IP , port ) ,
2019-02-21 23:08:05 +01:00
} )
}
}
}
2019-07-10 09:26:04 +02:00
return & dynamic . Service {
2019-08-26 10:30:05 +02:00
LoadBalancer : & dynamic . ServersLoadBalancer {
2019-02-21 23:08:05 +01:00
Servers : servers ,
2019-09-30 18:12:04 +02:00
PassHostHeader : func ( v bool ) * bool { return & v } ( true ) ,
2019-02-21 23:08:05 +01:00
} ,
} , nil
}
2019-07-10 09:26:04 +02:00
func ( p * Provider ) loadConfigurationFromIngresses ( ctx context . Context , client Client ) * dynamic . Configuration {
conf := & dynamic . Configuration {
HTTP : & dynamic . HTTPConfiguration {
Routers : map [ string ] * dynamic . Router { } ,
Middlewares : map [ string ] * dynamic . Middleware { } ,
Services : map [ string ] * dynamic . Service { } ,
2019-03-14 09:30:04 +01:00
} ,
2019-07-10 09:26:04 +02:00
TCP : & dynamic . TCPConfiguration { } ,
2019-02-21 23:08:05 +01:00
}
ingresses := client . GetIngresses ( )
2019-06-27 23:58:03 +02:00
tlsConfigs := make ( map [ string ] * tls . CertAndStores )
2019-02-21 23:08:05 +01:00
for _ , ingress := range ingresses {
ctx = log . With ( ctx , log . Str ( "ingress" , ingress . Name ) , log . Str ( "namespace" , ingress . Namespace ) )
if ! shouldProcessIngress ( p . IngressClass , ingress . Annotations [ annotationKubernetesIngressClass ] ) {
continue
}
err := getTLS ( ctx , ingress , client , tlsConfigs )
if err != nil {
log . FromContext ( ctx ) . Errorf ( "Error configuring TLS: %v" , err )
}
if len ( ingress . Spec . Rules ) == 0 {
if ingress . Spec . Backend != nil {
2019-03-14 09:30:04 +01:00
if _ , ok := conf . HTTP . Services [ "default-backend" ] ; ok {
2019-02-21 23:08:05 +01:00
log . FromContext ( ctx ) . Error ( "The default backend already exists." )
continue
}
service , err := loadService ( client , ingress . Namespace , * ingress . Spec . Backend )
if err != nil {
log . FromContext ( ctx ) .
WithField ( "serviceName" , ingress . Spec . Backend . ServiceName ) .
WithField ( "servicePort" , ingress . Spec . Backend . ServicePort . String ( ) ) .
Errorf ( "Cannot create service: %v" , err )
continue
}
2019-10-09 17:16:07 +02:00
conf . HTTP . Routers [ "default-router" ] = & dynamic . Router {
2019-02-21 23:08:05 +01:00
Rule : "PathPrefix(`/`)" ,
Priority : math . MinInt32 ,
Service : "default-backend" ,
}
2019-03-14 09:30:04 +01:00
conf . HTTP . Services [ "default-backend" ] = service
2019-02-21 23:08:05 +01:00
}
}
for _ , rule := range ingress . Spec . Rules {
if err := checkStringQuoteValidity ( rule . Host ) ; err != nil {
log . FromContext ( ctx ) . Errorf ( "Invalid syntax for host: %s" , rule . Host )
continue
}
for _ , p := range rule . HTTP . Paths {
service , err := loadService ( client , ingress . Namespace , p . Backend )
if err != nil {
log . FromContext ( ctx ) .
WithField ( "serviceName" , p . Backend . ServiceName ) .
WithField ( "servicePort" , p . Backend . ServicePort . String ( ) ) .
Errorf ( "Cannot create service: %v" , err )
continue
}
if err = checkStringQuoteValidity ( p . Path ) ; err != nil {
log . FromContext ( ctx ) . Errorf ( "Invalid syntax for path: %s" , p . Path )
continue
}
2019-10-14 11:18:04 +02:00
serviceName := provider . Normalize ( ingress . Namespace + "-" + p . Backend . ServiceName + "-" + p . Backend . ServicePort . String ( ) )
2019-02-21 23:08:05 +01:00
var rules [ ] string
if len ( rule . Host ) > 0 {
rules = [ ] string { "Host(`" + rule . Host + "`)" }
}
if len ( p . Path ) > 0 {
rules = append ( rules , "PathPrefix(`" + p . Path + "`)" )
}
2019-10-14 11:18:04 +02:00
routerKey := strings . TrimPrefix ( provider . Normalize ( rule . Host + p . Path ) , "-" )
2019-09-13 20:44:04 +02:00
conf . HTTP . Routers [ routerKey ] = & dynamic . Router {
2019-02-21 23:08:05 +01:00
Rule : strings . Join ( rules , " && " ) ,
Service : serviceName ,
}
2019-08-14 11:16:06 -06:00
if len ( ingress . Spec . TLS ) > 0 {
// TLS enabled for this ingress, add TLS router
2019-09-13 20:44:04 +02:00
conf . HTTP . Routers [ routerKey + "-tls" ] = & dynamic . Router {
2019-08-14 11:16:06 -06:00
Rule : strings . Join ( rules , " && " ) ,
Service : serviceName ,
TLS : & dynamic . RouterTLSConfig { } ,
}
}
2019-03-14 09:30:04 +01:00
conf . HTTP . Services [ serviceName ] = service
2019-02-21 23:08:05 +01:00
}
2019-03-18 10:10:04 +01:00
err := p . updateIngressStatus ( ingress , client )
if err != nil {
log . FromContext ( ctx ) . Errorf ( "Error while updating ingress status: %v" , err )
}
2019-02-21 23:08:05 +01:00
}
}
2019-06-27 23:58:03 +02:00
certs := getTLSConfig ( tlsConfigs )
if len ( certs ) > 0 {
2019-07-10 09:26:04 +02:00
conf . TLS = & dynamic . TLSConfiguration {
2019-06-27 23:58:03 +02:00
Certificates : certs ,
}
}
2019-02-21 23:08:05 +01:00
return conf
}
func shouldProcessIngress ( ingressClass string , ingressClassAnnotation string ) bool {
return ingressClass == ingressClassAnnotation ||
( len ( ingressClass ) == 0 && ingressClassAnnotation == traefikDefaultIngressClass )
}
2019-06-27 23:58:03 +02:00
func getTLS ( ctx context . Context , ingress * v1beta1 . Ingress , k8sClient Client , tlsConfigs map [ string ] * tls . CertAndStores ) error {
2019-02-21 23:08:05 +01:00
for _ , t := range ingress . Spec . TLS {
if t . SecretName == "" {
log . FromContext ( ctx ) . Debugf ( "Skipping TLS sub-section: No secret name provided" )
continue
}
2019-09-13 20:44:04 +02:00
configKey := ingress . Namespace + "-" + t . SecretName
2019-02-21 23:08:05 +01:00
if _ , tlsExists := tlsConfigs [ configKey ] ; ! tlsExists {
secret , exists , err := k8sClient . GetSecret ( ingress . Namespace , t . SecretName )
if err != nil {
return fmt . Errorf ( "failed to fetch secret %s/%s: %v" , ingress . Namespace , t . SecretName , err )
}
if ! exists {
return fmt . Errorf ( "secret %s/%s does not exist" , ingress . Namespace , t . SecretName )
}
cert , key , err := getCertificateBlocks ( secret , ingress . Namespace , t . SecretName )
if err != nil {
return err
}
2019-06-27 23:58:03 +02:00
tlsConfigs [ configKey ] = & tls . CertAndStores {
Certificate : tls . Certificate {
2019-02-21 23:08:05 +01:00
CertFile : tls . FileOrContent ( cert ) ,
KeyFile : tls . FileOrContent ( key ) ,
} ,
}
}
}
return nil
}
2019-06-27 23:58:03 +02:00
func getTLSConfig ( tlsConfigs map [ string ] * tls . CertAndStores ) [ ] * tls . CertAndStores {
2019-02-21 23:08:05 +01:00
var secretNames [ ] string
for secretName := range tlsConfigs {
secretNames = append ( secretNames , secretName )
}
sort . Strings ( secretNames )
2019-06-27 23:58:03 +02:00
var configs [ ] * tls . CertAndStores
2019-02-21 23:08:05 +01:00
for _ , secretName := range secretNames {
configs = append ( configs , tlsConfigs [ secretName ] )
}
return configs
}
func getCertificateBlocks ( secret * corev1 . Secret , namespace , secretName string ) ( string , string , error ) {
var missingEntries [ ] string
tlsCrtData , tlsCrtExists := secret . Data [ "tls.crt" ]
if ! tlsCrtExists {
missingEntries = append ( missingEntries , "tls.crt" )
}
tlsKeyData , tlsKeyExists := secret . Data [ "tls.key" ]
if ! tlsKeyExists {
missingEntries = append ( missingEntries , "tls.key" )
}
if len ( missingEntries ) > 0 {
return "" , "" , fmt . Errorf ( "secret %s/%s is missing the following TLS data entries: %s" ,
namespace , secretName , strings . Join ( missingEntries , ", " ) )
}
cert := string ( tlsCrtData )
if cert == "" {
missingEntries = append ( missingEntries , "tls.crt" )
}
key := string ( tlsKeyData )
if key == "" {
missingEntries = append ( missingEntries , "tls.key" )
}
if len ( missingEntries ) > 0 {
return "" , "" , fmt . Errorf ( "secret %s/%s contains the following empty TLS data entries: %s" ,
namespace , secretName , strings . Join ( missingEntries , ", " ) )
}
return cert , key , nil
}
2019-03-18 10:10:04 +01:00
func ( p * Provider ) updateIngressStatus ( i * v1beta1 . Ingress , k8sClient Client ) error {
// Only process if an EndpointIngress has been configured
if p . IngressEndpoint == nil {
return nil
}
if len ( p . IngressEndpoint . PublishedService ) == 0 {
if len ( p . IngressEndpoint . IP ) == 0 && len ( p . IngressEndpoint . Hostname ) == 0 {
return errors . New ( "publishedService or ip or hostname must be defined" )
}
return k8sClient . UpdateIngressStatus ( i . Namespace , i . Name , p . IngressEndpoint . IP , p . IngressEndpoint . Hostname )
}
serviceInfo := strings . Split ( p . IngressEndpoint . PublishedService , "/" )
if len ( serviceInfo ) != 2 {
return fmt . Errorf ( "invalid publishedService format (expected 'namespace/service' format): %s" , p . IngressEndpoint . PublishedService )
}
serviceNamespace , serviceName := serviceInfo [ 0 ] , serviceInfo [ 1 ]
service , exists , err := k8sClient . GetService ( serviceNamespace , serviceName )
if err != nil {
return fmt . Errorf ( "cannot get service %s, received error: %s" , p . IngressEndpoint . PublishedService , err )
}
if exists && service . Status . LoadBalancer . Ingress == nil {
// service exists, but has no Load Balancer status
log . Debugf ( "Skipping updating Ingress %s/%s due to service %s having no status set" , i . Namespace , i . Name , p . IngressEndpoint . PublishedService )
return nil
}
if ! exists {
return fmt . Errorf ( "missing service: %s" , p . IngressEndpoint . PublishedService )
}
return k8sClient . UpdateIngressStatus ( i . Namespace , i . Name , service . Status . LoadBalancer . Ingress [ 0 ] . IP , service . Status . LoadBalancer . Ingress [ 0 ] . Hostname )
}
2019-08-30 06:16:04 -04:00
func throttleEvents ( ctx context . Context , throttleDuration time . Duration , stop chan bool , eventsChan <- chan interface { } ) chan interface { } {
if throttleDuration == 0 {
return nil
}
// Create a buffered channel to hold the pending event (if we're delaying processing the event due to throttling)
eventsChanBuffered := make ( chan interface { } , 1 )
// Run a goroutine that reads events from eventChan and does a
// non-blocking write to pendingEvent. This guarantees that writing to
// eventChan will never block, and that pendingEvent will have
// something in it if there's been an event since we read from that channel.
go func ( ) {
for {
select {
case <- stop :
return
case nextEvent := <- eventsChan :
select {
case eventsChanBuffered <- nextEvent :
default :
// We already have an event in eventsChanBuffered, so we'll
// do a refresh as soon as our throttle allows us to. It's fine
// to drop the event and keep whatever's in the buffer -- we
// don't do different things for different events
log . FromContext ( ctx ) . Debugf ( "Dropping event kind %T due to throttling" , nextEvent )
}
}
}
} ( )
return eventsChanBuffered
}