Learn how to implement VerneMQ plugins using the Lua Scripting Language.
Developing VerneMQ plugins in Erlang is the most powerful way to extend the functionality of a VerneMQ broker but might be a barrier for developers not familiar with Erlang. For this reason, we've implemented a VerneMQ extension that allows you to develop plugins using the Lua scripting language. This extension is called vmq_diversity and is shipped as part of VerneMQ.
vmq_diversity uses the Luerl Project, which is an implementation of Lua 5.2 in pure Erlang instead of the official Lua interpreter.
Moreover vmq_diversity provides simple Lua libraries to communicate with MySQL, PostgreSQL, MongoDB, and Redis within your Lua VerneMQ plugins. An additional Json encoding/decoding library as well as a generic HTTP client library provide your Lua scripts a great way to talk to external services.
To enable vmq_diversity
make sure to set:
To specify a script to load when VerneMQ starts can be done like this:
It is also possible to dynamically load a Lua script using vmq-admin
:
To reload a script after a change:
If the vmq_diversity
plugin is enabled the folder ./share/lua
folder is scanned for Lua scripts to automatically load during startup. The automatic load folder can be configured in the vernemq.conf
file by changing the vmq_diversity.script
setting.
A VerneMQ plugin typically consists of one or more implemented VerneMQ hooks. We tried to keep the differences between the traditional Erlang based and Lua based plugins as small as possible. Please check out the Plugin Development Guide for more information about the different flows and a description of the different hooks.
Let's start with a first very basic example that implements a basic authentication and authorization scheme.
It is also possible to try the next plugin in the chain (see: Chaining) by returning next
instead of false
.
This subsection describes the data providers currently available to a Lua script. Every data provider is backed by a connection pool that has to be configured by your script.
ensure_pool
Ensures that the connection pool named config.pool_id
is setup in the system. The config
argument is a Lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 5).
user
: MySQL account name for login
password
: MySQL account password for login (in clear text).
host
: Host name for the MySQL server (default is localhost)
port
: Port that the MySQL server is listening on (default is 3306).
database
: MySQL database name.
encoding
: Encoding (default is latin1)
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
execute
Executes the provided SQL statement using a connection from the connection pool.
pool_id
: Name of the connection pool to use for this statement.
stmt
: A valid MySQL statement.
args...
: A variable number of arguments can be passed to substitute statement parameters.
Depending on the statement this call returns true
or false
or a Lua array containing the resulting rows (as Lua tables). In case the statement cannot be executed a badarg error is thrown.
ensure_pool
Ensures that the connection pool named config.pool_id
is setup in the system. The config
argument is a Lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 5).
user
: Postgres account name for login
password
: Postgres account password for login (in clear text).
host
: Host name for the Postgres server (default is localhost)
port
: Port that the Postgres server is listening on (default is 5432).
database
: Postgres database name.
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
execute
Executes the provided SQL statement using a connection from the connection pool.
pool_id
: Name of the connection pool to use for this statement.
stmt
: A valid MySQL statement.
args...
: A variable number of arguments can be passed to substitute statement parameters.
Depending on the statement this call returns true
or false
or a Lua array containing the resulting rows (as Lua tables). In case the statement cannot be executed a badarg error is thrown.
ensure_pool
Ensures that the connection pool named config.pool_id
is setup in the system. The config
argument is a Lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 5).
login
: MongoDB login name
password
: MongoDB password for login.
host
: Host name for the MongoDB server (default is localhost)
port
: Port that the MongoDB server is listening on (default is 27017).
database
: MongoDB database name.
w_mode
: Set mode for writes either to "unsafe" or "safe".
r_mode
: Set mode for reads either to "master" or "slave_ok".
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
insert
Insert the provided document (or list of documents) into the collection.
pool_id
: Name of the connection pool to use for this statement.
collection
: Name of a MongoDB collection.
doc_or_docs
: A single Lua table or a Lua array containing multiple Lua tables.
The provided document can set the document id using the _id
key. If the id isn't provided one gets autogenerated. The call returns the inserted document(s) or throws a badarg error if it cannot insert the document(s).
update
Updates all documents in the collection that match the given selector.
pool_id
: Name of the connection pool to use for this statement.
collection
: Name of a MongoDB collection.
selector
: A single Lua table containing the filter properties.
doc
: A single Lua table containing the update properties.
The call returns true
or throws a badarg error if it cannot update the document(s).
delete
Deletes all documents in the collection that match the given selector.
pool_id
: Name of the connection pool to use for this statement.
collection
: Name of a MongoDB collection.
selector
: A single Lua table containing the filter properties.
The call returns true
or throws a badarg error if it cannot delete the document(s).
find
Finds all documents in the collection that match the given selector.
pool_id
: Name of the connection pool to use for this statement.
collection
: Name of a MongoDB collection.
selector
: A single Lua table containing the filter properties.
args
: A Lua table that currently supports an optional projector=LuaTable
element.
The call returns a MongoDB cursor or throws a badarg error if it cannot setup the iterator.
next
Fetches next available document given a cursor object obtained via find
.
The call returns the next available document or false
if all documents have been fetched.
take
Fetches the next nr_of_docs
documents given a cursor object obtained via find
.
The call returns a Lua array containing the documents or false
if all documents have been fetched.
close
Closes and cleans up a cursor object obtained via find
.
The call returns true
.
find_one
Finds the first document in the collection that matches the given selector.
pool_id
: Name of the connection pool to use for this statement.
collection
: Name of a MongoDB collection.
selector
: A single Lua table containing the filter properties.
args
: A Lua table that currently supports an optional projector=LuaTable
element.
The call returns the matched document or false
in case no document was found.
ensure_pool
Ensures that the connection pool named config.pool_id
is setup in the system. The config
argument is a Lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 5).
password
: Redis password for login.
host
: Host name for the Redis server (default is localhost)
port
: Port that the Redis server is listening on (default is 6379).
database
: Redis database (default is 0).
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
cmd
Executes the given Redis command.
pool_id
: Name of the connection pool
command
: Redis command string.
args...
: Extra args.
This call returns a Lua table, true
, false
, or nil
. In case it cannot parse the command a badarg error is thrown.
ensure_pool
Ensures that the pool named config.pool_id
is setup in the system, The config
argument is a lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 5).
host
: Host name for the Memcached server (default is localhost)
port
: Port that the Redis server is listening on (default is 11211).
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
flush_all(pool_id)
Flushes all data from the Memcached server. Use with care.
Returns true
.
get(pool_id, key)
Get data for key key
.
Returns the data for the key and otherwise false
.
set(pool_id, key, value, expiration)
Unconditionally set a value for a key.
key
: Key.
value
: Value.
expiration
time until key/value pair is deleted in seconds. This
parameter is optional with default value 0
(no expiration).
Returns value
.
add(pool_id, key, value, expiration)
Add a key/value pair if the key doesn't already exist.
key
: Key.
value
: Value.
expiration
time until key/value pair is deleted in seconds. This
parameter is optional with default value 0
(no expiration).
Returns value
if key
didn't already exist, false
otherwise.
replace(pool_id, key, value, expiration)
Replace a key/value pair if the key already exists.
key
: Key.
value
: Value.
expiration
time until key/value pair is deleted in seconds. This
parameter is optional with default value 0
(no expiration).
Returns value
if key
already exists, false
otherwise.
delete(pool_id, key)
Delete key
and the associated value.
Returns true
if the key/value pair was deleted, false
otherwise
ensure_pool
Ensures that the connection pool named config.pool_id
is setup in the system. The config
argument is a Lua table holding the following keys:
pool_id
: Name of the connection pool (mandatory).
size
: Size of the connection pool (default is 10).
This call throws a badarg error in case it cannot setup the pool otherwise it returns true
.
get, put, post, delete
Executes a HTTP request with the given url and args.
url
: A valid http url.
body
: optional body to be included in the request.
headers
: optional Lua table containing extra headers to be included in the request.
This call returns false
in case of an error or a Lua table of the form:
body
Fetches the response body given a client ref obtained via the response Lua table.
This call returns false
in case of an error or the response body.
encode
Encodes a Lua value to a JSON string.
This call returns false if it cannot encode the given value.
decode
Decodes a JSON string to a Lua value.
This call returns false if it cannot decode the JSON string.
Uses the VerneMQ logging infrastructure to log the given log_string
.