devel:config-engines
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
devel:config-engines [2016/04/12 13:49] – [Lua Config KEMI Engine] miconda | devel:config-engines [2017/11/20 13:31] (current) – miconda | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== Configuration File Engines ====== | ====== Configuration File Engines ====== | ||
- | Kamailio implements from scratch the interpreter for its configuration file (kamailio.cfg). | + | Kamailio implements from scratch the interpreter for its configuration file native scripting language |
- | Starting with v5.0.0, the routing blocks can be written in some of its embedded | + | Starting with v5.0.0, the routing blocks can be written in some other (well known) scripting |
- | * Lua, implemented by app_lua module, as " | + | * JavaScript - implemented by app_jsdt module, as " |
+ | * Lua - implemented by app_lua module, as "lua" config engine | ||
+ | * Python - implemented by app_python module, as " | ||
Setting a configuration engine can be done with the global setting: | Setting a configuration engine can be done with the global setting: | ||
Line 25: | Line 27: | ||
</ | </ | ||
- | The aim is that the runtime active part of kamailio.cfg (the routing blocks) to be implemented in an embedded scripting language. | + | The aim is that the runtime active part of kamailio.cfg (the routing blocks) to be implemented in an embedded scripting language. |
- | + | ||
- | Internally, the support for implementing routing logic in an embedded language is codenamed **kemi** - Kamailio EMbedded Interface. | + | Among benefits of using different scripting languages: |
+ | |||
+ | * ability to benefit from what a well established language already provides: | ||
+ | * more people are expected to be familiar with | ||
+ | * an extended set of data types, expressions and statements already available | ||
+ | * a large set of extensions and libraries already available | ||
+ | * good documentation about language itself and its extensions | ||
+ | * reload the SIP routing logic without restarting Kamailio (implemented for Lua) | ||
+ | |||
+ | Internally, the support for implementing routing logic in an embedded language is codenamed **KEMI** - Kamailio EMbedded Interface. | ||
===== Exporting Functions To KEMI ===== | ===== Exporting Functions To KEMI ===== | ||
- | The current | + | Because Kamailio needs to load modules in order to export useful functions to KEMI, statical wrappers to C functions implemented in other modules cannot be used, because they will introduce dependencies on each embedded interpreter for all modules. |
+ | |||
+ | The implementation relies on defining a set of generic functions that are exported | ||
+ | |||
+ | Currently the association table size is 1024 (it means that there can be maximum 1024 Kamailio | ||
Each existing component of Kamailio (e.g., module), can export new functions to KEMI in the following way: | Each existing component of Kamailio (e.g., module), can export new functions to KEMI in the following way: | ||
Line 84: | Line 99: | ||
Exported functions must take first parameter as ' | Exported functions must take first parameter as ' | ||
+ | |||
+ | The functions exported by Kamailio core are listed inside the array **_sr_kemi_core** from the file **kemi.c**. | ||
+ | |||
+ | Not all combinations of extra (after sip_msg_t*) parameters types are supported right now - currently the are: | ||
+ | |||
+ | * 1 param - can be int of str* | ||
+ | * 2 params - any combination of int or str* | ||
+ | * 3 params - any combination of int or str* | ||
+ | * 4 params - all have to be str* (other combinations to be added as needed) | ||
+ | * 5 params - all have to be str* (other combinations to be added as needed) | ||
+ | * 6 params - all have to be str* (other combinations to be added as needed) | ||
+ | |||
===== Lua Config KEMI Engine ===== | ===== Lua Config KEMI Engine ===== | ||
Line 93: | Line 120: | ||
* **ksr_request_route()** - is executed by Kamailio core every time a SIP request is received. If this function is not defined, then Kamailio will write error messages. This is equivalent of request_route {} from kamailio.cfg. | * **ksr_request_route()** - is executed by Kamailio core every time a SIP request is received. If this function is not defined, then Kamailio will write error messages. This is equivalent of request_route {} from kamailio.cfg. | ||
* **ksr_reply_route()** - is executed by Kamailio core every time a SIP Response (reply) is received. If this function is not defined, then Kamailio will not write error messages. This is equivalent of reply_route {} from kamailio.cfg. | * **ksr_reply_route()** - is executed by Kamailio core every time a SIP Response (reply) is received. If this function is not defined, then Kamailio will not write error messages. This is equivalent of reply_route {} from kamailio.cfg. | ||
- | * TBD: the options for branch_route, | + | |
+ | * branch route callback - the name of the Lua function to be executed instead of a branch route has to be provided as parameter to KSR.tm.t_on_branch(...) | ||
+ | * onreply route callback - the name of the Lua function to be executed instead of an onreply route has to be provided as parameter to KSR.tm.t_on_reply(...) | ||
+ | * failure route callback - the name of the Lua function to be executed instead of a failure route has to be provided as parameter to KSR.tm.t_on_failure(...) | ||
+ | * branch failure route callback - the name of the Lua function to be executed instead of an event route for branch failure has to be provided as parameter to KSR.tm.t_on_branch_failure(...) | ||
+ | | ||
The following objects are available inside the Lua script: | The following objects are available inside the Lua script: | ||
* **sr** - provided by the old way of exporting functions to Lua (https:// | * **sr** - provided by the old way of exporting functions to Lua (https:// | ||
- | * **KSR** - provided via KEMI interface | + | * **KSR** - provided via KEMI interface. The functions exported to KEMI are accessible as KSR.submodule.function(...). If submodule name is empty (reserved for core functions), then they are available as KSR.function(...). |
==== Lua Embedded Config Example ==== | ==== Lua Embedded Config Example ==== | ||
Line 139: | Line 171: | ||
# ----- mi_fifo params ----- | # ----- mi_fifo params ----- | ||
- | modparam(" | + | modparam(" |
Line 165: | Line 197: | ||
<code lua> | <code lua> | ||
+ | -- Kamailio - equivalent of routing blocks in Lua | ||
+ | -- KSR - the new dynamic object exporting Kamailio functions | ||
+ | -- sr - the old static object exporting Kamailio functions | ||
+ | -- | ||
-- SIP request routing | -- SIP request routing | ||
+ | -- equivalent of request_route{} | ||
function ksr_request_route() | function ksr_request_route() | ||
KSR.info(" | KSR.info(" | ||
if sr.maxfwd.process_maxfwd(10) < 0 then | if sr.maxfwd.process_maxfwd(10) < 0 then | ||
- | sr.sl.send_reply(483, | + | KSR.sl.send_reply(483, |
return; | return; | ||
end | end | ||
- | KSR.sl.sreply(200, | + | -- KSR.sl.sreply(200, |
+ | |||
+ | sr.pv.sets(" | ||
+ | KSR.tm.t_on_branch(" | ||
+ | KSR.tm.t_on_reply(" | ||
+ | KSR.tm.t_on_failure(" | ||
+ | |||
+ | if KSR.tm.t_relay() < 0 then | ||
+ | KSR.sl.send_reply(500, | ||
+ | end | ||
end | end | ||
-- SIP response routing | -- SIP response routing | ||
+ | -- equivalent of reply_route{} | ||
function ksr_reply_route() | function ksr_reply_route() | ||
KSR.info(" | KSR.info(" | ||
end | end | ||
+ | |||
+ | -- branch route callback | ||
+ | -- equivalent of a branch_route{} | ||
+ | function ksr_branch_route_one() | ||
+ | KSR.info(" | ||
+ | end | ||
+ | |||
+ | -- onreply route callback | ||
+ | -- equivalent of an onreply_route{} | ||
+ | function ksr_onreply_route_one() | ||
+ | KSR.info(" | ||
+ | end | ||
+ | |||
+ | -- failure route callback | ||
+ | -- equivalent of a failure_route{} | ||
+ | function ksr_failure_route_one() | ||
+ | KSR.info(" | ||
+ | end | ||
+ | |||
</ | </ | ||
+ | |||
+ | ===== Python Config KEMI Engine ===== | ||
+ | |||
+ | The **app_python** module must be loaded and the Python script with routing logic must be set to its **script_name** parameter. | ||
+ | |||
+ | The **kemi** engine is built reusing the exiting **app_python** way of executing C code from Kamailio. In the Python script you have to declare the global **mod_init()** method where to instantiate an object of a class that implements the other callback methods (functions) to be executed by Kamailio. | ||
+ | |||
+ | Inside the new class, the following methods are relevant: | ||
+ | |||
+ | * **ksr_request_route(self, | ||
+ | * **ksr_reply_route(self, | ||
+ | * **ksr_onsend_route(self, | ||
+ | * branch route callback - the name of the Python function to be executed instead of a branch route has to be provided as parameter to KSR.tm.t_on_branch(...) | ||
+ | * onreply route callback - the name of the Python function to be executed instead of an onreply route has to be provided as parameter to KSR.tm.t_on_reply(...) | ||
+ | * failure route callback - the name of the Python function to be executed instead of a failure route has to be provided as parameter to KSR.tm.t_on_failure(...) | ||
+ | * branch failure route callback - the name of the Python function to be executed instead of an event route for branch failure has to be provided as parameter to KSR.tm.t_on_branch_failure(...) | ||
+ | * TBD: the options for specific event_route blocks. Meanwhile, should work using hybrid configuration with request_route/ | ||
+ | |||
+ | The following objects are available inside the Python script: | ||
+ | |||
+ | * **Router** - provided by the old way of exporting functions to Python (https:// | ||
+ | * **KSR** - provided via KEMI. The functions exported to KEMI are accessible as KSR.submodule.function(...). If submodule name is empty (reserved for core functions), then they are available as KSR.function(...). | ||
+ | |||
+ | |||
+ | ==== Python Embedded Config Example ==== | ||
+ | |||
+ | The kamailio.cfg with the global/core and modules settings: | ||
+ | |||
+ | <code c> | ||
+ | #!KAMAILIO | ||
+ | # | ||
+ | # Kamailio (OpenSER) SIP Server v5.0 | ||
+ | # - web: http:// | ||
+ | # | ||
+ | ####### Global Parameters ######### | ||
+ | |||
+ | debug=4 | ||
+ | log_stderror=yes | ||
+ | fork=yes | ||
+ | children=2 | ||
+ | |||
+ | memdbg=5 | ||
+ | memlog=5 | ||
+ | |||
+ | auto_aliases=no | ||
+ | |||
+ | listen=udp: | ||
+ | |||
+ | mpath=" | ||
+ | |||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | loadmodule " | ||
+ | |||
+ | # ----------------- setting module-specific parameters --------------- | ||
+ | |||
+ | # ----- mi_fifo params ----- | ||
+ | modparam(" | ||
+ | |||
+ | |||
+ | # ----- tm params ----- | ||
+ | # auto-discard branches from previous serial forking leg | ||
+ | modparam(" | ||
+ | # default retransmission timeout: 30sec | ||
+ | modparam(" | ||
+ | # default invite retransmission timeout after 1xx: 120sec | ||
+ | modparam(" | ||
+ | |||
+ | # ----- debugger params ----- | ||
+ | modparam(" | ||
+ | |||
+ | ####### Routing Logic ######## | ||
+ | |||
+ | modparam(" | ||
+ | |||
+ | cfgengine " | ||
+ | </ | ||
+ | |||
+ | The **/ | ||
+ | |||
+ | <code python> | ||
+ | import sys | ||
+ | import Router.Logger as Logger | ||
+ | import KSR as KSR | ||
+ | |||
+ | def dumpObj(obj): | ||
+ | for attr in dir(obj): | ||
+ | # KSR.info(" | ||
+ | Logger.LM_INFO(" | ||
+ | |||
+ | def mod_init(): | ||
+ | KSR.info(" | ||
+ | # dumpObj(KSR); | ||
+ | return kamailio(); | ||
+ | |||
+ | class kamailio: | ||
+ | def __init__(self): | ||
+ | KSR.info(' | ||
+ | |||
+ | def child_init(self, | ||
+ | KSR.info(' | ||
+ | return 0 | ||
+ | |||
+ | def ksr_request_route(self, | ||
+ | KSR.info(" | ||
+ | msg.rewrite_ruri(" | ||
+ | KSR.tm.t_on_branch(" | ||
+ | KSR.tm.t_on_reply(" | ||
+ | KSR.tm.t_on_failure(" | ||
+ | KSR.sl.send_reply(100, | ||
+ | if KSR.tm.t_relay() < 0 : | ||
+ | KSR.sl.send_reply(500, | ||
+ | return 1; | ||
+ | |||
+ | def ksr_reply_route(self, | ||
+ | KSR.info(" | ||
+ | return 1; | ||
+ | |||
+ | def ksr_branch_route_one(self, | ||
+ | KSR.info(" | ||
+ | return 1; | ||
+ | |||
+ | def ksr_onreply_route_one(self, | ||
+ | KSR.info(" | ||
+ | return 1; | ||
+ | |||
+ | def ksr_failure_route_one(self, | ||
+ | KSR.info(" | ||
+ | return 1; | ||
+ | </ | ||
+ | |||
+ | ===== JavaScript Config KEMI Engine ===== | ||
+ | |||
+ | The **app_jsdt** module must be loaded and the JavaScript script with routing logic must be set to its **load** parameter. | ||
+ | |||
+ | Inside the JavaScript script, following functions are relevant: | ||
+ | |||
+ | * **ksr_request_route()** - is executed by Kamailio core every time a SIP request is received. If this function is not defined, then Kamailio will write error messages. This is equivalent of request_route {} from kamailio.cfg. | ||
+ | * **ksr_reply_route()** - is executed by Kamailio core every time a SIP Response (reply) is received. If this function is not defined, then Kamailio will not write error messages. This is equivalent of reply_route {} from kamailio.cfg. | ||
+ | * **ksr_onsend_route()** - is executed by Kamailio core every time a SIP request (and optionally for a response) is sent out. If this function is not defined, then Kamailio will not write error messages. This is equivalent of onsend_route {} from kamailio.cfg. | ||
+ | * branch route callback - the name of the Lua function to be executed instead of a branch route has to be provided as parameter to KSR.tm.t_on_branch(...) | ||
+ | * onreply route callback - the name of the Lua function to be executed instead of an onreply route has to be provided as parameter to KSR.tm.t_on_reply(...) | ||
+ | * failure route callback - the name of the Lua function to be executed instead of a failure route has to be provided as parameter to KSR.tm.t_on_failure(...) | ||
+ | * branch failure route callback - the name of the Lua function to be executed instead of an event route for branch failure has to be provided as parameter to KSR.tm.t_on_branch_failure(...) | ||
+ | * TBD: the options for specific event_route blocks. Meanwhile, should work using hybrid configuration with request_route/ | ||
+ | |||
+ | The following objects are available inside the Lua script: | ||
+ | |||
+ | * **KSR** - provided via KEMI interface. The functions exported to KEMI are accessible as KSR.submodule.function(...). If submodule name is empty (reserved for core functions), then they are available as KSR.function(...). | ||
+ | |||
+ | ===== Basic IP Telephony Config Example ===== | ||
+ | |||
+ | Some examples of configuration files using the native, Lua or Python interpreter are available in Kamailio source tree inside the **examples/ | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | The configuration file to start with is **kamailio-basic-kemi.cfg**: | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | You can either rename it to kamailio.cfg or use **-f** command line option to point to it. | ||
+ | |||
+ | If started without any change and no define in command line, then Kamailio will continue to load the routing blocks in the native language, stored in the file **kamailio-basic-kemi-native.cfg**: | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | If you define WITH_CFGLUA inside **kamailio-basic-kemi.cfg** or provide the command line parameter **-A WITH_CFGLUA**, | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | If you define WITH_CFGJSDT inside **kamailio-basic-kemi.cfg** or provide the command line parameter **-A WITH_CFGJSDT**, | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | If you define WITH_CFGPYTHON inside **kamailio-basic-kemi.cfg** or provide the command line parameter **-A WITH_CFGPYTHON**, | ||
+ | |||
+ | * https:// | ||
+ | |||
+ | Combining **kamailio-basic-kemi.cfg** with **kamailio-basic-kemi-native.cfg** results more or less in the **kamailio-basic.cfg** from the **etc/** folder in Kamailio source tree. The Lua, JavaScript and Python scripts are offering the same features, but written in another language. | ||
+ | |||
+ | Note that you need to copy these files at the location of the configuration file for your Kamailio configuration and eventually adjust the paths to them inside **kamailio-basic-kemi.cfg**. | ||
+ | |||
+ | Let's say you installed from sources, the config files are located inside the **/ | ||
+ | |||
+ | Then you can start Kamailio with native routing blocks: | ||
+ | |||
+ | <code c> | ||
+ | / | ||
+ | </ | ||
+ | |||
+ | To start it with Lua routing blocks: | ||
+ | |||
+ | <code c> | ||
+ | / | ||
+ | </ | ||
+ | |||
+ | To start it with Python routing blocks: | ||
+ | |||
+ | <code c> | ||
+ | / | ||
+ | </ | ||
+ | |||
+ | To start it with JavaScript routing blocks: | ||
+ | |||
+ | <code c> | ||
+ | / | ||
+ | </ | ||
+ | |||
+ | If you want to print the log messages to the terminal, add the extra parameters **-E -e -ddd** - this will print up to debug level which can be too verbose. Using up to info level can be better, use the extra parameters | ||
+ | |||
+ | When you have at least log level info, with kamailio-basic-kemi.cfg you will notice that a log message is printed showing the duration in microseconds of executing the main routing blocks, no matter they were in native language, Lua, JavaScript or Python. If the logs were printed to the terminal, they look like: | ||
+ | |||
+ | < | ||
+ | LUA {REGISTER}: 13(28737) INFO: < | ||
+ | </ | ||
+ | |||
+ | Depending on the scripting language used for routing blocks, the log messages start with: | ||
+ | |||
+ | * NAT - for native interpreter | ||
+ | * LUA - for LUA | ||
+ | * JSC - for JavaScript | ||
+ | * PYT - for Python | ||
+ | |||
+ | Again, these are specific to the config options in the kamailio-basic-kemi.cfg file -- look for log_prefix inside it for more details. | ||
+ | |||
+ | ==== Interpreters Performances ==== | ||
+ | |||
+ | Relaying on the log messages that print the execution time for main routing blocks handling the SIP requests and responses, some statistics were gathered to see how the execution time compares for each of the interpreters. | ||
+ | |||
+ | The tests were not targeting to measure the capacity of Kamailio, but the impact of using different interpreters for same kind of features. | ||
+ | |||
+ | The system used for testing was a VirtualBox with 2GB or RAM and 2 processors, having Linux Mint as guest OS. The host was a MAC OS X running on a Mid 2012 model of Macbook Pro. | ||
+ | |||
+ | Testing was done for registrations with user authentication, | ||
+ | |||
+ | The values are in micro-seconds (1 / 1 000 000 of a second) and represents the average execution time, the minimum execution time and the maximum execution time for request_route{...} block (or the equivalent of). | ||
+ | |||
+ | The results were: | ||
+ | |||
+ | < | ||
+ | INTERPRETER - AVERAGE - MIN - MAX | ||
+ | |||
+ | NATIVE | ||
+ | |||
+ | LUA - 308.32 | ||
+ | |||
+ | PYTHON | ||
+ | |||
+ | </ | ||
+ | |||
+ | Remarks: | ||
+ | |||
+ | * the average was between 300 and 400 micro-seconds, | ||
+ | * the test was run many times, the Native and Lua execution were close to each other, many times Lua being faster, but again, at small difference | ||
+ | * no big surprise as Lua is calling directly the C bindings and its interpreter is written also in C. With the config used in the tests, the Lua script doesn' | ||
+ | * Python is slower comparing to Native and Lua, but not that much as one may expect. It can still deal with 2500 REGISTERs/ | ||
+ | * in some tests, the maximum execution time was between 5000 and 10000 micro-second, | ||
+ | |||
+ | ===== To-Do ===== | ||
+ | |||
+ | * see if same dynamic object mechanism can be implemented for Perl, Mono, Java, etc. | ||
+ | * discuss and decide the way to generate documentation of available functions via KEMI | ||
+ | * analyze if worth using a declarative language for specifying the exported functions and then generate necessary C code from them (could be useful for generating docs as well) |
devel/config-engines.1460468999.txt.gz · Last modified: 2016/04/12 13:49 by miconda