Reading emails and processing the information within the mail, is a common requirement in today's business processes.
I have always found OSB to be easier in implementing any functionality compared to SOA. Email protocol proves it right for email reading.Unlike SOA, where email adapters, drivers need to be created to read mails, OSB provides a simple protocol to take care of this feature.
In my example, I am going to read mails from a gmail account and process the details in the mail. Before proceeding on how to create an OSB for email reading, lets get the mailbox details ready.
IMAP or POP3 -
In order to read any mail box, we would need to connect to the mailbox through one of the two protocols - imap or pop3. imap connects over port 993 where as pop3 connects over port 995. Lets check if the gmail box is imap enabled. Else, we would need to enable it.
1) Login to the gmail box. Click on Settings.
2) Next click on "Forwarding and POP/IMAP".
Now the gmail box is IMAP enabled and we can use this protocol to OSB to access.
OSB project -
1. Create a simple service bus project. I have named it as PGEmailProject.
2. Create a Service Account with static mappings. This service account should contain the username and password to the gmail account you would listen to. Let me call mine as testgmail@gmail.com.
3. Next step is to start with the proxy service. Right click on the services section of the composite.xml. From the Transports section, choose Email.
4. A Proxy service popup would open. Provide a name for the proxy service. Transport is defaulted to email. Generate pipeline is ticked by default. Click on next.
5. The next wizard allows you to choose the Service Type. In my example, I can have any type of input coming. It need not be an xml. Hence, I chose Messaging service with Request as Text and Response None. Go to next section.
6. Final section is the Transport section where you need to enter the end point uri. The end point uri in Email protocol is nothing but the host and port of the mailbox from which the mails have to be read. In our case, it is imap.gmail.com:993. If we are connecting through pop, it would be pop.gmail.com:995. Click on Finish and we are done with the proxy service creation.
7. Now to set the transport parameters right. Click on the proxy service and move to Transport Details section. This section would be empty showing lot of errors. Time to fill up each of the applicable parameters.
- SSL Required - As gmail is https, check the checkbox.
- Service Account - Choose the one we created at the start of the project - PGEmailServiceAccount.sa
- Managed Server - All polling services, be it email, FTP, listen through one managed server of a clustered environment. We need to choose one of the managed servers. Mine being machine1.
- Polling Interval - Time between each email poll - 10 seconds for me
- Email protocol - As mentioned in earlier section, you can choose between imap and pop3. I choose imap.
- Read Limit - The number of mails to be read at a time - I left it as 10
- Pass By Reference - When enabled, the body is staged in archive location and the file location is passed as part of header. This is applicable only for Request type xml. As my reference type is Text, I have it disabled.
- Pass Attachments by Reference - Attachments are staged in archive location and file location is sent as part of headers. This is not applicable in my case and hence goes disabled.
- Post Read Action - What do you want to do with the mail once read? I want it to be deleted and hence my option is delete.
- IMAP Move Folder - Empty. As I didnt choose my Post Read Action as Move.
- Download, Archive and Error directory - The path to which the mails should be archived, downloaded. I have /test/email/download/ and /test/email/error/ locations in my server and hence have given this location. You can give the location based on your file structure.
- Request Encoding - Left to the default ISO-8859-1
Finally the Transport details page looks as follows -
8. Time to fill up the pipeline with some activities to check the mails that are read. For simplicity, I have added two alerts in the pipeline to print the body and the headers once they are read.
9. Deploy the project to your server and now send a mail to testgmail@gmail.com mailbox. This is the mailbox to which the proxy is listening to.
10. I sent a mail with Subject and Body as "Test Mail".
Once OSB reads the mails, alerts will be published with the following information.
PG Email Body:
This will contain the content of the body in the email.
<soapenv:Body xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/">Test Email</soapenv:Body>
PG Email Inbound:
All the header information including who sent the mail, subject, etc are returning as part of inbound headers.
<con:endpoint name="ProxyService$PGEmailProject$PGEmailProxyService" xmlns:con="http://www.bea.com/wli/sb/context"> <con:service/> <con:transport> <con:uri>mailfrom:imap.gmail.com:993</con:uri> <con:mode>request</con:mode> <con:qualityOfService>exactly-once</con:qualityOfService> <con:request xsi:type="ema:EmailRequestMetaData" xmlns:ema="http://www.bea.com/wli/sb/transports/email" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <tran:headers xsi:type="ema:EmailRequestHeaders" xmlns:tran="http://www.bea.com/wli/sb/transports"> <tran:user-header name="MIME-Version" value="1.0"/> <tran:user-header name="X-MS-Exchange-Organization-MessageDirectionality" value="Incoming"/> <tran:user-header name="X-MS-Exchange-Organization-AuthSource" value="BN1AFFO11FD012.protection.gbl"/> <tran:user-header name="X-EOPAttributedMessage" value="0"/> <tran:user-header name="X-DkimResult-Test" value="Passed"/> <tran:user-header name="X-EOPTenantAttributedMessage" value="9274ee3f-9425-4109-a27f-9fb15c10675d:0"/> <tran:user-header name="SpamDiagnosticOutput" value="1:99"/> <tran:user-header name="SpamDiagnosticMetadata" value="NSPM"/> <tran:user-header name="X-MS-Exchange-CrossTenant-Id" value="9274ee3f-9425-4109-a27f-9fb15c10675d"/> <tran:user-header name="X-MS-Exchange-Transport-EndToEndLatency" value="00:00:00.9743427"/> <tran:user-header name="X-MS-PublicTrafficType" value="Email"/> <tran:user-header name="X-MS-Office365-Filtering-HT" value="Tenant"/> <tran:user-header name="X-MS-Office365-Filtering-Correlation-Id" value="9948914d-638a-4764-c610-08d4a3f44afd"/> <tran:user-header name="Message-ID" value="<CABOwZYhyJ3enfdVQRMXG=3Ha=YFf_fzEeMYFJSat9gVywytRRg@mail.gmail.com>"/> <tran:user-header name="Return-Path" value="prasannarani.g@gmail.com"/> <tran:user-header name="X-MS-Exchange-CrossTenant-OriginalArrivalTime" value="26 May 2017 05:01:39.0171 (UTC)"/> <ema:To>Prasanna <testgmail@gmail.com></ema:To> <ema:From>Prasanna G <testprasanna@gmail.com></ema:From> <ema:Date>Fri May 26 01:01:38 EDT 2017</ema:Date> <ema:Subject>Test Mail</ema:Subject> <ema:Content-Type>multipart/alternative; boundary=94eb2c1499a87e29340550663b14</ema:Content-Type> </tran:headers> <tran:encoding xmlns:tran="http://www.bea.com/wli/sb/transports">utf-8</tran:encoding> </con:request> </con:transport> <con:security> <con:transportClient> <con:username><anonymous></con:username> </con:transportClient> </con:security> </con:endpoint>
Reading Attachments in the email :
The way attachments are read using email protocol depends on Pass Attachment by reference option set at the Email Transport Configuration level.
If the Messaging type is xml, we do get an option to enable Pass Attachment By Reference. In this case, the content in the attachment is staged in archive location and only reference to the location is sent as part of headers.
When the Messaging type is text, Pass By Reference and Pass attachment by Reference options cannot be enabled. In this case, the attachments are sent as part of $attachment variable.
I added two attachments - a png file and an xlsx file as attachments to another mail. The attachment variable looks as follows.
<con:attachments xmlns:con="http://www.bea.com/wli/sb/context"> <con:attachment> <con:Content-Type>image/png; name="Test.png"</con:Content-Type> <con:Content-Disposition>attachment; filename="Test.png"</con:Content-Disposition> <con:Content-Transfer-Encoding>base64</con:Content-Transfer-Encoding> <con:body> <con:binary-content ref="cid:-6ed90ee0:15c2ed5f20c:-7b17"/> </con:body> </con:attachment> <con:attachment> <con:Content-Type>application/vnd.openxmlformats-officedocument.spreadsheetml.sheet; name="Book1.xlsx"</con:Content-Type> <con:Content-Disposition>attachment; filename="Book1.xlsx"</con:Content-Disposition> <con:Content-Transfer-Encoding>base64</con:Content-Transfer-Encoding> <con:body> <con:binary-content ref="cid:2862c084:15c2ed5ba2e:-7a65"/> </con:body> </con:attachment> </con:attachments>
Note that for each attachment, the <con:attachment> section is repeated. You can find the Content type, name of the file, etc. But if you notice, the actual attachment is not present in base64 format, it is present as binary-content with reference cid. This is a feature in OSB, where the actual files are returned in binary-content format.
In order to further use the attachment content, it needs to be explicitly converted into base64 format. I would also quickly show you, on how to do that. I used a Java class to achieve this functionality and passed the binary content from OSB to java callout and got base64 format as response.
Here's the simple java code -
import weblogic.utils.encoders.BASE64Encoder; public class Base64Converter { public Base64Converter() { super(); } public static String encode(final byte[] bytes) { final BASE64Encoder encoder = new BASE64Encoder(); final String encodedString = encoder.encodeBuffer(bytes); return encodedString; } }
1. Create a simple Java class as above.
2. You need to import webservices-9.2.2.0.jar into the java project in order to make this class work.
3. Create a java jar of this class and add into OSB project.
4. Now add a Java Callout in the OSB.
5. The encode function of the class takes bytes array as input. From OSB, add the following binary-content as input to the java callout.
$attachments/ctx:attachment/ctx:body/ctx:binary-content
6. String returned as output of java callout will be in encoded base64 format.
Thus, emails and email attachments can be read through OSB email protocol.
Note: If you are using OSB 11g version, make sure that {MsgIdNotFound:READ_TIMEOUT} parameter is set to 0. Changing it any value is throwing a SocketTimeout Exception - Read TImeout exception. Oracle is looking into this issue.