本文档记录了开发 kong 插件所需注意的要点
开发插件之前,至少阅读一遍开发文档,卡壳儿的时候及时回顾文档
https://docs.konghq.com/0.14.x/plugin-development/
特别注意文档的版本,需要和你所使用的 kong 的版本相匹配,否则写出来的代码不兼容。
kong version
另外,可以参考 kong 内置的插件代码, 足够我们进行 copy&paste
find . -name handler.lua
通过 grep 发现,内置插件中并没有 http 请求的代码,补充如下:
  token, err = retrieve_token(conf)
  local resp = {}
  r, c, h = http.request {
    method = "GET",
    headers = {
      Authorization = token
    },
    url = "http://192.168.1.150/api/roles",
    sink=ltn12.sink.table(resp)
  }
  kong.log.inspect(r, c, h, resp)
注: 如果使用 docker/k8s 部署的 kong,请先登录容器然后再执行上述命令
使用luarocks 打包,文档里有写,这里给出一个实际例子。
打包前需要编写一个配置文件,参考如下
package = "kong-plugin-rbac"
version = "1.0.0-1"
-- The version '1.0.0' is the source code version, the trailing '1' is the version of this rockspec.
-- whenever the source version changes, the rockspec should be reset to 1. The rockspec version is only
-- updated (incremented) when this file changes, but the source remains the same.
-- TODO: This is the name to set in the Kong configuration `plugins` setting.
-- Here we extract it from the package name.
local pluginName = package:match("^kong%-plugin%-(.+)$")
supported_platforms = {"linux", "macosx"}
source = {
  url = "http://sky-cloud.net",
  tag = "1.0.0"
}
description = {
  summary = "rbac plugin for sky-cloud services",
  homepage = "http://sky-cloud.net",
  license = "private"
}
dependencies = {
}
build = {
  type = "builtin",
  modules = {
    -- TODO: add any additional files that the plugin consists of
    -- 根据自己的实际路径去修改
    ["kong.plugins."..pluginName..".handler"] = "kong/plugins/"..pluginName.."/handler.lua",
    ["kong.plugins."..pluginName..".schema"] = "kong/plugins/"..pluginName.."/schema.lua",
  }
}
打包命令
luarocks pack kong-plugin-rbac 1.0.0-1
上传到服务器,并安装,安装命令如下:
luarocks install kong-plugin-rbac-1.0.0-1.all.rock
在 docker 环境下 kong 插件的部署要较麻烦一些,需要做路径映射,将本地安装的 kong 插件映射到容器内,同时修改 lua 包的搜索路径。kong 的配置文件里的参数都是可以通过环境变量控制的,变量名的规律为 xx_xx, KONG_XX_XX, 比如 lua_package_path 参数对应的环境变量名为 KONG_LUA_PACKAGE_PATH.
    volumeMounts:
        - name: tz
          mountPath: /etc/localtime
        - name: rbac
          mountPath: /kong/plugins/rbac/
        - name: logs
          mountPath: /usr/local/kong/logs/
      volumes:
      - name: tz
        hostPath:
          path: /etc/localtime
      - name: rbac
        hostPath:
          path: /usr/local/share/lua/5.1/kong/plugins/rbac/
      - name : logs
        hostPath:
          path: /srv/nap/kong/logs/
如上所示,将本地的 rbac 插件目录,映射到了容器内。然后按照如下的方式修改注入环境变量。
先使用 luarocks 将代码打包, 打包需要一个配置文件,参考如下
        - name: KONG_LUA_PACKAGE_PATH
          value: /usr/lib64/lua/5.1/?.lua;/usr/lib64/lua/5.1/?/init.lua;/usr/share/lua/5.1/?.lua;/usr/share/lua/5.1/?/init.lua;./?.lua;./?/init.lua;/usr/local/share/lua/5.1/kong/plugins/rbac/?.lua;
        - name: KONG_LOG_LEVEL
          value: debug
        - name: KONG_PLUGINS
          value: rbac,jwt-keycloak,oidc,acl,aws-lambda,azure-functions,basic-auth,bot-detection,correlation-id,cors,datadog,file-log,hmac-auth,http-log,ip-restriction,jwt,key-auth,ldap-auth,loggly,oauth2,post-function,pre-function,prometheus,rate-limiting,request-size-limiting,request-termination,request-transformer,statsd,syslog,tcp-log,udp-log,zipkin
kong 的多个插件是按照优先级依次执行的,要注意这点,方便按照预期的方式调试/验证代码.
local RBACHandler =  BasePlugin:extend()
local priority_env_var = "KONG_RBAC_PRIORITY"
local priority
if os.getenv(priority_env_var) then
    priority = tonumber(os.getenv(priority_env_var))
else
    -- priority = 1500
    priority = 999
end
kong.log.debug('KONG_RBAC_PRIORITY: ' .. priority)
内置插件的优先级可以参考文档,保证自己开发的插件在合适的时机运行。
pdk 是 kong 提供的插件开发包,方便我们在不同插件之间共享数据,以及操作 db,使用方式比较简单,参考文档即可,但要注意文档的版本。