PermalinkPlugins in Other Languages
PermalinkIntroduction
External plugins are those that run on a process separate from Kong Gateway itself, enabling the use of any programming language for which an appropriate plugin server is available.
Each plugin server hosts one or more plugins and communicates with the main Kong Gateway process through Unix sockets. If so configured, Kong Gateway can manage those processes, starting, restarting and stopping as necessary.
Kong Gateway currently maintains a Go language plugin server, go-pluginserver and the corresponding PDK library package, go-pdk.
PermalinkKong Gateway plugin server configuration
The pluginserver_names
property is a comma-separated list of names, one
for each plugin server process. These names are used to group each process’
properties and to annotate log entries.
For each name, other properties can be defined:
Property | Description | Default |
---|---|---|
pluginserver_<NAME>_socket |
Unix socket path | /usr/local/kong/<NAME>.socket |
pluginserver_<NAME>_start_cmd |
Command to start the plugin server process | /usr/local/bin/<NAME> |
pluginserver_<NAME>_query_cmd |
Command to dump available plugins’ info | /usr/local/bin/query_<NAME> |
For example, you could set Go and Python plugins like this (assuming
an hypothetical Python plugin server called pypluginserver.py
):
pluginserver_names = go,python
pluginserver_go_socket = /usr/local/kong/go_pluginserver.sock
pluginserver_go_start_cmd = go-pluginserver -kong-prefix /usr/local/kong/ -plugins-directory /usr/local/kong/go-plugins
pluginserver_go_query_cmd = go-pluginserver -dump-all-plugins -plugins-directory /usr/local/kong/go-plugins
pluginserver_python_socket = /usr/local/kong/python_pluginserver.sock
pluginserver_python_start_cmd = pypluginserver.py
pluginserver_python_query_cmd = pypluginserver.py -dump
PermalinkLegacy configuration
Kong Gateway versions 2.0.x to 2.2.x supported only Go external plugins and a single plugin server using a different configuration style. Starting with Kong Gateway version 2.3, the old style is recognized and internally transformed to the new style.
If property pluginserver_names
isn’t defined, the legacy properties
go_plugins_dir
and go_pluginserver_exe
are tried:
Property | Description | Default |
---|---|---|
go_plugins_dir |
Directory with Go plugins | off , meaning to disable Go plugins |
go_pluginserver_exe |
Path to the go-pluginserver executable | /usr/local/bin/go-pluginserver |
Notes:
- The old style doesn’t allow multiple plugin servers.
- Version 0.5.0 of go-pluginserver requires the old style configuration.
- The new style configuration requires v0.6.0 of go-pluginserver
PermalinkDeveloping Go plugins
Kong Gateway support for the Go language consist of two parts:
- go-pdk as a library, provides Go functions to access Kong Gateway features of the PDK.
- go-pluginserver an executable to dynamically load plugins written in Go.
Notes:
The Kong Gateway version 2.3 allows multiple plugin servers; in particular it’s now possible to write single-plugin servers, in effect plugins as microservices. To help with this, version v0.6.0 of the go-pdk package includes an optional plugin server. See Embedded Server for more information.
The go-pluginserver process is still supported. Its main advantage is that it’s a single process for any number of plugins, but the dynamic loading of plugins has proven challenging under the Go language (unlike the microservice architecture, which is well supported by the language and tools).
PermalinkDevelopment
To write a Kong Gateway plugin in Go, you need to:
- Define a structure type to hold configuration.
- Write a
New()
function to create instances of your structure. -
Add methods on that structure to handle phases.
If you want a dynamically-loaded plugin to be used with go-pluginserver:
- Compile your Go plugin with
go build -buildmode plugin
. -
Put the resulting library (the
.so
file) into thego_plugins_dir
directory.If you want a standalone plugin microservice:
- Include the
go-pdk/server
sub-library. - Add a
main()
function that callsserver.StartServer(New, Version, Priority)
. - Compile as an executable with
go build
.
Note: Check out this repository for example Go plugins.
Permalink1. Configuration Structure
Plugins written in Lua define a schema to specify how to read and validate configuration data coming from the datastore or the Admin API. Since Go is a statically-typed language, all that specification is handled by defining a configuration structure:
type MyConfig struct {
Path string
Reopen bool
}
Public fields (that is, those starting with a capital letter) will be filled
with configuration data. If you want them to have a different name in the
datastore, add field tags as defined in the encoding/json
package:
type MyConfig struct {
Path string `json:my_file_path`
Reopen bool `json:reopen`
}
Permalink2. New() Constructor
Your plugin must define a function called New
that creates an instance of this type
and returns as an interface{}
. In most cases, it’s just this:
func New() interface{} {
return &MyConfig{}
}
You can add more fields to the structure and they’ll be passed around, but there are no guarantees about the lifetime or quantity of configuration instances.
Permalink3. Phase Handlers
Similarly to Kong Gateway Lua plugins, you can implement custom logic to be executed at
various points of the request processing lifecycle. For example, to execute
custom Go code in the access phase, define a function named Access
:
func (conf *MyConfig) Access (kong *pdk.PDK) {
...
}
The phases you can implement custom logic for are, and the expected function signature is the same for all of them:
Certificate
Rewrite
Access
Response
Preread
Log
Similar to Lua plugins, the presence of the Response
handler automatically enables the buffered proxy mode.
PermalinkEmbedded server
Each plugin can be a microservice, compiled as a standalone executable.
To use the embedded server, include github.com/Kong/go-pdk/server
in
the imports list, and add a main()
function:
func main () {
server.StartServer(New, Version, Priority)
}
Note that the main()
function must have a package main
line at the
top of the file.
Then, a standard Go build creates an executable. There are no extra go-pluginserver, no plugin loading, and no compiler/library/environment compatibility issues.
The resulting executable can be placed somewhere in your path (for example,
/usr/local/bin
). The common -h
flag shows a usage help message:
$ my-plugin -h
Usage of my-plugin:
-dump
Dump info about plugins
-help
Show usage info
-kong-prefix string
Kong prefix path (specified by the -p argument commonly used in the Kong CLI) (default "/usr/local/kong")
When run without arguments, it will create a socket file with the
kong-prefix
and the executable name, appending .socket
. For example,
if the executable is my-plugin
, it would be
/usr/local/kong/my-plugin.socket
by default.
PermalinkExample configuration
Two standalone plugins, called my-plugin
and other-one
:
pluginserver_names = my-plugin,other-one
pluginserver_my_plugin_socket = /usr/local/kong/my-plugin.socket
pluginserver_my_plugin_start_cmd = /usr/local/bin/my-plugin
pluginserver_my_plugin_query_cmd = /usr/local/bin/my-plugin -dump
pluginserver_other_one_socket = /usr/local/kong/other-one.socket
pluginserver_other_one_start_cmd = /usr/local/bin/other-one
pluginserver_other_one_query_cmd = /usr/local/bin/other-one -dump
Note that the socket and start command settings coincide with their defaults, so they can be omitted:
pluginserver_names = my-plugin,other-one
pluginserver_my_plugin_query_cmd = /usr/local/bin/my-plugin -dump
pluginserver_other_one_query_cmd = /usr/local/bin/other-one -dump