mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-22 15:12:38 +00:00 
			
		
		
		
	git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@6246 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
			
				
	
	
		
			297 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			297 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2007 Alfresco Software Limited.
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
| 
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
| 
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 | |
| 
 | |
|  * As a special exception to the terms and conditions of version 2.0 of 
 | |
|  * the GPL, you may redistribute this Program in connection with Free/Libre 
 | |
|  * and Open Source Software ("FLOSS") applications as described in Alfresco's 
 | |
|  * FLOSS exception.  You should have recieved a copy of the text describing 
 | |
|  * the FLOSS exception, and it is also available here: 
 | |
|  * http://www.alfresco.com/legal/licensing"
 | |
|  */
 | |
| package org.alfresco.repo.content.metadata;
 | |
| 
 | |
| import java.io.IOException;
 | |
| import java.io.InputStream;
 | |
| import java.io.Serializable;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Arrays;
 | |
| import java.util.Calendar;
 | |
| import java.util.HashSet;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| 
 | |
| import org.alfresco.service.cmr.repository.ContentIOException;
 | |
| import org.alfresco.service.cmr.repository.ContentReader;
 | |
| import org.apache.poi.poifs.eventfilesystem.POIFSReader;
 | |
| import org.apache.poi.poifs.eventfilesystem.POIFSReaderEvent;
 | |
| import org.apache.poi.poifs.eventfilesystem.POIFSReaderListener;
 | |
| import org.apache.poi.poifs.filesystem.DocumentInputStream;
 | |
| 
 | |
| /**
 | |
|  * Outlook format email meta-data extractor extracting the following values:
 | |
|  * <pre>
 | |
|  *   <b>sentDate:</b>               --      cm:sentdate
 | |
|  *   <b>originator:</b>             --      cm:originator,    cm:author
 | |
|  *   <b>addressee:</b>              --      cm:addressee
 | |
|  *   <b>addressees:</b>             --      cm:addressees
 | |
|  *   <b>subjectLine:</b>            --      cm:subjectline,   cm:description
 | |
|  * </pre>
 | |
|  * 
 | |
|  * @since 2.1
 | |
|  * @author Kevin Roast
 | |
|  */
 | |
| public class MailMetadataExtracter extends AbstractMappingMetadataExtracter
 | |
| {
 | |
|     private static final String KEY_SENT_DATE = "sentDate";
 | |
|     private static final String KEY_ORIGINATOR = "originator";
 | |
|     private static final String KEY_ADDRESSEE = "addressee";
 | |
|     private static final String KEY_ADDRESSEES = "addressees";
 | |
|     private static final String KEY_SUBJECT = "subjectLine";
 | |
| 
 | |
|     public static String[] SUPPORTED_MIMETYPES = new String[] {"message/rfc822"};
 | |
|     
 | |
|     private static final String STREAM_PREFIX = "__substg1.0_";
 | |
|     private static final int STREAM_PREFIX_LENGTH = STREAM_PREFIX.length();
 | |
| 
 | |
|     // the CC: email addresses
 | |
|     private ThreadLocal<List<String>> receipientEmails = new ThreadLocal<List<String>>();
 | |
|     
 | |
|     public MailMetadataExtracter()
 | |
|     {
 | |
|         super(new HashSet<String>(Arrays.asList(SUPPORTED_MIMETYPES)));
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Map<String, Serializable> extractRaw(ContentReader reader) throws Throwable
 | |
|     {
 | |
|         final Map<String, Serializable> rawProperties = newRawMap();
 | |
|         
 | |
|         POIFSReaderListener readerListener = new POIFSReaderListener()
 | |
|         {
 | |
|             public void processPOIFSReaderEvent(final POIFSReaderEvent event)
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     if (event.getName().startsWith(STREAM_PREFIX))
 | |
|                     {
 | |
|                         StreamHandler handler = new StreamHandler(event.getName(), event.getStream());
 | |
|                         handler.process(rawProperties);
 | |
|                     }
 | |
|                 }
 | |
|                 catch (Exception ex)
 | |
|                 {
 | |
|                     throw new ContentIOException("Property set stream: " + event.getPath() + event.getName(), ex);
 | |
|                 }
 | |
|             }
 | |
|         };
 | |
|         
 | |
|         InputStream is = null;
 | |
|         try
 | |
|         {
 | |
|             this.receipientEmails.set(new ArrayList<String>());
 | |
|             
 | |
|             is = reader.getContentInputStream();
 | |
|             POIFSReader poiFSReader = new POIFSReader();
 | |
|             poiFSReader.registerListener(readerListener);
 | |
|             
 | |
|             try
 | |
|             {
 | |
|                 poiFSReader.read(is);
 | |
|             }
 | |
|             catch (IOException err)
 | |
|             {
 | |
|                 // probably not an Outlook format MSG - ignore for now
 | |
|                 if (logger.isWarnEnabled())
 | |
|                     logger.warn("Unable to extract meta-data from message: " + err.getMessage());
 | |
|             }
 | |
|             
 | |
|             // store multi-value extracted property
 | |
|             if (this.receipientEmails.get().size() != 0)
 | |
|             {
 | |
|                 putRawValue(KEY_ADDRESSEES, (Serializable)receipientEmails.get(), rawProperties);
 | |
|             }
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             if (is != null)
 | |
|             {
 | |
|                 try { is.close(); } catch (IOException e) {}
 | |
|             }
 | |
|         }
 | |
|         // Done
 | |
|         return rawProperties;
 | |
|     }
 | |
|     
 | |
|     private static String convertExchangeAddress(String email)
 | |
|     {
 | |
|         if (email.lastIndexOf("/CN=") == -1)
 | |
|         {
 | |
|             return email;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // found a full Exchange format To header
 | |
|             return email.substring(email.lastIndexOf("/CN=") + 4);
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private static final String ENCODING_TEXT = "001E";
 | |
|     private static final String ENCODING_BINARY = "0102";
 | |
|     private static final String ENCODING_UNICODE = "001F";
 | |
|     
 | |
|     @SuppressWarnings("unused")
 | |
|     private static final String SUBSTG_MESSAGEBODY = "1000";
 | |
|     private static final String SUBSTG_RECIPIENTEMAIL = "39FE";      // 7bit email address
 | |
|     private static final String SUBSTG_RECIPIENTSEARCH = "300B";     // address 'search' variant
 | |
|     private static final String SUBSTG_RECEIVEDEMAIL = "0076";
 | |
|     private static final String SUBSTG_SENDEREMAIL = "0C1F";
 | |
|     private static final String SUBSTG_DATE = "0047";
 | |
|     private static final String SUBSTG_SUBJECT = "0037";
 | |
|     
 | |
|     /**
 | |
|      * Class to handle stream types. Can process and extract specific streams.
 | |
|      */
 | |
|     private class StreamHandler
 | |
|     {
 | |
|         StreamHandler(String name, DocumentInputStream stream)
 | |
|         {
 | |
|             this.type = name.substring(STREAM_PREFIX_LENGTH, STREAM_PREFIX_LENGTH + 4);
 | |
|             this.encoding = name.substring(STREAM_PREFIX_LENGTH + 4, STREAM_PREFIX_LENGTH + 8);
 | |
|             this.stream = stream;
 | |
|         }
 | |
|         
 | |
|         void process(final Map<String, Serializable> destination)
 | |
|             throws IOException
 | |
|         {
 | |
|             if (type.equals(SUBSTG_SENDEREMAIL))
 | |
|             {
 | |
|                 putRawValue(KEY_ORIGINATOR, convertExchangeAddress(extractText()), destination);
 | |
|             }
 | |
|             else if (type.equals(SUBSTG_RECIPIENTEMAIL))
 | |
|             {
 | |
|                 receipientEmails.get().add(convertExchangeAddress(extractText()));
 | |
|             }
 | |
|             else if (type.equals(SUBSTG_RECIPIENTSEARCH))
 | |
|             {
 | |
|                 String email = extractText(ENCODING_TEXT);
 | |
|                 int smptIndex = email.indexOf("SMTP:");
 | |
|                 if (smptIndex != -1)
 | |
|                 {
 | |
|                     /* also may be used for SUBSTG_RECIPIENTTRANSPORT = "5FF7"; 
 | |
|                        with search for SMPT followed by a null char */
 | |
|                     
 | |
|                     // this is a secondary mechanism for encoding a receipient email address
 | |
|                     // the 7 bit email address may not have been set by Outlook - so this is needed instead
 | |
|                     // handle null character at end of string
 | |
|                     int endIndex = email.length();
 | |
|                     if (email.codePointAt(email.length() - 1) == 0)
 | |
|                     {
 | |
|                         endIndex--;
 | |
|                     }
 | |
|                     email = email.substring(smptIndex + 5, endIndex);
 | |
|                     receipientEmails.get().add(email);
 | |
|                 }
 | |
|             }
 | |
|             else if (type.equals(SUBSTG_RECEIVEDEMAIL))
 | |
|             {
 | |
|                 putRawValue(KEY_ADDRESSEE, convertExchangeAddress(extractText()), destination);
 | |
|             }
 | |
|             else if (type.equals(SUBSTG_SUBJECT))
 | |
|             {
 | |
|                 putRawValue(KEY_SUBJECT, extractText(), destination);
 | |
|             }
 | |
|             else if (type.equals(SUBSTG_DATE))
 | |
|             {
 | |
|                 // the date is not "really" plain text - but it's appropriate to parse as such
 | |
|                 String date = extractText(ENCODING_TEXT);
 | |
|                 int valueIndex = date.indexOf("l=");
 | |
|                 if (valueIndex != -1)
 | |
|                 {
 | |
|                     int dateIndex = date.indexOf('-', valueIndex);
 | |
|                     if (dateIndex != -1)
 | |
|                     {
 | |
|                         dateIndex++;
 | |
|                         final Calendar c = Calendar.getInstance();
 | |
|                         String strYear = date.substring(dateIndex, dateIndex + 2);
 | |
|                         c.set(Calendar.YEAR, Integer.parseInt(strYear) + (2000 - 1900));
 | |
|                         String strMonth = date.substring(dateIndex + 2, dateIndex + 4);
 | |
|                         c.set(Calendar.MONTH, Integer.parseInt(strMonth) - 1);
 | |
|                         String strDay = date.substring(dateIndex + 4, dateIndex + 6);
 | |
|                         c.set(Calendar.DAY_OF_MONTH, Integer.parseInt(strDay));
 | |
|                         String strHour = date.substring(dateIndex + 6, dateIndex + 8);
 | |
|                         c.set(Calendar.HOUR, Integer.parseInt(strHour));
 | |
|                         String strMinute = date.substring(dateIndex + 10, dateIndex + 12);
 | |
|                         c.set(Calendar.MINUTE, Integer.parseInt(strMinute));
 | |
|                         c.set(Calendar.SECOND, 0);
 | |
|                         putRawValue(KEY_SENT_DATE, c.getTime(), destination);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         /**
 | |
|          * Extract the text from the stream based on the encoding
 | |
|          * 
 | |
|          * @return String
 | |
|          * 
 | |
|          * @throws IOException
 | |
|          */
 | |
|         private String extractText()
 | |
|             throws IOException
 | |
|         {
 | |
|             return extractText(this.encoding);
 | |
|         }
 | |
|         
 | |
|         /**
 | |
|          * Extract the text from the stream based on the encoding
 | |
|          * 
 | |
|          * @return String
 | |
|          * 
 | |
|          * @throws IOException
 | |
|          */
 | |
|         private String extractText(String encoding)
 | |
|             throws IOException
 | |
|         {
 | |
|             byte[] data = new byte[stream.available()];
 | |
|             stream.read(data);
 | |
|             
 | |
|             if (encoding.equals(ENCODING_TEXT) || encoding.equals(ENCODING_BINARY))
 | |
|             {
 | |
|                 return new String(data);
 | |
|             }
 | |
|             else if (encoding.equals(ENCODING_UNICODE))
 | |
|             {
 | |
|                 // convert double-byte encoding to single byte for String conversion
 | |
|                 byte[] b = new byte[data.length >> 1];
 | |
|                 for (int i=0; i<b.length; i++)
 | |
|                 {
 | |
|                     b[i] = data[i << 1];
 | |
|                 }
 | |
|                 return new String(b);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return new String(data);
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         private String type;
 | |
|         private String encoding;
 | |
|         private DocumentInputStream stream;
 | |
|     }
 | |
| }
 |