Let me start this post with a conversation between an end-user (Sarah) and a developer (Harry):
End-user: "Hey there, I'm receiving an error in the app."
Developer: "Oh, sorry to hear that. What is the message saying?"
End-user: "Unable to process row of table EBA_PROJ_STATUS_CATS. ORA-02292: integrity constraint (XXX.SYS_C0090660) violated - child record found"
Developer: "Oh, what are you trying to do?"
End-user: "I'm trying to delete a category."
Developer: "Oh, most likely this category is in use, so you can't delete the category, you first need ..."
End-user: "Ehh?!"
If the application is running in APEX 18.1, it's a different story. The screenshot will look like this:
APEX 18.1 actually enhanced the default error message. The user gets fewer details and sees a debug id. With this debug id the developer can get actually more info in Your App > Utilities > Debug Messages:
You might also want to check this blog post by Joel Kallman where to find more info when receiving an internal error with debug id.
Although APEX 18.1 captures more info, there's a more recommended way to deal with errors.
The error handling function has this definition:
function apex_error_handling (p_error in apex_error.t_error )
return apex_error.t_error_result
The example used in P-Track gives a good overview (read the comments in the package) of the different errors you want to capture:
function apex_error_handling (
p_error in apex_error.t_error )
return apex_error.t_error_result
is
l_result apex_error.t_error_result;
l_constraint_name varchar2(255);
begin
l_result := apex_error.init_error_result (
p_error => p_error );
-- If it is an internal error raised by APEX, like an invalid statement or
-- code which can not be executed, the error text might contain security sensitive
-- information. To avoid this security problem we can rewrite the error to
-- a generic error message and log the original error message for further
-- investigation by the help desk.
if p_error.is_internal_error then
-- mask all errors that are not common runtime errors (Access Denied
-- errors raised by application / page authorization and all errors
-- regarding session and session state)
if not p_error.is_common_runtime_error then
add_error_log( p_error );
-- Change the message to the generic error message which doesn't expose
-- any sensitive information.
l_result.message := 'An unexpected internal application error has occurred.';
l_result.additional_info := null;
end if;
else
-- Always show the error as inline error
-- Note: If you have created manual tabular forms (using the package
-- apex_item/htmldb_item in the SQL statement) you should still
-- use "On error page" on that pages to avoid loosing entered data
l_result.display_location := case
when l_result.display_location = apex_error.c_on_error_page then apex_error.c_inline_in_notification
else l_result.display_location
end;
-- If it's a constraint violation like
--
-- -) ORA-00001: unique constraint violated
-- -) ORA-02091: transaction rolled back (can hide a deferred constraint)
-- -) ORA-02290: check constraint violated
-- -) ORA-02291: integrity constraint violated - parent key not found
-- -) ORA-02292: integrity constraint violated - child record found
--
-- we try to get a friendly error message from our constraint lookup configuration.
-- If we don't find the constraint in our lookup table we fallback to
-- the original ORA error message.
if p_error.ora_sqlcode in (-1, -2091, -2290, -2291, -2292) then
l_constraint_name := apex_error.extract_constraint_name (
p_error => p_error );
begin
select message
into l_result.message
from eba_proj_error_lookup
where constraint_name = l_constraint_name;
exception when no_data_found then null; -- not every constraint has to be in our lookup table
end;
end if;
-- If an ORA error has been raised, for example a raise_application_error(-20xxx)
-- in a table trigger or in a PL/SQL package called by a process and we
-- haven't found the error in our lookup table, then we just want to see
-- the actual error text and not the full error stack
if p_error.ora_sqlcode is not null and l_result.message = p_error.message then
l_result.message := apex_error.get_first_ora_error_text (
p_error => p_error );
end if;
-- If no associated page item/tabular form column has been set, we can use
-- apex_error.auto_set_associated_item to automatically guess the affected
-- error field by examine the ORA error for constraint names or column names.
if l_result.page_item_name is null and l_result.column_alias is null then
apex_error.auto_set_associated_item (
p_error => p_error,
p_error_result => l_result );
end if;
end if;
return l_result;
end apex_error_handling;
When defining this error handling function the error the user gets is more like a notification message and embedded in your app. You can also define a custom message, in the above package there's a lookup in an error_lookup table, but as it can't find the constraint name, it falls back to the normal message.
The real power comes when you start to combine the error handling function with a call to also log session state information. Then you know exactly which record this error was produced for.
There are a couple of ways to include the session state:
Team Development
I typically include a feedback page in my apps. When the user logs feedback by clicking on the feedback link, this is saved in Team Development. The really cool thing is that whenever feedback is logged, automatically the session state of items and some other info like the browser that was being used at the moment of the logging is included. But you can also log feedback through an APEX API:
apex_util.submit_feedback (
p_comment => 'Unexpected Error',
p_type => 3,
p_application_id => v('APP_ID'),
p_page_id => v('APP_PAGE_ID'),
p_email => v('APP_USER'),
p_label_01 => 'Session',
p_attribute_01 => v('APP_SESSION'),
p_label_02 => 'Language',
p_attribute_02 => v('AI_LANGUAGE'),
p_label_03 => 'Error orq_sqlcode',
p_attribute_03 => p_error.ora_sqlcode,
p_label_04 => 'Error message',
p_attribute_04 => p_error.message,
p_label_05 => 'UI Error message',
p_attribute_05 => l_result.message
);
Logger
Logger is a PL/SQL logging and debugging framework. If you don't know it yet, you should definitely check it out. In my opinion, Logger is the best way to instrument your PL/SQL code. Logger has many cool features, one of them is the ability to log your APEX items:
logger.log_apex_items('Debug Items from Error log');
With the above methods, you know which record the end-user was looking at and what the context was. Note that you might find this information too if you look at their session, but it would take more time to figure things out.Be pro-active
Now, to prevent the conversation from happening again, you can take it one step further and start logging and monitoring those errors. Whenever errors happen you can, for example, log it in your own error table, or in your support ticket system and send yourself an email or notification.
Then instead of the end-user calling you, you call them and say "Hey, I saw you had some issues...".
By monitoring errors in your application, you can pro-actively take actions :)
Note that APEX itself also stores Application Errors. You find under Monitor Activity > Application Errors:
The report gives the error and the session, so you look further into what happened:
So, even when you didn't have an error handling function in place, you can still start monitoring errors that happen in your app. I know the readers of this blog are really smart so you might not see any errors, but still, it might be worthwhile to check it once and a while :)
You find another example of the error handling function in my Git account. I included an example of logging in your own error table and sending an email.
create or replace package app_error_pkg | |
as | |
-- | |
-- Function: apex_error_handling | |
-- Purpose: Try to elegantly handle errors that occur while using the application. | |
-- | |
function apex_error_handling ( | |
p_error in apex_error.t_error ) | |
return apex_error.t_error_result; | |
-- | |
-- Procedure: add_error_log | |
-- Purpose: logs application errors | |
-- | |
procedure add_error_log ( | |
p_error in varchar2 default null, | |
p_procedure_name in varchar2 default null, | |
p_page_item_name in varchar2 default null, | |
p_region_id in varchar2 default null, | |
p_column_alias in varchar2 default null, | |
p_row_num in varchar2 default null, | |
p_apex_error_code in varchar2 default null, | |
p_ora_sqlcode in varchar2 default null, | |
p_ora_sqlerrm in varchar2 default null, | |
p_error_backtrace in varchar2 default null, | |
p_arg1_name in varchar2 default null, | |
p_arg1_val in varchar2 default null, | |
p_arg2_name in varchar2 default null, | |
p_arg2_val in varchar2 default null, | |
p_arg3_name in varchar2 default null, | |
p_arg3_val in varchar2 default null, | |
p_arg4_name in varchar2 default null, | |
p_arg4_val in varchar2 default null, | |
p_arg5_name in varchar2 default null, | |
p_arg5_val in varchar2 default null, | |
p_arg6_name in varchar2 default null, | |
p_arg6_val in varchar2 default null, | |
p_arg7_name in varchar2 default null, | |
p_arg7_val in varchar2 default null, | |
p_arg8_name in varchar2 default null, | |
p_arg8_val in varchar2 default null, | |
p_arg9_name in varchar2 default null, | |
p_arg9_val in varchar2 default null, | |
p_arg10_name in varchar2 default null, | |
p_arg10_val in varchar2 default null | |
); | |
end app_error_pkg; | |
/ | |
create or replace package body app_error_pkg | |
as | |
-- | |
-- Function: apex_error_handling | |
-- Purpose: Try to elegantly handle errors that occur while using the application. | |
-- | |
function apex_error_handling ( p_error in apex_error.t_error ) | |
return apex_error.t_error_result is | |
l_result apex_error.t_error_result; | |
l_constraint_name varchar2(255); | |
procedure add_error_log ( p_error in apex_error.t_error ) is | |
pragma autonomous_transaction; | |
begin | |
-- Log the error. | |
insert into app_error ( | |
err_time, | |
app_id, | |
app_page_id, | |
app_user, | |
user_agent, | |
ip_address, | |
ip_address2, | |
message, | |
page_item_name, | |
region_id, | |
column_alias, | |
row_num, | |
apex_error_code, | |
ora_sqlcode, | |
ora_sqlerrm, | |
error_backtrace ) | |
values ( | |
sysdate, | |
sys_context('APEX$SESSION','APP_ID'), | |
sys_context('APEX$SESSION','APP_PAGE_ID'), | |
sys_context('APEX$SESSION','APP_USER'), | |
owa_util.get_cgi_env('HTTP_USER_AGENT'), | |
owa_util.get_cgi_env('REMOTE_ADDR'), | |
sys_context('USERENV', 'IP_ADDRESS'), | |
substr(p_error.message,1,4000), | |
p_error.page_item_name, | |
p_error.region_id, | |
p_error.column_alias, | |
p_error.row_num, | |
p_error.apex_error_code, | |
p_error.ora_sqlcode, | |
substr(p_error.ora_sqlerrm,1,4000), | |
substr(p_error.error_backtrace,1,4000) | |
); | |
commit; | |
end add_error_log; | |
begin | |
l_result := apex_error.init_error_result(p_error => p_error ); | |
-- If it is an internal error raised by APEX, like an invalid statement or | |
-- code which can not be executed, the error text might contain security sensitive | |
-- information. To avoid this security problem we can rewrite the error to | |
-- a generic error message and log the original error message for further | |
-- investigation by the help desk. | |
if p_error.is_internal_error then | |
-- mask all errors that are not common runtime errors (Access Denied | |
-- errors raised by application / page authorization and all errors | |
-- regarding session and session state) | |
if not p_error.is_common_runtime_error then | |
-- Submit to own error logging table | |
add_error_log( p_error ); | |
-- Submit into Team Development as feedback | |
apex_util.submit_feedback ( | |
p_comment => 'Unexpected Error', | |
p_type => 3, | |
p_application_id => v('APP_ID'), | |
p_page_id => v('APP_PAGE_ID'), | |
p_email => v('APP_USER'), | |
p_label_01 => 'Session', | |
p_attribute_01 => v('APP_SESSION'), | |
p_label_02 => 'Language', | |
p_attribute_02 => v('AI_LANGUAGE'), | |
p_label_03 => 'Error orq_sqlcode', | |
p_attribute_03 => p_error.ora_sqlcode, | |
p_label_04 => 'Error message', | |
p_attribute_04 => p_error.message, | |
p_label_05 => 'UI Error message', | |
p_attribute_05 => l_result.message | |
); | |
-- Logger | |
logger.log_error('There is an error in xxx'); | |
logger.log_apex_items('Debug Items from Error log'); | |
-- Send email | |
l_subject := 'Issue in XXX'; | |
l_body := p_error.ora_sqlcode || CHR(10) || p_error.message; | |
l_body := l_body || CHR(10) || 'App User: ' || v('APP_USER'); | |
l_body := l_body || CHR(10) || 'Session: ' || v('APP_SESSION'); | |
l_body := l_body || CHR(10) || 'App: ' || v('APP_ID'); | |
l_body := l_body || CHR(10) || 'Page: ' || v('APP_PAGE_ID'); | |
wwv_flow_api.set_security_group_id; | |
apex_mail.send( | |
p_from => 'xxx', | |
p_to => 'xxx', | |
p_subj => l_subject, | |
p_body => l_body, | |
p_body_html => l_body); | |
apex_mail.push_queue; | |
-- Log an Issues Developer Portal / JIRA | |
-- see other entries in Gist Dimitri Gielis | |
--https://gist.github.com/dgielis/e97c94391058dcacb4a2b50e355d9445 | |
-- Change the message to the generic error message which doesn't expose | |
-- any sensitive information. | |
l_result.message := 'An unexpected internal application error has occurred: ' || substr(p_error.message,0,3500); | |
l_result.additional_info := null; | |
end if; | |
else | |
-- Always show the error as inline error | |
-- Note: If you have created manual tabular forms (using the package | |
-- apex_item/htmldb_item in the SQL statement) you should still | |
-- use "On error page" on that pages to avoid loosing entered data | |
l_result.display_location := case | |
when l_result.display_location = apex_error.c_on_error_page then apex_error.c_inline_in_notification | |
else l_result.display_location | |
end; | |
-- If it's a constraint violation like | |
-- | |
-- -) ORA-00001: unique constraint violated | |
-- -) ORA-02091: transaction rolled back (-> can hide a deferred constraint) | |
-- -) ORA-02290: check constraint violated | |
-- -) ORA-02291: integrity constraint violated - parent key not found | |
-- -) ORA-02292: integrity constraint violated - child record found | |
-- | |
-- we try to get a friendly error message from our constraint lookup configuration. | |
-- If we don't find the constraint in our lookup table we fallback to | |
-- the original ORA error message. | |
if p_error.ora_sqlcode in (-1, -2091, -2290, -2291, -2292) then | |
l_constraint_name := apex_error.extract_constraint_name ( | |
p_error => p_error ); | |
begin | |
select message | |
into l_result.message | |
from app_error_lookup | |
where constraint_name = l_constraint_name; | |
exception when no_data_found then null; -- not every constraint has to be in our lookup table | |
end; | |
end if; | |
-- If an ORA error has been raised, for example a raise_application_error(-20xxx, '...') | |
-- in a table trigger or in a PL/SQL package called by a process and we | |
-- haven't found the error in our lookup table, then we just want to see | |
-- the actual error text and not the full error stack with all the ORA error numbers. | |
if p_error.ora_sqlcode is not null and l_result.message = p_error.message then | |
l_result.message := apex_error.get_first_ora_error_text ( | |
p_error => p_error ); | |
end if; | |
-- If no associated page item/tabular form column has been set, we can use | |
-- apex_error.auto_set_associated_item to automatically guess the affected | |
-- error field by examine the ORA error for constraint names or column names. | |
if l_result.page_item_name is null and l_result.column_alias is null then | |
apex_error.auto_set_associated_item ( | |
p_error => p_error, | |
p_error_result => l_result ); | |
end if; | |
end if; | |
return l_result; | |
end apex_error_handling; | |
-- | |
-- Procedure: add_error_log | |
-- Purpose: logs application errors | |
-- | |
procedure add_error_log ( | |
p_error in varchar2 default null, | |
p_procedure_name in varchar2 default null, | |
p_page_item_name in varchar2 default null, | |
p_region_id in varchar2 default null, | |
p_column_alias in varchar2 default null, | |
p_row_num in varchar2 default null, | |
p_apex_error_code in varchar2 default null, | |
p_ora_sqlcode in varchar2 default null, | |
p_ora_sqlerrm in varchar2 default null, | |
p_error_backtrace in varchar2 default null, | |
p_arg1_name in varchar2 default null, | |
p_arg1_val in varchar2 default null, | |
p_arg2_name in varchar2 default null, | |
p_arg2_val in varchar2 default null, | |
p_arg3_name in varchar2 default null, | |
p_arg3_val in varchar2 default null, | |
p_arg4_name in varchar2 default null, | |
p_arg4_val in varchar2 default null, | |
p_arg5_name in varchar2 default null, | |
p_arg5_val in varchar2 default null, | |
p_arg6_name in varchar2 default null, | |
p_arg6_val in varchar2 default null, | |
p_arg7_name in varchar2 default null, | |
p_arg7_val in varchar2 default null, | |
p_arg8_name in varchar2 default null, | |
p_arg8_val in varchar2 default null, | |
p_arg9_name in varchar2 default null, | |
p_arg9_val in varchar2 default null, | |
p_arg10_name in varchar2 default null, | |
p_arg10_val in varchar2 default null ) | |
is | |
begin | |
-- Remove old errors | |
--delete from app_errors where err_time <= localtimestamp - 21; | |
-- Log the error. | |
insert into app_error ( | |
app_id, | |
app_page_id, | |
app_user, | |
user_agent, | |
-- | |
ip_address, | |
ip_address2, | |
message, | |
page_item_name, | |
-- | |
region_id, | |
column_alias, | |
row_num, | |
apex_error_code, | |
-- | |
ora_sqlcode, | |
ora_sqlerrm, | |
error_backtrace, | |
arg1_name, | |
arg1_val, | |
arg2_name, | |
arg2_val, | |
arg3_name, | |
arg3_val, | |
arg4_name, | |
arg4_val, | |
arg5_name, | |
arg5_val, | |
arg6_name, | |
arg6_val, | |
arg7_name, | |
arg7_val, | |
arg8_name, | |
arg8_val, | |
arg9_name, | |
arg9_val, | |
arg10_name, | |
arg10_val ) | |
select v('APP_ID'), | |
v('APP_PAGE_ID'), | |
v('APP_USER'), | |
owa_util.get_cgi_env('HTTP_USER_AGENT'), | |
-- | |
owa_util.get_cgi_env('REMOTE_ADDR'), | |
sys_context('USERENV', 'IP_ADDRESS'), | |
substr(p_error,0,4000), | |
substr(p_page_item_name,0,4000), | |
-- | |
p_region_id, | |
p_column_alias, | |
p_row_num, | |
p_apex_error_code, | |
-- | |
p_ora_sqlcode, | |
substr(p_ora_sqlerrm,1,4000), | |
substr(p_error_backtrace,1,4000), | |
p_arg1_name, | |
p_arg1_val, | |
p_arg2_name, | |
p_arg2_val, | |
p_arg3_name, | |
p_arg3_val, | |
p_arg4_name, | |
p_arg4_val, | |
p_arg5_name, | |
p_arg5_val, | |
p_arg6_name, | |
p_arg6_val, | |
p_arg7_name, | |
p_arg7_val, | |
p_arg8_name, | |
p_arg8_val, | |
p_arg9_name, | |
p_arg9_val, | |
p_arg10_name, | |
p_arg10_val | |
from dual; | |
commit; | |
end add_error_log; | |
end app_error_pkg; | |
/ |
8 comments:
Great blog on Oracle Apex. Appreciated!! Learn more about Oracle Apex to gain the basic knowledge!!
Thankyou
can u share query for eba_proj_error_lookup table , i am not getting how to define field for it
Dimitri,
Thanks! Is there a reason you aren't trying to catch and replace not null constraints (you aren't looking for sqlcodes -1400 and -1407)?
Hi Dimitri,
Just implemented this.
Thanks!
Since I won't have access to production (including production's logger_logs) (at least for now), I extended the package slightly so that the email sends a full list of page items, application items, and apex_collections.
Dear Dimitri,
thanks a lot for the code on GIST, I would really like to use it in our project but I don't find any license information. Could you please tell me the constraints of use?
Thanks a lot,
Stefan
Hi Dimitri,
I notice this code is being used in the official Oracle documentation
https://docs.oracle.com/en/database/oracle/apex/22.1/aeapi/Example-of-an-Error-Handling-Function.html
and we have implemented a version of it for quite a while in our apps... but...
We just discovered that there is an issue with it masking errors on the way into pages... this link on apex.oracle.com demonstrates the problem
https://apex.oracle.com/pls/apex/r/gussay/demo-error-masking-on-page-entry
Username Jane
Password U$er_Jan3
I propose you mention this issue in the Official Oracle APEX Forum or log a support request.
Might be an issue / behavior change in APEX 22.2
An SR has been logged
SR 3-31246104701 : Apex Errors are not being shown during Pre Rendering when using Apex Error Handler
Post a Comment