Skip to Main Content

How do I use Oracle's UTL_SMTP to send a message body that included an embedded image

PhilMan2Dec 19 2014 — edited Dec 27 2014

I'm attempting to use Oracle's 11.g's UTL_SMTP to send a message body that includes an embedded image. I have this process working for a standard text body, and for an eMail that may contain attachments (Word doc, JPG image, etc.). I'm stuck on making this work with an image embedded in the body of the eMail. When I include and embedded image, the eMail displays in Outlook 2013 as a red X where the image should be. The rest of the text appears fine. The html body of the eMail is passed in to the procedure via "p_html".

My Procedure looks like this:

create or replace procedure html_email_attachments(
  p_to 
in varchar2,
  p_from 
in varchar2,
  p_subject 
in varchar2,
  p_text 
in varchar2 default null,
  p_html 
in varchar2 default null,
  p_smtp_hostname 
in varchar2,
  p_smtp_portnum 
in varchar2,
  p_smtp_username 
in varchar2,
  p_smtp_password 
in varchar2,
  p_event_pkey 
in number
  
)
is
  l_boundary varchar2
(255) default 'a1b2c3d4e3f2g1';
  l_connection utl_smtp
.connection;
  l_body_html clob
:= empty_clob; --This LOB will be the email message
  l_offset number
;
  l_ammount number
;
  l_temp varchar2
(32767) default null;
  l_encoded_username varchar2
(2048);
  l_encoded_password varchar2
(2048);
  l_attach_mime varchar2
(1024);
  l_attach_name varchar2
(1024);
  l_attach_blob blob
;
  l_attachment_pkeys DBMS_SQL
.varchar2_table;
  l_step PLS_INTEGER 
:= 12000; -- A multiple of 3 and <= 24573
  crlf varchar2
(2) := CHR(13) || CHR(10);
  diag varchar2
(4000);
  v_raw raw
(57);
  v_length integer
:= 0;
  v_buffer_size integer
:= 57;
  v_offset integer
:= 1;
begin
-- Store all the attachment primary keys into l_attachment_pkeys 
  
Select ma.prim_key  Bulk Collect Into l_attachment_pkeys
  
From  mail_attachment ma
  
Where ma.event_fkey = p_event_pkey;
-- Encode p_smtp_username and p_smtp_password
  l_encoded_username
:= UTL_RAW.cast_to_varchar2
  
(UTL_ENCODE.base64_encode(UTL_RAW.cast_to_raw(p_smtp_username)));
  l_encoded_password
:= UTL_RAW.cast_to_varchar2
  
(UTL_ENCODE.base64_encode(UTL_RAW.cast_to_raw(p_smtp_password)));
-- set the l_connection variable
  l_connection
:= UTL_SMTP.open_connection(p_smtp_hostname, p_smtp_portnum);
-- begin the connection
  utl_smtp
.ehlo(l_connection, p_smtp_hostname);--DO NOT USE HELO
-- authenticate to the server using the encoded username and password
  utl_smtp
.command(l_connection, 'AUTH', 'LOGIN');
  utl_smtp
.command(l_connection, l_encoded_username);
  utl_smtp
.command(l_connection, l_encoded_password);
-- First, establish the mail's From and To. This is essential
  utl_smtp
.mail( l_connection, p_from );
  utl_smtp
.rcpt( l_connection, p_to );
-- Define the Header
  l_temp
:= l_temp || 'MIME-Version: 1.0'   || crlf;
  l_temp
:= l_temp || 'To: '   || p_to  || crlf;
  l_temp
:= l_temp || 'From: '   || p_from  || crlf;
  l_temp
:= l_temp || 'Subject: '  || p_subject  || crlf;
  l_temp
:= l_temp || 'Reply-To: ' || p_from  || crlf;
  l_temp
:= l_temp || 'Content-Type: multipart/mixed; boundary=' ||
  chr
(34) || l_boundary ||  chr(34) || crlf;

  
----------------------------------------------------
  
-- Write the headers
  dbms_lob
.createtemporary( l_body_html, false, 10 );
  dbms_lob
.write(l_body_html,length(l_temp),1,l_temp);

  
----------------------------------------------------
  
-- Write the HTML boundary
  
-- Identify the content type as text/html
  
-- Determine the offset (initially it's 1)
  
-- Write out
  l_temp 
:= crlf || crlf ||'--' || l_boundary || crlf;
  l_temp 
:= l_temp || 'content-type: text/html;' || crlf || crlf;
  l_offset
:= dbms_lob.getlength(l_body_html) + 1;
  dbms_lob
.write(l_body_html,length(l_temp),l_offset,l_temp);

  
----------------------------------------------------
  
-- Write the HTML portion of the message
  l_offset
:= dbms_lob.getlength(l_body_html) + 1;
  dbms_lob
.write(l_body_html,length(p_html),l_offset,p_html);
  
----------------------------------------------------
  
-- Send the email body in 1900 byte chunks to UTL_SMTP
  l_offset 
:= 1;
  l_ammount
:= 1900;
  utl_smtp
.open_data(l_connection);

  
while l_offset < dbms_lob.getlength(l_body_html) loop
  utl_smtp
.write_data(l_connection,
  dbms_lob
.substr(l_body_html,l_ammount,l_offset));
  l_offset 
:= l_offset + l_ammount ;
  l_ammount
:= least(1900,dbms_lob.getlength(l_body_html) - l_ammount);
  
end loop;
  
--The following crlf is necessary after the body is written
  utl_smtp
.write_data(l_connection, crlf);
  
----------------------------------------------------
  
-- Write the attachments of the message
  
-- Loop through all the selected primary keys in l_attachment_pkeys
  
While l_attachment_pkeys is null
  Loop
  
For i IN l_attachment_pkeys.FIRST .. l_attachment_pkeys.LAST
  Loop
  
-- Determine the attachment variables for each instance of attachments
  
Select mime_type  into l_attach_mime
  
from mail_attachment where prim_key = l_attachment_pkeys(i);
  
Select file_name  into l_attach_name
  
from mail_attachment where prim_key = l_attachment_pkeys(i);
  
Select attachment into l_attach_blob
  
from mail_attachment where prim_key = l_attachment_pkeys(i);
  v_length
:= dbms_lob.getlength(l_attach_blob);

  
-- Write out the boundary
  UTL_SMTP
.write_data(l_connection, '--' || l_boundary || crlf);
  
-- Write out the attachment metadata
  UTL_SMTP
.write_data(l_connection, 'Content-Type: ' || l_attach_mime ||
  
'; name=' || chr(34) || l_attach_name || chr(34) || crlf);
  UTL_SMTP
.write_data(l_connection, 'Content-Transfer-Encoding: base64' ||
  crlf
);
  UTL_SMTP
.write_data(l_connection, 'Content-Disposition: attachment;
  filename="'
|| l_attach_name || '"' || crlf || crlf);
  
--Write out the attachment blob in portions of l_step length
  
FOR k IN 0 .. TRUNC((DBMS_LOB.getlength(l_attach_blob) - 1 )/l_step)
  LOOP
  UTL_SMTP
.write_data(l_connection, UTL_RAW.cast_to_varchar2
  
(UTL_ENCODE.base64_encode(DBMS_LOB.substr(l_attach_blob,
  l_step
, k * l_step + 1))));
  
END LOOP;
  UTL_SMTP
.write_data(l_connection, crlf);
  
End Loop;
  
End Loop;

  
----------------------------------------------------
  
-- Write the final html boundary
  utl_smtp
.write_data(l_connection,
  crlf
|| '--' || l_boundary || '--' || crlf);
  
----------------------------------------------------
  
-- Close the connection and end
  utl_smtp
.close_data(l_connection);
  utl_smtp
.quit( l_connection );
  dbms_lob
.freetemporary(l_body_html);
end;


A sample "p_html" field looks like this:

<p>
  Text
right before the image.</p>
<p>
  
<img alt="HarryPotter" src="" /></p>
<p>
  Text
right after the image.</p>

Comments
Locked Post
New comments cannot be posted to this locked post.
Post Details
Locked due to inactivity on Jan 24 2015
Added on Dec 19 2014
7 comments
2,749 views