LILAM is a high-performance logging, monitoring and event-driven orchestration framework for Oracle PL/SQL. It provides deep real-time insights into process metrics and utilizes a dynamic JSON-based rule engine to trigger autonomous responses and coordinate complex workflows. Its simple API allows for seamless integration into existing applications with minimal overhead.
LILAM utilizes autonomous transactions to ensure that process states, log entries, and performance metrics are persisted independently of the main execution flow. This decoupled approach guarantees a complete audit trail and reliable monitoring data, even if the primary business process undergoes a rollback.
LILAM is developed by a developer who hates over-engineered tools. Focus: 5 minutes to integrate, 100% visibility.
LILAM comes ready to test right out of the box, so no custom implementation or coding is required to see the framework in action immediately after setup. First code impressions you can find here: learn_lilam.
The following points complement the Key Features and provide a deeper insight into the architectural decisions and technical innovations of LILAM.
LILAM introduces a high-performance Client-Server architecture using Oracle Pipes. This allows for asynchronous log processing and cross-session monitoring
LILAM offers two execution models that can be used interchangeably:
lilam.new_session. LILAM acts as embedded library, Log and Metric calls are executed immediately within your current database session. This is ideal for straightforward debugging and ensuring data is persisted synchronously.lilam.start_server('SERVER_NAME');. These background processes register under a custom name and monitor for incoming commands. You can scale by running multiple servers for the same name or use different names for logical separation.lilam.server_new_session('SERVER_NAME');. LILAM automatically identifies and connects to the specified available server.[!IMPORTANT] Unified API: Regardless of the chosen mode, the logging API remains identical. You use the same
lilam.log(...)calls throughout your application. The only difference is the initial setup (lilam.new_sessionfor vs.lilam.server_new_sessionfor Decoupled mode).
LILAM prioritizes the stability of your application. It uses a Hybrid Model to balance speed and system integrity:
LILAM strictly utilizes PRAGMA AUTONOMOUS_TRANSACTION. This guarantees that log entries and monitoring data are permanently stored in the database, even if the calling main transaction performs a ROLLBACK due to an error. This ensures the root cause remains available for post-mortem analysis.
By leveraging the UTL_CALL_STACK, LILAM automatically captures the exact program execution path. Instead of just logging a generic error, it documents the entire call chain, significantly accelerating the debugging process in complex, nested PL/SQL environments.
To minimize the impact on the main application’s overhead, LILAM features an internal buffering system. Log writing is processed efficiently, offering a decisive performance advantage over simple, row-by-row logging methods, especially in high-load production environments.
LILAM is designed to be “invisible.” The framework ensures that an internal error during the logging process (e.g., table space issues or configuration errors) doesn’t crash the calling application logic. Exceptions within LILAM are caught and handled internally, prioritizing the stability of your business transaction over the logging activity itself.
LILAMs decoupled architecture is designed for seamless integration with modern monitoring stacks. Its structured data format allows for the easy creation of adapters.
LILAM is a specialized framework for deep process insights. Using MARK_EVENT and TRACE functionality, named actions are monitored independently. The framework automatically tracks metrics per action and context:
MARK_EVENT.TRACE_START and TRACE_STOP for precise measurement of work blocks, ensuring clear visibility into long-running tasks.Instead of performing expensive aggregations across millions of monitor records, LILAM uses an incremental calculation mechanism. Metrics like averages and counters are updated on-the-fly. This ensures that monitoring dashboards (e.g., in Grafana, APEX, or Oracle Jet) remain highly responsive even with massive datasets.
By avoiding file system dependencies (UTL_FILE) and focusing on native database features, LILAM is 100% compatible with Oracle Autonomous Database and optimized for scalable cloud infrastructures.
LILAM promotes a standardized error-handling and monitoring culture within development teams. Its easy-to-use API allows for a “zero-config” start, enabling developers to implement professional observability in just a few minutes. No excessive DBA grants or infrastructure overhead required — just provide standard PL/SQL permissions, deploy the package, and start logging immediately.
LILAM categorizes data by its intended use to ensure maximum performance for status queries and analysis:
LILAM doesn’t just log data; it evaluates it. Using versioned JSON Rule-Sets, LILAM monitors process changes and business transactions in real-time.
To illustrate how LILAM works, imagine monitoring a subway system:
Process (TRACK_LINE_4): The overall mission or service run of a specific line.
Event (CLOSE_DOOR): A discrete point in time. We mark this event at a specific station (STATION_ID_400). If a mandatory event is missing, LILAM can trigger an alert.
Trace/Transaction (TRACK_SECTION): A time-based segment representing the travel between two points (e.g. SECTION_ID_402). By using trace_start and trace_stop, we automatically measure the travel time (latency).
l_processId NUMBER;
-- Start the mission (as a new Process/Session.
-- This and all other calls return in microseconds, as the LILAM proxy instantly offloads the workload to the asynchronous worker.
-- Optional group-based isolation: LILAM servers can be assigned to specific groups to ensure strict workload isolation
l_processId := lilam.server_new_session(p_processName => 'TRACK_LINE_4', p_groupName => 'UNDERGROUND_MONITORING', p_logLevel => lilam.logLevelMonitor);
-- set number of steps this mission needs to be finished correctly
-- in our sample there are only two steps: leaving station and arriving station
lilam.set_steps_todo(p_processId => l_processId, p_stepsToDo => 2);
-- leave station
lilam.step_done(p_processId => l_processId); -- increments step-counter into `1`
-- doors must be closed (Event)
lilam.mark_event(p_processId => l_processId, p_actionName => 'CLOSE_DOOR', p_contextName => 'STATION_ID_400);
-- log travel start
lilam.info(p_processId => l_processId, p_logText => 'Line 4 leaving base');
-- travel the segment (trace Transaction by starting and stopping)
lilam.trace_start(p_processId => l_processId, p_actionName => 'TRACK_SECTION', p_contextName => 'SECTION_ID_402');
dbms_session.sleep(30); -- the train needed 30 seconds
lilam.trace_stop(p_processId => l_processId, p_actionName => 'TRACK_SECTION', p_contextName => 'SECTION_ID_402');
-- the mission of line is very! short - only one section; so the mission ends here
-- ! missed code: lilam.step_done(p_processId => l_processId); -- increments step-counter into `2`
lilam.info(p_processId => l_processId, p_logText => 'Line 4 is back');
lilam.close_session(p_processId => l_processId);
-- the step-counter still is `1`. If there was an implemented rule-set which awaits 2 steps
-- at the end of mission, LILAM would raise an `ALERT`
SELECT id, status, last_update, ... FROM lilam_log WHERE process_name = ... (provides the current status of the process)
ID PROCESS_NAME PROCESS_START PROCESS_END LAST_UPDATE STEPS_TO_DO STEPS_DONE STATUS INFO 1 my application 12.01.26 18:17:51,… 12.01.26 18:18:53,… 12.01.26 18:18:53,… 100 99 2 ERROR
SELECT * FROM lilam_log_detail WHERE process_id = <id> AND monitoring = 0
The monitoring table consists of two parts: the ‘left’ one is dedicated to logging, the ‘right’ one is dedicated to monitoring.
PROCESS_ID NO INFO LOG_LEVEL SESSION_TIME SESSION_USER HOST_NAME ERR_STACK ERR_BACKTRACE ERR_CALLSTACK 1 1 Start INFO 13.01.26 10:… SCOTT SERVER1 NULL NULL NULL 1 2 Function A DEBUG 13.01.26 11:… SCOTT SERVER1 NULL NULL ”— PL/SQL …” 1 3 Something happened ERROR 13.01.26 12:… SCOTT SERVER1 ”— PL/SQL …” ”— PL/SQL …” ”— PL/SQL …”
SELECT * FROM lilam_log_detail WHERE process_id = <id> AND monitoring = 1
PROCESS_ID MON_TYPE ACTION CONTEXT START_TIME STOP_TIME STEPS_DONE USED_MILLIS AVG_MILLIS ACTION_COUNT 1 0 MY_ACTION MY_CONTEXT 13.01.26 10:.. NULL NULL 402 0 1 0 MY_ACTION MY_CONTEXT 2 NULL 1500 510 501 505 1 1 TRANS_ACT ROUTE_1 5 1000 1 490 500 499
Locations of the core components:
LILAM is designed for high-concurrency environments. The following results were achieved on standard Consumer Hardware (Fujitsu LIFEBOOK A-Series) running an Oracle Database inside VirtualBox. This demonstrates the massive efficiency of the Pipe-to-Bulk architecture, even when facing significant virtualization overhead (I/O emulation and CPU scheduling):
| Configuration | Throughput | Status |
|---|---|---|
| Exclusive Server (1 Client) | ~1.6k msg/s | Finished in 30m |
| Shared Server (2 Clients) | ~2.2k msg/s | Finished in 45m |
Key Takeaway: Even on mobile hardware, LILAM handles millions of records without blocking the application sessions. On enterprise-grade server hardware with NVMe storage, throughput is expected to scale significantly higher.
LILAM was developed and stress-tested on a consumer-grade laptop using Oracle Database 23ai Free. To provide a realistic assessment of its capabilities, a rigorous test scenario was designed to push the entire system to its physical limits under these conditions.
For a detailed analysis of throughput, latency, and resource efficiency, please refer to the full reports:
This project is dual-licensed:
If you wish to use LILAM in a proprietary environment without the GPL “copyleft” obligations, please contact me for a commercial license.
process_idDo you find LILAM useful? Consider sponsoring the project to support its ongoing development and long-term maintenance.