- Overview
- Specifying an
Interceptor
- Webhook
Interceptors
- GitHub
Interceptors
- GitLab
Interceptors
- Bitbucket
Interceptors
- CEL
Interceptors
- Implementing custom
Interceptors
An Interceptor
is a "catch-all" event processor for a specific platform that runs before the TriggerBinding.
It allows you to perform payload filtering, verification (using a secret),
transformation, define and test trigger conditions, and implement other useful processing. Once the event data passes through an Interceptor
, it then goes to the Trigger
before you pass
the payload data to the TriggerBinding
. You can also use an Interceptor
to modify the behavior of the associated Trigger
.
Tekton Triggers currently supports two distinct Interceptor
implementations:
- Standalone
Interceptors
, which are instances of theClusterInterceptor
Custom Resource Definition (CRD). You specify theseInterceptors
by referencing them, along with the desired parameters, within yourEventListener
. You can use theClusterInterceptor
CRD to implement your own customInterceptors
. - Legacy
Interceptors
, which you define entirely as part of of theEventListener
definition. This implementation will eventually be deprecated, so please consider transitioning to standaloneInterceptors
as soon as possible. See TEP-0026 for more context on this change.
Tekton Triggers ships with the following Interceptors
to help you get started:
- Webhook
Interceptors
- GitHub
Interceptors
- GitLab
Interceptors
- Bitbucket
Interceptors
- CEL
Interceptors
To specify an Interceptor
within your EventListener
, create an interceptors:
field with the following sub-fields:
name
- (optional) a name that uniquely identifies thisInterceptor
definitionref
- a reference to aClusterInterceptor
object with the following fields:name
- the name of the referencedClusterInterceptor
kind
- (optional) specifies that the referenced Kubernetes object is aClusterInterceptor
objectapiVersion
- (optional) specifies the target API version, for exampletriggers.tekton.dev/v1alpha1
params
-name
/value
pairs that specify the parameters you want to pass to theClusterInterceptor
params
- (optional)name
/value
pairs that specify the desired parameters for theInterceptor
; thename
field takes a string, while thevalue
field takes a valid JSON object
Below is an example standalone Interceptor
reference within an EventListener
definition:
interceptors:
- name: "validate GitHub payload and filter on eventType"
ref:
name: "github"
params:
- name: "secretRef"
value:
secretName: github-secret
secretKey: secretToken
- name: "eventTypes"
value: ["pull_request"]
- name: "CEL filter: only when PRs are opened"
ref:
name: "cel"
params:
- name: "filter"
value: "body.action in ['opened', 'reopened']"
Note: Tekton Triggers ships with only a legacy Webhook Interceptor
. If you want to implement it using
the standalone model, see Implementing custom Interceptors
.
A Webhook Interceptor
allows you to process your event payload by an external Kubernetes object containing
custom business logic. The Kubernetes object, exposed via a Kubernetes Service, receives event payload
data from your EventListener
via HTTP, applies its business logic to it, and returns the processed payload
(both headers plus body) via a HTTP 200 response to the EventListener
.
This payload can then continue on to the TriggerBinding
specified in the EventListener
. If processing
is not successful, the Interceptor
recognizes that the returned HTTP response code is not 200 and halts
further processing of the event payload.
You can optionally specify additional data to merge with the event payload before sending out for processing
by adding the data as canonical key/value pairs
in the Interceptor's
header
field.
Your external Kubernetes object must meet the following criteria:
- Fronted by a regular Kubernetes v1 Service running on HTTP port 80,
- Accepts JSON payloads over HTTP,
- Accepts HTTP POST requests with JSON payloads,
- Returns an HTTP 200 OK response when you want the
EventListener
to continue processing the event, - Returns a JSON object in the HTTP response body.
- Returns headers expected by subsequently chained
Interceptors
and the associatedTriggerBinding
.
Note: If your business logic does not modify either the HTTP payload's header or body, simply return the same HTTP header or body that you received.
Below is an example Webhook Interceptor
definition:
---
apiVersion: triggers.tekton.dev/v1alpha1
kind: EventListener
metadata:
name: listener-interceptor
spec:
serviceAccountName: tekton-triggers-example-sa
triggers:
- name: foo-trig
interceptors:
- webhook:
header:
- name: Foo-Trig-Header1
value: string-value
- name: Foo-Trig-Header2
value:
- array-val1
- array-val2
objectRef:
kind: Service
name: gh-validate
apiVersion: v1
namespace: default
bindings:
- ref: pipeline-binding
template:
ref: pipeline-template
A GitHub Interceptor
contains logic that validates and filters GitHub webhooks.
It can validate the webhook's origin as described in Securing your webhooks
as well as filter incoming events by the criteria you specify. The GitHub Interceptor
always preserves the payload data (both header and body) in its responses.
To use a GitHub Interceptor
as a GitHub webhook validator, do the following:
- Create a secret string value.
- Configure the
GitHub
webhook with that value. - Create a Kubernetes secret containing your secret value.
- Pass the Kubernetes secret as a reference to your GitHub
Interceptor
.
To use a GitHub Interceptor
as a filter for event data, specify the event types
you want the Interceptor
to accept in the eventTypes
field. The Interceptor
accepts data event types listed in Event types and payloads.
Below is an example GitHub Interceptor
reference:
triggers:
- name: github-listener
interceptors:
- ref:
name: "github"
kind: ClusterInterceptor
apiVersion: triggers.tekton.dev
params:
- name: "secretRef"
value:
secretName: github-secret
secretKey: secretToken
- name: "eventTypes"
value: ["pull_request"]
For reference, below is an example legacy GitHub Interceptor
definition:
triggers:
- name: github-listener
interceptors:
- github:
secretRef:
secretName: github-secret
secretKey: secretToken
eventTypes:
For more information, see our example of using this Interceptor
.
A GitLab Interceptor
contains logic that validates and filters GitLab webhooks.
It can validate the webhook's origin as described in Webhooks
as well as filter incoming events by the criteria you specify. The GitLab Interceptor
always preserves the payload data (both header and body) in its responses.
To use a GitLab Interceptor
as a GitLab webhook validator, do the following:
- Create a secret string value.
- Configure the GitLab webhook with that value.
- Create a Kubernetes secret containing your secret value.
- Pass the Kubernetes secret as a reference to your GitLab
Interceptor
.
To use a GitLab Interceptor
as a filter for event data, specify the event types
you want the Interceptor
to accept in the eventTypes
field. The Interceptor
accepts data event types listed in Events.
Below is an example GitLab Interceptor
reference:
interceptors:
- ref:
name: "gitlab"
params:
- name: "secretRef"
value:
secretName: foo
secretKey: bar
- name: "eventTypes"
value: ["Push Hook"]
For reference, below is an example legacy GitLab Interceptor
definition:
apiVersion: triggers.tekton.dev/v1alpha1
kind: EventListener
metadata:
name: gitlab-listener-interceptor
spec:
serviceAccountName: tekton-triggers-example-sa
triggers:
- name: foo-trig
interceptors:
- gitlab:
secretRef:
secretName: foo
secretKey: bar
eventTypes:
- Push Hook
bindings:
- ref: pipeline-binding
template:
ref: pipeline-template
A Bitbucket Interceptor
contains logic that validates and filters Bitbucket server webhooks.
It can validate the webhook's origin as described in Webhooks. It does
not support Bitbucket cloud due to lack of support for validating secrets; if you need this
functionality with Bitbucket, use a CEL Interceptor
instead. The Bitbucket Interceptor
always preserves the payload
data (both header and body) in its responses.
To use a Bitbucket Interceptor
as a Bitbucket server webhook validator, do the following:
- Create a secret string value.
- Configure the Bitbucket server webhook with that value.
- Create a Kubernetes secret containing your secret value.
- Pass the Kubernetes secret as a reference to your Bitbucket
Interceptor
.
To use a Bitbucket Interceptor
as a filter for event data, specify the event types
you want the Interceptor
to accept in the eventTypes
field. The Interceptor
accepts data event types listed in Event payload.
Below is an example Bitbucket Interceptor
reference:
interceptors:
- ref:
name: "bitbucket"
params:
- name: secretRef
value:
secretName: bitbucket-secret
secretKey: secretToken
- name: eventTypes
value:
- repo:refs_changed
For reference, below is an example legacy Bitbucket Interceptor
definition:
---
apiVersion: triggers.tekton.dev/v1alpha1
kind: EventListener
metadata:
name: bitbucket-listener
spec:
serviceAccountName: tekton-triggers-example-sa
triggers:
- name: bitbucket-triggers
interceptors:
- ref:
name: "bitbucket"
params:
- name: secretRef
value:
secretName: bitbucket-secret
secretKey: secretToken
- name: eventTypes
value:
- repo:refs_changed
bindings:
- ref: bitbucket-binding
template:
ref: bitbucket-template
A CEL Interceptor
allows you to filter and modify the payloads of incoming events using
the [CEL]https://github.com/google/cel-spec/blob/master/doc/langdef.md) expression language.
CEL Interceptors
support overlays
, which are CEL expressions that Tekton Triggers adds
to the event payload in the top-level extensions
field. overlays
are accessible from
TriggerBindings
.
In the example overlays
definition below, the Interceptor
adds two new fields to the
event payload that the corresponding TriggerBinding
will receive in addition to the standard
header
and body fields
: extensions.truncated_sha
and extensions.branch_name
:
triggers:
- name: cel-trig
interceptors:
- ref:
name: cel
params:
- name: "overlays"
value:
- key: truncated_sha
expression: "body.pull_request.head.sha.truncate(7)"
- key: branch_name
expression: "body.ref.split('/')[2]"
Below is the same example as a legacy Interceptor
:
triggers:
- name: cel-trig
interceptors:
- cel:
overlays:
- key: truncated_sha
expression: "body.pull_request.head.sha.truncate(7)"
- key: branch_name
expression: "body.ref.split('/')[2]"
You can use the key
element within the overlays
definition to create new or replace existing
elements within the extensions
field. This does not modify the body of the event payload, but instead
adds the extra fields to the top-level extensions
field.
For example, the following expression:
- key: short_sha
expression: "truncate(body.pull_request.head.sha, 7)"
can access the short_sha
field and its value that have been created in extensions
field:
{
"body": {
"ref": "refs/heads/master",
"pull_request": {
"head": {
"sha": "6113728f27ae82c7b1a177c8d03f9e96e0adf246"
}
}
},
"extensions": {
"short_sha": "6113728"
}
}
Note: You can also replace existing fields by specifying a key that matches the path to an existing field/value pair.
You can access the extra fields added by a CEL Interceptor
from your TriggerBinding
as follows:
apiVersion: triggers.tekton.dev/v1alpha1
kind: TriggerBinding
metadata:
name: pipeline-binding-with-cel-extensions
spec:
params:
- name: gitrevision
value: $(extensions.short_sha)
- name: branch
value: $(extensions.branch_name)
In the example CEL Interceptor
definition below, the cel-trig-with-matches
Trigger
filters events that don't have an 'X-GitHub-Event'
header matching 'pull_request'
and
adds an extra key to the JSON body of the payload with a truncated string derived from the hook
body:
triggers:
- name: cel-trig-with-matches
interceptors:
- ref:
name: "cel"
params:
- name: "filter"
value: "header.match('X-GitHub-Event', 'pull_request')"
- name: "overlays"
value:
- key: truncated_sha
expression: "body.pull_request.head.sha.truncate(7)"
bindings:
- name: sha
value: $(extensions.truncated_sha)
Below is the same example as a legacy Interceptor
:
triggers:
- name: cel-trig-with-matches
interceptors:
- cel:
filter: "header.match('X-GitHub-Event', 'pull_request')"
overlays:
- key: truncated_sha
expression: "body.pull_request.head.sha.truncate(7)"
bindings:
- name: sha
value: $(extensions.truncated_sha)
In the example CEL Interceptor
definition below, the filter
expression must
return a true
value for this Trigger
to execute and apply the specified overlays
:
You also have the option to omit the filter
expression entirely, in which case the Interceptor
applies the specified overlays
to the payload's body:
apiVersion: triggers.tekton.dev/v1alpha1
kind: EventListener
metadata:
name: cel-eventlistener-no-filter
spec:
serviceAccountName: tekton-triggers-example-sa
triggers:
- name: cel-trig
interceptors:
- ref:
name: "cel"
params:
- name: "overlays"
value:
- key: extensions.truncated_sha
expression: "body.pull_request.head.sha.truncate(7)"
bindings:
- ref: pipeline-binding
template:
ref: pipeline-template
You can chain Interceptors
with the following constraints:
-
ClusterInterceptors
do not modify the body of the event payload; instead, they add extra fields to the top-levelextensions
field. -
Webhook
Interceptors
can modify the body of the event payload, but cannot access the top-levelextensions
field.
Each ClusterInterceptor can return values in the InterceptorResponse within the extensions
field. These values are then added to the extensions
field of the InterceptorRequest that is sent to the next interceptor in the chain.
If two interceptors return an extensions field with the same name, the latter one will overwrite the one from the previous one i.e. if interceptors A and B both return foo
in the Extensions field of the InterceptorResponse, the values written by B will overwrite the ones written by A. To prevent this, it is recommended that each cluster interceptor write to its own top level field i.e A returns A.foo
and B return B.foo
in the InterceptorResponse.
Note: We are working on changing the behavior of Webhook Interceptors
to match that of CEL Interceptors
so that both Interceptor
types can share data via the top-level extensions
field.
Since Webhook Interceptors
cannot access the extensions
field, the EventListener
adds the extensions
field to the body of the event payload before passing it to any Webhook Interceptor
.
In the example Interceptor
chain shown below, the first CEL Interceptor
adds the truncated_sha
field to the extensions
field and the EventListener
adds it to the body of the payload so that
the Webhook Interceptor
can access that data:
interceptors:
- cel:
overlays:
- key: "truncated_sha"
expression: "body.sha.truncate(5)"
- webhook:
objectRef:
kind: Service
name: some-interceptor
apiVersion: v1
- cel:
filter: "body.extensions.truncated_sha == \"abcde\"" # Can also be extensions.truncated_sha == \"abcde\"
As a result, the body of the payload sent to the Webhook Interceptor
is as follows:
{
"sha": "abcdefghi", // Original field
"extensions": {
"truncated_sha": "abcde"
}
}
As long as the Webhook Interceptor
does not modify the body of the payload, the last CEL interceptor in the chain and the target TriggerBinding
can access the truncated_sha
field both in the body of the payload as well as via the extra fields added to the top-level extension
field, namely $(body.extensions.truncated_sha)
as well as $(extensions.truncated_sha)
.
Tekton Triggers ships with the ClusterInterceptor
Custom Resource Definition (CRD), which you can use to implement custom Interceptors
. See ClusterInterceptors
for more information.