How to use Oracle UTL_SMTP to send a message body that includes an inline image

I'm trying to use Oracle 11.g UTL_SMTP to send a message body that includes an inline image. I have this process that works for standard text body and for email that may contain attachments (Word doc, JPG image, etc.). I am stuck making this work with an image embedded in the email body. When I include and embed the image, the eMail appears in Outlook 2013 as a red X where the image should be. The rest of the text looks great. The body of the html email is passed 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 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 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;

      

The sample field "p_html" looks like this:

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

      

+3


source to share


2 answers


the following code works for me:

DECLARE
    conn utl_smtp.connection;
    BOUNDARY  VARCHAR2 (256) := '-----090303020209010600070908';
    i         pls_integer;
    len       pls_integer;
    buff_size pls_integer := 57;
    l_raw     raw(57);
    p_image blob;
    MailServer          VARCHAR2(50) := 'xxxxxx.xxxxx.xxxxxxx.xxxxx';
     l_message            VARCHAR2(32767) :='<html>
<body>
<img src="cid:banner" alt="banner"/>
<br>
Test HTML with Embedded Image-chk latest
<p>And here it is:</p>
<p>The end.</p>
</body>
</html>
';
begin
SELECT PIC
INTO p_image
FROM TEMP_MAIL_PIC_BLOB
WHERE PIC_ID = 1;
conn := utl_smtp.open_connection(MailServer, 25);
UTL_SMTP.helo (conn, MailServer);
UTL_SMTP.mail (conn, 'from@abc.com');
UTL_SMTP.rcpt (conn, 'to@abc.com');
UTL_SMTP.open_data (conn);
UTL_SMTP.write_data (conn, 'From' || ': ' || 'from@test.com'|| UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, 'To' || ': ' || 'to@test.com'|| UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, 'MIME-Version: 1.0' || UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, 'Subject: image inline testing' || UTL_TCP.CRLF) ;
UTL_SMTP.write_data (conn, 'Content-Type: multipart/mixed; boundary="' || BOUNDARY || '"' || UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, UTL_TCP.CRLF);
UTL_SMTP.write_data (conn,  '--' || BOUNDARY || UTL_TCP.CRLF );
UTL_SMTP.write_data (conn,  'Content-Type: text/html; charset=US-ASCII'|| UTL_TCP.CRLF );
UTL_SMTP.write_data (conn, UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, l_message);
UTL_SMTP.write_data (conn, UTL_TCP.CRLF);
UTL_SMTP.write_data (conn,  '--' || BOUNDARY || UTL_TCP.CRLF );
UTL_SMTP.write_data (conn,  'Content-Type: image/jpg;'|| UTL_TCP.CRLF );
UTL_SMTP.write_data (conn, 'Content-Disposition: inline; filename="banner.jpg"' || UTL_TCP.CRLF);
UTL_SMTP.WRITE_DATA (conn, 'Content-ID: <banner> ' || UTL_TCP.CRLF);   
UTL_SMTP.write_data (conn, 'Content-Transfer-Encoding' || ': ' || 'base64' || UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, UTL_TCP.CRLF);
i := 1;
len := dbms_lob.getlength(p_image);
while i < len
loop
  dbms_lob.read(p_image, buff_size, i, l_raw);
  utl_smtp.write_raw_data(conn, utl_encode.base64_encode(l_raw));
  utl_smtp.write_data(conn, utl_tcp.crlf);
  i := i + buff_size;
end loop;
utl_smtp.write_data(conn, utl_tcp.crlf);
    UTL_SMTP.write_data (conn, '--' || BOUNDARY || '--' || UTL_TCP.CRLF);
UTL_SMTP.write_data (conn, UTL_TCP.CRLF);
UTL_SMTP.close_data (conn);
UTL_SMTP.quit (conn);
end;

      



Giova relationship

+1


source


Phil,

I am facing the same problems. Showing the red X in perspective. My code is stored in a CLOB column.



How did you solve this? Can I share a code that works for you?

Thanks Aj

0


source







All Articles