-
-
Notifications
You must be signed in to change notification settings - Fork 246
HOMER LUA Scripting
One of hottest features in HOMER 7 is lightspeed packet manipulation using LuaJIT 💥
This feature delivers unprecedented power and flexibility in processing HEP messages to extract custom headers, anonymize or encrypt user data, tag traffic based on custom logic and beyond only limited by your imagination and skills - all performing at native speed!
This idea has been incubating for a few years and @adubovikov now released it to heplify-server ❤️
- heplify-server w/ LuaJIT (git)
- liblua5.1-dev
A few simple steps are required to enable and configure your first HEP Lua script.
Enable scripting and configure your entrypoint LuaJIT script, by default inside the lua
directory.
ScriptEnable = true
ScriptFolder = "lua/"
The entrypoint LuaJIT script will be called for each matching HEP packet on its way to the core. The following default example provides a few usage examples for HEP Type 1 (SIP) to extract a custom header using Regex, Add custom tags and Replace existing message fields, all commented -- inline.
-- this function will be executed first
function checkRAW()
local protoType = scriptEngine.GetHEPProtoType()
-- Check if we have SIP type
if protoType ~= 1 then
return
end
-- original SIP message Payload
local raw = scriptEngine.GetRawMessage()
-- scriptEngine.Logp("DEBUG", "raw", raw)
-- Create lua table
local headers = {}
headers["X-test"] = "Super TEST Header"
-- local _, _, name, value = string.find(raw, "(Call-ID:)%s*:%s*(.+)")
local name, value = raw:match("(CSeq):%s+(.-)\n")
if name == "CSeq" then
headers[name] = value
end
scriptEngine.SetCustomSIPHeader(headers)
return
end
-- this function will be executed second
function checkSIP()
-- get the parsed SIP struct
local sip = scriptEngine.GetSIPStruct()
-- a struct can be nil so better check it
if (sip == nil or sip == '') then
return
end
if sip.FromHost == "127.0.0.1" then
-- scriptEngine.Logp("ERROR", "found User-Agent:", sip.UserAgent)
end
scriptEngine.SetSIPHeader("FromHost", "1.1.1.1")
-- Full SIP messsage can be changed
-- scriptEngine.SetRawMessage("SIP 2/0")
return
end
-- this function will be executed third
function changeNodeIDtoNameFast()
-- get only nodeID
local nodeID = scriptEngine.GetHEPNodeID()
if nodeID == 0 then
scriptEngine.SetHEPField("NodeName","TestNode")
end
return
end
-- this function will be executed fourth
function changeNodeIDtoNameSlow()
-- get the parsed HEP struct
local hep = scriptEngine.GetHEPStruct()
-- a struct can be nil so better check it
if (hep == nil or hep == '') then
return
end
if hep.NodeID == 0 then
hep.NodeName="TestNode"
end
return
end
For Docker it can be handy to pass only a flag or enviroment variable. To do that you can base64 encode your lua function and set it as ScriptBase64 config option. Let's set ScriptBase64 to following lua function.
function md5SumToCID()
local sum = scriptEngine.Hash("[email protected]", "md5")
scriptEngine.SetHEPField("CID", sum)
return
end
ScriptBase64 = "ZnVuY3Rpb24gbWQ1U3VtVG9DSUQoKQoJCglsb2NhbCBzdW0gPSBzY3JpcHRFbmdpbmUuSGFzaCgiNjczMTg3Y2VhZmM1NzlmYWI3OGNjODRjYjEwNzdhM2ZAMC4wLjAuMCIsICJtZDUiKQoJc2NyaXB0RW5naW5lLlNldEhFUEZpZWxkKCJDSUQiLCBzdW0pCgoJcmV0dXJuIAoKZW5k"
In order to display and search any added field, implement in Mapping for the desired transaction type:
{
"id": "CSeq",
"type": "string",
"index": "secondary",
"name": "CSeq Extraction",
"form_type": "input",
"position": 99,
"sid_type": true,
"hide": false
}
- Checkout the list of standard preparsed headers
- Thinking of complex logic and routing? The entrypoint script can also include and use sub-scripts.
That's it, You're ready to go!
If you find this powerful feature interesting and use it in your setup, please report any bugs and contribute your scripts or changes back to the community to grow a fantastic library of HEPxperiments!
You can reload your lua script without interruption with following command:
killall -HUP heplify-server
The following functions can be used.
scriptEngine.GetHEPStruct()
scriptEngine.GetSIPStruct()
scriptEngine.GetHEPProtoType()
scriptEngine.GetHEPSrcIP()
scriptEngine.GetHEPSrcPort()
scriptEngine.GetHEPDstIP()
scriptEngine.GetHEPDstPort()
scriptEngine.GetHEPTimeSeconds()
scriptEngine.GetHEPTimeUseconds()
scriptEngine.GetHEPNodeID()
scriptEngine.GetRawMessage()
scriptEngine.SetRawMessage(value string)
scriptEngine.SetCustomSIPHeader(map luatable)
scriptEngine.SetHEPField(field string, value string)
scriptEngine.SetSIPHeader(header string, value string)
scriptEngine.Hash(text string, hashfunc string)
scriptEngine.Logp(level string, message string, data interface{})
scriptEngine.Print(text string)
The following headers can be directly manipulated.
For custom extractions, use Regex against the RAW header.
"ViaOne"
"FromUser"
"FromHost"
"FromTag"
"ToUser"
"ToHost"
"ToTag"
"CallID"
"XCallID"
"ContactUser"
"ContactHost"
"Authorization.Username"
"UserAgent"
"Server"
"PaiUser"
"PaiHost"
(C) 2008-2023 QXIP BV
HEP/EEP Agent Examples:
- CaptAgent
- HEPlify
- Kamailio
- OpenSIPS
- FreeSwitch
- Asterisk
- sipgrep
- sngrep
- RTPEngine
- RTPProxy
- Oracle ACME SBC
- Sonus SBC
- Avaya SM
- Sansay SBC
HEP/EEP Agent Examples (LOGS):
HEP/EEP Proxy:
Extra Examples:
- Custom JSON Stats
- RTCP-XR Stats
- GEO IP Maps
- Janus/Meetecho-WebRTC
- Cloudshark Export
- Encrypted HEP Tunneling
- SNMP Monitoring
- FreeSWITCH ESL Monitoring
- Kazoo Monitoring
- Speech-to-Text-to-HEP
Extra Resources: