aboutsummaryrefslogtreecommitdiffstats
path: root/poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java
diff options
context:
space:
mode:
Diffstat (limited to 'poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java')
-rw-r--r--poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java1317
1 files changed, 670 insertions, 647 deletions
diff --git a/poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java b/poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java
index c71f2a8e9b..5ffa6d0191 100644
--- a/poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java
+++ b/poi-scratchpad/src/main/java/org/apache/poi/hsmf/MAPIMessage.java
@@ -58,668 +58,691 @@ import static org.apache.logging.log4j.util.Unbox.box;
/**
* Reads an Outlook MSG File in and provides hooks into its data structure.
- *
+ * <p>
* If you want to develop with HSMF, you might find it worth getting
- * some of the Microsoft public documentation, such as:
- *
+ * some of the Microsoft public documentation, such as:
+ * <p>
* [MS-OXCMSG]: Message and Attachment Object Protocol Specification
*/
public class MAPIMessage extends POIReadOnlyDocument {
- /**
- * A MAPI file can be an email (NOTE) or a number of other types
- */
- public enum MESSAGE_CLASS {
- APPOINTMENT,
- CONTACT,
- NOTE,
- POST,
- STICKY_NOTE,
- TASK,
- UNKNOWN,
- UNSPECIFIED
- }
-
- /** For logging problems we spot with the file */
- private static final Logger LOG = LogManager.getLogger(MAPIMessage.class);
+ /**
+ * A MAPI file can be an email (NOTE) or a number of other types
+ */
+ public enum MESSAGE_CLASS {
+ APPOINTMENT,
+ CONTACT,
+ NOTE,
+ POST,
+ STICKY_NOTE,
+ TASK,
+ UNKNOWN,
+ UNSPECIFIED
+ }
+
+ /**
+ * For logging problems we spot with the file
+ */
+ private static final Logger LOG = LogManager.getLogger(MAPIMessage.class);
+
+ private static final Pattern GUESS_7_BIT_ENCODING_PATTERN = Pattern.compile("content-type:.*?charset=[\"']?([^;'\"]+)[\"']?", Pattern.CASE_INSENSITIVE);
+
private Chunks mainChunks;
- private NameIdChunks nameIdChunks;
- private RecipientChunks[] recipientChunks;
- private AttachmentChunks[] attachmentChunks;
-
- private boolean returnNullOnMissingChunk;
-
- /**
- * Constructor for creating new files.
- */
- public MAPIMessage() {
- // TODO - make writing possible
- super(new POIFSFileSystem());
- }
-
-
- /**
- * Constructor for reading MSG Files from the file system.
- *
- * @param filename Name of the file to read
- * @throws IOException on errors reading, or invalid data
- * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
- * input format
- */
- public MAPIMessage(String filename) throws IOException {
- this(new File(filename));
- }
- /**
- * Constructor for reading MSG Files from the file system.
- *
- * @param file The file to read from
- * @throws IOException on errors reading, or invalid data
- * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
- * input format
- */
- public MAPIMessage(File file) throws IOException {
- this(new POIFSFileSystem(file));
- }
-
- /**
- * Constructor for reading MSG Files from an input stream.
- *
- * <p>Note - this will buffer the whole message into memory
- * in order to process. For lower memory use, use {@link #MAPIMessage(File)}
- *
- * @param in The InputStream to buffer and then read from
- * @throws IOException on errors reading, or invalid data
- * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
- * input format
- */
- public MAPIMessage(InputStream in) throws IOException {
- this(new POIFSFileSystem(in));
- }
- /**
- * Constructor for reading MSG Files from a POIFS filesystem
- *
- * @param fs Open POIFS FileSystem containing the message
- * @throws IOException on errors reading, or invalid data
- * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
- * input format
- */
- public MAPIMessage(POIFSFileSystem fs) throws IOException {
- this(fs.getRoot());
- }
- /**
- * Constructor for reading MSG Files from a certain
- * point within a POIFS filesystem
- * @param poifsDir Directory containing the message
- * @throws IOException on errors reading, or invalid data
- * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
- * input format
- */
- public MAPIMessage(DirectoryNode poifsDir) throws IOException {
- super(poifsDir);
-
- // Grab all the chunks
- ChunkGroup[] chunkGroups = POIFSChunkParser.parse(poifsDir);
-
- // Grab interesting bits
- ArrayList<AttachmentChunks> attachments = new ArrayList<>();
- ArrayList<RecipientChunks> recipients = new ArrayList<>();
- for(ChunkGroup group : chunkGroups) {
- // Should only ever be one of each of these
- if(group instanceof Chunks) {
- mainChunks = (Chunks)group;
- } else if(group instanceof NameIdChunks) {
- nameIdChunks = (NameIdChunks)group;
- } else if(group instanceof RecipientChunks) {
- recipients.add( (RecipientChunks)group );
- }
-
- // Can be multiple of these - add to list(s)
- if(group instanceof AttachmentChunks) {
- attachments.add( (AttachmentChunks)group );
- }
- }
- attachmentChunks = attachments.toArray(new AttachmentChunks[0]);
- recipientChunks = recipients.toArray(new RecipientChunks[0]);
-
- // Now sort these chunks lists so they're in ascending order,
- // rather than in random filesystem order
- Arrays.sort(attachmentChunks, new AttachmentChunksSorter());
- Arrays.sort(recipientChunks, new RecipientChunksSorter());
- }
-
-
- /**
- * Gets a string value based on the passed chunk.
- * @throws ChunkNotFoundException if the chunk isn't there
- */
- public String getStringFromChunk(StringChunk chunk) throws ChunkNotFoundException {
- if(chunk == null) {
- if(returnNullOnMissingChunk) {
- return null;
- } else {
- throw new ChunkNotFoundException();
- }
- }
- return chunk.getValue();
- }
-
-
- /**
- * Gets the plain text body of this Outlook Message
- * @return The string representation of the 'text' version of the body, if available.
- * @throws ChunkNotFoundException If the text-body chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getTextBody() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getTextBodyChunk());
- }
-
- /**
- * Gets the html body of this Outlook Message, if this email
- * contains a html version.
- * @return The string representation of the 'html' version of the body, if available.
- * @throws ChunkNotFoundException If the html-body chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getHtmlBody() throws ChunkNotFoundException {
- ByteChunk htmlBodyBinaryChunk = mainChunks.getHtmlBodyChunkBinary();
- if (htmlBodyBinaryChunk != null) {
- List<PropertyValue> cpid = mainChunks.getProperties().get(MAPIProperty.INTERNET_CPID);
- if (cpid != null && !cpid.isEmpty()) {
- int codepage = ((LongPropertyValue) cpid.get(0)).getValue();
- try {
- String encoding = CodePageUtil.codepageToEncoding(codepage, true);
- byte[] htmlBodyBinary = htmlBodyBinaryChunk.getValue();
- return new String(htmlBodyBinary, encoding);
- } catch (UnsupportedEncodingException e) {
- LOG.atWarn().log("HTML body binary: Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.INTERNET_CPID);
+ private NameIdChunks nameIdChunks;
+ private RecipientChunks[] recipientChunks;
+ private AttachmentChunks[] attachmentChunks;
+
+ private boolean returnNullOnMissingChunk;
+
+ /**
+ * Constructor for creating new files.
+ */
+ public MAPIMessage() {
+ // TODO - make writing possible
+ super(new POIFSFileSystem());
+ }
+
+
+ /**
+ * Constructor for reading MSG Files from the file system.
+ *
+ * @param filename Name of the file to read
+ * @throws IOException on errors reading, or invalid data
+ * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
+ * input format
+ */
+ public MAPIMessage(String filename) throws IOException {
+ this(new File(filename));
+ }
+
+ /**
+ * Constructor for reading MSG Files from the file system.
+ *
+ * @param file The file to read from
+ * @throws IOException on errors reading, or invalid data
+ * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
+ * input format
+ */
+ public MAPIMessage(File file) throws IOException {
+ this(new POIFSFileSystem(file));
+ }
+
+ /**
+ * Constructor for reading MSG Files from an input stream.
+ *
+ * <p>Note - this will buffer the whole message into memory
+ * in order to process. For lower memory use, use {@link #MAPIMessage(File)}
+ *
+ * @param in The InputStream to buffer and then read from
+ * @throws IOException on errors reading, or invalid data
+ * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
+ * input format
+ */
+ public MAPIMessage(InputStream in) throws IOException {
+ this(new POIFSFileSystem(in));
+ }
+
+ /**
+ * Constructor for reading MSG Files from a POIFS filesystem
+ *
+ * @param fs Open POIFS FileSystem containing the message
+ * @throws IOException on errors reading, or invalid data
+ * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
+ * input format
+ */
+ public MAPIMessage(POIFSFileSystem fs) throws IOException {
+ this(fs.getRoot());
+ }
+
+ /**
+ * Constructor for reading MSG Files from a certain
+ * point within a POIFS filesystem
+ *
+ * @param poifsDir Directory containing the message
+ * @throws IOException on errors reading, or invalid data
+ * @throws RuntimeException a number of runtime exceptions can be thrown, especially if there are problems with the
+ * input format
+ */
+ public MAPIMessage(DirectoryNode poifsDir) throws IOException {
+ super(poifsDir);
+
+ // Grab all the chunks
+ ChunkGroup[] chunkGroups = POIFSChunkParser.parse(poifsDir);
+
+ // Grab interesting bits
+ ArrayList<AttachmentChunks> attachments = new ArrayList<>();
+ ArrayList<RecipientChunks> recipients = new ArrayList<>();
+ for (ChunkGroup group : chunkGroups) {
+ // Should only ever be one of each of these
+ if (group instanceof Chunks) {
+ mainChunks = (Chunks) group;
+ } else if (group instanceof NameIdChunks) {
+ nameIdChunks = (NameIdChunks) group;
+ } else if (group instanceof RecipientChunks) {
+ recipients.add((RecipientChunks) group);
}
- }
- return htmlBodyBinaryChunk.getAs7bitString();
- }
- return getStringFromChunk(mainChunks.getHtmlBodyChunkString());
- }
-
- /**
- * Gets the RTF Rich Message body of this Outlook Message, if this email
- * contains a RTF (rich) version.
- * @return The string representation of the 'RTF' version of the body, if available.
- * @throws ChunkNotFoundException If the rtf-body chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getRtfBody() throws ChunkNotFoundException {
- ByteChunk chunk = mainChunks.getRtfBodyChunk();
- if(chunk == null) {
- if(returnNullOnMissingChunk) {
- return null;
- } else {
- throw new ChunkNotFoundException();
- }
- }
-
- try {
- MAPIRtfAttribute rtf = new MAPIRtfAttribute(
- MAPIProperty.RTF_COMPRESSED, Types.BINARY.getId(), chunk.getValue()
- );
- return rtf.getDataString();
- } catch(IOException e) {
- throw new RuntimeException("Shouldn't happen", e);
- }
- }
-
- /**
- * Gets the subject line of the Outlook Message
- * @throws ChunkNotFoundException If the subject-chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getSubject() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getSubjectChunk());
- }
-
- /**
- * Gets the display value of the "FROM" line of the outlook message
- * This is not the actual address that was sent from but the formatted display of the user name.
- * @throws ChunkNotFoundException If the from-chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getDisplayFrom() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getDisplayFromChunk());
- }
-
- /**
- * Gets the display value of the "TO" line of the outlook message.
- * If there are multiple recipients, they will be separated
- * by semicolons.
- * This is not the actual list of addresses/values that will be
- * sent to if you click Reply in the email - those are stored
- * in {@link RecipientChunks}.
- * @throws ChunkNotFoundException If the to-chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getDisplayTo() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getDisplayToChunk());
- }
-
- /**
- * Gets the display value of the "CC" line of the outlook message.
- * If there are multiple recipients, they will be separated
- * by semicolons.
- * This is not the actual list of addresses/values that will be
- * sent to if you click Reply in the email - those are stored
- * in {@link RecipientChunks}.
- * @throws ChunkNotFoundException If the cc-chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getDisplayCC() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getDisplayCCChunk());
- }
-
- /**
- * Gets the display value of the "BCC" line of the outlook message.
- * If there are multiple recipients, they will be separated
- * by semicolons.
- * This is not the actual list of addresses/values that will be
- * sent to if you click Reply in the email - those are stored
- * in {@link RecipientChunks}.
- * This will only be present in sent emails, not received ones!
- * @throws ChunkNotFoundException If the bcc-chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getDisplayBCC() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getDisplayBCCChunk());
- }
-
- /**
- * Returns all the recipients' email address, separated by
- * semicolons. Checks all the likely chunks in search of
- * the addresses.
- */
- public String getRecipientEmailAddress() throws ChunkNotFoundException {
- return toSemicolonList(getRecipientEmailAddressList());
- }
- /**
- * Returns an array of all the recipient's email address, normally
- * in TO then CC then BCC order.
- * Checks all the likely chunks in search of the addresses.
- */
- public String[] getRecipientEmailAddressList() throws ChunkNotFoundException {
- if(recipientChunks == null || recipientChunks.length == 0) {
- throw new ChunkNotFoundException("No recipients section present");
- }
-
- String[] emails = new String[recipientChunks.length];
- for(int i=0; i<emails.length; i++) {
- RecipientChunks rc = recipientChunks[i];
- String email = rc.getRecipientEmailAddress();
- if(email != null) {
- emails[i] = email;
- } else {
- if(returnNullOnMissingChunk) {
- emails[i] = null;
+
+ // Can be multiple of these - add to list(s)
+ if (group instanceof AttachmentChunks) {
+ attachments.add((AttachmentChunks) group);
+ }
+ }
+ attachmentChunks = attachments.toArray(new AttachmentChunks[0]);
+ recipientChunks = recipients.toArray(new RecipientChunks[0]);
+
+ // Now sort these chunks lists so they're in ascending order,
+ // rather than in random filesystem order
+ Arrays.sort(attachmentChunks, new AttachmentChunksSorter());
+ Arrays.sort(recipientChunks, new RecipientChunksSorter());
+ }
+
+
+ /**
+ * Gets a string value based on the passed chunk.
+ *
+ * @throws ChunkNotFoundException if the chunk isn't there
+ */
+ public String getStringFromChunk(StringChunk chunk) throws ChunkNotFoundException {
+ if (chunk == null) {
+ if (returnNullOnMissingChunk) {
+ return null;
} else {
- throw new ChunkNotFoundException("No email address holding chunks found for the " + (i+1) + "th recipient");
+ throw new ChunkNotFoundException();
}
- }
- }
-
- return emails;
- }
-
-
- /**
- * Returns all the recipients' names, separated by
- * semicolons. Checks all the likely chunks in search of
- * the names.
- * See also {@link #getDisplayTo()}, {@link #getDisplayCC()}
- * and {@link #getDisplayBCC()}.
- */
- public String getRecipientNames() throws ChunkNotFoundException {
- return toSemicolonList(getRecipientNamesList());
- }
- /**
- * Returns an array of all the recipient's names, normally
- * in TO then CC then BCC order.
- * Checks all the likely chunks in search of the names.
- * See also {@link #getDisplayTo()}, {@link #getDisplayCC()}
- * and {@link #getDisplayBCC()}.
- */
- public String[] getRecipientNamesList() throws ChunkNotFoundException {
- if(recipientChunks == null || recipientChunks.length == 0) {
- throw new ChunkNotFoundException("No recipients section present");
- }
-
- String[] names = new String[recipientChunks.length];
- for(int i=0; i<names.length; i++) {
- RecipientChunks rc = recipientChunks[i];
- String name = rc.getRecipientName();
- if(name != null) {
- names[i] = name;
- } else {
- throw new ChunkNotFoundException("No display name holding chunks found for the " + (i+1) + "th recipient");
- }
- }
-
- return names;
- }
-
- /**
- * Tries to identify the correct encoding for 7-bit (non-unicode)
- * strings in the file.
- * <p>Many messages store their strings as unicode, which is
- * nice and easy. Some use one-byte encodings for their
- * strings, but don't always store the encoding anywhere
- * helpful in the file.</p>
- * <p>This method checks for codepage properties, and failing that
- * looks at the headers for the message, and uses these to
- * guess the correct encoding for your file.</p>
- * <p>Bug #49441 has more on why this is needed</p>
- */
- public void guess7BitEncoding() {
- String generalcodepage = null;
- String htmlbodycodepage = null;
- String bodycodepage = null;
- //
- // General codepage: Message codepage property.
- //
- List<PropertyValue> val = mainChunks.getProperties().get(MAPIProperty.MESSAGE_CODEPAGE);
- if (val != null && !val.isEmpty()) {
- int codepage = ((LongPropertyValue) val.get(0)).getValue();
- try {
- String encoding = CodePageUtil.codepageToEncoding(codepage, true);
- generalcodepage = encoding;
- } catch (UnsupportedEncodingException e) {
- LOG.atWarn().log("Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.MESSAGE_CODEPAGE);
- }
- }
- //
- // General codepage fallback: Message locale ID property.
- //
- if (generalcodepage == null) {
- val = mainChunks.getProperties().get(MAPIProperty.MESSAGE_LOCALE_ID);
- if (val != null && !val.isEmpty()) {
- int lcid = ((LongPropertyValue) val.get(0)).getValue();
- int codepage = LocaleUtil.getDefaultCodePageFromLCID(lcid);
- try {
- if (codepage != 0) {
- String encoding = CodePageUtil.codepageToEncoding(codepage, true);
- generalcodepage = encoding;
- }
- } catch (UnsupportedEncodingException e) {
- LOG.atWarn().log("Invalid codepage ID {}from locale ID{} set for the message via {}, ignoring", box(codepage),box(lcid), MAPIProperty.MESSAGE_LOCALE_ID);
- }
- }
- }
- //
- // General codepage fallback: Charset on a content type header.
- //
- if (generalcodepage == null) {
- try {
- String[] headers = getHeaders();
- if (headers != null && headers.length > 0) {
- Pattern p = Pattern.compile("content-type:.*?charset=[\"']?([^;'\"]+)[\"']?", Pattern.CASE_INSENSITIVE);
- for (String header : headers) {
- if (header.toLowerCase(LocaleUtil.getUserLocale()).startsWith("content-type")) {
- Matcher m = p.matcher(header);
- if (m.matches()) {
- String encoding = m.group(1);
- generalcodepage = encoding;
- }
- }
- }
- }
- } catch (ChunkNotFoundException e) {
- }
- }
- //
- // HTML and text body encoding: Internet CPID property.
- // UTF-8 is ignored for text body. This seems to be a special Outlook behavior.
- //
- val = mainChunks.getProperties().get(MAPIProperty.INTERNET_CPID);
- if (val != null && !val.isEmpty()) {
- int codepage = ((LongPropertyValue) val.get(0)).getValue();
- try {
- String encoding = CodePageUtil.codepageToEncoding(codepage, true);
- htmlbodycodepage = encoding;
- if (!encoding.equalsIgnoreCase("utf-8")) {
- bodycodepage = encoding;
- }
- } catch (UnsupportedEncodingException e) {
- LOG.atWarn().log("Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.INTERNET_CPID);
- }
- }
- //
- // Apply encoding
- //
- set7BitEncoding(generalcodepage, htmlbodycodepage, bodycodepage);
- }
-
- /**
- * Many messages store their strings as unicode, which is
- * nice and easy. Some use one-byte encodings for their
- * strings, but don't easily store the encoding anywhere
- * in the file!
- * If you know what the encoding is of your file, you can
- * use this method to set the 7 bit encoding for all
- * the non unicode strings in the file.
- * @see #guess7BitEncoding()
- */
- public void set7BitEncoding(String charset) {
- set7BitEncoding(charset, charset, charset);
- }
- public void set7BitEncoding(String generalcharset, String htmlbodycharset, String bodycharset) {
- for(Chunk c : mainChunks.getChunks()) {
- if(c instanceof StringChunk) {
- if (c.getChunkId() == MAPIProperty.BODY_HTML.id) {
- if (htmlbodycharset != null) {
- ((StringChunk)c).set7BitEncoding(htmlbodycharset);
- }
- }
- else if (c.getChunkId() == MAPIProperty.BODY.id) {
- if (bodycharset != null) {
- ((StringChunk)c).set7BitEncoding(bodycharset);
- }
- }
- else if (generalcharset != null) {
- ((StringChunk)c).set7BitEncoding(generalcharset);
- }
- }
- }
- if (generalcharset != null) {
- if (nameIdChunks!=null) {
- for(Chunk c : nameIdChunks.getChunks()) {
- if(c instanceof StringChunk) {
- ((StringChunk)c).set7BitEncoding(generalcharset);
- }
- }
}
- for(RecipientChunks rc : recipientChunks) {
- for(Chunk c : rc.getAll()) {
- if(c instanceof StringChunk) {
- ((StringChunk)c).set7BitEncoding(generalcharset);
- }
- }
+ return chunk.getValue();
+ }
+
+
+ /**
+ * Gets the plain text body of this Outlook Message
+ *
+ * @return The string representation of the 'text' version of the body, if available.
+ * @throws ChunkNotFoundException If the text-body chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getTextBody() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getTextBodyChunk());
+ }
+
+ /**
+ * Gets the html body of this Outlook Message, if this email
+ * contains a html version.
+ *
+ * @return The string representation of the 'html' version of the body, if available.
+ * @throws ChunkNotFoundException If the html-body chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getHtmlBody() throws ChunkNotFoundException {
+ ByteChunk htmlBodyBinaryChunk = mainChunks.getHtmlBodyChunkBinary();
+ if (htmlBodyBinaryChunk != null) {
+ List<PropertyValue> cpid = mainChunks.getProperties().get(MAPIProperty.INTERNET_CPID);
+ if (cpid != null && !cpid.isEmpty()) {
+ int codepage = ((LongPropertyValue) cpid.get(0)).getValue();
+ try {
+ String encoding = CodePageUtil.codepageToEncoding(codepage, true);
+ byte[] htmlBodyBinary = htmlBodyBinaryChunk.getValue();
+ return new String(htmlBodyBinary, encoding);
+ } catch (UnsupportedEncodingException e) {
+ LOG.atWarn().log("HTML body binary: Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.INTERNET_CPID);
+ }
+ }
+ return htmlBodyBinaryChunk.getAs7bitString();
}
- }
- }
-
- /**
- * Does this file contain any strings that
- * are stored as 7 bit rather than unicode?
- */
- public boolean has7BitEncodingStrings() {
- for(Chunk c : mainChunks.getChunks()) {
- if(c instanceof StringChunk) {
- if( c.getType() == Types.ASCII_STRING ) {
- return true;
+ return getStringFromChunk(mainChunks.getHtmlBodyChunkString());
+ }
+
+ /**
+ * Gets the RTF Rich Message body of this Outlook Message, if this email
+ * contains a RTF (rich) version.
+ *
+ * @return The string representation of the 'RTF' version of the body, if available.
+ * @throws ChunkNotFoundException If the rtf-body chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getRtfBody() throws ChunkNotFoundException {
+ ByteChunk chunk = mainChunks.getRtfBodyChunk();
+ if (chunk == null) {
+ if (returnNullOnMissingChunk) {
+ return null;
+ } else {
+ throw new ChunkNotFoundException();
}
- }
- }
-
- if (nameIdChunks!=null) {
- for(Chunk c : nameIdChunks.getChunks()) {
- if(c instanceof StringChunk) {
- if( c.getType() == Types.ASCII_STRING ) {
- return true;
- }
+ }
+
+ try {
+ MAPIRtfAttribute rtf = new MAPIRtfAttribute(
+ MAPIProperty.RTF_COMPRESSED, Types.BINARY.getId(), chunk.getValue()
+ );
+ return rtf.getDataString();
+ } catch (IOException e) {
+ throw new RuntimeException("Shouldn't happen", e);
+ }
+ }
+
+ /**
+ * Gets the subject line of the Outlook Message
+ *
+ * @throws ChunkNotFoundException If the subject-chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getSubject() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getSubjectChunk());
+ }
+
+ /**
+ * Gets the display value of the "FROM" line of the outlook message
+ * This is not the actual address that was sent from but the formatted display of the user name.
+ *
+ * @throws ChunkNotFoundException If the from-chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getDisplayFrom() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getDisplayFromChunk());
+ }
+
+ /**
+ * Gets the display value of the "TO" line of the outlook message.
+ * If there are multiple recipients, they will be separated
+ * by semicolons.
+ * This is not the actual list of addresses/values that will be
+ * sent to if you click Reply in the email - those are stored
+ * in {@link RecipientChunks}.
+ *
+ * @throws ChunkNotFoundException If the to-chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getDisplayTo() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getDisplayToChunk());
+ }
+
+ /**
+ * Gets the display value of the "CC" line of the outlook message.
+ * If there are multiple recipients, they will be separated
+ * by semicolons.
+ * This is not the actual list of addresses/values that will be
+ * sent to if you click Reply in the email - those are stored
+ * in {@link RecipientChunks}.
+ *
+ * @throws ChunkNotFoundException If the cc-chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getDisplayCC() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getDisplayCCChunk());
+ }
+
+ /**
+ * Gets the display value of the "BCC" line of the outlook message.
+ * If there are multiple recipients, they will be separated
+ * by semicolons.
+ * This is not the actual list of addresses/values that will be
+ * sent to if you click Reply in the email - those are stored
+ * in {@link RecipientChunks}.
+ * This will only be present in sent emails, not received ones!
+ *
+ * @throws ChunkNotFoundException If the bcc-chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getDisplayBCC() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getDisplayBCCChunk());
+ }
+
+ /**
+ * Returns all the recipients' email address, separated by
+ * semicolons. Checks all the likely chunks in search of
+ * the addresses.
+ */
+ public String getRecipientEmailAddress() throws ChunkNotFoundException {
+ return toSemicolonList(getRecipientEmailAddressList());
+ }
+
+ /**
+ * Returns an array of all the recipient's email address, normally
+ * in TO then CC then BCC order.
+ * Checks all the likely chunks in search of the addresses.
+ */
+ public String[] getRecipientEmailAddressList() throws ChunkNotFoundException {
+ if (recipientChunks == null || recipientChunks.length == 0) {
+ throw new ChunkNotFoundException("No recipients section present");
+ }
+
+ String[] emails = new String[recipientChunks.length];
+ for (int i = 0; i < emails.length; i++) {
+ RecipientChunks rc = recipientChunks[i];
+ String email = rc.getRecipientEmailAddress();
+ if (email != null) {
+ emails[i] = email;
+ } else {
+ if (returnNullOnMissingChunk) {
+ emails[i] = null;
+ } else {
+ throw new ChunkNotFoundException("No email address holding chunks found for the " + (i + 1) + "th recipient");
+ }
}
- }
- }
-
- for(RecipientChunks rc : recipientChunks) {
- for(Chunk c : rc.getAll()) {
- if(c instanceof StringChunk) {
- if( c.getType() == Types.ASCII_STRING ) {
- return true;
- }
+ }
+
+ return emails;
+ }
+
+
+ /**
+ * Returns all the recipients' names, separated by
+ * semicolons. Checks all the likely chunks in search of
+ * the names.
+ * See also {@link #getDisplayTo()}, {@link #getDisplayCC()}
+ * and {@link #getDisplayBCC()}.
+ */
+ public String getRecipientNames() throws ChunkNotFoundException {
+ return toSemicolonList(getRecipientNamesList());
+ }
+
+ /**
+ * Returns an array of all the recipient's names, normally
+ * in TO then CC then BCC order.
+ * Checks all the likely chunks in search of the names.
+ * See also {@link #getDisplayTo()}, {@link #getDisplayCC()}
+ * and {@link #getDisplayBCC()}.
+ */
+ public String[] getRecipientNamesList() throws ChunkNotFoundException {
+ if (recipientChunks == null || recipientChunks.length == 0) {
+ throw new ChunkNotFoundException("No recipients section present");
+ }
+
+ String[] names = new String[recipientChunks.length];
+ for (int i = 0; i < names.length; i++) {
+ RecipientChunks rc = recipientChunks[i];
+ String name = rc.getRecipientName();
+ if (name != null) {
+ names[i] = name;
+ } else {
+ throw new ChunkNotFoundException("No display name holding chunks found for the " + (i + 1) + "th recipient");
}
- }
- }
- return false;
- }
-
- /**
- * Returns all the headers, one entry per line
- */
- public String[] getHeaders() throws ChunkNotFoundException {
- String headers = getStringFromChunk(mainChunks.getMessageHeaders());
- if(headers == null) {
- return null;
- }
- return headers.split("\\r?\\n");
- }
-
- /**
- * Gets the conversation topic of the parsed Outlook Message.
- * This is the part of the subject line that is after the RE: and FWD:
- * @throws ChunkNotFoundException If the conversation-topic chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public String getConversationTopic() throws ChunkNotFoundException {
- return getStringFromChunk(mainChunks.getConversationTopic());
- }
-
- /**
- * Gets the message class of the parsed Outlook Message.
- * (Yes, you can use this to determine if a message is a calendar
- * item, note, or actual outlook Message)
- * For emails the class will be IPM.Note
- *
- * @throws ChunkNotFoundException If the message-class chunk does not exist and
- * returnNullOnMissingChunk is set
- */
- public MESSAGE_CLASS getMessageClassEnum() throws ChunkNotFoundException {
- String mc = getStringFromChunk(mainChunks.getMessageClass());
- if (mc == null || mc.trim().length() == 0) {
- return MESSAGE_CLASS.UNSPECIFIED;
- } else if (mc.equalsIgnoreCase("IPM.Note")) {
- return MESSAGE_CLASS.NOTE;
- } else if (mc.equalsIgnoreCase("IPM.Contact")) {
- return MESSAGE_CLASS.CONTACT;
- } else if (mc.equalsIgnoreCase("IPM.Appointment")) {
- return MESSAGE_CLASS.APPOINTMENT;
- } else if (mc.equalsIgnoreCase("IPM.StickyNote")) {
- return MESSAGE_CLASS.STICKY_NOTE;
- } else if (mc.equalsIgnoreCase("IPM.Task")) {
- return MESSAGE_CLASS.TASK;
- } else if (mc.equalsIgnoreCase("IPM.Post")) {
- return MESSAGE_CLASS.POST;
- } else {
- LOG.atWarn().log("I don't recognize message class '{}'. Please open an issue on POI's bugzilla", mc);
- return MESSAGE_CLASS.UNKNOWN;
- }
- }
- /**
- * Gets the date that the message was accepted by the
- * server on.
- */
- public Calendar getMessageDate() throws ChunkNotFoundException {
- if (mainChunks.getSubmissionChunk() != null) {
- return mainChunks.getSubmissionChunk().getAcceptedAtTime();
- }
- else {
- // Try a few likely suspects...
- for (MAPIProperty prop : new MAPIProperty[] {
- MAPIProperty.CLIENT_SUBMIT_TIME, MAPIProperty.LAST_MODIFICATION_TIME,
- MAPIProperty.CREATION_TIME
- }) {
- List<PropertyValue> val = mainChunks.getProperties().get(prop);
+ }
+
+ return names;
+ }
+
+ /**
+ * Tries to identify the correct encoding for 7-bit (non-unicode)
+ * strings in the file.
+ * <p>Many messages store their strings as unicode, which is
+ * nice and easy. Some use one-byte encodings for their
+ * strings, but don't always store the encoding anywhere
+ * helpful in the file.</p>
+ * <p>This method checks for codepage properties, and failing that
+ * looks at the headers for the message, and uses these to
+ * guess the correct encoding for your file.</p>
+ * <p>Bug #49441 has more on why this is needed</p>
+ */
+ public void guess7BitEncoding() {
+ String generalcodepage = null;
+ String htmlbodycodepage = null;
+ String bodycodepage = null;
+ //
+ // General codepage: Message codepage property.
+ //
+ List<PropertyValue> val = mainChunks.getProperties().get(MAPIProperty.MESSAGE_CODEPAGE);
+ if (val != null && !val.isEmpty()) {
+ int codepage = ((LongPropertyValue) val.get(0)).getValue();
+ try {
+ String encoding = CodePageUtil.codepageToEncoding(codepage, true);
+ generalcodepage = encoding;
+ } catch (UnsupportedEncodingException e) {
+ LOG.atWarn().log("Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.MESSAGE_CODEPAGE);
+ }
+ }
+ //
+ // General codepage fallback: Message locale ID property.
+ //
+ if (generalcodepage == null) {
+ val = mainChunks.getProperties().get(MAPIProperty.MESSAGE_LOCALE_ID);
if (val != null && !val.isEmpty()) {
- return ((TimePropertyValue)val.get(0)).getValue();
+ int lcid = ((LongPropertyValue) val.get(0)).getValue();
+ int codepage = LocaleUtil.getDefaultCodePageFromLCID(lcid);
+ try {
+ if (codepage != 0) {
+ String encoding = CodePageUtil.codepageToEncoding(codepage, true);
+ generalcodepage = encoding;
+ }
+ } catch (UnsupportedEncodingException e) {
+ LOG.atWarn().log("Invalid codepage ID {}from locale ID{} set for the message via {}, ignoring", box(codepage), box(lcid), MAPIProperty.MESSAGE_LOCALE_ID);
+ }
}
- }
- }
-
- if(returnNullOnMissingChunk)
- return null;
- throw new ChunkNotFoundException();
- }
-
-
- /**
- * Gets the main, core details chunks
- */
- public Chunks getMainChunks() {
- return mainChunks;
- }
- /**
- * Gets all the recipient details chunks.
- * These will normally be in the order of:
- * * TO recipients, in the order returned by {@link #getDisplayTo()}
- * * CC recipients, in the order returned by {@link #getDisplayCC()}
- * * BCC recipients, in the order returned by {@link #getDisplayBCC()}
- */
- public RecipientChunks[] getRecipientDetailsChunks() {
- return recipientChunks;
- }
- /**
- * Gets the Name ID chunks, or
- * null if there aren't any
- */
- public NameIdChunks getNameIdChunks() {
- return nameIdChunks;
- }
- /**
- * Gets the message attachments.
- */
- public AttachmentChunks[] getAttachmentFiles() {
- return attachmentChunks;
- }
-
-
- /**
- * Will you get a null on a missing chunk, or a
- * {@link ChunkNotFoundException} (default is the
- * exception).
- */
- public boolean isReturnNullOnMissingChunk() {
- return returnNullOnMissingChunk;
- }
-
- /**
- * Sets whether on asking for a missing chunk,
- * you get back null or a {@link ChunkNotFoundException}
- * (default is the exception).
- */
- public void setReturnNullOnMissingChunk(boolean returnNullOnMissingChunk) {
- this.returnNullOnMissingChunk = returnNullOnMissingChunk;
- }
-
-
- private String toSemicolonList(String[] l) {
- StringBuilder list = new StringBuilder();
- boolean first = true;
-
- for(String s : l) {
- if(s == null) continue;
- if(first) {
- first = false;
- } else {
- list.append("; ");
- }
- list.append(s);
- }
-
- return list.toString();
- }
+ }
+ //
+ // General codepage fallback: Charset on a content type header.
+ //
+ if (generalcodepage == null) {
+ try {
+ String[] headers = getHeaders();
+ if (headers != null && headers.length > 0) {
+ for (String header : headers) {
+ if (header.toLowerCase(LocaleUtil.getUserLocale()).startsWith("content-type")) {
+ Matcher m = GUESS_7_BIT_ENCODING_PATTERN.matcher(header);
+ if (m.matches()) {
+ String encoding = m.group(1);
+ generalcodepage = encoding;
+ }
+ }
+ }
+ }
+ } catch (ChunkNotFoundException e) {
+ }
+ }
+ //
+ // HTML and text body encoding: Internet CPID property.
+ // UTF-8 is ignored for text body. This seems to be a special Outlook behavior.
+ //
+ val = mainChunks.getProperties().get(MAPIProperty.INTERNET_CPID);
+ if (val != null && !val.isEmpty()) {
+ int codepage = ((LongPropertyValue) val.get(0)).getValue();
+ try {
+ String encoding = CodePageUtil.codepageToEncoding(codepage, true);
+ htmlbodycodepage = encoding;
+ if (!encoding.equalsIgnoreCase("utf-8")) {
+ bodycodepage = encoding;
+ }
+ } catch (UnsupportedEncodingException e) {
+ LOG.atWarn().log("Invalid codepage ID {} set for the message via {}, ignoring", box(codepage), MAPIProperty.INTERNET_CPID);
+ }
+ }
+ //
+ // Apply encoding
+ //
+ set7BitEncoding(generalcodepage, htmlbodycodepage, bodycodepage);
+ }
+
+ /**
+ * Many messages store their strings as unicode, which is
+ * nice and easy. Some use one-byte encodings for their
+ * strings, but don't easily store the encoding anywhere
+ * in the file!
+ * If you know what the encoding is of your file, you can
+ * use this method to set the 7 bit encoding for all
+ * the non unicode strings in the file.
+ *
+ * @see #guess7BitEncoding()
+ */
+ public void set7BitEncoding(String charset) {
+ set7BitEncoding(charset, charset, charset);
+ }
+
+ public void set7BitEncoding(String generalcharset, String htmlbodycharset, String bodycharset) {
+ for (Chunk c : mainChunks.getChunks()) {
+ if (c instanceof StringChunk) {
+ if (c.getChunkId() == MAPIProperty.BODY_HTML.id) {
+ if (htmlbodycharset != null) {
+ ((StringChunk) c).set7BitEncoding(htmlbodycharset);
+ }
+ } else if (c.getChunkId() == MAPIProperty.BODY.id) {
+ if (bodycharset != null) {
+ ((StringChunk) c).set7BitEncoding(bodycharset);
+ }
+ } else if (generalcharset != null) {
+ ((StringChunk) c).set7BitEncoding(generalcharset);
+ }
+ }
+ }
+ if (generalcharset != null) {
+ if (nameIdChunks != null) {
+ for (Chunk c : nameIdChunks.getChunks()) {
+ if (c instanceof StringChunk) {
+ ((StringChunk) c).set7BitEncoding(generalcharset);
+ }
+ }
+ }
+ for (RecipientChunks rc : recipientChunks) {
+ for (Chunk c : rc.getAll()) {
+ if (c instanceof StringChunk) {
+ ((StringChunk) c).set7BitEncoding(generalcharset);
+ }
+ }
+ }
+ }
+ }
+
+ /**
+ * Does this file contain any strings that
+ * are stored as 7 bit rather than unicode?
+ */
+ public boolean has7BitEncodingStrings() {
+ for (Chunk c : mainChunks.getChunks()) {
+ if (c instanceof StringChunk) {
+ if (c.getType() == Types.ASCII_STRING) {
+ return true;
+ }
+ }
+ }
+
+ if (nameIdChunks != null) {
+ for (Chunk c : nameIdChunks.getChunks()) {
+ if (c instanceof StringChunk) {
+ if (c.getType() == Types.ASCII_STRING) {
+ return true;
+ }
+ }
+ }
+ }
+
+ for (RecipientChunks rc : recipientChunks) {
+ for (Chunk c : rc.getAll()) {
+ if (c instanceof StringChunk) {
+ if (c.getType() == Types.ASCII_STRING) {
+ return true;
+ }
+ }
+ }
+ }
+ return false;
+ }
+
+ /**
+ * Returns all the headers, one entry per line
+ */
+ public String[] getHeaders() throws ChunkNotFoundException {
+ String headers = getStringFromChunk(mainChunks.getMessageHeaders());
+ if (headers == null) {
+ return null;
+ }
+ return headers.split("\\r?\\n");
+ }
+
+ /**
+ * Gets the conversation topic of the parsed Outlook Message.
+ * This is the part of the subject line that is after the RE: and FWD:
+ *
+ * @throws ChunkNotFoundException If the conversation-topic chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public String getConversationTopic() throws ChunkNotFoundException {
+ return getStringFromChunk(mainChunks.getConversationTopic());
+ }
+
+ /**
+ * Gets the message class of the parsed Outlook Message.
+ * (Yes, you can use this to determine if a message is a calendar
+ * item, note, or actual outlook Message)
+ * For emails the class will be IPM.Note
+ *
+ * @throws ChunkNotFoundException If the message-class chunk does not exist and
+ * returnNullOnMissingChunk is set
+ */
+ public MESSAGE_CLASS getMessageClassEnum() throws ChunkNotFoundException {
+ String mc = getStringFromChunk(mainChunks.getMessageClass());
+ if (mc == null || mc.trim().length() == 0) {
+ return MESSAGE_CLASS.UNSPECIFIED;
+ } else if (mc.equalsIgnoreCase("IPM.Note")) {
+ return MESSAGE_CLASS.NOTE;
+ } else if (mc.equalsIgnoreCase("IPM.Contact")) {
+ return MESSAGE_CLASS.CONTACT;
+ } else if (mc.equalsIgnoreCase("IPM.Appointment")) {
+ return MESSAGE_CLASS.APPOINTMENT;
+ } else if (mc.equalsIgnoreCase("IPM.StickyNote")) {
+ return MESSAGE_CLASS.STICKY_NOTE;
+ } else if (mc.equalsIgnoreCase("IPM.Task")) {
+ return MESSAGE_CLASS.TASK;
+ } else if (mc.equalsIgnoreCase("IPM.Post")) {
+ return MESSAGE_CLASS.POST;
+ } else {
+ LOG.atWarn().log("I don't recognize message class '{}'. Please open an issue on POI's bugzilla", mc);
+ return MESSAGE_CLASS.UNKNOWN;
+ }
+ }
+
+ /**
+ * Gets the date that the message was accepted by the
+ * server on.
+ */
+ public Calendar getMessageDate() throws ChunkNotFoundException {
+ if (mainChunks.getSubmissionChunk() != null) {
+ return mainChunks.getSubmissionChunk().getAcceptedAtTime();
+ } else {
+ // Try a few likely suspects...
+ for (MAPIProperty prop : new MAPIProperty[]{
+ MAPIProperty.CLIENT_SUBMIT_TIME, MAPIProperty.LAST_MODIFICATION_TIME,
+ MAPIProperty.CREATION_TIME
+ }) {
+ List<PropertyValue> val = mainChunks.getProperties().get(prop);
+ if (val != null && !val.isEmpty()) {
+ return ((TimePropertyValue) val.get(0)).getValue();
+ }
+ }
+ }
+
+ if (returnNullOnMissingChunk)
+ return null;
+ throw new ChunkNotFoundException();
+ }
+
+
+ /**
+ * Gets the main, core details chunks
+ */
+ public Chunks getMainChunks() {
+ return mainChunks;
+ }
+
+ /**
+ * Gets all the recipient details chunks.
+ * These will normally be in the order of:
+ * * TO recipients, in the order returned by {@link #getDisplayTo()}
+ * * CC recipients, in the order returned by {@link #getDisplayCC()}
+ * * BCC recipients, in the order returned by {@link #getDisplayBCC()}
+ */
+ public RecipientChunks[] getRecipientDetailsChunks() {
+ return recipientChunks;
+ }
+
+ /**
+ * Gets the Name ID chunks, or
+ * null if there aren't any
+ */
+ public NameIdChunks getNameIdChunks() {
+ return nameIdChunks;
+ }
+
+ /**
+ * Gets the message attachments.
+ */
+ public AttachmentChunks[] getAttachmentFiles() {
+ return attachmentChunks;
+ }
+
+
+ /**
+ * Will you get a null on a missing chunk, or a
+ * {@link ChunkNotFoundException} (default is the
+ * exception).
+ */
+ public boolean isReturnNullOnMissingChunk() {
+ return returnNullOnMissingChunk;
+ }
+
+ /**
+ * Sets whether on asking for a missing chunk,
+ * you get back null or a {@link ChunkNotFoundException}
+ * (default is the exception).
+ */
+ public void setReturnNullOnMissingChunk(boolean returnNullOnMissingChunk) {
+ this.returnNullOnMissingChunk = returnNullOnMissingChunk;
+ }
+
+
+ private String toSemicolonList(String[] l) {
+ StringBuilder list = new StringBuilder();
+ boolean first = true;
+
+ for (String s : l) {
+ if (s == null) continue;
+ if (first) {
+ first = false;
+ } else {
+ list.append("; ");
+ }
+ list.append(s);
+ }
+
+ return list.toString();
+ }
}