-
Notifications
You must be signed in to change notification settings - Fork 750
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
re-architecting around gateways and sensors #92
Merged
Changes from 1 commit
Commits
Show all changes
37 commits
Select commit
Hold shift + click to select a range
12af598
added gateway for processing outside events, making gateway long runn…
VaibhavPage 4bb4457
refactoring k8s manifests
VaibhavPage 95c2548
adding s3 gateway
VaibhavPage 5839ee9
making s3 and webhook gateway mult configurable
VaibhavPage c3542da
refactor of calendar, s3 and webhook gateway done. started with streams
VaibhavPage 543f698
added nats gateway
VaibhavPage 44956f4
renamed signals to gateways. moved out-of-box gateways to core.
VaibhavPage 78843be
refactor gateways
VaibhavPage cdc20f3
removing signals folder
VaibhavPage 2ab9407
added kafka gateway
VaibhavPage 7caee70
gateway and sensor dirs under a common controllers and clients directory
VaibhavPage 2db02fe
adding resource, amqp and mqtt gateways. refactored sensor watcher
VaibhavPage 62ccce9
adding gateway examples, readme etc
VaibhavPage 721bac5
adding gateway docs. refactor gateway and sensor examples
VaibhavPage 57791a9
added watcher for sensor. Completion count for sensor. Updated gatewa…
VaibhavPage ee5fefb
sensor's signal are gateway-name/configuration-name
VaibhavPage a039fa1
making gateway processor as gRPC server
VaibhavPage 16d175a
giving user options to implement gateway either by mimicing core gate…
VaibhavPage 66d3247
gateway spec changed. divided extending gateways into 3 parts. starte…
VaibhavPage 28db2b8
updated gateway examples with deploySpec. started writing custom gate…
VaibhavPage 11110d3
adding wiki for writing custom gateways
VaibhavPage 8947fe5
updating wiki. grpc gateway updated DialOptions
VaibhavPage 6828b2e
added REST gateways. wiki to follow
VaibhavPage 50ec67c
refactor rest gateways
VaibhavPage b8c79c0
added k8 events for gateway. Gateway configuration state is now maint…
VaibhavPage 99b04d5
minor refactor in k8 events
VaibhavPage 655028a
added gateways as watchers for gateway along with sensors. started ad…
VaibhavPage 3b0c7b0
added k8 event for escalation. nodes are deleted from gateway resourc…
VaibhavPage b43c934
added unit tests. refactored sensor notification handler. added k8 ev…
VaibhavPage 970c63b
adding test cases for sensor controller, operator and notification ha…
VaibhavPage 9ec8c44
updating webhook gateway
VaibhavPage 6433b77
fixed issue with syncing gateway nodes and gateway configmap
VaibhavPage 6691b07
test: cleaning up test cases
VaibhavPage d08ac54
updating dependencies
VaibhavPage 92b31a5
update docker registry to argoproj
VaibhavPage ac90e6b
exposing gateway and sensor validate method
VaibhavPage edfa3a1
update dependencies
VaibhavPage File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
package common | ||
|
||
import ( | ||
corev1 "k8s.io/api/core/v1" | ||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||
"time" | ||
"k8s.io/client-go/kubernetes" | ||
) | ||
|
||
// K8Event abstracts kubernetes event. | ||
type K8Event struct { | ||
// The object that this event is about. | ||
Name string | ||
// Namespace where event should be created | ||
Namespace string | ||
// What action was taken/failed regarding to the Regarding object. | ||
Action string | ||
// This should be a short, machine understandable string that gives the reason | ||
// for the transition into the object's current status. | ||
Reason string | ||
// Kind of component generating this event | ||
Kind string | ||
// Name of the controller that emitted this Event | ||
ReportingController string | ||
// ID of the controller instance | ||
ReportingInstance string | ||
// Type of this event (Normal, Warning), new types could be added in the future | ||
Type string | ||
// Map of string keys and values that can be used to organize and categorize | ||
// (scope and select) objects. | ||
Labels map[string]string | ||
} | ||
|
||
// CreateK8Event returns a kubernetes event object | ||
func GetK8Event(event *K8Event) *corev1.Event{ | ||
return &corev1.Event{ | ||
Reason: event.Reason, | ||
Type: event.Type, | ||
Action: event.Action, | ||
EventTime: metav1.MicroTime{ | ||
Time: time.Now(), | ||
}, | ||
ObjectMeta: metav1.ObjectMeta{ | ||
Namespace: event.Namespace, | ||
Name: event.Name + "-" + RandomStringGenerator(), | ||
Labels: event.Labels, | ||
}, | ||
InvolvedObject: corev1.ObjectReference{ | ||
Namespace: event.Namespace, | ||
Name: event.Name, | ||
Kind: event.Kind, | ||
}, | ||
Source: corev1.EventSource{ | ||
Component: event.Name, | ||
}, | ||
ReportingInstance: event.ReportingInstance, | ||
ReportingController: event.ReportingController, | ||
} | ||
} | ||
|
||
// CreateK8Event creates a kubernetes event resource | ||
func CreateK8Event(event *corev1.Event, clientset kubernetes.Interface) error { | ||
_, err := clientset.CoreV1().Events(event.ObjectMeta.Namespace).Create(event) | ||
return err | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,38 +25,121 @@ import ( | |
v1alpha "github.com/argoproj/argo-events/pkg/apis/sensor/v1alpha1" | ||
"github.com/tidwall/gjson" | ||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||
"time" | ||
"github.com/argoproj/argo-events/common" | ||
) | ||
|
||
// createEscalationEvent creates a k8 event for escalation | ||
func (se *sensorExecutor) createEscalationEvent(policy *v1alpha1.EscalationPolicy, signalFilterName string) error { | ||
se.log.Info().Interface("policy", policy).Msg("printing policy") | ||
event := common.GetK8Event(&common.K8Event{ | ||
Name: policy.Name, | ||
Namespace: se.sensor.Namespace, | ||
ReportingInstance: se.sensor.Name, | ||
ReportingController: se.sensor.Name, | ||
Labels: map[string]string{ | ||
common.LabelEventSeen: "", | ||
common.LabelSignalName: signalFilterName, | ||
}, | ||
Type: common.LabelArgoEventsEscalationKind, | ||
Action: string(policy.Level), | ||
Reason: policy.Message, | ||
}) | ||
err := common.CreateK8Event(event, se.kubeClient) | ||
return err | ||
} | ||
|
||
// apply the signal filters to an event | ||
func filterEvent(f v1alpha1.SignalFilter, event *v1alpha.Event) (bool, error) { | ||
dataRes, err := filterData(f.Data, event) | ||
return filterTime(f.Time, &event.Context.EventTime) && filterContext(f.Context, &event.Context) && dataRes, err | ||
func (se *sensorExecutor) filterEvent(f v1alpha1.SignalFilter, event *v1alpha.Event) (bool, error) { | ||
dataRes, err := se.filterData(f.Data.Filters, event) | ||
// generate sensor failure event and mark sensor as failed | ||
if err != nil { | ||
return false, err | ||
} | ||
if !dataRes { | ||
err = se.createEscalationEvent(f.Data.EscalationPolicy, f.Name) | ||
if err != nil { | ||
return false, err | ||
} | ||
} | ||
timeRes, err := se.filterTime(f.Time, &event.Context.EventTime) | ||
// generate sensor failure event and mark sensor as failed | ||
if err != nil { | ||
return false, err | ||
} | ||
if !timeRes { | ||
err = se.createEscalationEvent(f.Time.EscalationPolicy, f.Name) | ||
if err != nil { | ||
return false, err | ||
} | ||
} | ||
ctxRes := se.filterContext(f.Context, &event.Context) | ||
if !ctxRes { | ||
err = se.createEscalationEvent(f.Context.EscalationPolicy, f.Name) | ||
if err != nil { | ||
return false, err | ||
} | ||
} | ||
return timeRes && ctxRes && dataRes, err | ||
} | ||
|
||
// applyTimeFilter checks the eventTime against the timeFilter: | ||
// 1. the eventTime is greater than or equal to the start time | ||
// 2. the eventTime is less than the end time | ||
// returns true if 1 and 2 are true and false otherwise | ||
func filterTime(timeFilter *v1alpha1.TimeFilter, eventTime *metav1.Time) bool { | ||
if timeFilter != nil && eventTime != nil { | ||
if timeFilter.Start != nil && timeFilter.Stop != nil { | ||
return (timeFilter.Start.Before(eventTime) || timeFilter.Start.Equal(eventTime)) && eventTime.Before(timeFilter.Stop) | ||
func (se *sensorExecutor) filterTime(timeFilter *v1alpha1.TimeFilter, eventTime *metav1.Time) (bool, error) { | ||
if timeFilter != nil { | ||
currentT := time.Now().UTC() | ||
se.log.Info().Str("current-time", currentT.String()).Msg("current time") | ||
currentTStr := fmt.Sprintf("%d-%s-%d", currentT.Year(), int(currentT.Month()), currentT.Day()) | ||
|
||
if timeFilter.Start != "" && timeFilter.Stop != "" { | ||
se.log.Info().Str("start time format", currentTStr + " " + timeFilter.Start).Msg("start time format") | ||
startTime, err := time.Parse("2006-01-02 15:04:05", currentTStr + " " + timeFilter.Start) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. let's make the format string There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. done 👍 |
||
if err != nil { | ||
fmt.Println(err) | ||
return false, err | ||
} | ||
se.log.Info().Str("start time", startTime.String()).Msg("start time") | ||
startTime = startTime.UTC() | ||
se.log.Info().Str("stop time format", currentTStr + " " + timeFilter.Stop).Msg("stop time format") | ||
stopTime, err := time.Parse("2006-01-02 15:04:05", currentTStr + " " + timeFilter.Stop) | ||
if err != nil { | ||
fmt.Println(err) | ||
return false, err | ||
} | ||
se.log.Info().Str("stop time", stopTime.String()).Msg("stop time") | ||
stopTime = stopTime.UTC() | ||
return (startTime.Before(eventTime.Time) || stopTime.Equal(eventTime.Time)) && eventTime.Time.Before(stopTime), nil | ||
} | ||
if timeFilter.Start != nil { | ||
if timeFilter.Start != "" { | ||
// stop is nil - does not have an end | ||
return timeFilter.Start.Before(eventTime) || timeFilter.Start.Equal(eventTime) | ||
startTime, err := time.Parse("2006-01-02 15:04:05", currentTStr + " " + timeFilter.Start) | ||
if err != nil { | ||
return false, err | ||
} | ||
se.log.Info().Str("start time", startTime.String()).Msg("start time") | ||
startTime = startTime.UTC() | ||
return startTime.Before(eventTime.Time) || startTime.Equal(eventTime.Time), nil | ||
} | ||
if timeFilter.Stop != nil { | ||
return eventTime.Before(timeFilter.Stop) | ||
if timeFilter.Stop != "" { | ||
stopTime, err := time.Parse("2016-01-02 15:04:05", currentTStr + " " + timeFilter.Stop) | ||
if err != nil { | ||
return false, err | ||
} | ||
se.log.Info().Str("stop time", stopTime.String()).Msg("stop time") | ||
stopTime = stopTime.UTC() | ||
return eventTime.Time.Before(stopTime), nil | ||
} | ||
} | ||
return true | ||
se.log.Info().Msg("NO time filter") | ||
return true, nil | ||
} | ||
|
||
// applyContextFilter checks the expected EventContext against the actual EventContext | ||
// values are only enforced if they are non-zero values | ||
// map types check that the expected map is a subset of the actual map | ||
func filterContext(expected *v1alpha.EventContext, actual *v1alpha.EventContext) bool { | ||
func (se *sensorExecutor) filterContext(expected *v1alpha.EventContext, actual *v1alpha.EventContext) bool { | ||
if expected == nil { | ||
return true | ||
} | ||
|
@@ -89,9 +172,12 @@ func filterContext(expected *v1alpha.EventContext, actual *v1alpha.EventContext) | |
// applyDataFilter runs the dataFilter against the event's data | ||
// returns (true, nil) when data passes filters, false otherwise | ||
// TODO: split this function up into smaller pieces | ||
func filterData(dataFilters []*v1alpha1.DataFilter, event *v1alpha.Event) (bool, error) { | ||
func (se *sensorExecutor) filterData(dataFilters []*v1alpha1.DataFilter, event *v1alpha.Event) (bool, error) { | ||
// TODO: use the event.Context.SchemaURL to figure out correct data format to unmarshal to | ||
// for now, let's just use a simple map[string]interface{} for arbitrary data | ||
if dataFilters == nil { | ||
return true, nil | ||
} | ||
if event == nil { | ||
return false, fmt.Errorf("nil event") | ||
} | ||
|
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can you not just use
GenerateName
here instead?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yup, had it changed in my local. I'll push some new changes with test cases