Oracle PL/SQL E-mail Construction and Transmission with HTML Format Message Body and Optional Attachments
Clone this repository or download it as a zip archive.
Note: plsql_utilties and app_html_table_pkg are provided as submodules, so use the clone command with recurse-submodules option:
git clone --recurse-submodules https://github.com/lee-lindley/html_email.git
or download them separately as zip archives and extract the content of root folder into plsql_utilities and app_html_table_pkg folders respectively.
Follow the instructions in install.sql
Note that you do not absolutely require either submodule. The only essential element is arr_varchar2_udt which is simple enough to create yourself as noted in install.sql.
An Oracle Object Type for constructing and sending an HTML email, optionally with attachments. The best way to explain is with a usage example.
This example is recorded as a comment in the type specification, modified slightly here using my test case values:
DECLARE
v_email html_email_udt;
v_src SYS_REFCURSOR;
v_query VARCHAR2(32767) := q'!SELECT --+ no_parallel
v.view_name AS "View Name"
,c.comments AS "Comments"
FROM dictionary d
INNER JOIN all_views v
ON v.view_name = d.table_name
LEFT OUTER JOIN all_tab_comments c
ON c.table_name = v.view_name
WHERE d.table_name LIKE 'ALL%'
ORDER BY v.view_name
FETCH FIRST 40 ROWS ONLY!';
--
-- Because you cannot CLOSE/ReOPEN a dynamic sys_refcursor variable directly,
-- you must regenerate it and assign it. Weird restriction, but do not
-- try to fight it by opening it in the main code twice. Get a fresh copy from a function.
FUNCTION l_getcurs RETURN SYS_REFCURSOR IS
l_src SYS_REFCURSOR;
BEGIN
OPEN l_src FOR v_query;
RETURN l_src;
END;
BEGIN
v_email := html_email_udt(
p_to_list => '[email protected], [email protected]'
,p_from_email_addr => '[email protected]'
,p_reply_to => '[email protected]'
,p_subject => 'A sample email from html_email_udt'
,p_smtp_server => 'localhost'
);
v_email.add_paragraph('We constructed and sent this email with html_email_udt.');
v_src := l_getcurs;
-- convert sql query result set into HTML table
v_email.add_table_to_body(p_refcursor => v_src, p_caption => 'DBA Views');
-- need to close it because we are going to open again.
-- The called package may have closed it, but must be sure or nasty
-- bugs/caching can happen.
BEGIN
CLOSE v_src;
EXCEPTION WHEN invalid_cursor THEN NULL;
END;
-- add a spreadsheet attachment
-- https://github.com/mbleron/ExcelGen
DECLARE
l_xlsx_blob BLOB;
l_ctxId ExcelGen.ctxHandle;
l_sheet_handle BINARY_INTEGER;
BEGIN
v_src := l_getcurs;
l_ctxId := ExcelGen.createContext();
l_sheet_handle := ExcelGen.addSheetFromCursor(l_ctxId, 'DBA Views', v_src, p_tabColor => 'green');
BEGIN
CLOSE v_src;
EXCEPTION WHEN invalid_cursor THEN NULL;
END;
ExcelGen.setHeader(l_ctxId, l_sheet_handle, p_frozen => TRUE);
--------
v_email.add_attachment(
p_file_name => 'dba_views.xlsx'
,p_blob_content => ExcelGen.getFileContent(l_ctxId)
);
--------
excelGen.closeContext(l_ctxId);
END;
v_email.add_paragraph('The attached spreadsheet should match what is in the html table above');
--dbms_output.put_line(v_email.body);
v_email.send;
END;
A snapshot from my email client follows. It is mentioned in the type definition comments about cursor_to_table that spaces in the column names will be munged to x0020. I obviously forgot and the "View Name" column heading is jacked. The attachment opens in Excel as expected.
The attachment mime type values can be looked up from the file name extension using the package mime_type_pkg that is optionally installed along with the object. If you choose not to install that package, it will default to text/plain for CLOB attachments and application/octet-stream for BLOBs. Honestly, modern email clients on MS Windows seem to pay attention almost exclusively to the file extension, so I do not think you will suffer any harm by sticking with these two defaults. Nevertheless, we provide a way to do it right, and that is the default behavior if you use "install.sql".
Although object attributes cannot be made private, you have no need for them. The object interface is through the methods.
The type is marked as NOT FINAL so that you can derive a subtype from it. Possible uses involve adding methods for local conventions, such as a signature block, and perhaps overriding or supplementing methods with localized html conventions.
It is recommended that you edit install.sql before deploying to set suitable values for smtp_server, reply_to and from_email_addr define variables. The DEFAULT values on the constructor parameters below are those currently defined in "install.sql".
The procedure version does the work so that if the type html_email_udt is inherited, the child class can call it.
CONSTRUCTOR FUNCTION html_email_udt(
-- these take strings that can have multiple comma separated email addresses
p_to_list VARCHAR2 DEFAULT NULL
,p_cc_list VARCHAR2 DEFAULT NULL
,p_bcc_list VARCHAR2 DEFAULT NULL
--
,p_from_email_addr VARCHAR2 DEFAULT '[email protected]'
,p_reply_to VARCHAR2 DEFAULT '[email protected]'
,p_smtp_server VARCHAR2 DEFAULT 'localhost'
,p_subject VARCHAR2 DEFAULT NULL
,p_body CLOB DEFAULT NULL
-- compile time decision whether attribute 'log' is included
,p_log app_log_udt DEFAULT NULL
) RETURN SELF AS RESULT
,FINAL MEMBER PROCEDURE html_email_constructor(
SELF IN OUT NOCOPY html_email_udt
,p_to_list VARCHAR2 DEFAULT NULL
,p_cc_list VARCHAR2 DEFAULT NULL
,p_bcc_list VARCHAR2 DEFAULT NULL
,p_from_email_addr VARCHAR2 DEFAULT '&&from_email_addr'
,p_reply_to VARCHAR2 DEFAULT '&&reply_to'
,p_smtp_server VARCHAR2 DEFAULT '&&smtp_server'
,p_subject VARCHAR2 DEFAULT NULL
,p_body CLOB DEFAULT NULL
-- compile time decision whether attribute 'log' is included
,p_log app_log_udt DEFAULT NULL
A string that contains one or more email addresses separated with comma and optional white space for the TO addresses, Carbon Copy addresses and Blind Carbon Copy addresses, respectively.
A string containing a single email address that you likely configured at compile time. The parameter allows this to be overridden.
A string containing a single email address that you likely configured at compile time. The parameter allows this to be overridden. This is the REPLY-TO address of the standard and can be different than the displayed FROM email address. Typically set to a no-reply style address because you do not want replies coming to the database server, but perhaps you have a mailbox configured for this.
Name of the mail server to contact on port 25. You will have configured this on install and are unlikely to change it on the fly; however, a scenario where you have different email servers is possible.
The email subject line. You can also add it later with a method.
The HTML content of the email body. You can put the entire content here, start it here and add to it, or put nothing here and build it entirely with methods (most common case).
Logging object instance (if compiled with the 'log' attribute).
--
-- best explanation of method chaining rules I found is
-- https://stevenfeuersteinonplsql.blogspot.com/2019/09/object-type-methods-part-3.html
--
,FINAL MEMBER PROCEDURE send(SELF IN html_email_udt) -- cannot be in/out if we allow chaining it.
Opens the port to the email server, negotiates, sends the gnarly insides of an email negotiation including the header with addresses and boundary definition, sends the HTML body, then adds any attachments. Closes the connection.
At that point the email object is done. I have not experimented with reusing it with a changed address list or server. Perhaps it will work. Generally at this point you let the object go.
,MEMBER PROCEDURE add_paragraph(SELF IN OUT NOCOPY html_email_udt , p_clob CLOB)
,MEMBER FUNCTION add_paragraph(p_clob CLOB) RETURN html_email_udt
If the email body is not empty, adds '<br>' to separate from the last set of email. Yes I know it shouldn't be necessary, but email clients are HTML stupid.
Adds '<p>' followed by your text. It does not bother with the closing '</p>' tag.
,MEMBER PROCEDURE add_code_block(SELF IN OUT NOCOPY html_email_udt , p_clob CLOB)
,MEMBER FUNCTION add_code_block(p_clob CLOB) RETURN html_email_udt
If the email body is not empty, adds '<br>' tag.
Adds '<pre><code>', your text, then '</code></pre><br>'.
,MEMBER PROCEDURE add_to_body(SELF IN OUT NOCOPY html_email_udt, p_clob CLOB)
,MEMBER FUNCTION add_to_body(p_clob CLOB) RETURN html_email_udt
Simple append of your text to the email body. Does not add any tags.
,MEMBER PROCEDURE add_table_to_body( -- see cursor_to_table
SELF IN OUT NOCOPY html_email_udt
,p_sql_string CLOB := NULL
,p_refcursor SYS_REFCURSOR := NULL
,p_caption VARCHAR2 := NULL
)
,MEMBER FUNCTION add_table_to_body( -- see cursor_to_table
p_sql_string CLOB := NULL
,p_refcursor SYS_REFCURSOR := NULL
,p_caption VARCHAR2 := NULL
) RETURN html_email_udt
Given a string containing a SQL query, or a SYS_REFCURSOR (but not both), run the query through a DBMS_XMLGEN, XMLTYPE.transform operation to produce an HTML table that is inserted into your email. This works well enough, but is lacking in formatting control and has a few rough edges like translating spaces in your column aliases to '_x0020_'.
See app_html_table_pkg which is included as a submodule and which you can optionally compile as a more sophisticated alternative.
-- these take strings that can have multiple comma separated email addresses
,MEMBER PROCEDURE add_to(SELF IN OUT NOCOPY html_email_udt, p_to VARCHAR2)
,MEMBER FUNCTION add_to(p_to VARCHAR2) RETURN html_email_udt
,MEMBER PROCEDURE add_cc(SELF IN OUT NOCOPY html_email_udt, p_cc VARCHAR2)
,MEMBER FUNCTION add_cc(p_cc VARCHAR2) RETURN html_email_udt
,MEMBER PROCEDURE add_bcc(SELF IN OUT NOCOPY html_email_udt, p_bcc VARCHAR2)
,MEMBER FUNCTION add_bcc(p_bcc VARCHAR2) RETURN html_email_udt
You can put them all in the constructor and/or add addresses at any time during construction of the email prior to send. You might conditionally add an addressee while constructing the email after determining an edge condition.
As with the constructor, these take string with comma separated email addresses. Leading and trailing spaces are removed from each one.
,MEMBER PROCEDURE add_subject(SELF IN OUT NOCOPY html_email_udt, p_subject VARCHAR2)
,MEMBER FUNCTION add_subject(p_subject VARCHAR2) RETURN html_email_udt
If you already added one in the constructor, this replaces it
,MEMBER PROCEDURE add_attachment(
SELF IN OUT NOCOPY html_email_udt
,p_file_name VARCHAR2
,p_clob_content CLOB DEFAULT NULL
,p_blob_content BLOB DEFAULT NULL
-- looks up the mime type from the file_name extension
)
,MEMBER FUNCTION add_attachment(
p_file_name VARCHAR2
,p_clob_content CLOB DEFAULT NULL
,p_blob_content BLOB DEFAULT NULL
-- looks up the mime type from the file_name extension
) RETURN html_email_udt
,MEMBER PROCEDURE add_attachment( -- just in case you need fine control
SELF IN OUT NOCOPY html_email_udt
,p_attachment email_attachment_udt
)
,MEMBER FUNCTION add_attachment( -- just in case you need fine control
p_attachment email_attachment_udt
) RETURN html_email_udt
The attachment itself is just data in the email body, but it has an attribute called 'Content-Disposition: attachment; filename=YOUR_NAME'. This is what appears in email clients as the suggested filename for the attachment.
(I cannot tell you why I used p_file_name instead of the more common p_filename.)
Provide a CLOB if you have a text attachment like a CSV file.
Provide a BLOB if you have a binar file attachment like an XSLT file.
Note that you cannot provide both p_clob_content and p_blob_content. One of them must be null.
Attachments are object types that contain file_name, clob_content, blob_content, and mime_type member attributes. If you want fine grained control over the mime_type rather than letting html_email_udt determine it from the filename extension, you can construct the object yourself and provide it.
--Note: that if the cursor does not return any rows, we silently pass back an empty clob
,STATIC FUNCTION cursor_to_table(
-- pass in a string.
-- Unfortunately any tables that are not in your schema
-- will need to be fully qualified with the schema name.
p_sql_string CLOB := NULL
-- pass in an open cursor. This is better for my money.
,p_refcursor SYS_REFCURSOR := NULL
-- if provided, will be the caption on the table, generally centered on the top of it
-- by most renderers.
,p_caption VARCHAR2 := NULL
-- compile time decision whether attribute 'log' is included
,p_log app_log_udt DEFAULT NULL
) RETURN CLOB
As a bonus it provides this static function to convert a cursor or query string into a CLOB containing an HTML table. You can include that in the email or use it for a different purpose. It is called by member method add_table_to_body.
See app_html_table_pkg for a better option.
The privs required to use UTL_SMTP and to enable access to a network port for your schema may require some work from your DBA, and maybe the firewall and/or network team. There is some information in comments in install.sql about Access Control Lists for the network priv.
A compile time option allows configuring with invoker rights (AUTHID CURRENT_USER) in which case the calling schemas will require the network access control in addition to the compiling schema.
The Example shows using sendmail listening on port 25 on my local RHL database server (localhost), but you will likely need to use a company relay such as smtp.mycompany.com. The administrator of the relay may even need to authorize your database machine as a client. If the relay server complain about certificates, there is Oracle documentation about configuring them. I have not done so.
One more example showing chaining of methods without ever declaring a variable as one might use in an EXCEPTION block:
DECLARE
l_var VARCHAR2(1);
BEGIN
l_var := 'too much for the var size';
EXCEPTION WHEN OTHERS THEN
html_email_udt(
p_to_list => '[email protected]'
,p_subject => 'error from procedure'
,p_smtp_server => 'localhost'
,p_body => 'Exception failed job xyz with the following information:'
).add_code_block(
'sqlerrm : '||SQLERRM
).add_code_block(
'backtrace : '||DBMS_UTILITY.FORMAT_ERROR_BACKTRACE
).send
;
RAISE;
END;
See the documentation in the package README.md.
Runs each of these scripts in correct order and with compile options. There are a set of six sqlplus "define" commands at the top that populate compile directives in PLSQL_CCFLAGS and control whether or not deployment scripts are executed.
ALTER SESSION SET PLSQL_CCFLAGS='use_app_log:TRUE,use_app_parameter:FALSE,use_mime_type:TRUE,use_invoker_rights:FALSE';
There are three "define" statements for default values for the html_email_udt constructor that should also be set appropriately.
install.sql has directions allowing choice on whether to use anything from the submodule. None is required except for the type arr_varchar2_udt which is easy enough to replace. If you already have a 'TABLE OF VARCHAR2(4000)' object deployed, look in the install script for the define of array_varchar2_type. Set the variables to FALSE for any components you do not want.
Likewise, the define named compile_app_html_table_pkg is set to FALSE as the default. Change it to TRUE if you want to add that package to the database.