Skip to main content

Lua Support

Nauthilus has Lua 5.1 support in all areas of the service. To understand the interfaces, you must first get an idea of what happens with an incoming authentication request.

Authentication workflow

An incoming authentication request first enters the brute_force check. After that it continues with the features pipeline. After that has past, it continues to process the request in a password backend. When the final result for the request was obtained, it passes filters.

Filters may change the backend result in one or the other way (accepting a formely rejected message or vice versa). This is especially useful for other remote services that can influence the authentication process.

After all this has finished, it is possible to do some post actions, which are run independent of all other steps in the whole pipeline and therefore can not influence the final result anymore.

In the following sequence diagram you can see the processing of the request in more detail.

Additional things to know

When starting the server, it is possible to call an init script, which may be used to register prometheus elements, start connection tracker or define custom redis pools. The latter is interesting, if you prefer using other redis servers for all your custom Lua scripts.

While runtime...

When an incoming authentication request is started, a Lua context is created.

All parts of a request share that common request context. Lua scripts can set arbitrary data in the context and read/delete things from there.

Lua scripts can modify the final log line by adding key-value pairs from each script.


Configuration

For the configuration, please have a look for the configuration file document.


Lua components

Each component does provide a set of global functions, constants, ... and requires a well-defined response from each request.

Every Lua script that has been configured is pre-compiled and kept in memory for future use. To make script changes, you must reload the service.


Lua libraries

Nauthilus does not automatically preload Lua modules. Therefor, a dynamic loader has been written, which must be run before requireing a module.

Example:

dynamic_loader("foo")
local foo = require("foo")

A module is preloaded for a single Lua state, which is shared accross all scripts that make use of it. Let's say, there have been defined three filter scripts and 5 post-action scripts, then each class (filter and post-action) share the same Lua state and receive the same preloaded modules.

Modules are never preloaded twice.

This is the list of modules that are currently available:

Loader nameDescription
nauthilus_mailE-Mail functions
nauthilus_passwordPassword compare and validation functions
nauthilus_redisRedis related functions
nauthilus_miscCountry code and sleep functions
nauthilus_contextGlobal Lua context accross all States in Nauthilus
nauthilus_ldapLDAP related functions
nauthilus_backendBackend related functions
nauthilus_http_requestHTTP request header functions
nauthilus_gluacryptogluacrpyto project on Github
nauthilus_gll_plugingopher-lua-libs project on Github
nauthilus_gll_argparsegopher-lua-libs project on Github
nauthilus_gll_base64gopher-lua-libs project on Github
nauthilus_gll_cert_utilgopher-lua-libs project on Github
nauthilus_gll_chefgopher-lua-libs project on Github
nauthilus_gll_cloudwatchgopher-lua-libs project on Github
nauthilus_gll_cmdgopher-lua-libs project on Github
nauthilus_gll_cryptogopher-lua-libs project on Github
nauthilus_gll_dbgopher-lua-libs project on Github
nauthilus_gll_filepathgopher-lua-libs project on Github
nauthilus_gll_goosgopher-lua-libs project on Github
nauthilus_gll_httpgopher-lua-libs project on Github
nauthilus_gll_humanizegopher-lua-libs project on Github
nauthilus_gll_inspectgopher-lua-libs project on Github
nauthilus_gll_ioutilgopher-lua-libs project on Github
nauthilus_gll_jsongopher-lua-libs project on Github
nauthilus_gll_loggopher-lua-libs project on Github
nauthilus_gll_pbgopher-lua-libs project on Github
nauthilus_gll_pprofgopher-lua-libs project on Github
nauthilus_gll_prometheusgopher-lua-libs project on Github
nauthilus_gll_regexpgopher-lua-libs project on Github
nauthilus_gll_runtimegopher-lua-libs project on Github
nauthilus_gll_shellescapegopher-lua-libs project on Github
nauthilus_gll_statsgopher-lua-libs project on Github
nauthilus_gll_storagegopher-lua-libs project on Github
nauthilus_gll_stringsgopher-lua-libs project on Github
nauthilus_gll_tacgopher-lua-libs project on Github
nauthilus_gll_tcpgopher-lua-libs project on Github
nauthilus_gll_telegramgopher-lua-libs project on Github
nauthilus_gll_templategopher-lua-libs project on Github
nauthilus_gll_timegopher-lua-libs project on Github
nauthilus_gll_xmlpathgopher-lua-libs project on Github
nauthilus_gll_yamlgopher-lua-libs project on Github
nauthilus_gll_zabbixgopher-lua-libs project on Github

Example:

dynamic_loader("nauthilus_redis")
local nauthilus_redis = require("nauthilus_redis")

dynamic_loader("nauthilus_context")
local nauthilus_builtin_context = require("nauthilus_context")

dynamic_loader("nauthilus_gluacrypto")
local crypto = require("crypto")

dynamic_loader("nauthilus_gll_db")
local db = require("db")

dynamic_loader("nauthilus_gll_time")
local time = require("time")

Actions

Whenever a brute froce attack is recognized, actions may be called. The request will wait until all requests have finished. Actions are processed by a central action worker. No results are returned to the regular request, so actions in general do their own logging!

Also, features may call actions if they were triggered. The request is waiting to finish all actions by the worker process.

Features

Besides the well known features geoip, rbl, tls_encryption and relay_domains, a new feature has been integrated: lua. This feature is processed before all other features (in fact, you might replace all these features with pure Lua...). Lua-features are run one after another. As soon as a feature has triggered, the request will reject the authentication process. Furthermore, Lua features can set a flag to bypass all built-in features.

Lua backend

A new backend has been implemented. It can be used for all features that Nauthilus currently supports: Checking passwords, running different modes (no-auth, list-accounts), adding TOTP...

The backend can accept a request or reject it. It has full access to all meta information that are delivered from the incoming request.

Filters

There may exist remote services that may be contacted after the main backend authentication proccess returned its first
result. Think of something like GeoIP service or some IP white/blacklisting. Even a request that might have authenticated correctly may be rejected to a policy violation from such a service. Therefor filters have the power to overwrite the result from a backend.

You can also use filters to retrieve additional information from databases or LDAP and add additional attributes to the remaining result. This is useful for setups, where Nauthilus may also take the role of a Dovecot proxy. Users may get routed to different mail stores upon successful authentication. For this, you may retrieve the current backend server list with servers that have been checked as being alive with the backend_server_monitoring feature and select nominate it for the current client request.

info

Filters never affect caching! This is important, because otherwise valid credentials might result in storing them in the negative password cache or vice versa for invalid credentials.

warning

You always have to deal with the "request.authenticated" flag! If you don't care enough, you might acidentially reject legitimate authenticated users or allow bad guys.

Post actions

Post actions are actions, which run after the request hast come to its final result. Its main purpose is to start some automated things like doing statistics stuff, sending messages to operators or anything else that does not require fast instant processing.

As an example have a look at the telegram script. Lua scripts in earlier stages of the process may provide some information by using the Lua context. The telegram script may pick up these information and decide to send out some notifications to an operator channel.


Required functions and constants

Every Lua script must provide a pre-defined Lua function with a request parameter. Concerning the actual script, there is a requried return statement.

Nauthilus will look for these functions and parses the results.


Common request fields for all Lua scripts

The following request fields are supported

NameTypePrecenseAdditional info
debugboolalways-
repeatingboolmaybe-
user_foundboolmaybe-
authenticatedboolmaybe-
no_authboolalwaystrue, if the reuqest is used to retrieve user information
servicestringalwaysNauthilus endpoint like "dovecot" or "nginx"
sessionstringalways-
client_ipstringalways-
client_portstringalways-
client_netstringmaybeAvailable in conjunction with brute-force-actions
client_idstringmaybe-
user_agentstringmaybe-
local_ipstringalways-
local_portstringalways-
usernamestringalways-
accountstringmaybeFilter and post actions
unique_user_idstringmaybeUsed with OIDC subject
display_namestringmaybe-
passwordstringalways-
protocolstringalways-
brute_force_bucketstringmaybeAvailable in conjunction with brute-force-actions
featurestringmaybeIn actions, if a feature has triggered
status_messagestringalwaysCurrent status message returned to client, if auth request failed
sslstringmaybeHAproxy: %[ssl_fc]
ssl_session_idstringmaybeHAproxy: %[ssl_fc_session_id,hex]
ssl_client_verifystringmaybeHAproxy: %[ssl_c_verify]
ssl_client_dnstringmaybeHAproxy: %{+Q}[ssl_c_s_dn]
ssl_client_cnstringmaybeHAproxy: %{+Q}[ssl_c_s_dn(cn)]
ssl_issuerstringmaybeHAproxy: %{+Q}[ssl_c_i_dn]
ssl_client_not_beforestringmaybeHAproxy: %{+Q}[ssl_c_notbefore]
ssl_client_not_afterstringmaybeHAproxy: %{+Q}[ssl_c_notafter]
ssl_subject_dnstringmaybeHAproxy: %{+Q}[ssl_c_s_dn]
ssl_issuer_dnstringmaybeHAproxy: %{+Q}[ssl_c_i_dn]
ssl_client_subject_dnstringmaybeHAproxy: %{+Q}[ssl_c_s_dn]
ssl_client_issuer_dnstringmaybeHAproxy: %{+Q}[ssl_c_i_dn]
ssl_protocolstringmaybeHAproxy: %[ssl_fc_protocol]
ssl_cipherstringmaybeHAproxy: %[ssl_fc_cipher]
note

TLS-related values may be retrieved from Nginx and as a fallback tried to be retrieved from HAproxy headers.

tip

It is always a good idea to check the value of a request field, before using it.

Features

A Lua feature script must provide the following function:

---@param request table
---@return number, number, number
function nauthilus_call_feature(request)
return trigger, skip_flag, failure_info -- See details below
end
important

It must return three values: The trigger state, a flag that indicates, if other features shall be skipped and a third value which is an indicator for errors that occurred in the script itself.

Constants for the returned result

ConstantMeaningValueCategory
nauthilus_builtin.FEATURE_TRIGGER_NOThe feature has not been triggered0trigger
nauthilus_builtin.FEATURE_TRIGGER_YESThe feature has been triggered and the request must be rejected1trigger
nauthilus_builtin.FEATURES_ABORT_NOProcess other built-in features0skip_flag
nauthilus_builtin.FEATURES_ABORT_YESAfter finishing the script, skip all other built-in features1skip_flag
nauthilus_builtin.FEATURE_RESULT_OKThe script finished without errors0failure_info
nauthilus_builtin.FEATURE_RESULT_FAILSomething went wrong while executing the script1failure_info

Request fields

Only common request fields are present.

Filters

A Lua filter script must provide the following function:

---@param request table
---@return number, number, number
function nauthilus_call_filter(request)
if request.authenticated then
-- do something
end

return filter_action, failure_info -- See details below
end
important

It must return three values: The trigger state, a flag that indicates, if other features shall be skipped and a third value which is an indicator for errors that occurred in the script itself.

Constants for the returned result

ConstantMeaningValueCategory
nauthilus_builtin.FILTER_ACTION_ACCEPTThe request must be accepted0filter_action
nauthilus_builtin.FILTER_ACTION_REJECTThe request has to be rejected1filter_action
nauthilus_builtin.FILTER_RESULT_OKThe script finished without errors0filter_info
nauthilus_builtin.FILTER_RESULT_FAILSomething went wrong while executing the script1filter_info

Request fields

Only common request fields are present.

Actions (including post)

A Lua action script must provide the following function:

---@param request table
---@return number
function nauthilus_call_action(request)
if request.no_auth then
-- Example post action: Store request information in database
end

return failure_info -- See details below
end
important

Actions must return the script status constant.

Constants for the returned result

ConstantMeaningValueCategory
nauthilus_builtin.ACTION_RESULT_OKThe script finished without errors0failure_info
nauthilus_builtin.ACTION_RESULT_FAILThe script finished with errors1failure_info

Request fields

Only common request fields are available.

Lua Backend

The Lua backend script must provide the following function:

---@param request table
---@return number, userdata
function nauthilus_backend_verify_password(request)
local backend_result_object = backend_result:new()
-- Do something with backend_result_object

return failure_info, backend_result_object -- See details below
end

For user account listing, the following function is required:

---@param request table
------@return number, table
function nauthilus_backend_list_accounts(request)
local accounts = {}

return failure_info, accounts -- See details below
end

If you plan on adding TOTP-keys for your users, you must provide the follwing function:

---@param request table
---@return number
function nauthilus_backend_add_totp(request)
return failure_info -- See details below
end
important

The backend must return the result status constant and a backend result object

Constants for the returned result

ConstantMeaningValueCategory
nauthilus_builtin.BACKEND_RESULT_OKThe script finished without errors0failure_info
nauthilus_builtin.BACKEND_RESULT_FAILThe script finished with errors1failure_info

Request fields

Function nauthilus_backend_verify_password request fields

Only common request fields are used.

Function nauthilus_backend_list_accounts request fields

Only "debug" and "session" from the common requests are available.

Function nauthilus_backend_add_totp request fields

Only "debug" and "session" from the common requests as well as "totp_secret" (string) are available.


UserData object backend_result

The nauthilus_backend_result object can be initialized in the Lua backend and in Lua filters. The following methods exist:

backend

NameMeaning
authenticatedSet or get the authentication status
user_foundSet or get the user found flag which indicated, if the backend found the user
account_fieldSet or get the account field name, which must have been added to a list of result attributes
totp_secret_fieldSet or get the TOTP secret field name, which must have been added to the result attributes
totp_recovery_fieldNot yet implemented
unique_user_id_fieldSet or get the unique user id field, which must have been added to the result attributes
display_name_fieldSet or get the display name field, which must have been added to the result attributes
attributesSet or get the result attributes as a Lua table

filters

Filters only have an "attributes" method. While Lua backends do return a nauthlus_backend_result directly, filters can only apply it with a Lua function called "nauthilus_backend.apply_backend_result(backend_result_object)".

Attributes can not overwrite existing attributes!

Example usage for nauthilus_backend_result

local attributes = {}
attributes["account"] = "bob"

local b = nauthilus_backend_result.new()
b:attributes(attributes) -- Add the table
b:account_field("account") -- Attributes contain a key "account" for the account field
b:authenticated(true) -- User is authenticated
b:user_found(true) -- The user was found

Endpoints /api/v1/mail/dovecot, /api/v1/generic/user and /api/v1/generic/json

"attributes" represent a common result store for a backend query. All fields that have been set by *_field methods will be used for further internal processing, while all other attributes will be converted to HTTP-response-headers, which will be sent back to the client application that talked to Nauthilus. These headeres will be prefixed with X-Nauthilus-.

For the generic endpoints, "attributes" will bew returned in the JSON respone.


Additional notes

Nauthilus uses the gopher-lua-libs library in all Lua scripts. Please have a look at their documentation for all the modules that can directly be used in Nauthilus scripts:

gopher-lua-libs on Github