This feature makes it possible to write Tyk middleware using your favorite languages.
Python support is available, more notes here.
Lua support is available as well. Notes here.
Tyk provides support for gRPC, you may use any of the gRPC supported languages (e.g. Ruby, Java, etc.). Check the gRPC support README.
The ID extractor is a very useful mechanism that will let you cache your authentication IDs and prevent certain requests from hitting your CP backend. It takes a set of rules from your API configuration (the rules are set per API).
A sample usage will look like this:
"custom_middleware": {
"pre": [
{
"name": "MyPreMiddleware",
"require_session": false
}
],
"id_extractor": {
"extract_from": "header",
"extract_with": "value",
"extractor_config": {
"header_name": "Authorization"
}
},
"driver": "grpc"
},
Tyk provides a set of ID extractors that aim to cover the most common use cases, a very simple one is the value extractor.
This feature implements an in-process message passing mechanism, based on Protocol Buffers, any supported languages should provide a function to receive, unmarshal and process this kind of messages.
The main interoperability task is achieved by using cgo as a bridge between a supported language -like Python- and the Go codebase.
Your C bridge function must accept and return a CoProcessMessage
data structure like the one described in api.h
, where p_data
is a pointer to the serialized data and length
indicates the length of it.
struct CoProcessMessage {
void* p_data;
int length;
};
The unpacked data will hold the actual CoProcessObject
data structure, where HookType
represents the hook type (see below), Request
represents the HTTP request and Session
is the Tyk session data.
The Spec
field holds the API specification data, like organization ID, API ID, etc.
type CoProcessObject struct {
HookType string
Request CoProcessMiniRequestObject
Session SessionState
Metadata map[string]string
Spec map[string]string
}
coprocess.Dispatcher
describes a very simple interface for implementing the dispatcher logic, the required methods are: Dispatch
, DispatchEvent
and Reload
.
Dispatch
accepts a pointer to a struct CoProcessObject
(as described above) and must return an object of the same type. This method will be called for every configured hook, on every request. Traditionally this method will perform a single function call on the target language side (like Python_DispatchHook
in coprocess_python
), and the corresponding logic will be handled from there (mostly because different languages have different ways of loading, referencing or calling middlewares).
DispatchEvent
provides a way of dispatching Tyk events to a target language. This method doesn't return any variables but does receive a JSON-encoded object containing the event data. For extensibility purposes, this method doesn't use Protocol Buffers, the input is a []byte
, the target language will take this (as a char
) and perform the JSON decoding operation.
Reload
is called when triggering a hot reload, this method could be useful for reloading scripts or modules in the target language.
This component is in charge of dispatching your HTTP requests to the custom middlewares, in the right order. The dispatcher follows the standard middleware chain logic and provides a simple mechanism for "hooking" your custom middleware behavior, the supported hooks are:
Pre: gets executed before any authentication information is extracted from the header or parameter list of the request.
Post: gets executed after the authentication, validation, throttling, and quota-limiting middleware has been executed, just before the request is proxied upstream. Use this to post-process a request before sending it to your upstream API.
PostKeyAuth: gets executed right after the autentication process.
CustomAuthCheck: gets executed as a custom authentication middleware, instead of the standard ones provided by Tyk. Use this to provide your own authentication mechanism.
coprocess_api.go
provides a bridge between the gateway API and C, any function that needs to be exported should have the export
keyword:
//export TykTriggerEvent
func TykTriggerEvent( CEventName *C.char, CPayload *C.char ) {
eventName := C.GoString(CEventName)
payload := C.GoString(CPayload)
FireSystemEvent(tykcommon.TykEvent(eventName), EventMetaDefault{
Message: payload,
})
}
You should also expect a header file declaration of this function in api.h
, like this:
#ifndef TYK_COPROCESS_API
#define TYK_COPROCESS_API
extern void TykTriggerEvent(char* event_name, char* payload);
#endif
The language binding will include this header file (or declare the function inline) and perform the necessary steps to call it with the appropriate arguments (like a ffi
mechanism could do). As a reference, this is how this could be achieved if you're building a Cython module:
cdef extern:
void TykTriggerEvent(char* event_name, char* payload);
def call():
event_name = 'my event'.encode('utf-8')
payload = 'my payload'.encode('utf-8')
TykTriggerEvent( event_name, payload )
The intended way of using a Coprocess middleware is to specify it as part of an API definition:
"custom_middleware": {
"pre": [
{
"name": "MyPreMiddleware",
"require_session": false
},
{
"name": "AnotherPreMiddleware",
"require_session": false
}
],
"post": [
{
"name": "MyPostMiddleware",
"require_session": false
}
],
"post_key_auth": [
{
"name": "MyPostKeyAuthMiddleware",
"require_session": true
}
],
"auth_check": {
"name": "MyAuthCheck"
},
"driver": "python"
}
It's important to note that all hook types support chaining except the custom auth check (auth_check
).
It's possible to use a build tag:
go build -tags 'coprocess python'
go build -tags 'coprocess somelanguage'
Each language should implement a CoProcessInit
function, this will be called from the main function when the coprocess
build tag is used.
Using the coprocess
build tag with no language tag will fail.
A standard build is still possible:
go build
coprocess_dummy.go
provides a dummy CoProcessInit
function that will be called if you perform a standard Tyk build. This file will be ignored when using the coprocess
build tag, as we expect it to be implemented by a language.
You must use the coprocess
build tag to run the tests:
go test -tags 'coprocess'
go test -run CoProcess -tags 'coprocess'