2021-12-23 19:08:04 +01:00
# Profiler (FB 5.0)
The profiler allows users to measure performance cost of SQL and PSQL code.
It's implemented with a system package in the engine passing data to a profiler plugin.
2022-05-16 16:00:18 +02:00
This documentation treats the engine and plugin parts as a single thing, in the way the default profiler (`Default_Profiler`) is going to be used.
2021-12-23 19:08:04 +01:00
The `RDB$PROFILER` package allows to profile execution of PSQL code collecting statistics of how many times each line was executed along with its minimum, maximum and accumulated execution times (with nanoseconds precision), as well open and fetch statistics of implicit and explicit SQL cursors.
2022-06-03 03:42:42 +02:00
To collect profile data, an user must first start a profile session with `RDB$PROFILER.START_SESSION` . This function returns an profile session ID which is later stored in the profiler snapshot tables to be queried and analyzed by the user. A profiler session may be local (same attachment) or remote (another attachment).
2021-12-23 19:08:04 +01:00
2022-06-03 03:42:42 +02:00
Remote profiling just forwards commands to the remote attachment. So it's possible that a client simultaneous profile multiple attachments. It's also possible that a locally or remotely started profile session have commands issued by another attachment.
Remote issued commands needs that the target attachment be in an idle state, i.e., not executing others requests. When they are not idle the call blocks waiting for that state.
After a session is started, PSQL and SQL statements statistics starts to be collected in memory. Note that a profile session collects data only of statements executed in the same attachment associated with the session.
2021-12-23 19:08:04 +01:00
Data is aggregated and stored per requests (i.e. a statement execution). When querying snapshot tables, user may do extra aggregation per statements or use the auxiliary views that do that automatically.
A session may be paused to temporary disable statistics collecting. It may be resumed later to return statistics collection in the same session.
A new session may be started when a session is already active. In this case it has the same semantics of finishing the current session with `RDB$PROFILER.FINISH_SESSION(FALSE)` so snapshots tables are not updated in the same moment.
2022-06-04 02:52:15 +02:00
To analyze the collected data, the user must flush the data to the snapshot tables, which may be done finishing or pausing a session (with `FLUSH` parameter set to `TRUE` ) or calling `RDB$PROFILER.FLUSH` . Data is flushed using an autonomous transaction (a transaction started and finished for the specific purpose of profiler data update).
2021-12-23 19:08:04 +01:00
Following is a sample profile session and queries for data analysis.
```
-- Preparation - create table and routines that will be analyzed
create table tab (
id integer not null,
val integer not null
);
set term !;
create or alter function mult(p1 integer, p2 integer) returns integer
as
begin
return p1 * p2;
end!
create or alter procedure ins
as
declare n integer = 1;
begin
while (n < = 1000)
do
begin
if (mod(n, 2) = 1) then
insert into tab values (:n, mult(:n, 2));
n = n + 1;
end
end!
set term ;!
-- Start profiling
2022-05-14 03:20:10 +02:00
select rdb$profiler.start_session('Profile Session 1') from rdb$database;
2021-12-23 19:08:04 +01:00
set term !;
execute block
as
begin
execute procedure ins;
delete from tab;
end!
set term ;!
execute procedure rdb$profiler.finish_session(true);
execute procedure ins;
2022-05-14 03:20:10 +02:00
select rdb$profiler.start_session('Profile Session 2') from rdb$database;
2021-12-23 19:08:04 +01:00
select mod(id, 5),
sum(val)
from tab
where id < = 50
group by mod(id, 5)
order by sum(val);
execute procedure rdb$profiler.finish_session(true);
-- Data analysis
2022-06-04 02:52:15 +02:00
commit; -- start new transaction
2022-05-11 20:42:18 +02:00
select * from plg$prof_sessions;
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
select * from plg$prof_psql_stats_view;
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
select * from plg$prof_record_source_stats_view;
2021-12-23 19:08:04 +01:00
select preq.*
2022-05-11 20:42:18 +02:00
from plg$prof_requests preq
join plg$prof_sessions pses
2022-05-12 03:31:33 +02:00
on pses.profile_id = preq.profile_id and
2021-12-23 19:08:04 +01:00
pses.description = 'Profile Session 1';
select pstat.*
2022-05-11 20:42:18 +02:00
from plg$prof_psql_stats pstat
join plg$prof_sessions pses
2022-05-12 03:31:33 +02:00
on pses.profile_id = pstat.profile_id and
2021-12-23 19:08:04 +01:00
pses.description = 'Profile Session 1'
2022-05-12 03:31:33 +02:00
order by pstat.profile_id,
2021-12-23 19:08:04 +01:00
pstat.request_id,
pstat.line_num,
pstat.column_num;
select pstat.*
2022-05-11 20:42:18 +02:00
from plg$prof_record_source_stats pstat
join plg$prof_sessions pses
2022-05-12 03:31:33 +02:00
on pses.profile_id = pstat.profile_id and
2021-12-23 19:08:04 +01:00
pses.description = 'Profile Session 2'
2022-05-12 03:31:33 +02:00
order by pstat.profile_id,
2021-12-23 19:08:04 +01:00
pstat.request_id,
pstat.cursor_id,
pstat.record_source_id;
```
## Function `START_SESSION`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.START_SESSION` starts a new profiler session, turns it the current session (of the given `ATTACHMENT_ID` ) and return its identifier.
2021-12-23 19:08:04 +01:00
2022-05-14 03:20:10 +02:00
If `PLUGIN_NAME` is `NULL` (the default) it uses the database configuration `DefaultProfilerPlugin` .
2022-05-16 16:00:18 +02:00
`PLUGIN_OPTIONS` is plugin specific options and currently should be `NULL` for `Default_Profiler` plugin.
2021-12-23 19:08:04 +01:00
Input parameters:
2022-05-14 03:20:10 +02:00
- `DESCRIPTION` type `VARCHAR(255) CHARACTER SET UTF8` default `NULL`
2022-06-03 03:42:42 +02:00
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2022-05-14 03:20:10 +02:00
- `PLUGIN_NAME` type `VARCHAR(255) CHARACTER SET UTF8` default `NULL`
2022-05-16 16:00:18 +02:00
- `PLUGIN_OPTIONS` type `VARCHAR(255) CHARACTER SET UTF8` default `NULL`
2021-12-23 19:08:04 +01:00
Return type: `BIGINT NOT NULL` .
## Procedure `PAUSE_SESSION`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.PAUSE_SESSION` pauses the current profiler session (of the given `ATTACHMENT_ID` ) so the next executed statements statistics are not collected.
2021-12-23 19:08:04 +01:00
If `FLUSH` is `TRUE` the snapshot tables are updated with data up to the current moment. Otherwise data remains only in memory for later update.
2022-06-03 03:42:42 +02:00
Calling `RDB$PROFILER.PAUSE_SESSION(TRUE)` has the same semantics of calling `RDB$PROFILER.PAUSE_SESSION(FALSE)` followed by `RDB$PROFILER.FLUSH` (using the same `ATTACHMENT_ID` ).
2021-12-23 19:08:04 +01:00
Input parameters:
2022-05-14 03:29:37 +02:00
- `FLUSH` type `BOOLEAN NOT NULL` default `FALSE`
2022-06-03 03:42:42 +02:00
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2021-12-23 19:08:04 +01:00
## Procedure `RESUME_SESSION`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.RESUME_SESSION` resumes the current profiler session (of the given `ATTACHMENT_ID` ) if it was paused so the next executed statements statistics are collected again.
Input parameters:
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2021-12-23 19:08:04 +01:00
## Procedure `FINISH_SESSION`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.FINISH_SESSION` finishes the current profiler session (of the given `ATTACHMENT_ID` ).
2021-12-23 19:08:04 +01:00
If `FLUSH` is `TRUE` the snapshot tables are updated with data of the finished session (and old finished sessions not yet present in the snapshot). Otherwise data remains only in memory for later update.
2022-06-03 03:42:42 +02:00
Calling `RDB$PROFILER.FINISH_SESSION(TRUE)` has the same semantics of calling `RDB$PROFILER.FINISH_SESSION(FALSE)` followed by `RDB$PROFILER.FLUSH` (using the same `ATTACHMENT_ID` ).
2021-12-23 19:08:04 +01:00
Input parameters:
2022-05-14 03:29:37 +02:00
- `FLUSH` type `BOOLEAN NOT NULL` default `TRUE`
2022-06-03 03:42:42 +02:00
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2021-12-23 19:08:04 +01:00
2022-05-12 14:55:08 +02:00
## Procedure `CANCEL_SESSION`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.CANCEL_SESSION` cancels the current profiler session (of the given `ATTACHMENT_ID` ).
2022-05-12 14:55:08 +02:00
All session data present in the profiler plugin is discarded and will not be flushed.
Data already flushed is not deleted automatically.
2022-06-03 03:42:42 +02:00
Input parameters:
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2022-05-13 02:43:53 +02:00
## Procedure `DISCARD`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.DISCARD` removes all sessions (of the given `ATTACHMENT_ID` ) from memory, without flushing them.
2022-05-13 02:43:53 +02:00
If there is a active session, it is cancelled.
2022-06-03 03:42:42 +02:00
Input parameters:
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2021-12-23 19:08:04 +01:00
## Procedure `FLUSH`
2022-06-03 03:42:42 +02:00
`RDB$PROFILER.FLUSH` updates the snapshot tables with data from the profile sessions (of the given `ATTACHMENT_ID` ) in memory.
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
After update data is stored in tables `PLG$PROF_SESSIONS` , `PLG$PROF_STATEMENTS` , `PLG$PROF_RECORD_SOURCES` , `PLG$PROF_REQUESTS` , `PLG$PROF_PSQL_STATS` and `PLG$PROF_RECORD_SOURCE_STATS` and may be read and analyzed by the user.
2021-12-23 19:08:04 +01:00
2022-06-04 02:52:15 +02:00
Data is updated using an autonomous transaction, so if the procedure is called in a snapshot transaction, data will not be directly readable in the same transaction.
2021-12-23 19:08:04 +01:00
2022-06-04 02:52:15 +02:00
Once flush happens, finished sessions are removed from memory.
2022-06-03 03:42:42 +02:00
Input parameters:
- `ATTACHMENT_ID` type `BIGINT NOT NULL` default `CURRENT_CONNECTION`
2021-12-23 19:08:04 +01:00
# Snapshot tables
Snapshot tables (as well views and sequence) are automatically created in the first usage of the profiler. They are owned by the current user with read/write permissions for `PUBLIC` .
When a session is deleted the related data in others profiler snapshot tables are automatically deleted too through foregin keys with `DELETE CASCADE` option.
Below is the list of tables that stores profile data.
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_SESSIONS`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `ATTACHMENT_ID` type `BIGINT` - Attachment ID
- `USER_NAME` type `CHAR(63) CHARACTER SET UTF8` - User name
- `DESCRIPTION` type `VARCHAR(255) CHARACTER SET UTF8` - Description passed in `RDB$PROFILER.START_SESSION`
- `START_TIMESTAMP` type `TIMESTAMP WITH TIME ZONE` - Moment the profile session was started
- `FINISH_TIMESTAMP` type `TIMESTAMP WITH TIME ZONE` - Moment the profile session was finished (NULL when not finished)
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID`
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_STATEMENTS`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `STATEMENT_ID` type `BIGINT` - Statement ID
- `PARENT_STATEMENT_ID` type `BIGINT` - Parent statement ID - related to sub routines
- `STATEMENT_TYPE` type `VARCHAR(20) CHARACTER SET UTF8` - BLOCK, FUNCTION, PROCEDURE or TRIGGER
- `PACKAGE_NAME` type `CHAR(63) CHARACTER SET UTF8` - Package of FUNCTION or PROCEDURE
- `ROUTINE_NAME` type `CHAR(63) CHARACTER SET UTF8` - Routine name of FUNCTION, PROCEDURE or TRIGGER
- `SQL_TEXT` type `BLOB subtype TEXT CHARACTER SET UTF8` - SQL text for BLOCK
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID, STATEMENT_ID`
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_RECORD_SOURCES`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `STATEMENT_ID` type `BIGINT` - Statement ID
- `CURSOR_ID` type `BIGINT` - Cursor ID
- `RECORD_SOURCE_ID` type `BIGINT` - Record source ID
- `PARENT_RECORD_SOURCE_ID` type `BIGINT` - Parent record source ID
- `ACCESS_PATH` type `VARCHAR(255) CHARACTER SET UTF8` - Access path for the record source
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID, STATEMENT_ID, CURSOR_ID, RECORD_SOURCE_ID`
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_REQUESTS`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `REQUEST_ID` type `BIGINT` - Request ID
- `STATEMENT_ID` type `BIGINT` - Statement ID
- `CALLER_REQUEST_ID` type `BIGINT` - Caller request ID
- `START_TIMESTAMP` type `TIMESTAMP WITH TIME ZONE` - Moment this request was first gathered profile data
- `FINISH_TIMESTAMP` type `TIMESTAMP WITH TIME ZONE` - Moment this request was finished
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID, REQUEST_ID`
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_PSQL_STATS`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `REQUEST_ID` type `BIGINT` - Request ID
- `LINE_NUM` type `INTEGER` - Line number of the statement
- `COLUMN_NUM` type `INTEGER` - Column number of the statement
- `STATEMENT_ID` type `BIGINT` - Statement ID
- `COUNTER` type `BIGINT` - Number of executed times of the statement
- `MIN_TIME` type `BIGINT` - Minimal time (in nanoseconds) of a statement execution
- `MAX_TIME` type `BIGINT` - Maximum time (in nanoseconds) of a statement execution
- `TOTAL_TIME` type `BIGINT` - Accumulated execution time (in nanoseconds) of the statement
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID, REQUEST_ID, LINE_NUM, COLUMN_NUM`
2021-12-23 19:08:04 +01:00
2022-05-11 20:42:18 +02:00
## Table `PLG$PROF_RECORD_SOURCE_STATS`
2021-12-23 19:08:04 +01:00
2022-05-12 03:31:33 +02:00
- `PROFILE_ID` type `BIGINT` - Profile session ID
2021-12-23 19:08:04 +01:00
- `REQUEST_ID` type `BIGINT` - Request ID
- `CURSOR_ID` type `BIGINT` - Cursor ID
- `RECORD_SOURCE_ID` type `BIGINT` - Record source ID
- `STATEMENT_ID` type `BIGINT` - Statement ID
- `OPEN_COUNTER` type `BIGINT` - Number of open times of the record source
- `OPEN_MIN_TIME` type `BIGINT` - Minimal time (in nanoseconds) of a record source open
- `OPEN_MAX_TIME` type `BIGINT` - Maximum time (in nanoseconds) of a record source open
- `OPEN_TOTAL_TIME` type `BIGINT` - Accumulated open time (in nanoseconds) of the record source
- `FETCH_COUNTER` type `BIGINT` - Number of fetch times of the record source
- `FETCH_MIN_TIME` type `BIGINT` - Minimal time (in nanoseconds) of a record source fetch
- `FETCH_MAX_TIME` type `BIGINT` - Maximum time (in nanoseconds) of a record source fetch
- `FETCH_TOTAL_TIME` type `BIGINT` - Accumulated fetch time (in nanoseconds) of the record source
2022-05-12 03:31:33 +02:00
- Primary key: `PROFILE_ID, REQUEST_ID, CURSOR_ID, RECORD_SOURCE_ID`
2021-12-23 19:08:04 +01:00
# Auxiliary views
These views help profile data extraction aggregated at statement level.
They should be the preferred way to analyze the collected data. They can also be used together with the tables to get additional data not present on the views.
After hot spots are found, one can drill down in the data at the request level through the tables.
2022-05-11 20:42:18 +02:00
## View `PLG$PROF_PSQL_STATS_VIEW`
2021-12-23 19:08:04 +01:00
```
2022-05-12 03:31:33 +02:00
select pstat.profile_id,
2021-12-23 19:08:04 +01:00
pstat.statement_id,
sta.statement_type,
sta.package_name,
sta.routine_name,
sta.parent_statement_id,
sta_parent.statement_type parent_statement_type,
sta_parent.routine_name parent_routine_name,
(select sql_text
2022-05-11 20:42:18 +02:00
from plg$prof_statements
2022-05-12 03:31:33 +02:00
where profile_id = pstat.profile_id and
2021-12-23 19:08:04 +01:00
statement_id = coalesce(sta.parent_statement_id, pstat.statement_id)
) sql_text,
pstat.line_num,
pstat.column_num,
sum(pstat.counter) counter,
min(pstat.min_time) min_time,
max(pstat.max_time) max_time,
sum(pstat.total_time) total_time,
sum(pstat.total_time) / nullif(sum(pstat.counter), 0) avg_time
2022-05-11 20:42:18 +02:00
from plg$prof_psql_stats pstat
join plg$prof_statements sta
2022-05-12 03:31:33 +02:00
on sta.profile_id = pstat.profile_id and
2021-12-23 19:08:04 +01:00
sta.statement_id = pstat.statement_id
2022-05-11 20:42:18 +02:00
left join plg$prof_statements sta_parent
2022-05-12 03:31:33 +02:00
on sta_parent.profile_id = sta.profile_id and
2021-12-23 19:08:04 +01:00
sta_parent.statement_id = sta.parent_statement_id
2022-05-12 03:31:33 +02:00
group by pstat.profile_id,
2021-12-23 19:08:04 +01:00
pstat.statement_id,
sta.statement_type,
sta.package_name,
sta.routine_name,
sta.parent_statement_id,
sta_parent.statement_type,
sta_parent.routine_name,
pstat.line_num,
pstat.column_num
order by sum(pstat.total_time) desc
```
2022-05-11 20:42:18 +02:00
## View `PLG$PROF_RECORD_SOURCE_STATS_VIEW`
2021-12-23 19:08:04 +01:00
```
2022-05-12 03:31:33 +02:00
select rstat.profile_id,
2021-12-23 19:08:04 +01:00
rstat.statement_id,
sta.statement_type,
sta.package_name,
sta.routine_name,
sta.parent_statement_id,
sta_parent.statement_type parent_statement_type,
sta_parent.routine_name parent_routine_name,
(select sql_text
2022-05-11 20:42:18 +02:00
from plg$prof_statements
2022-05-12 03:31:33 +02:00
where profile_id = rstat.profile_id and
2021-12-23 19:08:04 +01:00
statement_id = coalesce(sta.parent_statement_id, rstat.statement_id)
) sql_text,
rstat.cursor_id,
rstat.record_source_id,
recsrc.parent_record_source_id,
recsrc.access_path,
sum(rstat.open_counter) open_counter,
min(rstat.open_min_time) open_min_time,
max(rstat.open_max_time) open_max_time,
sum(rstat.open_total_time) open_total_time,
sum(rstat.open_total_time) / nullif(sum(rstat.open_counter), 0) open_avg_time,
sum(rstat.fetch_counter) fetch_counter,
min(rstat.fetch_min_time) fetch_min_time,
max(rstat.fetch_max_time) fetch_max_time,
sum(rstat.fetch_total_time) fetch_total_time,
sum(rstat.fetch_total_time) / nullif(sum(rstat.fetch_counter), 0) fetch_avg_time,
coalesce(sum(rstat.open_total_time), 0) + coalesce(sum(rstat.fetch_total_time), 0) open_fetch_total_time
2022-05-11 20:42:18 +02:00
from plg$prof_record_source_stats rstat
join plg$prof_record_sources recsrc
2022-05-12 03:31:33 +02:00
on recsrc.profile_id = rstat.profile_id and
2021-12-23 19:08:04 +01:00
recsrc.statement_id = rstat.statement_id and
recsrc.cursor_id = rstat.cursor_id and
recsrc.record_source_id = rstat.record_source_id
2022-05-11 20:42:18 +02:00
join plg$prof_statements sta
2022-05-12 03:31:33 +02:00
on sta.profile_id = rstat.profile_id and
2021-12-23 19:08:04 +01:00
sta.statement_id = rstat.statement_id
2022-05-11 20:42:18 +02:00
left join plg$prof_statements sta_parent
2022-05-12 03:31:33 +02:00
on sta_parent.profile_id = sta.profile_id and
2021-12-23 19:08:04 +01:00
sta_parent.statement_id = sta.parent_statement_id
2022-05-12 03:31:33 +02:00
group by rstat.profile_id,
2021-12-23 19:08:04 +01:00
rstat.statement_id,
sta.statement_type,
sta.package_name,
sta.routine_name,
sta.parent_statement_id,
sta_parent.statement_type,
sta_parent.routine_name,
rstat.cursor_id,
rstat.record_source_id,
recsrc.parent_record_source_id,
recsrc.access_path
order by coalesce(sum(rstat.open_total_time), 0) + coalesce(sum(rstat.fetch_total_time), 0) desc
```