Create a custom OEM uptime report for business hours only
There was a need to create a custom Information Publisher Report in Oracle Enterprise Manager 13c to provide uptime metrics. These uptime metrics were to be calculated weekly, from Monday to Friday, only for the hours of 8am to 5pm, and for specific targets. Any target status of Up, System Error, Agent Down, Blackout, Status Pending would be considered up. Only a target status of Down would be classified as down.
Yes, I do understand that Information Publisher is deprecated in lieu of BI Publisher, but it's just so easy to use.
Creating the Report
Here is the output of that report:
Unfortunately, creating this was more complicated than it looked.
The query used in the report is the following:
As you can see from the query, I am simply selecting from a function:
SELECT
TO_CHAR(oem_week, 'YYYY-MM-DD') "Week Of",
SUBSTR(oem_target,INSTR(oem_target,'/',-1)+1,LENGTH(oem_target)) "Target",
TO_CHAR(TRUNC(oem_uptime, 2), 'FM999.00') || '%' "Uptime"
FROM
TABLE("AHMED.ABOULNAGA".get_uptime_weekly())
ORDER BY
TO_CHAR(oem_week,'YYYYMMDD') DESC,
oem_target
This function simply returns exactly the output needed, allowing the query to be as simple as possible. Here is the output of the query when you run it manually:
Since I'm running the report as my own user, not SYSMAN, I have to provide a few grants. Run these as SYSTEM or SYSMAN on the database:
GRANT select ON sysman.mgmt$availability_history TO "AHMED.ABOULNAGA";
GRANT select ON sysman.MGMT$TARGET TO "AHMED.ABOULNAGA";
GRANT create type TO "AHMED.ABOULNAGA";
Now, logging in as my database user AHMED.ABOULNAGA, I create a couple of types which are used by my function, which includes the exact output returned by the function:
CREATE TYPE custom_uptime_weekly_type AS OBJECT (
oem_target VARCHAR2(100),
oem_week DATE,
oem_uptime VARCHAR(10)
);
CREATE TYPE custom_uptime_weekly_table AS TABLE OF custom_uptime_weekly_type;
Now here is the dreaded query.
You can customize the list of targets, hours of operation, and start and end dates (query is hardcoded to start from the first day of the calendar year up to today's date).
CREATE OR REPLACE FUNCTION ati_ahmed_get_uptime_weekly RETURN ati_ahmed_uptime_weekly_table
PIPELINED
IS
vcurrday NUMBER;
vstarttime DATE;
vendtime DATE;
vuptime VARCHAR2(10);
TYPE vtargetarray_type IS
VARRAY(50) OF VARCHAR2(100);
vtarget vtargetarray_type;
vrecord ati_ahmed_uptime_weekly_type := ati_ahmed_uptime_weekly_type(NULL, NULL, NULL);
vuptime_weekly NUMBER;
vday_count INT;
BEGIN
-- ----------------------------------------
-- This is the list of targets
-- ----------------------------------------
vtarget := vtargetarray_type(
'/prod_soa_domain/soa_domain/osb_server1',
'/prod_soa_domain/soa_domain/osb_server2',
'/prod_soa_domain/soa_domain/soa_server1',
'/prod_soa_domain/soa_domain/soa_server2',
'/prod_soa_domain/soa_domain/ess_server1',
'/prod_soa_domain/soa_domain/ess_server2',
'/prod_soa_domain/soa_domain/bam_server1',
'/prod_soa_domain/soa_domain/bam_server2'
);
vcurrday := to_number(to_char(sysdate, 'DDD'));
-- ----------------------------------------
-- Start from the first day of the year
-- ----------------------------------------
FOR j IN 1..vtarget.count LOOP
vuptime_weekly := 0;
vday_count := 0;
FOR i IN 1..vcurrday LOOP IF to_char(sysdate - vcurrday + i, 'DY') NOT IN (
'SAT',
'SUN'
) THEN
-- ----------------------------------------
-- This is the hours of operation
-- ----------------------------------------
vstarttime := to_date(to_char(sysdate - vcurrday + i, 'DD-MON-YYYY')
|| ' 08:00:00', 'DD-MON-YYYY HH24:MI:SS');
vendtime := to_date(to_char(sysdate - vcurrday + i, 'DD-MON-YYYY')
|| ' 17:00:00', 'DD-MON-YYYY HH24:MI:SS');
SELECT
100 - SUM(a.value) "Uptime"
INTO vuptime
FROM
(
( SELECT
'Down' availability_status,
0 value,
2 order_col
FROM
dual
UNION ALL
SELECT
'Up' availability_status,
0 value,
1 order_col_txt_id
FROM
dual
UNION ALL
SELECT
'System Error' availability_status,
0 value,
5 order_col_txt_id
FROM
dual
UNION ALL
SELECT
'Agent Down' availability_status,
0 value,
4 order_col_txt_id
FROM
dual
UNION ALL
SELECT
'Blackout' availability_status,
0 value,
3 order_col_txt_id
FROM
dual
UNION ALL
SELECT
'Status Pending' availability_status,
0 value,
6 order_col_txt_id
FROM
dual
)
UNION ALL
SELECT
decode(lower(availability_status),
'target down', 'Down',
'target up', 'Up',
'metric error', 'System Error',
'agent down', 'Agent Down',
'unreachable', 'Unreachable',
'blackout', 'Blackout',
'pending/unknown', 'Status Pending')
availability_status_txt_id
,
round(SUM(least(nvl(end_timestamp, sysdate), vendtime) -
greatest(start_timestamp, vstarttime)) * 100 / vcurrday
, 2) AS value_txt_id,
decode(lower(availability_status),
'target down', 2,
'target up', 1,
'metric error', 5,
'agent down', 4,
'unreachable', 7,
'blackout', 3,
'pending/unknown', 6) order_col_txt_id
FROM
sysman.mgmt$availability_history b,
sysman.mgmt$target t
WHERE
b.target_name = vtarget(j)
AND b.target_guid = t.target_guid
AND lower(availability_status) != 'unreachable'
AND b.start_timestamp <= vendtime
AND ( b.end_timestamp >= vstarttime
OR b.end_timestamp IS NULL )
GROUP BY
lower(availability_status),
decode(lower(availability_status),
'target down', 2,
'target up', 1,
'metric error', 5,
'agent down', 4,
'unreachable', 7,
'blackout', 3,
'pending/unknown', 6)
) a
WHERE
-- ----------------------------------------
-- Only report target status 'Down' as down
-- ----------------------------------------
lower(availability_status) = 'down'
GROUP BY
a.availability_status,
a.order_col
ORDER BY
a.order_col;
IF to_char(sysdate - vcurrday + i, 'Dy') = 'Mon' THEN
vuptime_weekly := 0;
vday_count := 0;
END IF;
vuptime_weekly := vuptime_weekly + vuptime;
vday_count := vday_count + 1;
IF to_char(sysdate - vcurrday + i, 'Dy') = 'Fri' THEN
vrecord.oem_target := vtarget(j);
vrecord.oem_week := vstarttime - 4;
vrecord.oem_uptime := vuptime_weekly / vday_count;
PIPE ROW ( vrecord );
END IF;
END IF;
END LOOP;
END LOOP;
END;
Now a final grant is needed:
GRANT execute ON "AHMED.ABOULNAGA".custom_get_uptime_weekly TO mgmt_view;
Keep in mind that you can run this report for practically any target type.
This was perhaps the messiest query I created for an OEM Information Publisher report, but only because the specific requirements became too complicated to return in a SQL query alone coupled with some limitations in Information Publisher.