Skip to Main Content

SQL & PL/SQL

Announcement

For appeals, questions and feedback about Oracle Forums, please email oracle-forums-moderators_us@oracle.com. Technical questions should be asked in the appropriate category. Thank you!

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

Saubhik

This thread may help you.

PhilMan2

Hi,

I looked at this thread and its links.  It doesn't help, unfortunately.  I'm still looking for an answer of how to send an embedded image in the body of an html eMail using UTL_SMTP.

Has anyone had any experience with this?

Phil

Stako Botev

Hi,

Have you considered that this could be due to your e-mail client?

I just tried your sample message body with the embedded image in it - I sent it through Oracle UTL_SMTP and received it successfully on my local e-mail client.

The image was fine and displayed as normal picture. However when I sent the same email on my gmail account, the picture was not shown.

So this is most probably due to configuration of the mail client.

Cheerz,

Stako

PhilMan2

Hi 2772777.  Thanks for looking at this.  I really appreciate it.

I get similar results when I switch to my hotmail account.  I retrieve the eMail using Outlook 2013 as well as through a FIrefox browser to outlook.com.  All attempts do NOT show the image.  I'm wondering if it's the way I'm calling UTL_SMTP.

The code that actually sends html body of the eMail is:

----------------------------------------------------
-- 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);
----------------------------------------------------

Any ideas?  Again, thanks for your help.

Phil

Billy Verreynne

If you create a valid MIME body as per RFC specifications, it will render correctly in the vast majority (if not all), of e-mail clients.

PhilMan2

Hi Billy.  Thanks for replying.  I looked at the link you provided.  I'm trying to figure out where the body I'm sending is not a valid MIME body.  Is there some sort of validation site that can tell me what's improper about my current message?

Thanks for your help.

Phil

Billy Verreynne

No. Not that I'm aware of.

When creating e-mail bodies, I prefer to letting an e-mail client do it for me. I will use my e-mail client, create the type of e-mail I want, and then mail it to myself. I'll then view the incoming e-mail in its raw format and use that as a template for my code.

Just keep in mind that not all mail clients and mail servers are equal. MS Exchange for example for some idiotic reason rips the raw e-mail apart and rewrites the entire thing - to the extent of replacing your boundary tags with its boundary tags, and base64 encoding plain/text MIME areas. So do not expect your e-mail to arrive in the actual format that you send it. Which is why it is best to ensure your e-mail is as vanilla MIME as possible. 

1 - 7
Locked Post
New comments cannot be posted to this locked post.

Post Details

Locked on Jan 24 2015
Added on Dec 19 2014
7 comments
2,794 views