Иерархический запрос Oracle с применением иерархии к подмножествам

Я создаю специализированный инструмент для мониторинга блокировки БД для Oracle SE. По сути, у меня есть задание планировщика, которое каждые 10 секунд делает снимок информации о сеансе gv$ и сохраняет его в пользовательской таблице ashstat_data.

И теперь я пытаюсь создать представление для отображения цепочек блокировки сеансов в иерархическом формате для удобства. Представление основано на иерархическом выборе, который показывает заблокированные сеансы как дочерние элементы под родительским блокирующим сеансом:

with l_snap as
(select * 
from ashstat_data
where sample_time>sysdate - interval '10' second )
SELECT 
s.sample_time,
' '||LPAD('-', (LEVEL-1), '-' ) || SESSION_ID asid,
s.wait_class,
s.event,
s.LOGON_TIME,
s.sql_id
FROM l_snap s
natural join dba_users d
inner join dba_objects do on s.LOCKED_OBJECT_ID=do.object_id
WHERE s.SESSION_ID IN 
(SELECT blocking_session FROM l_snap where blocking_session is not null)
OR s.blocking_session IS NOT NULL
CONNECT BY PRIOR 
 s.SESSION_ID=s.blocking_session
START WITH s.blocking_session IS NULL
;

И пока это работает просто идеально, только если я выбираю данные из одного снимка (where sample_time>sysdate - interval '10' second):

SAMPLE_TIME              ASID   WAIT_CLASS  LOGON_TIME          SQL_ID
5/9/2018 16:13:08.173    302    Idle        05/09/2018 11:33:57 
5/9/2018 16:13:08.173    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:13:08.173    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx

Но если я выбираю записи из нескольких снимков (т. е. (where sample_time>sysdate - interval '12' second), они упорядочиваются на основе соединения по условию:

SAMPLE_TIME              ASID   WAIT_CLASS  LOGON_TIME          SQL_ID
5/9/2018 16:17:18.166    302    Idle        05/09/2018 11:33:57 
5/9/2018 16:17:18.166    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:08.170    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:18.166    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx
5/9/2018 16:17:08.170    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx
5/9/2018 16:17:08.170    302    Idle        05/09/2018 11:33:57 
5/9/2018 16:17:18.166    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:08.170    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:18.166    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx
5/9/2018 16:17:08.170    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx

Чего я хотел бы добиться, так это того, чтобы цепочка блокировки была представлена ​​для каждого снимка отдельно, поэтому последний результат выбора будет выглядеть так для всех собранных данных:

SAMPLE_TIME              ASID   WAIT_CLASS  LOGON_TIME          SQL_ID
5/9/2018 16:17:18.166    302    Idle        05/09/2018 11:33:57 
5/9/2018 16:17:18.166    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:18.166    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx
5/9/2018 16:17:08.170    302    Idle        05/09/2018 11:33:57 
5/9/2018 16:17:08.170    -594   Application 05/09/2018 11:34:01 0wvbggc3p3swx
5/9/2018 16:17:08.170    -646   Application 05/09/2018 11:34:07 0wvbggc3p3swx

person Jānis Krišāns    schedule 09.05.2018    source источник


Ответы (1)


Вероятно, вам нужно добавить sample_time в предложение CONNECT BY:

with l_snap as
(select * 
from ashstat_data
where sample_time>sysdate - interval '10' second )
SELECT 
s.sample_time,
' '||LPAD('-', (LEVEL-1), '-' ) || SESSION_ID asid,
s.wait_class,
s.event,
s.LOGON_TIME,
s.sql_id
FROM l_snap s
natural join dba_users d
inner join dba_objects do on s.LOCKED_OBJECT_ID=do.object_id
WHERE s.SESSION_ID IN 
(SELECT blocking_session FROM l_snap where blocking_session is not null)
OR s.blocking_session IS NOT NULL
CONNECT BY PRIOR 
 s.SESSION_ID=s.blocking_session
 and prior s.sample_time = s.sample_time
START WITH s.blocking_session IS NULL
;
person Sentinel    schedule 09.05.2018
comment
Спасибо, это работает. Я уже пробовал что-то подобное, просто не отметил s.sample_time = s.sample_time предыдущим предложением. - person Jānis Krišāns; 10.05.2018