Lugate is a library for building JSON-RPC 2.0 Gateway API just inside of your NGINX configuration file
location / {
# MIME type determined by default_type:
default_type 'application/json';
content_by_lua_block {
-- Load lugate module
local Lugate = require "lugate"
-- Get new lugate instance
local lugate = Lugate:init({
json = require "rapidjson", -- 1. Require wrapper to work with json (should be installed)
ngx = ngx, -- 2. Require nginx instance
cache = {'redis', '127.0.0.1', 6379}, -- 3. Configure cache wrapper (redis and dummy cache modules are currently available)
routes = { -- 4. Routing rules
['v1%.([^%.]+).*'] = '/v1/%1', -- 4.1 v1.math.subtract -> /v1/math (for example)
['v2%.([^%.]+).*'] = '/v2/%1', -- 4.2 v2.math.addition -> /v2/math (for example)
},
debug = true, -- 5. Enable debug mode (write all requests and responses to the nginx error log)
})
-- Send multi requst and get multi response
lugate:run()
-- Print responses out
lugate:print_responses()
}
}
When we talk about Microservices Architecture pattern there is a thing there called API Gateway which is the single entry point into the system.
Lugate is a binding over OpenResty's ngx_http_lua_module module. The library provides features for request parsing, validating and routing. Lugate acts like a caching proxy over the JSON-RPC 2.0 protocol and is meant to be used like an entry point for your application.
The autogenerated API reference is available at: API reference.
First of all, you have to have nginx compiled with
ngx_http_lua_module module.
Take a look at the example/provision.sh
file to get a tip on how to do it.
Lugate by itself can be installed via the luarocks package manager. Just run:
luarocks install lugate
in work...
Lugate brings its own protocol to live. It's goal is transferring, routing and distributing cache control logic over JSON-RPC 2.0 request.
The params member of the request object is wrapped in additional structure:
- route - The routing destination
- params - The original parameters member value should be moved here for now
- cache - Cache control logic
- ttl - Cache time to life
- key - Cache key
- tags - Cache tags
When the request is processed by the lugate proxy, the route and cache values are removed from the params member and the nested params value is expanded to fill the whole parent params field.
The request routing plan:
- Request is preprocessed by the the lugate proxy. Location: gateway.lugate.loc:
{
"jsonrpc": "2.0",
"method": "subtract",
"params": {
"cache": {
"ttl": 3600,
"key": "foobar",
"tags": ["news_list", "top7"]
},
"route": "v2.substract",
"params": [42, 23]
},
"id": 1
}
- Request is send to the proper service (deciding of the
route
param). Location: /v2/substract:
{
"jsonrpc": "2.0",
"method": "substract",
"params": [42, 23],
"id": 1
}
- Request is sent back to client:
{
"jsonrpc": "2.0",
"result": -19,
"id": 2
}
The routing destination. The value is processed by the lua gsub function.
The original params values.
The time-to-life value is used to set cache expiration time. If ttl is false, no cache is set.
The cache key is calculated by client. This approach gives more flexibility for debugging and development goals.
The cache tags. Cache Keys Stored in special Tag sets. Use it for cache invalidation.
You can use the pre
and post
hooks to paste additional logic to the run
method:
pre(lugate)
is evaluated before all requests are processed;post(lugate)
- after all requests are processed;cache(lugate, response)
- before cache is stored;
The lugate
instance is passed to the callback as the only parameter. If false
is returned the futher evaluation is stopped.
Example on how to define hooks:
local lugate = Lugate:init({
json = require "rapidjson",
ngx = ngx,
cache = {'redis', '127.0.0.1', 6379},
routes = {
['v1%.([^%.]+).*'] = '/v1/%1', -- v1.math.subtract -> /v1/math
['v2%.([^%.]+).*'] = '/v2/%1', -- v2.math.addition -> /v2/math
},
pre = function(lugate)
ngx.say("Hi there!")
ngx.exit(ngx.HTTP_OK)
end
})
Use busted for running tests (from the root directory of the project).
The example project uses vagrant provision. So you need to have vagrant and virtualbox installed.
Launch project by running this command from the example directory:
vagrant up
Add this line to your local /etc/hosts
file:
192.168.50.47 gateway.lugate.loc service01.lugate.loc service02.lugate.loc service03.lugate.loc
- HTTP errors handling (#4)
- Better request/response factories testing
- Invalid JSON detection
- Debug log and version header features
- Prevent invalid JSON on empty responses in batch
- Support for cache tagging
- Hooks feature added (middlewares)
- Redis cache support