diff options
Diffstat (limited to 'net-mail/exim')
-rw-r--r-- | net-mail/exim/Manifest | 15 | ||||
-rw-r--r-- | net-mail/exim/files/exiscan-acl-4.20-09.patch | 6707 |
2 files changed, 7 insertions, 6715 deletions
diff --git a/net-mail/exim/Manifest b/net-mail/exim/Manifest index f230c0738f51..97df6c148bcc 100644 --- a/net-mail/exim/Manifest +++ b/net-mail/exim/Manifest @@ -3,14 +3,13 @@ MD5 b4546401419ab569939d6a7b97b117a0 exim-4.10.ebuild 6209 MD5 552b6ba253f6d49c3ae0a92e3c867271 exim-4.12.ebuild 5436 MD5 cb909c0ba85b24cd0c917682f780bee7 exim-4.14.ebuild 5483 MD5 6a279805f089c006ac5a0c242d68d86c exim-4.20-r1.ebuild 6235 -MD5 3cf81a767f62a64891a39ee698821f05 exim-4.20-r2.ebuild 6328 MD5 fb49b47a884b048542837810acb08633 exim-4.20.ebuild 5569 +MD5 ea63797589f6933c74605b04fe13c299 metadata.xml 725 +MD5 3cf81a767f62a64891a39ee698821f05 exim-4.20-r2.ebuild 6328 MD5 76850d186609333f5b5858a5d512a364 exim-4.21.ebuild 6141 MD5 e5754fa101873ae8932f66793cc50626 exim-4.22.ebuild 6356 -MD5 ea63797589f6933c74605b04fe13c299 metadata.xml 725 MD5 1cbb9e2c12ce43f00937397e98cdac76 exim-4.24-r1.ebuild 6365 MD5 5a76bab84c947dc380e3ce396016ea35 exim-4.24.ebuild 6356 -MD5 117455c076e246e7c02d5fa464dd4be7 files/digest-exim-4.24 132 MD5 68fc403ba2c98ccba281939085cb9052 files/auth_conf.sub 775 MD5 0cb2ffe88c81fd7ac8429b5a19d58b35 files/configure 16346 MD5 34def7b529693beda5b84c422d25a878 files/digest-exim-4.10 128 @@ -18,17 +17,17 @@ MD5 100ab1bbdfea50083ecce4bf78af5cc9 files/digest-exim-4.12 129 MD5 091d3610d9a2a0b879f543cc070ef933 files/digest-exim-4.14 128 MD5 04ba8d65b73d23eb58c6d3e4accbed7f files/digest-exim-4.20 198 MD5 04ba8d65b73d23eb58c6d3e4accbed7f files/digest-exim-4.20-r1 198 -MD5 0cc15aecabaeace6ff4cdd1a462b9eb0 files/digest-exim-4.20-r2 62 -MD5 e62c9a744fd2c944af3dbc881156574b files/digest-exim-4.21 132 -MD5 9be54e4b44d4b7b6899302a1f3f74007 files/digest-exim-4.22 132 MD5 369b42cafcbe5631c2d03cbbfd4aeddd files/exim-4.10-gentoo.diff 1889 -MD5 5d5aabea7d56a91803df1312c04d32c0 files/exim-4.14-tail.patch 446 MD5 622b726ea7b32aae93a8fe9f3c2af9cd files/exim-4.20-maildir.patch 478 MD5 f442b68d435598831bab8536ade071b8 files/exim.confd 62 MD5 f0676c8d629efaa211aebbe01750eb41 files/exim.rc5 585 MD5 c90e5fcc2a5dfdd3095c5619d9dcc389 files/exim.rc6 509 -MD5 5898fa2e00e85c771cffe741f3198c07 files/exiscan-acl-4.20-09.patch 263216 MD5 eb249c90af3ab11e5a4d307e184a75ac files/exiscan.conf 22113 MD5 d230e0fa45f2b65d5bc50c0879c40148 files/pam.d-exim 101 MD5 0883fe67a34142a57e6f39ed9419cbee files/system_filter.exim 8118 +MD5 5d5aabea7d56a91803df1312c04d32c0 files/exim-4.14-tail.patch 446 +MD5 0cc15aecabaeace6ff4cdd1a462b9eb0 files/digest-exim-4.20-r2 62 +MD5 e62c9a744fd2c944af3dbc881156574b files/digest-exim-4.21 132 +MD5 9be54e4b44d4b7b6899302a1f3f74007 files/digest-exim-4.22 132 +MD5 117455c076e246e7c02d5fa464dd4be7 files/digest-exim-4.24 132 MD5 117455c076e246e7c02d5fa464dd4be7 files/digest-exim-4.24-r1 132 diff --git a/net-mail/exim/files/exiscan-acl-4.20-09.patch b/net-mail/exim/files/exiscan-acl-4.20-09.patch deleted file mode 100644 index 42d62d26be3f..000000000000 --- a/net-mail/exim/files/exiscan-acl-4.20-09.patch +++ /dev/null @@ -1,6707 +0,0 @@ -diff -urN exim-4.20-orig/OS/Makefile-Base exim-4.20/OS/Makefile-Base ---- exim-4.20-orig/OS/Makefile-Base Mon May 12 15:39:15 2003 -+++ exim-4.20/OS/Makefile-Base Wed May 14 12:07:55 2003 -@@ -246,14 +246,14 @@ - # Targets for final binaries; the main one has a build number which is - # updated each time. We don't bother with that for the auxiliaries. - --OBJ_EXIM = acl.o child.o crypt16.o daemon.o dbfn.o debug.o deliver.o \ -+OBJ_EXIM = acl.o child.o crypt16.o daemon.o dbfn.o debug.o deliver.o demime.o \ - directory.o dns.o drtables.o enq.o exim.o expand.o filter.o \ - filtertest.o globals.o \ -- header.o host.o ip.o log.o lss.o match.o moan.o \ -+ header.o host.o ip.o log.o lss.o malware.o match.o moan.o \ - os.o parse.o queue.o \ -- rda.o readconf.o receive.o retry.o rewrite.o \ -- route.o search.o smtp_in.o smtp_out.o spool_in.o spool_out.o \ -- store.o string.o tls.o tod.o transport.o tree.o verify.o \ -+ rda.o readconf.o receive.o regex.o retry.o rewrite.o \ -+ route.o search.o smtp_in.o smtp_out.o spam.o spool_in.o spool_mbox.o spool_out.o \ -+ store.o string.o tls.o tnef.o tod.o transport.o tree.o verify.o \ - local_scan.o $(EXIM_PERL) - - exim: pcre/libpcre.a lookups/lookups.a auths/auths.a \ -@@ -465,6 +465,7 @@ - dbfn.o: $(HDRS) dbfn.c - debug.o: $(HDRS) debug.c - deliver.o: $(HDRS) deliver.c -+demime.o: $(HDRS) demime.c - directory.o: $(HDRS) directory.c - dns.o: $(HDRS) dns.c - enq.o: $(HDRS) enq.c -@@ -478,6 +479,7 @@ - ip.o: $(HDRS) ip.c - log.o: $(HDRS) log.c - lss.o: $(HDRS) lss.c -+malware.o: $(HDRS) malware.c - match.o: $(HDRS) match.c - moan.o: $(HDRS) moan.c - os.o: $(HDRS) os.c -@@ -486,17 +488,21 @@ - rda.o: $(HDRS) rda.c - readconf.o: $(HDRS) readconf.c - receive.o: $(HDRS) receive.c -+regex.o: $(HDRS) regex.c - retry.o: $(HDRS) retry.c - rewrite.o: $(HDRS) rewrite.c - route.o: $(HDRS) route.c - search.o: $(HDRS) search.c - smtp_in.o: $(HDRS) smtp_in.c - smtp_out.o: $(HDRS) smtp_out.c -+spam.o: $(HDRS) spam.c - spool_in.o: $(HDRS) spool_in.c -+spool_mbox.o: $(HDRS) spool_mbox.c - spool_out.o: $(HDRS) spool_out.c - store.o: $(HDRS) store.c - string.o: $(HDRS) string.c - tls.o: $(HDRS) tls.c tls-gnu.c tls-openssl.c -+tnef.o: $(HDRS) tnef.c - tod.o: $(HDRS) tod.c - transport.o: $(HDRS) transport.c - tree.o: $(HDRS) tree.c -diff -urN exim-4.20-orig/README.EXISCAN exim-4.20/README.EXISCAN ---- exim-4.20-orig/README.EXISCAN Thu Jan 1 01:00:00 1970 -+++ exim-4.20/README.EXISCAN Wed May 14 12:04:24 2003 -@@ -0,0 +1 @@ -+Please refer to doc/exiscan-acl-spec.txt -diff -urN exim-4.20-orig/doc/exiscan-acl-examples.txt exim-4.20/doc/exiscan-acl-examples.txt ---- exim-4.20-orig/doc/exiscan-acl-examples.txt Thu Jan 1 01:00:00 1970 -+++ exim-4.20/doc/exiscan-acl-examples.txt Fri Jun 6 13:18:19 2003 -@@ -0,0 +1,440 @@ -+-------------------------------------------------------------- -+exiscan-acl example configurations / FAQ -+-------------------------------------------------------------- -+ -+Author: Tom Kistner <tom@duncanthrax.net> -+ -+The exiscan website is at http://duncanthrax.net/exiscan/. You -+will find the latest patch versions, as well as links to the -+mailing list and its archives there. -+ -+This document shows some example configuration snippets: -+ -+1. Basic sitewide virus and spam filtering by rejecting -+ matching messages after DATA. -+2. Adding a cryptographic "checks done" header that will -+ prevent re-scanning when the message re-visits one of your -+ mail servers, and the body size did not change. -+3. Marking spam-suspicious messages with extra headers and a -+ tag in the subject. -+4. Having more than one spam threshold to act on. -+5. Redirecting matching messages to special accounts while -+ preserving envelope recipient information. -+6. A multi-profile configuration for sites where different -+ "customers" (or users) have different content scanning -+ preferences. -+ -+These examples serve as a guideline and should give you some -+pointers that can help you to create your own configuration. -+Please do not copy these examples verbatim. You really need to -+know what you are doing. The content scanning topic is really -+complex and you can screw up your mail server easily if you do -+not get it "right". -+ -+I recommend to read the exiscan documentation on the above -+mentioned website before trying to make sense of the following -+examples. -+ -+Each example shows part of a DATA ACL definition, unless -+otherwise noted. -+ -+-------------------------------------------------------------- -+1. Basic setup for simple site-wide filtering -+-------------------------------------------------------------- -+The following example only shows the most basic use of the -+exiscan content filtering features. You should see it as a -+base that you can build on. However, it may be all you need -+for smaller systems with only a few users. -+ -+/* ----------------- -+# Do not scan messages submitted from our own hosts -+# and locally submitted messages. Since the DATA ACL -+# is not called for messages not submitted via SMTP -+# protocols, we do not need to check for an empty -+# host field. -+accept hosts = 127.0.0.1:+relay_from_hosts -+ -+# Unpack MIME containers and reject file extensions -+# used by worms. Note that the extension list may be -+# incomplete. -+deny message = $found_extension files are not accepted here -+ demime = com:vbs:bat:pif:scr -+ -+# Reject messages that have serious MIME errors. -+# This calls the demime condition again, but it -+# will return cached results. -+deny message = Serious MIME defect detected ($demime_reason) -+ demime = * -+ condition = ${if >{$demime_errorlevel}{2}{1}{0}} -+ -+# Reject messages containing malware. -+deny message = This message contains malware ($malware_name) -+ malware = * -+ -+# Reject spam messages. Remember to tweak your -+# site-wide SA profile. Do not spam-scan messages -+# larger than eighty kilobytes. -+deny message = Classified as spam (score $spam_score) -+ condition = ${if <{$message_size}{80k}{1}{0}} -+ spam = nobody -+ -+# Finally accept all other messages that have -+# made it to this point -+accept -+------------------ */ -+ -+ -+ -+-------------------------------------------------------------- -+2. Adding a cryptographic "scanning done" header -+-------------------------------------------------------------- -+ -+If you have a mail setup where the same message may pass your -+server twice (redirects from other servers), or you have -+multiple mail servers, you may want to make sure that each -+message is only checked once, to save processing time. Here is -+how to do it: -+ -+At the very beginning of your DATA ACL, put this: -+ -+/* ----------------- -+# Check our crytographic header. If it matches, accept -+# the message. -+accept condition = ${if eq {${hmac{md5}\ -+ {mysecret}\ -+ {$body_linecount}}}\ -+ {$h_X-Scan-Signature:} {1}{0}} -+------------------ */ -+ -+At the end, just before the final "accept" verb, put this: -+ -+/* ----------------- -+# Add the cryptographic header. -+warn message = X-Scan-Signature: ${hmac{md5}{mysecret}\ -+ {$body_linecount}} -+------------------ */ -+ -+Notice the two "mysecret" strings? Replace them with your own -+secret, and don't tell anyone :) The hash also includes the -+number of lines in the message body, to protect against -+message "modifications". -+ -+ -+-------------------------------------------------------------- -+3. Marking Spam messages with extra headers and subject tag -+-------------------------------------------------------------- -+ -+Since the false positive rate with spam scanning is high -+compared to virus scanning, it is wise to implement a scheme -+with two thresholds, where you reject messages with high -+scores and just mark messages with lower scores. End users can -+then set up filters in their Mail User Agents (MUAs). Since -+many MUAs can not filter on custom headers, it can be -+necessary to put a "spam tag" in the subject line. Since it is -+not (yet) possible to remove headers in Exims DATA ACL, we -+must do this in a system filter. Please see the Exim docs on -+how to set up a system filter. -+ -+The following example will unconditionally put two spam -+information headers in each message, if it is smaller than -+eighty kilobytes: -+ -+/* ----------------- -+# Always put X-Spam-Score header in the message. -+# It looks like this: -+# X-Spam-Score: 6.6 (++++++) -+# When a MUA cannot match numbers, it can match for an -+# equivalent number of '+' signs. -+# The 'true' makes sure that the header is always put -+# in, no matter what the score. -+warn message = X-Spam-Score: $spam_score ($spam_bar) -+ condition = ${if <{$message_size}{80k}{1}{0}} -+ spam = nobody:true -+ -+# Always put X-Spam-Report header in the message. -+# This is a multiline header that informs the user -+# which tests a message has "hit", and how much a -+# test has contributed to the score. -+warn message = X-Spam-Report: $spam_report -+ condition = ${if <{$message_size}{80k}{1}{0}} -+ spam = nobody:true -+------------------ */ -+ -+For the subject tag, we prepare a new subject header in the -+ACL, then swap it with the original Subject in the system -+filter. -+ -+In the DATA ACL, put this: -+/* ----------------- -+warn message = X-New-Subject: *SPAM* $h_subject: -+ spam = nobody -+------------------ */ -+ -+In the system filter, put this: -+/* ----------------- -+if "${if def:header_X-New-Subject: {there}}" is there -+then -+ headers remove subject -+ headers add "Subject: $h_X-New-Subject:" -+ headers remove X-New-Subject -+endif -+------------------ */ -+ -+ -+-------------------------------------------------------------- -+4. Defining multiple spam thresholds with different actions -+-------------------------------------------------------------- -+If you want to mark messages if they exceed your threshold, -+but also have a higher "cutoff" threshold where you reject -+messages, use the example above, plus this part: -+ -+/* ----------------- -+deny message = Spam score too high ($spam_score) -+ condition = ${if <{$message_size}{80k}{1}{0}} -+ spam = nobody:true -+ condition = ${if >{$spam_score_int}{100}{1}{0}} -+------------------ */ -+ -+The last condition is only true if the spam score exceeds 10.0 -+points (Keep in mind that $spam_score_int is the messages -+score multiplied by ten). -+ -+ -+ -+-------------------------------------------------------------- -+5. Redirect infected or spam messages to special accounts -+-------------------------------------------------------------- -+Sometimes it is desirable not to reject messages, but to stop -+them for inspection, and then decide wether to delete, bounce -+or pass them. -+ -+There are multiple ways to achieve this. The simplest way is -+to freeze suspicious messages, and then thaw or bounce them -+after a review. Here is a simple example that will freeze spam -+suspicious messages when they exceed the SA threshold: -+ -+/* ----------------- -+warn log_message = frozen by spam scanner, score $spam_score -+ spam = nobody -+ control = freeze -+------------------ */ -+ -+Another way is to redirect suspicious messages to special -+postmaster accounts, where they can be reviewed. This involves -+setting up a router for these special accounts that acts on a -+header set in the DATA ACL. -+ -+This is the DATA ACL entry: -+ -+/* ----------------- -+warn message = X-Redirect-To: spambox@mycompany.com -+ spam = nobody -+------------------ */ -+ -+This puts the target address in a special header, which can in -+turn be read with this router: -+ -+/* ----------------- -+scan_redirect: -+ driver = redirect -+ condition = ${if def:h_X-Redirect-To: {1}{0}} -+ headers_add = X-Original-Recipient: $local_part@$domain -+ data = $h_X-Redirect-To: -+ headers_remove = X-Redirect-To -+ redirect_router = my_second_router -+------------------ */ -+ -+This router should probably be your very first one, and you -+need to edit the last line (redirect_router = ) to replace -+"my_second_router" with the name of your original first -+router. Note that the original message recipient is saved in -+the "X-Original-Recipient" header, and the X-Redirect-To -+header line is removed. -+ -+ -+-------------------------------------------------------------- -+6. Having multiple content scanning profiles for several -+ users or domains. -+-------------------------------------------------------------- -+This is one of the most often asked questions, and it also has -+the most complicated answer. To understand the difficulties, -+you should first remember that the exiscan facilities are run -+in the DATA ACL. This ACL is called ONCE per message, after -+the sending server has transmitted the end-of-data marker. -+This gives us the very cool possibility to reject unwanted -+messages with a 5xx error code in response. The big drawback -+is that a message can have multiple recipients, and you can -+only reject or accept a message for ALL recipients, not -+individual ones. -+ -+I will first sum up the possible solutions to this dilemma: -+ -+ a. Make sure that each incoming message can have only one -+ envelope recipient. This is brutal, but effective and -+ reliably solves the problem on your end. :) Drawback: -+ Incoming mail to multiple recipients is slowed down. The -+ exact time depends on the retry strategies of the sending -+ hosts. -+ -+ b. Offer a limited number of "profiles" that your customers -+ can subscribe to. Then, similar to a.), only accept -+ recipients with the same profile in a single "batch", and -+ defer the others. This does improve on the drawback of -+ a.) a bit. -+ -+ c. Do scanning as usual, but never reject messages in the -+ DATA ACL. Instead put appropriate information in extra -+ headers and query those in routers or transports later. -+ Drawback: You'll have to send bounces yourself, and your -+ queue will fill up with frozen bounces. Advantage: clean -+ solution, protocol-wise. -+ -+As you see, you can't have your cake and eat it too. Now lets -+get into the details of each possible solution. -+ -+a.) Making sure each incoming message that will be scanned -+ only has one recipient. -+ -+ To use this scheme, you must make sure that you do not use -+ it on your +relay_from_hosts and authenticated senders. -+ Both of these may be MUAs who cannot cope with such a -+ thing. -+ -+ Here is a RCPT ACL that implements the behaviour -+ (shortened, do not copy 1:1!): -+ -+ /* ------------ -+ acl_check_rcpt: -+ -+ # accept local, relay-allowed -+ # and authenticated sources -+ -+ accept hosts = : -+ deny local_parts = ^.*[@%!/|] -+ accept hosts = 127.0.0.1:+relay_from_hosts -+ accept authenticated = * -+ -+ # the following treat non-local, -+ # non-authenticated sources -+ -+ defer message = only one recipient at a time -+ condition = ${if def:acl_m0 {1}{0}} -+ -+ # [ .. ] -+ # put RBLs etc. here -+ # [ .. ] -+ -+ accept domains = +local_domains -+ endpass -+ message = unknown user -+ verify = recipient -+ set acl_m0 = $local_part@$domain -+ -+ accept domains = +relay_to_domains -+ endpass -+ message = unrouteable address -+ verify = recipient -+ set acl_m0 = $domain -+ -+ deny message = relay not permitted -+ ------------ */ -+ -+ The lines which contain acl_m0 are the important ones. The -+ $acl_m0 variable gets set when a remote server -+ successfully sends one RCPT. Subsequent RCPT commands are -+ deferred if this variable is set. The $acl_m0 variable now -+ contains the single recipient domain, which you can use in -+ the DATA ACL to determine the scanning profile. -+ -+ This scheme is only recommended for small servers with a -+ low number of possible recipients, where recipients do not -+ belong to the same organization. An example would be a -+ multiuser shell server. -+ -+ -+b.) Having several scanning profiles that "customers" can -+ choose from. -+ -+ Suppose you want to offer three profiles. Lets call them -+ "reject-aggressive", "reject-conservative", and "warn -+ -only". Customers can select one of the profiles for each -+ of their domains. So you end up with a mapping like this: -+ -+ domain-a.com: reject-aggressive -+ domain-b.org: warn-only -+ domain-c.net: reject-aggressive -+ domain-d.com: reject-conservative -+ [ .. ] -+ -+ Suppose you put that in a file called /etc/exim/scanprefs -+ -+ Now we make a scheme similar to a.), but we do allow more -+ than one recipient if they have the same scanning profile -+ than the first recipient. -+ -+ Here is a RCPT ACL that implements the behaviour -+ (shortened, do not copy 1:1!): -+ -+ /* ------------ -+ acl_check_rcpt: -+ -+ # accept local, relay-allowed and authenticated sources -+ -+ accept hosts = : -+ deny local_parts = ^.*[@%!/|] -+ accept hosts = 127.0.0.1:+relay_from_hosts -+ accept authenticated = * -+ -+ # the following treat non-local, non-authenticated sources -+ -+ defer message = try this address in the next batch -+ condition = ${if eq {${acl_m0}}\ -+ {${lookup{$domain}\ -+ lsearch{/etc/exim/scanprefs}}}\ -+ {0}{1}} -+ -+ # [ .. ] -+ # put RBLs etc. here -+ # [ .. ] -+ -+ accept domains = +local_domains -+ endpass -+ message = unknown user -+ verify = recipient -+ set acl_m0 = $local_part@$domain -+ -+ accept domains = +relay_to_domains -+ endpass -+ message = unrouteable address -+ verify = recipient -+ set acl_m0 = ${lookup{$domain}\ -+ lsearch{/etc/exim/scanprefs}} -+ -+ deny message = relay not permitted -+ ------------ */ -+ -+ Now a recipient address get deferred if its scan profile -+ does not match the current batch profile. The $acl_m0 -+ variable contains the name of the profile, that can be -+ used for processing in the DATA ACL. -+ -+ This scheme works pretty well if you keep the number of -+ possible profiles low, since that will prevent -+ fragmentation of RCPT blocks. -+ -+ -+c.) Classic content scanning without the possibility of -+ rejects after DATA. -+ -+ This emulates the "classic" content scanning in routers -+ and transports. The difference is that we still do the -+ scan in the DATA ACL, but put the outcome of each facility -+ in message headers, that can the be evaluated in special -+ routers, individually for each recipient. -+ -+ A special approach can be taken for spam scanning, since -+ the $spam_score_int variable is also available in routers -+ and transports (it gets written to the spool files), so -+ you do not need to put that information in a header, but -+ rather act on $spam_score_int directly. -+ -diff -urN exim-4.20-orig/doc/exiscan-acl-spec.txt exim-4.20/doc/exiscan-acl-spec.txt ---- exim-4.20-orig/doc/exiscan-acl-spec.txt Thu Jan 1 01:00:00 1970 -+++ exim-4.20/doc/exiscan-acl-spec.txt Mon May 26 11:24:25 2003 -@@ -0,0 +1,426 @@ -+-------------------------------------------------------------- -+The exiscan-acl patch for exim4 - Documentation -+-------------------------------------------------------------- -+(c) Tom Kistner <tom@duncanthrax.net> 2003-???? -+License: GPL -+ -+The exiscan-acl patch adds content scanning to the exim4 ACL -+system. It supports the following scanning facilities: -+ -+ - MIME unpacking, sanity checking, file extension blocking -+ - Antivirus using 3rd party scanners -+ - Antispam using SpamAssassin -+ - Regular expression match against headers and body -+ -+These facilities are hooked into exim by adding new conditions -+to exim's ACL system. These conditions are designed to be used -+in the acl_smtp_data ACL. It is run when the sending host has -+completed the DATA phase and is waiting for our final response -+to his end-of-data marker. This allows us to reject messages -+containing unwanted content at that stage. -+ -+The exiscan-acl patch also defines several expansion variables -+that can be used to customise the error responses sent to the -+remote server. -+ -+The default exim configure file contains commented -+configuration examples for all facilites. -+ -+ -+0. Overall concept -+-------------------------------------------------------------- -+ -+The exiscan-acl patch adds the following conditions -+(facilities), which can ONLY be used in the ACL after DATA -+(acl_smtp_data): -+ -+- demime (MIME unpacking and file extension checks) -+- regex (match regular expressions against message headers -+ and body) -+- malware (attach 3rd party virus/malware scanner) -+- spam (attach SpamAssassin) -+ -+Each of these facilities has its own chapter further below in -+this document. There is also a commented sample configuration -+in the "configure" file of the exim distribution. -+ -+All facilites work on a MBOX copy of the message that is -+temporarily spooled up in a file called: -+ -+ <spool_directory>/scan/<message_id>/<message_id>.eml -+ -+The .eml extension is a friendly hint to virus scanners that -+they can expect an MBOX-like structure inside that file. The -+file is only spooled up once, when the first exiscan facility -+condition is called. Subsequent calls to exiscan conditions -+will just open the file again. The directory is recursively -+removed when the acl_smtp_data has finished running. When the -+"demime" condition has been used, this directory will also -+contain files produced by the MIME decoder. -+ -+ -+1. The "demime" facility -+ MIME unpacking, sanity checking and file extension blocking -+-------------------------------------------------------------- -+ -+The demime facility unpacks MIME containers in the message. It -+detects errors in MIME containers and can match file -+extensions found in the message against a list. Using this -+facility will produce additional files in the temporary scan -+directory that contain the unpacked MIME parts of the message. -+If you do antivirus scanning, it is recommened to use the -+"demime" condition before the antivirus ("malware") condition. -+ -+The condition name of this facility is "demime". On the right -+hand side, you can pass a colon-separated list of file -+extensions that it should match against. If one of the file -+extensions is found, the condition will return "OK" (or -+"true"), otherwise it will return FAIL (or "false"). If there -+was any TEMPORARY error while demimeing (mostly "disk full"), -+the condition will return DEFER, and the message will be -+temporarily rejected. -+ -+The right-hand side gets "expanded" before being treated as a -+list, so you can have conditions and lookups there. If it -+expands to an empty string, "false", or zero ("0"), no -+demimeing is done and the conditions returns FALSE. -+ -+A short example: -+ -+/* ------------ -+deny message = Found blacklisted file attachment -+ demime = vbs:com:bat:pif:prf:lnk -+--------------- */ -+ -+When the condition is run, it sets up the following expansion -+variables: -+ -+ $demime_errorlevel When an error was detected in a MIME -+ container, this variable contains the -+ "severity" of the error, as an integer -+ number. The higher the value, the -+ more severe the error. If this -+ variable is unset or zero, no error has -+ occured. -+ -+ $demime_reason When $demime_errorlevel is greater than -+ zero, this variable contains a human -+ -readable text string describing the -+ MIME error that occured. -+ -+ $found_extension When the "demime" condition returns -+ "true", this variable contains the file -+ extension it has found. -+ -+Both $demime_errorlevel and $demime_reason are set with the -+first call of the "demime" condition, and are not changed on -+subsequent calls. -+ -+If do not want to check for any file extensions, but rather -+use the demime facility for unpacking or error checking -+purposes, just pass "*" as the right-hand side value. -+ -+Here is a more elaborate example on how to use this facility: -+ -+/* ----------------- -+# Reject messages with serious MIME container errors -+deny message = Found MIME error ($demime_reason). -+ demime = * -+ condition = ${if >{$demime_errorlevel}{2}{1}{0}} -+ -+# Reject known virus spreading file extensions. -+# Accepting these is pretty much braindead. -+deny message = contains $found_extension file (blacklisted). -+ demime = com:vbs:bat:pif:scr -+ -+# Freeze .exe and .doc files. Postmaster can -+# examine them and eventually thaw them up. -+deny log_message = Another $found_extension file. -+ demime = exe:doc -+ control = freeze -+--------------------- */ -+ -+ -+ -+2. The "spam" facility -+ Antispam measures with SpamAssassin -+-------------------------------------------------------------- -+ -+The "spam" facility calls SpamAssassin's "spamd" daemon to get -+a spam-score and a report for the message. You must first -+install SpamAssassin. You can get it at -+http://www.spamassassin.org, or, if you have a working Perl -+installation, you can use CPAN by calling -+ -+perl -MCPAN -e 'install Mail::SpamAssassin' -+ -+SpamAssassin has it's own set of configuration files. Please -+review its documentation to see how you can tweak it. The -+default installation should work nicely, however. -+ -+After having installed and configured SpamAssassin, start the -+"spamd" daemon. By default, it listens on 127.0.0.1, TCP port -+783. If you use another host or port for spamd, you must set -+the spamd_address option in Section 1 of the exim -+configuration as follows (example): -+ -+spamd_address = 127.0.0.1 783 -+ -+If you use the above mentioned default, you do NOT need to set -+this option. -+ -+To use the antispam facility, put the "spam" condition in a -+DATA ACL block. Here is a very simple example: -+ -+/* --------------- -+deny message = This message was classified as SPAM -+ spam = joe -+---------------- */ -+ -+On the right-hand side of the spam condition, you can put the -+username that SpamAssassin should scan for. That allows you to -+use per-domain or per-user antispam profiles. The right-hand -+side is expanded before being used, so you can put lookups or -+conditions there. When the right-hand side evaluates to "0" or -+"false", no scanning will be done and the condition will fail -+immediately. -+ -+If you do not want to scan for a particular user, but rather -+use the SpamAssassin system-wide default profile, you can scan -+for an unknown user, or simply use "nobody". -+ -+The "spam" condition will return true if the threshold -+specified in the user's SpamAssassin profile has been matched -+or exceeded. If you want to use the spam condition for it's -+side effects (see the variables below), you can make it always -+return "true" by appending ":true" to the username. -+ -+When the condition is run, it sets up the following expansion -+variables: -+ -+ $spam_score The spam score of the message, for example -+ "3.4" or "30.5". This is useful for -+ inclusion in log or reject messages. -+ -+ $spam_score_int The spam score of the message, multiplied -+ by ten, as an integer value. For example -+ "34" or "305". This is useful for numeric -+ comparisons in conditions. See further -+ below for a more complicated example. This -+ variable is special, since it is written -+ to the spool file, so it can be used -+ during the whole life of the message on -+ your exim system, even in routers -+ or transports. -+ -+ $spam_bar A string consisting of a number of '+' or -+ '-' characters, representing the -+ spam_score value. A spam score of "4.4" -+ would have a spam_bar of '++++'. This is -+ useful for inclusion in warning headers, -+ since MUAs can match on such strings. -+ -+ $spam_report A multiline text table, containing the -+ full SpamAssassin report for the message. -+ Useful for inclusion in headers or reject -+ messages. -+ -+The spam condition caches its results. If you call it again -+with the same user name, it will not really scan again, but -+rather return the same values as before. -+ -+Finally, here is a commented example on how to use the spam -+condition: -+ -+/* ---------------- -+# put headers in all messages (no matter if spam or not) -+warn message = X-Spam-Score: $spam_score ($spam_bar) -+ spam = nobody:true -+warn message = X-Spam-Report: $spam_report -+ spam = nobody:true -+ -+# add second subject line with *SPAM* marker when message -+# is over threshold -+warn message = Subject: *SPAM* $h_Subject -+ spam = nobody -+ -+# reject spam at high scores (> 12) -+deny message = This message scored $spam_score spam points. -+ spam = nobody:true -+ condition = ${if >{$spam_score_int}{120}{1}{0}} -+----------------- */ -+ -+ -+ -+3. The "regex" facility -+ Match headers and body lines of the message against regular -+ expressions -+-------------------------------------------------------------- -+ -+The "regex" condition takes one or more regular expressions as -+arguments and matches them against the full message, that is -+all headers and the complete body. It is particularly useful -+to filter trash that cannot be recognized by the spam or -+malware conditions. With large messages, this condition can be -+fairly CPU-intensive. -+ -+The regular expressions are matched linewise, with a maximum -+line length of 32k characters. -+ -+The regular expressions are passed as a colon-separated list. -+To include a literal colon, you must double it. Since the -+whole right-hand side string is expanded before being used, -+you must also escape dollar ($) signs with backslashes. -+ -+Here is a simple example: -+ -+/* ---------------------- -+deny message = contains blacklisted regex ($regex_match_string) -+ regex = [Mm]ortgage : URGENT BUSINESS PROPOSAL -+----------------------- */ -+ -+The condition returns true if one of the regular expressions -+has matched a line of the message. The $regex_match_string -+variable is then set up and contains the matching regular -+expression. -+ -+ -+ -+4. The "malware" facility -+ Scan messages for viruses using an external virus scanner -+-------------------------------------------------------------- -+ -+This facility lets you connect virus scanner software to exim. -+It supports a "generic" interface to scanners called via the -+shell, and specialized interfaces for "daemon" type virus -+scanners, who are resident in memory and thus are much faster. -+ -+To use this facility, you MUST set the "av_scanner" option in -+section 1 of the exim config file. It specifies the scanner -+type to use, and any additional options it needs to run. The -+basic syntax is as follows: -+ -+ av_scanner = <scanner-type>:<option1>:<option2>:[...] -+ -+The following scanner-types are supported in this release: -+ -+ sophie Sophie is a daemon that uses Sophos' libsavi -+ library to scan for viruses. You can get Sophie -+ at http://www.vanja.com/tools/sophie/. The only -+ option for this scanner type is the path to the -+ UNIX socket that Sophie uses for client -+ communication. The default path is -+ /var/run/sophie, so if you are using this, you -+ can omit the option. Example: -+ -+ av_scanner = sophie:/tmp/sophie -+ -+ -+ kavdaemon Kapersky's kavdaemon is a daemon-type scanner. -+ You can get a trial version at -+ http://www.kapersky.com. This scanner type takes -+ one option, which is the path to the daemon's -+ UNIX socket. The default is "/var/run/AvpCtl". -+ Example: -+ -+ av_scanner = kavdaemon:/opt/AVP/AvpCtl -+ -+ -+ clamd Another daemon type scanner, this one is GPL and -+ free. Get it at http://clamav.elektrapro.com/. -+ Clamd does not seem to unpack MIME containers, -+ so it is recommended to use the demime facility -+ with it. It takes one option: either the path -+ and name of a UNIX socket file, or a -+ hostname/port pair, separated by space. If -+ unset, the default is "/tmp/clamd". Example: -+ -+ av_scanner = clamd:192.168.2.100 1234 -+ or -+ av_scanner = clamd:/opt/clamd/socket -+ -+ -+ cmdline This is the keyword for the generic command line -+ scanner interface. It can be used to attach -+ virus scanners that are invoked on the shell. -+ This scanner type takes 3 mantadory options: -+ -+ - full path and name of the scanner binary, with -+ all command line options and a placeholder -+ (%s) for the directory to scan. -+ -+ - A regular expression to match against the -+ STDOUT and STDERR output of the virus scanner. -+ If the expression matches, a virus was found. -+ You must make absolutely sure that this -+ expression only matches on "virus found". This -+ is called the "trigger" expression. -+ -+ - Another regular expression, containing exactly -+ ONE pair of braces, to match the name of the -+ virus found in the scanners output. This is -+ called the "name" expression. -+ -+ Example: -+ -+ Sophos Sweep reports a virus on a line like -+ this: -+ -+ Virus 'W32/Magistr-B' found in file ./those.bat -+ -+ For the "trigger" expression, we just use the -+ "found" word. For the "name" expression, we want -+ to get the W32/Magistr-B string, so we can match -+ for the single quotes left and right of it, -+ resulting in the regex '(.*)' (WITH the quotes!) -+ -+ Altogether, this makes the configuration -+ setting: -+ -+ av_scanner = cmdline:\ -+ /path/to/sweep -all -rec -archive %s:\ -+ found:'(.+)' -+ -+ -+When av_scanner is correcly set, you can use the "malware" -+condition in the DATA ACL. The condition takes a right-hand -+argument that is expanded before use. It can then be one of -+ -+ - "true", "*", or "1", in which case the message is scanned -+ for viruses. The condition will succeed if a virus was -+ found, or fail otherwise. This is the recommended usage. -+ -+ - "false" or "0", in which case no scanning is done and the -+ condition will fail immediately. -+ -+ - a regular expression, in which case the message is scanned -+ for viruses. The condition will succeed if a virus found -+ found and its name matches the regular expression. This -+ allows you to take special actions on certain types of -+ viruses. -+ -+When a virus was found, the condition sets up an expansion -+variable called $malware_name that contains the name of the -+virus found. You should use it in a "message" modifier that -+contains the error returned to the sender. -+ -+The malware condition caches its results, so when you use it -+multiple times, the actual scanning process is only carried -+out once. -+ -+If your virus scanner cannot unpack MIME and TNEF containers -+itself, you should use the demime condition prior to the -+malware condition. -+ -+Here is a simple example: -+ -+/* ---------------------- -+deny message = This message contains malware ($malware_name) -+ demime = * -+ malware = * -+---------------------- */ -+ -+ -+-------------------------------------------------------------- -+End of file -+-------------------------------------------------------------- -diff -urN exim-4.20-orig/exim_monitor/em_globals.c exim-4.20/exim_monitor/em_globals.c ---- exim-4.20-orig/exim_monitor/em_globals.c Mon May 12 15:39:23 2003 -+++ exim-4.20/exim_monitor/em_globals.c Wed May 14 12:04:24 2003 -@@ -133,6 +133,7 @@ - - BOOL local_error_message = FALSE; - uschar *local_scan_data = NULL; -+uschar *spam_score_int = NULL; - BOOL log_timezone = FALSE; - int message_age = 0; - uschar *message_id; -diff -urN exim-4.20-orig/scripts/MakeLinks exim-4.20/scripts/MakeLinks ---- exim-4.20-orig/scripts/MakeLinks Mon May 12 15:39:17 2003 -+++ exim-4.20/scripts/MakeLinks Wed May 14 12:04:24 2003 -@@ -166,6 +166,7 @@ - - ln -s ../src/dbfunctions.h dbfunctions.h - ln -s ../src/dbstuff.h dbstuff.h -+ln -s ../src/demime.h demime.h - ln -s ../src/exim.h exim.h - ln -s ../src/functions.h functions.h - ln -s ../src/globals.h globals.h -@@ -173,8 +174,10 @@ - ln -s ../src/macros.h macros.h - ln -s ../src/mytypes.h mytypes.h - ln -s ../src/osfunctions.h osfunctions.h -+ln -s ../src/spam.h spam.h - ln -s ../src/store.h store.h - ln -s ../src/structs.h structs.h -+ln -s ../src/tnef.h tnef.h - - ln -s ../src/acl.c acl.c - ln -s ../src/buildconfig.c buildconfig.c -@@ -184,6 +187,7 @@ - ln -s ../src/dbfn.c dbfn.c - ln -s ../src/debug.c debug.c - ln -s ../src/deliver.c deliver.c -+ln -s ../src/demime.c demime.c - ln -s ../src/directory.c directory.c - ln -s ../src/dns.c dns.c - ln -s ../src/drtables.c drtables.c -@@ -202,6 +206,7 @@ - ln -s ../src/ip.c ip.c - ln -s ../src/log.c log.c - ln -s ../src/lss.c lss.c -+ln -s ../src/malware.c malware.c - ln -s ../src/match.c match.c - ln -s ../src/moan.c moan.c - ln -s ../src/parse.c parse.c -@@ -210,19 +215,23 @@ - ln -s ../src/rda.c rda.c - ln -s ../src/readconf.c readconf.c - ln -s ../src/receive.c receive.c -+ln -s ../src/regex.c regex.c - ln -s ../src/retry.c retry.c - ln -s ../src/rewrite.c rewrite.c - ln -s ../src/route.c route.c - ln -s ../src/search.c search.c - ln -s ../src/smtp_in.c smtp_in.c - ln -s ../src/smtp_out.c smtp_out.c -+ln -s ../src/spam.c spam.c - ln -s ../src/spool_in.c spool_in.c -+ln -s ../src/spool_mbox.c spool_mbox.c - ln -s ../src/spool_out.c spool_out.c - ln -s ../src/store.c store.c - ln -s ../src/string.c string.c - ln -s ../src/tls.c tls.c - ln -s ../src/tls-gnu.c tls-gnu.c - ln -s ../src/tls-openssl.c tls-openssl.c -+ln -s ../src/tnef.c tnef.c - ln -s ../src/tod.c tod.c - ln -s ../src/transport.c transport.c - ln -s ../src/tree.c tree.c -diff -urN exim-4.20-orig/src/acl.c exim-4.20/src/acl.c ---- exim-4.20-orig/src/acl.c Mon May 12 15:39:17 2003 -+++ exim-4.20/src/acl.c Fri May 16 09:11:35 2003 -@@ -7,6 +7,8 @@ - - /* Code for handling Access Control Lists (ACLs) */ - -+/* This file has been modified by the exiscan-acl patch. */ -+ - #include "exim.h" - - -@@ -32,19 +34,19 @@ - /* ACL condition and modifier codes - keep in step with the table that - follows. */ - --enum { ACLC_ACL, ACLC_AUTHENTICATED, ACLC_CONDITION, ACLC_CONTROL, ACLC_DELAY, -+enum { ACLC_ACL, ACLC_AUTHENTICATED, ACLC_CONDITION, ACLC_CONTROL, ACLC_DELAY, ACLC_DEMIME, - ACLC_DNSLISTS, ACLC_DOMAINS, ACLC_ENCRYPTED, ACLC_ENDPASS, ACLC_HOSTS, -- ACLC_LOCAL_PARTS, ACLC_LOG_MESSAGE, ACLC_MESSAGE, ACLC_RECIPIENTS, -- ACLC_SENDER_DOMAINS, ACLC_SENDERS, ACLC_SET, ACLC_VERIFY }; -+ ACLC_LOCAL_PARTS, ACLC_LOG_MESSAGE, ACLC_MALWARE, ACLC_MESSAGE, ACLC_RECIPIENTS, -+ ACLC_REGEX, ACLC_SENDER_DOMAINS, ACLC_SENDERS, ACLC_SET, ACLC_SPAM, ACLC_VERIFY }; - - /* ACL conditions/modifiers: "delay", "control", "endpass", "message", --"log_message", and "set" are modifiers that look like conditions but always -+"log_message", "set" and "spam" are modifiers that look like conditions but always - return TRUE. They are used for their side effects. */ - - static uschar *conditions[] = { US"acl", US"authenticated", US"condition", -- US"control", US"delay", US"dnslists", US"domains", US"encrypted", -- US"endpass", US"hosts", US"local_parts", US"log_message", US"message", -- US"recipients", US"sender_domains", US"senders", US"set", US"verify" }; -+ US"control", US"delay", US"demime", US"dnslists", US"domains", US"encrypted", -+ US"endpass", US"hosts", US"local_parts", US"log_message", US"malware", US"message", -+ US"recipients", US"regex", US"sender_domains", US"senders", US"set", US"spam", US"verify" }; - - /* Flags to indicate for which conditions /modifiers a string expansion is done - at the outer level. In the other cases, expansion already occurs in the -@@ -56,6 +58,7 @@ - TRUE, /* condition */ - TRUE, /* control */ - TRUE, /* delay */ -+ FALSE, /* demime */ - TRUE, /* dnslists */ - FALSE, /* domains */ - FALSE, /* encrypted */ -@@ -63,11 +66,14 @@ - FALSE, /* hosts */ - FALSE, /* local_parts */ - TRUE, /* log_message */ -+ TRUE, /* malware */ - TRUE, /* message */ - FALSE, /* recipients */ -+ TRUE, /* regex */ - FALSE, /* sender_domains */ - FALSE, /* senders */ - TRUE, /* set */ -+ TRUE, /* spam */ - TRUE /* verify */ - }; - -@@ -79,6 +85,7 @@ - FALSE, /* condition */ - TRUE, /* control */ - TRUE, /* delay */ -+ FALSE, /* demime */ - FALSE, /* dnslists */ - FALSE, /* domains */ - FALSE, /* encrypted */ -@@ -86,11 +93,14 @@ - FALSE, /* hosts */ - FALSE, /* local_parts */ - TRUE, /* log_message */ -+ FALSE, /* malware */ - TRUE, /* message */ - FALSE, /* recipients */ -+ FALSE, /* regex */ - FALSE, /* sender_domains */ - FALSE, /* senders */ - TRUE, /* set */ -+ FALSE, /* spam */ - FALSE /* verify */ - }; - -@@ -109,6 +119,13 @@ - (1<<ACL_WHERE_STARTTLS)|(ACL_WHERE_VRFY), - - 0, /* delay */ -+ -+ (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* demime */ -+ (1<<ACL_WHERE_CONNECT)| -+ (1<<ACL_WHERE_ETRN)|(1<<ACL_WHERE_EXPN)| -+ (1<<ACL_WHERE_MAIL)|(1<<ACL_WHERE_STARTTLS)| -+ (1<<ACL_WHERE_VRFY), -+ - (1<<ACL_WHERE_NOTSMTP), /* dnslists */ - - (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* domains */ -@@ -131,6 +148,13 @@ - (1<<ACL_WHERE_VRFY), - - 0, /* log_message */ -+ -+ (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* malware */ -+ (1<<ACL_WHERE_CONNECT)| -+ (1<<ACL_WHERE_ETRN)|(1<<ACL_WHERE_EXPN)| -+ (1<<ACL_WHERE_MAIL)|(1<<ACL_WHERE_STARTTLS)| -+ (1<<ACL_WHERE_VRFY), -+ - 0, /* message */ - - (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* recipients */ -@@ -140,6 +164,12 @@ - (1<<ACL_WHERE_MAIL)|(1<<ACL_WHERE_STARTTLS)| - (1<<ACL_WHERE_VRFY), - -+ (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* regex */ -+ (1<<ACL_WHERE_CONNECT)| -+ (1<<ACL_WHERE_ETRN)|(1<<ACL_WHERE_EXPN)| -+ (1<<ACL_WHERE_MAIL)|(1<<ACL_WHERE_STARTTLS)| -+ (1<<ACL_WHERE_VRFY), -+ - (1<<ACL_WHERE_AUTH)|(1<<ACL_WHERE_CONNECT)| /* sender_domains */ - (1<<ACL_WHERE_HELO)| - (1<<ACL_WHERE_ETRN)|(1<<ACL_WHERE_EXPN)| -@@ -152,6 +182,12 @@ - - 0, /* set */ - -+ (1<<ACL_WHERE_NOTSMTP)|(1<<ACL_WHERE_AUTH)| /* spam */ -+ (1<<ACL_WHERE_CONNECT)| -+ (1<<ACL_WHERE_ETRN)|(1<<ACL_WHERE_EXPN)| -+ (1<<ACL_WHERE_MAIL)|(1<<ACL_WHERE_STARTTLS)| -+ (1<<ACL_WHERE_VRFY), -+ - /* Certain types of verify are always allowed, so we let it through - always and check in the verify function itself */ - -@@ -1096,6 +1132,23 @@ - rc = verify_check_dnsbl(&arg); - break; - -+ -+ case ACLC_DEMIME: -+ rc = demime(&arg); -+ break; -+ -+ case ACLC_MALWARE: -+ rc = malware(&arg); -+ break; -+ -+ case ACLC_SPAM: -+ rc = spam(&arg); -+ break; -+ -+ case ACLC_REGEX: -+ rc = regex(&arg); -+ break; -+ - case ACLC_DOMAINS: - rc = match_isinlist(addr->domain, &arg, 0, &domainlist_anchor, - addr->domain_cache, MCL_DOMAIN, TRUE, &deliver_domain_data); -@@ -1618,6 +1671,10 @@ - return DISCARD; - } - -+/* Remove spooled mbox and demimed files. -+Will immediately return if no files had been created */ -+unspool_mbox(); -+ - /* Before giving an error response, take a look at the length of any user - message, and split it up into multiple lines if possible. */ - -diff -urN exim-4.20-orig/src/configure.default exim-4.20/src/configure.default ---- exim-4.20-orig/src/configure.default Mon May 12 15:39:18 2003 -+++ exim-4.20/src/configure.default Tue May 27 13:32:26 2003 -@@ -108,6 +108,25 @@ - - # You should not change that setting until you understand how ACLs work. - -+# The following ACL entry is used if you want to do content scanning with the -+# exiscan-acl patch. When you uncomment this line, you must also review the -+# acl_check_content entry in the ACL section further below. -+ -+# acl_smtp_data = acl_check_content -+ -+# This configuration variable defines the virus scanner that is used with -+# the 'malware' ACL condition of the exiscan acl-patch. If you do not use -+# virus scanning, leave it commented. Please read doc/exiscan-acl-readme.txt -+# for a list of supported scanners. -+ -+# av_scanner = sophie:/var/run/sophie -+ -+# The following setting is only needed if you use the 'spam' ACL condition -+# of the exiscan-acl patch. It specifies on which host and port the SpamAssassin -+# "spamd" daemon is listening. If you do not use this condition, or you use -+# the default of "127.0.0.1 783", you can omit this option. -+ -+# spamd_address = 127.0.0.1 783 - - # Specify the domain you want to be added to all unqualified addresses - # here. An unqualified address is one that does not contain an "@" character -@@ -308,6 +327,52 @@ - deny message = relay not permitted - - -+# This access control list is used for content scanning with the exiscan-acl -+# patch. You must also uncomment the entry for acl_smtp_data (scroll up), -+# otherwise the ACL will not be used. IMPORTANT: the default entries here -+# should be treated as EXAMPLES. You MUST read the file doc/exiscan-acl-spec.txt -+# to fully understand what you are doing ... -+ -+acl_check_content: -+ -+ # First unpack MIME containers and reject serious errors. -+ deny message = This message contains a MIME error ($demime_reason) -+ demime = * -+ condition = ${if >{$demime_errorlevel}{2}{1}{0}} -+ -+ # Reject typically wormish file extensions. There is almost no -+ # sense in sending such files by email. -+ deny message = This message contains an unwanted file extension ($found_extension) -+ demime = scr:vbs:bat:lnk:pif -+ -+ # Reject virus infested messages. -+ deny message = This message contains malware ($malware_name) -+ malware = * -+ -+ # Reject messages containing "viagra" in all kinds of whitespace/case combinations -+ # WARNING: this is an example ! -+ deny message = This message matches a blacklisted regular expression ($regex_match_string) -+ regex = [Vv] *[Ii] *[Aa] *[Gg] *[Rr] *[Aa] -+ -+ # Always add X-Spam-Score and X-Spam-Report headers, using SA system-wide settings -+ # (user "nobody"), no matter if over threshold or not. -+ warn message = X-Spam-Score: $spam_score ($spam_bar) -+ spam = nobody:true -+ warn message = X-Spam-Report: $spam_report -+ spam = nobody:true -+ -+ # Add X-Spam-Flag if spam is over system-wide threshold -+ warn message = X-Spam-Flag: YES -+ spam = nobody -+ -+ # Reject spam messages with score over 10, using an extra condition. -+ deny message = This message scored $spam_score points. Congratulations! -+ spam = nobody:true -+ condition = ${if >{$spam_score_int}{100}{1}{0}} -+ -+ # finally accept all the rest -+ accept -+ - - ###################################################################### - # ROUTERS CONFIGURATION # -diff -urN exim-4.20-orig/src/demime.c exim-4.20/src/demime.c ---- exim-4.20-orig/src/demime.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/demime.c Fri May 16 17:08:52 2003 -@@ -0,0 +1,1243 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* Code for unpacking MIME containers. Called from acl.c. */ -+ -+#include "exim.h" -+#include "demime.h" -+ -+uschar demime_reason_buffer[1024]; -+int demime_ok = 0; -+struct file_extension *file_extensions = NULL; -+ -+int demime(uschar **listptr) { -+ int sep = 0; -+ uschar *list = *listptr; -+ uschar *option; -+ uschar option_buffer[64]; -+ unsigned long long mbox_size; -+ FILE *mbox_file; -+ uschar defer_error_buffer[1024]; -+ int demime_rc; -+ -+ /* reset found_extension variable */ -+ found_extension = NULL; -+ -+ /* try to find 1st option */ -+ if ((option = string_nextinlist(&list, &sep, -+ option_buffer, -+ sizeof(option_buffer))) != NULL) { -+ -+ /* parse 1st option */ -+ if ( (Ustrcmp(option,"false") == 0) || (Ustrcmp(option,"0") == 0) ) { -+ /* explicitly no demimeing */ -+ return FAIL; -+ }; -+ } -+ else { -+ /* no options -> no demimeing */ -+ return FAIL; -+ }; -+ -+ /* make sure the eml mbox file is spooled up */ -+ mbox_file = spool_mbox(&mbox_size); -+ -+ if (mbox_file == NULL) { -+ /* error while spooling */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "demime acl condition: error while creating mbox spool file"); -+ return DEFER; -+ }; -+ -+ /* call demimer if not already done earlier */ -+ if (!demime_ok) -+ demime_rc = mime_demux(mbox_file, defer_error_buffer); -+ -+ fclose(mbox_file); -+ -+ if (demime_rc == DEFER) { -+ /* temporary failure (DEFER => DEFER) */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "demime acl condition: %s", defer_error_buffer); -+ return DEFER; -+ }; -+ -+ /* set demime_ok to avoid unpacking again */ -+ demime_ok = 1; -+ -+ /* check for file extensions, if there */ -+ while (option != NULL) { -+ struct file_extension *this_extension = file_extensions; -+ -+ /* Look for the wildcard. If it is found, we always return true. -+ The user must then use a custom condition to evaluate demime_errorlevel */ -+ if (Ustrcmp(option,"*") == 0) { -+ found_extension = NULL; -+ return OK; -+ }; -+ -+ /* loop thru extension list */ -+ while (this_extension != NULL) { -+ if (strcmpic(option, this_extension->file_extension_string) == 0) { -+ /* found one */ -+ found_extension = this_extension->file_extension_string; -+ return OK; -+ }; -+ this_extension = this_extension->next; -+ }; -+ -+ /* grab next extension from option list */ -+ option = string_nextinlist(&list, &sep, -+ option_buffer, -+ sizeof(option_buffer)); -+ }; -+ -+ /* nothing found */ -+ return FAIL; -+} -+ -+ -+/************************************************* -+* unpack TNEF in given directory * -+*************************************************/ -+ -+int mime_unpack_tnef(uschar *directory) { -+ uschar filepath[1024]; -+ int n; -+ struct dirent *entry; -+ DIR *tempdir; -+ -+ /* open the dir */ -+ tempdir = opendir(CS directory); -+ if (tempdir == NULL) { -+ return -2; -+ }; -+ -+ /* loop thru dir */ -+ n = 0; -+ do { -+ entry = readdir(tempdir); -+ /* break on end of list */ -+ if (entry == NULL) break; -+ snprintf(CS filepath,1024,"%s/%s",directory,entry->d_name); -+ if ( (Ustrcmp(entry->d_name,"..") != 0) && (Ustrcmp(entry->d_name,".") != 0) ) { -+ TNEF_set_path(CS directory); -+ n = TNEF_main(CS filepath); -+ }; -+ } while (1); -+ -+ closedir(tempdir); -+ return 0; -+} -+ -+ -+/************************************************* -+* small hex_str -> integer conversion function * -+*************************************************/ -+ -+/* needed for quoted-printable -+*/ -+ -+unsigned int mime_hstr_i(uschar *cptr) { -+ unsigned int i, j = 0; -+ -+ while (cptr && *cptr && isxdigit(*cptr)) { -+ i = *cptr++ - '0'; -+ if (9 < i) i -= 7; -+ j <<= 4; -+ j |= (i & 0x0f); -+ } -+ -+ return(j); -+} -+ -+ -+/************************************************* -+* decode quoted-printable chars * -+*************************************************/ -+ -+/* gets called when we hit a = -+ returns: new pointer position -+ result code in c: -+ -2 - decode error -+ -1 - soft line break, no char -+ 0-255 - char to write -+*/ -+ -+uschar *mime_decode_qp(uschar *qp_p,int *c) { -+ uschar hex[] = {0,0,0}; -+ int nan = 0; -+ uschar *initial_pos = qp_p; -+ -+ /* advance one char */ -+ qp_p++; -+ -+ REPEAT_FIRST: -+ if ( (*qp_p == '\t') || (*qp_p == ' ') || (*qp_p == '\r') ) { -+ /* tab or whitespace may follow -+ just ignore it, but remember -+ that this is not a valid hex -+ encoding any more */ -+ nan = 1; -+ qp_p++; -+ goto REPEAT_FIRST; -+ } -+ else if ( (('0' <= *qp_p) && (*qp_p <= '9')) || (('A' <= *qp_p) && (*qp_p <= 'F')) || (('a' <= *qp_p) && (*qp_p <= 'f')) ) { -+ /* this is a valid hex char, if nan is unset */ -+ if (nan) { -+ /* this is illegal */ -+ *c = -2; -+ return initial_pos; -+ } -+ else { -+ hex[0] = *qp_p; -+ qp_p++; -+ }; -+ } -+ else if (*qp_p == '\n') { -+ /* hit soft line break already, continue */ -+ *c = -1; -+ return qp_p; -+ } -+ else { -+ /* illegal char here */ -+ *c = -2; -+ return initial_pos; -+ }; -+ -+ if ( (('0' <= *qp_p) && (*qp_p <= '9')) || (('A' <= *qp_p) && (*qp_p <= 'F')) || (('a' <= *qp_p) && (*qp_p <= 'f')) ) { -+ if (hex[0] > 0) { -+ hex[1] = *qp_p; -+ /* do hex conversion */ -+ *c = mime_hstr_i(hex); -+ qp_p++; -+ return qp_p; -+ } -+ else { -+ /* huh ? */ -+ *c = -2; -+ return initial_pos; -+ }; -+ } -+ else { -+ /* illegal char */ -+ *c = -2; -+ return initial_pos; -+ }; -+ -+} -+ -+ -+/************************************************* -+* open new dump file * -+*************************************************/ -+ -+/* open new dump file -+ returns: -2 soft error -+ or file #, FILE * in f -+*/ -+ -+int mime_get_dump_file(uschar *extension, FILE **f, uschar *info) { -+ uschar file_name[1024]; -+ int result; -+ unsigned int file_nr; -+ uschar default_extension[] = ".com"; -+ uschar *p; -+ -+ if (extension == NULL) -+ extension = default_extension; -+ -+ /* scan the proposed extension. -+ if it is longer than 4 chars, or -+ contains exotic chars, use the default extension */ -+ -+/* if (Ustrlen(extension) > 4) { -+ extension = default_extension; -+ }; -+*/ -+ -+ p = extension+1; -+ -+ while (*p != 0) { -+ *p = (uschar)tolower((uschar)*p); -+ if ( (*p < 97) || (*p > 122) ) { -+ extension = default_extension; -+ break; -+ }; -+ p++; -+ }; -+ -+ /* find a new file to write to */ -+ file_nr = 0; -+ do { -+ struct stat mystat; -+ -+ snprintf(CS file_name,1024,"%s/scan/%s/%s-%05u%s",spool_directory,message_id,message_id,file_nr,extension); -+ file_nr++; -+ if (file_nr >= MIME_SANITY_MAX_DUMP_FILES) { -+ /* max parts reached */ -+ mime_trigger_error(MIME_ERRORLEVEL_TOO_MANY_PARTS); -+ break; -+ }; -+ result = stat(CS file_name,&mystat); -+ } -+ while(result != -1); -+ -+ *f = fopen(CS file_name,"w"); -+ if (*f == NULL) { -+ /* cannot open new dump file, disk full ? -> soft error */ -+ snprintf(CS info, 1024,"unable to open dump file"); -+ return -2; -+ }; -+ -+ return file_nr; -+} -+ -+ -+/************************************************* -+* Find a string in a mime header * -+*************************************************/ -+ -+/* Find a string in a mime header, and optionally fill in -+ the value associated with it into *value -+ -+ returns: 0 - nothing found -+ 1 - found param -+ 2 - found param + value -+*/ -+ -+int mime_header_find(uschar *header, uschar *param, uschar **value) { -+ uschar *needle; -+ -+ needle = strstric(header,param,FALSE); -+ if (needle != NULL) { -+ if (value != NULL) { -+ needle += Ustrlen(param); -+ if (*needle == '=') { -+ uschar *value_start; -+ uschar *value_end; -+ -+ value_start = needle + 1; -+ value_end = strstric(value_start,US";",FALSE); -+ if (value_end != NULL) { -+ /* allocate mem for value */ -+ *value = (uschar *)malloc((value_end - value_start)+1); -+ if (*value == NULL) -+ return 0; -+ -+ Ustrncpy(*value,value_start,(value_end - value_start)); -+ (*value)[(value_end - value_start)] = '\0'; -+ return 2; -+ }; -+ }; -+ }; -+ return 1; -+ }; -+ return 0; -+} -+ -+ -+/************************************************* -+* Read a line of MIME input * -+*************************************************/ -+/* returns status code, one of -+ MIME_READ_LINE_EOF 0 -+ MIME_READ_LINE_OK 1 -+ MIME_READ_LINE_OVERFLOW 2 -+ -+ In header mode, the line will be "cooked". -+*/ -+ -+int mime_read_line(FILE *f, int mime_demux_mode, uschar *buffer, long *num_copied) { -+ int c; -+ int done = 0; -+ int header_value_mode = 0; -+ int header_open_brackets = 0; -+ -+ *num_copied = 0; -+ -+ while(!done) { -+ -+ c = fgetc(f); -+ if (c == EOF) break; -+ -+ /* --------- header mode -------------- */ -+ if (mime_demux_mode == MIME_DEMUX_MODE_MIME_HEADERS) { -+ -+ /* always skip CRs */ -+ if (c == '\r') continue; -+ -+ if (c == '\n') { -+ /* look if next char is '\t' or ' ' */ -+ c = fgetc(f); -+ if (c == EOF) break; -+ if ( (c == '\t') || (c == ' ') ) continue; -+ /* end of the header, terminate with ';' */ -+ ungetc(c,f); -+ c = ';'; -+ done = 1; -+ }; -+ -+ /* skip control characters */ -+ if (c < 32) continue; -+ -+ /* skip whitespace + tabs */ -+ if ( (c == ' ') || (c == '\t') ) -+ continue; -+ -+ if (header_value_mode) { -+ /* --------- value mode ----------- */ -+ /* skip quotes */ -+ if (c == '"') continue; -+ -+ /* leave value mode on ';' */ -+ if (c == ';') { -+ header_value_mode = 0; -+ }; -+ /* -------------------------------- */ -+ } -+ else { -+ /* -------- non-value mode -------- */ -+ if (c == '\\') { -+ /* quote next char. can be used -+ to escape brackets. */ -+ c = fgetc(f); -+ if (c == EOF) break; -+ } -+ else if (c == '(') { -+ header_open_brackets++; -+ continue; -+ } -+ else if ((c == ')') && header_open_brackets) { -+ header_open_brackets--; -+ continue; -+ } -+ else if ( (c == '=') && !header_open_brackets ) { -+ /* enter value mode */ -+ header_value_mode = 1; -+ }; -+ -+ /* skip chars while we are in a comment */ -+ if (header_open_brackets > 0) -+ continue; -+ /* -------------------------------- */ -+ }; -+ } -+ /* ------------------------------------ */ -+ else { -+ /* ----------- non-header mode -------- */ -+ /* break on '\n' */ -+ if (c == '\n') -+ done = 1; -+ /* ------------------------------------ */ -+ }; -+ -+ /* copy the char to the buffer */ -+ buffer[*num_copied] = (uschar)c; -+ /* raise counter */ -+ (*num_copied)++; -+ -+ /* break if buffer is full */ -+ if (*num_copied > MIME_SANITY_MAX_LINE_LENGTH-1) { -+ done = 1; -+ }; -+ } -+ -+ /* 0-terminate */ -+ buffer[*num_copied] = '\0'; -+ -+ if (*num_copied > MIME_SANITY_MAX_LINE_LENGTH-1) -+ return MIME_READ_LINE_OVERFLOW; -+ else -+ if (c == EOF) -+ return MIME_READ_LINE_EOF; -+ else -+ return MIME_READ_LINE_OK; -+} -+ -+ -+/************************************************* -+* Check for a MIME boundary * -+*************************************************/ -+ -+/* returns: 0 - no boundary found -+ 1 - start boundary found -+ 2 - end boundary found -+*/ -+ -+int mime_check_boundary(uschar *line, struct boundary *boundaries) { -+ struct boundary *thisboundary = boundaries; -+ -+ /* check for '--' first */ -+ if (Ustrncmp(line,"--",2) == 0) { -+ while(thisboundary != NULL) { -+ if (Ustrncmp(&line[2],thisboundary->boundary_string,Ustrlen(thisboundary->boundary_string)) == 0) { -+ if (Ustrncmp(&line[(2+Ustrlen(thisboundary->boundary_string))],"--",2) == 0) { -+ /* final boundary found */ -+ return 2; -+ }; -+ return 1; -+ }; -+ thisboundary = thisboundary->next; -+ }; -+ }; -+ -+ return 0; -+} -+ -+ -+/************************************************* -+* Check for start of a UUENCODE block * -+*************************************************/ -+ -+/* returns 0 for no hit, -+ >0 for hit -+*/ -+ -+int mime_check_uu_start(uschar *line, uschar *uu_file_extension, int *has_tnef) { -+ -+ if ( (strncmpic(line,US"begin ",6) == 0)) { -+ uschar *uu_filename = &line[6]; -+ -+ /* skip perms, if present */ -+ Ustrtoul(&line[6],&uu_filename,10); -+ -+ /* advance one char */ -+ uu_filename++; -+ -+ /* This should be the filename. -+ Check if winmail.dat is present, -+ which indicates TNEF. */ -+ if (strncmpic(uu_filename,US"winmail.dat",11) == 0) { -+ *has_tnef = 1; -+ }; -+ -+ /* reverse to dot if present, -+ copy up to 4 chars for the extension */ -+ if (Ustrrchr(uu_filename,'.') != NULL) -+ uu_filename = Ustrrchr(uu_filename,'.'); -+ -+ return sscanf(CS uu_filename, "%4[.0-9A-Za-z]",CS uu_file_extension); -+ } -+ else { -+ /* nothing found */ -+ return 0; -+ }; -+} -+ -+ -+/************************************************* -+* Decode a uu line * -+*************************************************/ -+ -+/* returns number of decoded bytes -+ -2 for soft errors -+*/ -+ -+int warned_about_uudec_line_sanity_1 = 0; -+int warned_about_uudec_line_sanity_2 = 0; -+long uu_decode_line(uschar *line, uschar **data, long line_len, uschar *info) { -+ uschar *p; -+ long num_decoded = 0; -+ uschar tmp_c; -+ uschar *work; -+ int uu_decoded_line_len, uu_encoded_line_len; -+ -+ /* allocate memory for data and work buffer */ -+ *data = (uschar *)malloc(line_len); -+ if (*data == NULL) { -+ snprintf(CS info, 1024,"unable to allocate %u bytes",line_len); -+ return -2; -+ }; -+ -+ work = (uschar *)malloc(line_len); -+ if (work == NULL) { -+ snprintf(CS info, 1024,"unable to allocate %u bytes",line_len); -+ return -2; -+ }; -+ -+ memcpy(work,line,line_len); -+ -+ /* First char is line length -+ This is microsofts way of getting it. Scary. */ -+ if (work[0] < 32) { -+ /* ignore this line */ -+ return 0; -+ } -+ else { -+ uu_decoded_line_len = uudec[work[0]]; -+ }; -+ -+ p = &work[1]; -+ -+ while (*p > 32) { -+ *p = uudec[*p]; -+ p++; -+ }; -+ -+ uu_encoded_line_len = (p - &work[1]); -+ p = &work[1]; -+ -+ /* check that resulting line length is a multiple of 4 */ -+ if ( ( uu_encoded_line_len % 4 ) != 0) { -+ if (!warned_about_uudec_line_sanity_1) { -+ mime_trigger_error(MIME_ERRORLEVEL_UU_MISALIGNED); -+ warned_about_uudec_line_sanity_1 = 1; -+ }; -+ return -1; -+ }; -+ -+ /* check that the line length matches */ -+ if ( ( (((uu_encoded_line_len/4)*3)-2) > uu_decoded_line_len ) || (((uu_encoded_line_len/4)*3) < uu_decoded_line_len) ) { -+ if (!warned_about_uudec_line_sanity_2) { -+ mime_trigger_error(MIME_ERRORLEVEL_UU_LINE_LENGTH); -+ warned_about_uudec_line_sanity_2 = 1; -+ }; -+ return -1; -+ }; -+ -+ while ( ((p - &work[1]) < uu_encoded_line_len) && (num_decoded < uu_decoded_line_len)) { -+ -+ /* byte 0 ---------------------- */ -+ if ((p - &work[1] + 1) >= uu_encoded_line_len) { -+ return 0; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 2; -+ -+ tmp_c = *(p+1); -+ tmp_c >>= 4; -+ (*data)[num_decoded] |= tmp_c; -+ -+ num_decoded++; -+ p++; -+ -+ /* byte 1 ---------------------- */ -+ if ((p - &work[1] + 1) >= uu_encoded_line_len) { -+ return 0; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 4; -+ -+ tmp_c = *(p+1); -+ tmp_c >>= 2; -+ (*data)[num_decoded] |= tmp_c; -+ -+ num_decoded++; -+ p++; -+ -+ /* byte 2 ---------------------- */ -+ if ((p - &work[1] + 1) >= uu_encoded_line_len) { -+ return 0; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 6; -+ -+ (*data)[num_decoded] |= *(p+1); -+ -+ num_decoded++; -+ p+=2; -+ -+ }; -+ -+ return uu_decoded_line_len; -+} -+ -+ -+/************************************************* -+* Decode a b64 or qp line * -+*************************************************/ -+ -+/* returns number of decoded bytes -+ -1 for hard errors -+ -2 for soft errors -+*/ -+ -+int warned_about_b64_line_length = 0; -+int warned_about_b64_line_sanity = 0; -+int warned_about_b64_illegal_char = 0; -+int warned_about_qp_line_sanity = 0; -+long mime_decode_line(int mime_demux_mode,uschar *line, uschar **data, long max_data_len, uschar *info) { -+ uschar *p; -+ long num_decoded = 0; -+ int offset = 0; -+ uschar tmp_c; -+ -+ /* allocate memory for data */ -+ *data = (uschar *)malloc(max_data_len); -+ if (*data == NULL) { -+ snprintf(CS info, 1024,"unable to allocate %u bytes",max_data_len); -+ return -2; -+ }; -+ -+ if (mime_demux_mode == MIME_DEMUX_MODE_BASE64) { -+ /* ---------------------------------------------- */ -+ -+ /* NULL out trailing '\r' and '\n' chars */ -+ while (Ustrrchr(line,'\r') != NULL) { -+ *(Ustrrchr(line,'\r')) = '\0'; -+ }; -+ while (Ustrrchr(line,'\n') != NULL) { -+ *(Ustrrchr(line,'\n')) = '\0'; -+ }; -+ -+ /* check maximum base 64 line length */ -+ if (Ustrlen(line) > MIME_SANITY_MAX_B64_LINE_LENGTH ) { -+ if (!warned_about_b64_line_length) { -+ mime_trigger_error(MIME_ERRORLEVEL_B64_LINE_LENGTH); -+ warned_about_b64_line_length = 1; -+ }; -+ }; -+ -+ p = line; -+ offset = 0; -+ while (*(p+offset) != '\0') { -+ /* hit illegal char ? */ -+ if (b64[*(p+offset)] == 128) { -+ if (!warned_about_b64_illegal_char) { -+ mime_trigger_error(MIME_ERRORLEVEL_B64_ILLEGAL_CHAR); -+ warned_about_b64_illegal_char = 1; -+ }; -+ offset++; -+ } -+ else { -+ *p = b64[*(p+offset)]; -+ p++; -+ }; -+ }; -+ *p = 255; -+ -+ /* check that resulting line length is a multiple of 4 */ -+ if ( ( (p - &line[0]) % 4 ) != 0) { -+ if (!warned_about_b64_line_sanity) { -+ mime_trigger_error(MIME_ERRORLEVEL_B64_MISALIGNED); -+ warned_about_b64_line_sanity = 1; -+ }; -+ }; -+ -+ /* line is translated, start bit shifting */ -+ p = line; -+ num_decoded = 0; -+ -+ while(*p != 255) { -+ -+ /* byte 0 ---------------------- */ -+ if (*(p+1) == 255) { -+ break; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 2; -+ -+ tmp_c = *(p+1); -+ tmp_c >>= 4; -+ (*data)[num_decoded] |= tmp_c; -+ -+ num_decoded++; -+ p++; -+ -+ /* byte 1 ---------------------- */ -+ if (*(p+1) == 255) { -+ break; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 4; -+ -+ tmp_c = *(p+1); -+ tmp_c >>= 2; -+ (*data)[num_decoded] |= tmp_c; -+ -+ num_decoded++; -+ p++; -+ -+ /* byte 2 ---------------------- */ -+ if (*(p+1) == 255) { -+ break; -+ } -+ -+ (*data)[num_decoded] = *p; -+ (*data)[num_decoded] <<= 6; -+ -+ (*data)[num_decoded] |= *(p+1); -+ -+ num_decoded++; -+ p+=2; -+ -+ }; -+ return num_decoded; -+ /* ---------------------------------------------- */ -+ } -+ else if (mime_demux_mode == MIME_DEMUX_MODE_QP) { -+ /* ---------------------------------------------- */ -+ p = line; -+ -+ while (*p != 0) { -+ if (*p == '=') { -+ int decode_qp_result; -+ -+ p = mime_decode_qp(p,&decode_qp_result); -+ -+ if (decode_qp_result == -2) { -+ /* Error from decoder. p is unchanged. */ -+ if (!warned_about_qp_line_sanity) { -+ mime_trigger_error(MIME_ERRORLEVEL_QP_ILLEGAL_CHAR); -+ warned_about_qp_line_sanity = 1; -+ }; -+ (*data)[num_decoded] = '='; -+ num_decoded++; -+ p++; -+ } -+ else if (decode_qp_result == -1) { -+ /* End of the line with soft line break. -+ Bail out. */ -+ goto QP_RETURN; -+ } -+ else if (decode_qp_result >= 0) { -+ (*data)[num_decoded] = decode_qp_result; -+ num_decoded++; -+ }; -+ } -+ else { -+ (*data)[num_decoded] = *p; -+ num_decoded++; -+ p++; -+ }; -+ }; -+ QP_RETURN: -+ return num_decoded; -+ /* ---------------------------------------------- */ -+ }; -+ -+ return 0; -+} -+ -+ -+ -+/************************************************* -+* Log demime errors and set mime error level * -+*************************************************/ -+ -+/* This sets the global demime_reason expansion -+variable and the demime_errorlevel gauge. */ -+ -+void mime_trigger_error(int level, uschar *format, ...) { -+ char *f; -+ va_list ap; -+ -+ if( (f = malloc(16384+23)) != NULL ) { -+ /* first log the incident */ -+ sprintf(f,"demime acl condition: "); -+ f+=22; -+ va_start(ap, format); -+ vsnprintf(f, 16383,(char *)format, ap); -+ va_end(ap); -+ f-=22; -+ log_write(0, LOG_MAIN|LOG_PANIC, f); -+ /* then copy to demime_reason_buffer if new -+ level is greater than old level */ -+ if (level > demime_errorlevel) { -+ demime_errorlevel = level; -+ Ustrcpy(demime_reason_buffer, US f); -+ demime_reason = demime_reason_buffer; -+ }; -+ free(f); -+ }; -+} -+ -+/************************************************* -+* Demultiplex MIME stream. * -+*************************************************/ -+ -+/* We can handle BASE64, QUOTED-PRINTABLE, and UUENCODE. -+ UUENCODE does not need to have a proper -+ transfer-encoding header, we detect it with "begin" -+ -+ This function will report human parsable errors in -+ *info. -+ -+ returns DEFER -> soft error (see *info) -+ OK -> EOF hit, all ok -+*/ -+ -+int mime_demux(FILE *f, uschar *info) { -+ int mime_demux_mode = MIME_DEMUX_MODE_MIME_HEADERS; -+ int uu_mode = MIME_UU_MODE_OFF; -+ FILE *mime_dump_file = NULL; -+ FILE *uu_dump_file = NULL; -+ uschar *line; -+ int mime_read_line_status = MIME_READ_LINE_OK; -+ long line_len; -+ struct boundary *boundaries = NULL; -+ struct mime_part mime_part_p; -+ int has_tnef = 0; -+ -+ /* allocate room for our linebuffer */ -+ line = (uschar *)malloc(MIME_SANITY_MAX_LINE_LENGTH); -+ if (line == NULL) { -+ snprintf(CS info, 1024,"unable to allocate %u bytes",MIME_SANITY_MAX_LINE_LENGTH); -+ return DEFER; -+ }; -+ -+ /* clear MIME header structure */ -+ memset(&mime_part_p,0,sizeof(mime_part)); -+ -+ /* ----------------------- start demux loop --------------------- */ -+ while (mime_read_line_status == MIME_READ_LINE_OK) { -+ -+ /* read a line of input. Depending on the mode we are in, -+ the returned format will differ. */ -+ mime_read_line_status = mime_read_line(f,mime_demux_mode,line,&line_len); -+ if (mime_read_line_status == MIME_READ_LINE_OVERFLOW) { -+ mime_trigger_error(MIME_ERRORLEVEL_LONG_LINE); -+ /* despite the error, continue .. */ -+ mime_read_line_status == MIME_READ_LINE_OK; -+ continue; -+ } -+ else if (mime_read_line_status == MIME_READ_LINE_EOF) { -+ break; -+ }; -+ -+ if (mime_demux_mode == MIME_DEMUX_MODE_MIME_HEADERS) { -+ /* -------------- header mode --------------------- */ -+ -+ /* Check for an empty line, which is the end of the headers. -+ In HEADER mode, the line is returned "cooked", with the -+ final '\n' replaced by a ';' */ -+ if (line_len == 1) { -+ int tmp; -+ -+ /* We have reached the end of the headers. Start decoding -+ with the collected settings. */ -+ if (mime_part_p.seen_content_transfer_encoding > 1) { -+ mime_demux_mode = mime_part_p.seen_content_transfer_encoding; -+ } -+ else { -+ /* default to plain mode if no specific encoding type found */ -+ mime_demux_mode = MIME_DEMUX_MODE_PLAIN; -+ }; -+ -+ /* open new dump file */ -+ tmp = mime_get_dump_file(mime_part_p.extension, &mime_dump_file, info); -+ if (tmp < 0) { -+ return DEFER; -+ }; -+ -+ /* clear out mime_part */ -+ memset(&mime_part_p,0,sizeof(mime_part)); -+ } -+ else { -+ /* Another header to check for file extensions, -+ encoding type and boundaries */ -+ if (strncmpic(US"content-type:",line,Ustrlen("content-type:")) == 0) { -+ /* ---------------------------- Content-Type header ------------------------------- */ -+ uschar *value = line; -+ -+ /* check for message/partial MIME type and reject it */ -+ if (mime_header_find(line,US"message/partial",NULL) > 0) -+ mime_trigger_error(MIME_ERRORLEVEL_MESSAGE_PARTIAL); -+ -+ /* check for TNEF content type, remember to unpack TNEF later. */ -+ if (mime_header_find(line,US"application/ms-tnef",NULL) > 0) -+ has_tnef = 1; -+ -+ /* find the file extension, but do not fill it in -+ it is already set, since content-disposition has -+ precedence. */ -+ if (mime_part_p.extension == NULL) { -+ if (mime_header_find(line,US"name",&value) == 2) { -+ if (Ustrlen(value) > MIME_SANITY_MAX_FILENAME) -+ mime_trigger_error(MIME_ERRORLEVEL_FILENAME_LENGTH); -+ mime_part_p.extension = value; -+ mime_part_p.extension = Ustrrchr(value,'.'); -+ if (mime_part_p.extension == NULL) { -+ /* file without extension, setting -+ NULL will use the default extension later */ -+ mime_part_p.extension = NULL; -+ } -+ else { -+ struct file_extension *this_extension = -+ (struct file_extension *)malloc(sizeof(file_extension)); -+ -+ this_extension->file_extension_string = -+ (uschar *)malloc(Ustrlen(mime_part_p.extension)+1); -+ Ustrcpy(this_extension->file_extension_string, -+ mime_part_p.extension+1); -+ this_extension->next = file_extensions; -+ file_extensions = this_extension; -+ }; -+ }; -+ }; -+ -+ /* find a boundary and add it to the list, if present */ -+ value = line; -+ if (mime_header_find(line,US"boundary",&value) == 2) { -+ struct boundary *thisboundary; -+ -+ if (Ustrlen(value) > MIME_SANITY_MAX_BOUNDARY_LENGTH) { -+ mime_trigger_error(MIME_ERRORLEVEL_BOUNDARY_LENGTH); -+ } -+ else { -+ thisboundary = (struct boundary*)malloc(sizeof(boundary)); -+ thisboundary->next = boundaries; -+ thisboundary->boundary_string = value; -+ boundaries = thisboundary; -+ }; -+ }; -+ -+ if (mime_part_p.seen_content_type == 0) { -+ mime_part_p.seen_content_type = 1; -+ } -+ else { -+ mime_trigger_error(MIME_ERRORLEVEL_DOUBLE_HEADERS); -+ }; -+ /* ---------------------------------------------------------------------------- */ -+ } -+ else if (strncmpic(US"content-transfer-encoding:",line,Ustrlen("content-transfer-encoding:")) == 0) { -+ /* ---------------------------- Content-Transfer-Encoding header -------------- */ -+ -+ if (mime_part_p.seen_content_transfer_encoding == 0) { -+ if (mime_header_find(line,US"base64",NULL) > 0) { -+ mime_part_p.seen_content_transfer_encoding = MIME_DEMUX_MODE_BASE64; -+ } -+ else if (mime_header_find(line,US"quoted-printable",NULL) > 0) { -+ mime_part_p.seen_content_transfer_encoding = MIME_DEMUX_MODE_QP; -+ } -+ else { -+ mime_part_p.seen_content_transfer_encoding = MIME_DEMUX_MODE_PLAIN; -+ }; -+ } -+ else { -+ mime_trigger_error(MIME_ERRORLEVEL_DOUBLE_HEADERS); -+ }; -+ /* ---------------------------------------------------------------------------- */ -+ } -+ else if (strncmpic(US"content-disposition:",line,Ustrlen("content-disposition:")) == 0) { -+ /* ---------------------------- Content-Disposition header -------------------- */ -+ uschar *value = line; -+ -+ if (mime_part_p.seen_content_disposition == 0) { -+ mime_part_p.seen_content_disposition = 1; -+ -+ if (mime_header_find(line,US"filename",&value) == 2) { -+ if (Ustrlen(value) > MIME_SANITY_MAX_FILENAME) -+ mime_trigger_error(MIME_ERRORLEVEL_FILENAME_LENGTH); -+ mime_part_p.extension = value; -+ mime_part_p.extension = Ustrrchr(value,'.'); -+ if (mime_part_p.extension == NULL) { -+ /* file without extension, setting -+ NULL will use the default extension later */ -+ mime_part_p.extension = NULL; -+ } -+ else { -+ struct file_extension *this_extension = -+ (struct file_extension *)malloc(sizeof(file_extension)); -+ -+ this_extension->file_extension_string = -+ (uschar *)malloc(Ustrlen(mime_part_p.extension)+1); -+ Ustrcpy(this_extension->file_extension_string, -+ mime_part_p.extension+1); -+ this_extension->next = file_extensions; -+ file_extensions = this_extension; -+ }; -+ }; -+ } -+ else { -+ mime_trigger_error(MIME_ERRORLEVEL_DOUBLE_HEADERS); -+ }; -+ /* ---------------------------------------------------------------------------- */ -+ }; -+ }; /* End of header checks */ -+ /* ------------------------------------------------ */ -+ } -+ else { -+ /* -------------- non-header mode ----------------- */ -+ int tmp; -+ -+ if (uu_mode == MIME_UU_MODE_OFF) { -+ uschar uu_file_extension[5]; -+ /* We are not currently decoding UUENCODE -+ Check for possible UUENCODE start tag. */ -+ if (mime_check_uu_start(line,uu_file_extension,&has_tnef)) { -+ /* possible UUENCODING start detected. -+ Set unconfirmed mode first. */ -+ uu_mode = MIME_UU_MODE_UNCONFIRMED; -+ /* open new uu dump file */ -+ tmp = mime_get_dump_file(uu_file_extension, &uu_dump_file, info); -+ if (tmp < 0) { -+ free(line); -+ return DEFER; -+ }; -+ }; -+ } -+ else { -+ uschar *data; -+ long data_len = 0; -+ -+ if (uu_mode == MIME_UU_MODE_UNCONFIRMED) { -+ /* We are in unconfirmed UUENCODE mode. */ -+ -+ data_len = uu_decode_line(line,&data,line_len,info); -+ -+ if (data_len == -2) { -+ /* temp error, turn off uudecode mode */ -+ if (uu_dump_file != NULL) { -+ fclose(uu_dump_file); uu_dump_file = NULL; -+ }; -+ uu_mode = MIME_UU_MODE_OFF; -+ return DEFER; -+ } -+ else if (data_len == -1) { -+ if (uu_dump_file != NULL) { -+ fclose(uu_dump_file); uu_dump_file = NULL; -+ }; -+ uu_mode = MIME_UU_MODE_OFF; -+ data_len = 0; -+ } -+ else if (data_len > 0) { -+ /* we have at least decoded a valid byte -+ turn on confirmed mode */ -+ uu_mode = MIME_UU_MODE_CONFIRMED; -+ }; -+ } -+ else if (uu_mode == MIME_UU_MODE_CONFIRMED) { -+ /* If we are in confirmed UU mode, -+ check for single "end" tag on line */ -+ if ((strncmpic(line,US"end",3) == 0) && (line[3] < 32)) { -+ if (uu_dump_file != NULL) { -+ fclose(uu_dump_file); uu_dump_file = NULL; -+ }; -+ uu_mode = MIME_UU_MODE_OFF; -+ } -+ else { -+ data_len = uu_decode_line(line,&data,line_len,info); -+ if (data_len == -2) { -+ /* temp error, turn off uudecode mode */ -+ if (uu_dump_file != NULL) { -+ fclose(uu_dump_file); uu_dump_file = NULL; -+ }; -+ uu_mode = MIME_UU_MODE_OFF; -+ return DEFER; -+ } -+ else if (data_len == -1) { -+ /* skip this line */ -+ data_len = 0; -+ }; -+ }; -+ }; -+ -+ /* write data to dump file, if available */ -+ if (data_len > 0) { -+ if (fwrite(data,1,data_len,uu_dump_file) < data_len) { -+ /* short write */ -+ snprintf(CS info, 1024,"short write on uudecode dump file"); -+ free(line); -+ return DEFER; -+ }; -+ }; -+ }; -+ -+ if (mime_demux_mode != MIME_DEMUX_MODE_SCANNING) { -+ /* Non-scanning and Non-header mode. That means -+ we are currently decoding data to the dump -+ file. */ -+ -+ /* Check for a known boundary. */ -+ tmp = mime_check_boundary(line,boundaries); -+ if (tmp == 1) { -+ /* We have hit a known start boundary. -+ That will put us back in header mode. */ -+ mime_demux_mode = MIME_DEMUX_MODE_MIME_HEADERS; -+ if (mime_dump_file != NULL) { -+ fclose(mime_dump_file); mime_dump_file = NULL; -+ }; -+ } -+ else if (tmp == 2) { -+ /* We have hit a known end boundary. -+ That puts us into scanning mode, which will end when we hit another known start boundary */ -+ mime_demux_mode = MIME_DEMUX_MODE_SCANNING; -+ if (mime_dump_file != NULL) { -+ fclose(mime_dump_file); mime_dump_file = NULL; -+ }; -+ } -+ else { -+ uschar *data; -+ long data_len; -+ -+ /* decode the line with the appropriate method */ -+ if (mime_demux_mode == MIME_DEMUX_MODE_PLAIN) { -+ /* in plain mode, just dump the line */ -+ data = line; -+ data_len = line_len; -+ } -+ else if ( (mime_demux_mode == MIME_DEMUX_MODE_QP) || (mime_demux_mode == MIME_DEMUX_MODE_BASE64) ) { -+ data_len = mime_decode_line(mime_demux_mode,line,&data,line_len,info); -+ if (data_len < 0) { -+ /* Error reported from the line decoder. */ -+ data_len = 0; -+ }; -+ }; -+ -+ /* write data to dump file */ -+ if (data_len > 0) { -+ if (fwrite(data,1,data_len,mime_dump_file) < data_len) { -+ /* short write */ -+ snprintf(CS info, 1024,"short write on dump file"); -+ free(line); -+ return DEFER; -+ }; -+ }; -+ -+ }; -+ } -+ else { -+ /* Scanning mode. We end up here after a end boundary. -+ This will usually be at the end of a message or at -+ the end of a MIME container. -+ We need to look for another start boundary to get -+ back into header mode. */ -+ if (mime_check_boundary(line,boundaries) == 1) { -+ mime_demux_mode = MIME_DEMUX_MODE_MIME_HEADERS; -+ }; -+ -+ }; -+ /* ------------------------------------------------ */ -+ }; -+ }; -+ /* ----------------------- end demux loop ----------------------- */ -+ -+ /* close files, they could still be open */ -+ if (mime_dump_file != NULL) -+ fclose(mime_dump_file); -+ if (uu_dump_file != NULL) -+ fclose(uu_dump_file); -+ -+ /* release line buffer */ -+ free(line); -+ -+ /* FIXME: release boundary buffers. -+ Not too much of a problem since -+ this instance of exim is not resident. */ -+ -+ if (has_tnef) { -+ uschar file_name[1024]; -+ /* at least one file could be TNEF encoded. -+ attempt to send all decoded files thru the TNEF decoder */ -+ -+ snprintf(CS file_name,1024,"%s/scan/%s",spool_directory,message_id); -+ mime_unpack_tnef(file_name); -+ }; -+ -+ return 0; -+} -+ -diff -urN exim-4.20-orig/src/demime.h exim-4.20/src/demime.h ---- exim-4.20-orig/src/demime.h Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/demime.h Wed May 14 12:04:24 2003 -@@ -0,0 +1,146 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* demime defines */ -+ -+#define MIME_DEMUX_MODE_SCANNING 0 -+#define MIME_DEMUX_MODE_MIME_HEADERS 1 -+#define MIME_DEMUX_MODE_BASE64 2 -+#define MIME_DEMUX_MODE_QP 3 -+#define MIME_DEMUX_MODE_PLAIN 4 -+ -+#define MIME_UU_MODE_OFF 0 -+#define MIME_UU_MODE_UNCONFIRMED 1 -+#define MIME_UU_MODE_CONFIRMED 2 -+ -+#define MIME_MAX_EXTENSION 128 -+ -+#define MIME_READ_LINE_EOF 0 -+#define MIME_READ_LINE_OK 1 -+#define MIME_READ_LINE_OVERFLOW 2 -+ -+#define MIME_SANITY_MAX_LINE_LENGTH 131071 -+#define MIME_SANITY_MAX_FILENAME 512 -+#define MIME_SANITY_MAX_HEADER_OPTION_VALUE 1024 -+#define MIME_SANITY_MAX_B64_LINE_LENGTH 76 -+#define MIME_SANITY_MAX_BOUNDARY_LENGTH 1024 -+#define MIME_SANITY_MAX_DUMP_FILES 1024 -+ -+ -+ -+/* MIME errorlevel settings */ -+ -+#define MIME_ERRORLEVEL_LONG_LINE 3,US"line length in message or single header size exceeds %u bytes",MIME_SANITY_MAX_LINE_LENGTH -+#define MIME_ERRORLEVEL_TOO_MANY_PARTS 3,US"too many MIME parts (max %u)",MIME_SANITY_MAX_DUMP_FILES -+#define MIME_ERRORLEVEL_MESSAGE_PARTIAL 3,US"'message/partial' MIME type" -+#define MIME_ERRORLEVEL_FILENAME_LENGTH 3,US"proposed filename exceeds %u characters",MIME_SANITY_MAX_FILENAME -+#define MIME_ERRORLEVEL_BOUNDARY_LENGTH 3,US"boundary length exceeds %u characters",MIME_SANITY_MAX_BOUNDARY_LENGTH -+#define MIME_ERRORLEVEL_DOUBLE_HEADERS 2,US"double headers (content-type, content-disposition or content-transfer-encoding)" -+#define MIME_ERRORLEVEL_UU_MISALIGNED 1,US"uuencoded line length is not a multiple of 4 characters" -+#define MIME_ERRORLEVEL_UU_LINE_LENGTH 1,US"uuencoded line length does not match advertised number of bytes" -+#define MIME_ERRORLEVEL_B64_LINE_LENGTH 1,US"base64 line length exceeds %u characters",MIME_SANITY_MAX_B64_LINE_LENGTH -+#define MIME_ERRORLEVEL_B64_ILLEGAL_CHAR 2,US"base64 line contains illegal character" -+#define MIME_ERRORLEVEL_B64_MISALIGNED 1,US"base64 line length is not a multiple of 4 characters" -+#define MIME_ERRORLEVEL_QP_ILLEGAL_CHAR 1,US"quoted-printable encoding contains illegal character" -+ -+ -+/* demime structures */ -+ -+typedef struct mime_part { -+ /* true if there was a content-type header */ -+ int seen_content_type; -+ /* true if there was a content-transfer-encoding header -+ contains the encoding type */ -+ int seen_content_transfer_encoding; -+ /* true if there was a content-disposition header */ -+ int seen_content_disposition; -+ /* pointer to a buffer with the proposed file extension */ -+ uschar *extension; -+} mime_part; -+ -+typedef struct boundary { -+ struct boundary *next; -+ uschar *boundary_string; -+} boundary; -+ -+typedef struct file_extension { -+ struct file_extension *next; -+ uschar *file_extension_string; -+} file_extension; -+ -+/* available functions for the TNEF library (tnef.c & tnef.h) */ -+ -+extern int TNEF_main( char *filename ); -+extern int TNEF_set_verbosity( int level ); -+extern int TNEF_set_debug( int level ); -+extern int TNEF_set_syslogging( int level ); -+extern int TNEF_set_stderrlogging( int level ); -+extern int TNEF_set_path( char *path ); -+ -+ -+ -+/* demime.c prototypes */ -+ -+int mime_unpack_tnef(uschar *); -+unsigned int mime_hstr_i(uschar *); -+uschar *mime_decode_qp(uschar *, int *); -+int mime_get_dump_file(uschar *, FILE **, uschar *); -+int mime_header_find(uschar *, uschar *, uschar **); -+int mime_read_line(FILE *, int, uschar *, long *); -+int mime_check_boundary(uschar *, struct boundary *); -+int mime_check_uu_start(uschar *, uschar *, int *); -+long uu_decode_line(uschar *, uschar **, long, uschar *); -+long mime_decode_line(int ,uschar *, uschar **, long, uschar *); -+void mime_trigger_error(int, uschar *, ...); -+int mime_demux(FILE *, uschar *); -+ -+ -+ -+/* BASE64 decoder matrix */ -+static unsigned char b64[256]={ -+/* 0 */ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+/* 16 */ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+/* 32 */ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 62, 128, 128, 128, 63, -+/* 48 */ 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 128, 128, 128, 255, 128, 128, -+/* 64 */ 128, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, -+/* 80 */ 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 128, 128, 128, 128, 128, -+/* 96 */ 128, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, -+ 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, -+ 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128 -+}; -+ -+ -+/* Microsoft-Style uudecode matrix */ -+static unsigned char uudec[256]={ -+/* 0 */ 0, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, -+/* 16 */ 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -+/* 32 */ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, -+/* 48 */ 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, -+/* 64 */ 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, -+/* 80 */ 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -+/* 96 */ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, -+/* 112 */ 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, -+/* 128 */ 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, -+/* 144 */ 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -+/* 160 */ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, -+/* 176 */ 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, -+/* 192 */ 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, -+/* 208 */ 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -+/* 224 */ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, -+/* 240 */ 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31 -+}; -+ -diff -urN exim-4.20-orig/src/expand.c exim-4.20/src/expand.c ---- exim-4.20-orig/src/expand.c Mon May 12 15:39:19 2003 -+++ exim-4.20/src/expand.c Wed May 14 12:04:24 2003 -@@ -218,11 +218,14 @@ - { "caller_uid", vtype_int, &real_uid }, - { "compile_date", vtype_stringptr, &version_date }, - { "compile_number", vtype_stringptr, &version_cnumber }, -+ { "demime_errorlevel", vtype_int, &demime_errorlevel }, -+ { "demime_reason", vtype_stringptr, &demime_reason }, - { "dnslist_domain", vtype_stringptr, &dnslist_domain }, - { "dnslist_text", vtype_stringptr, &dnslist_text }, - { "dnslist_value", vtype_stringptr, &dnslist_value }, - { "domain", vtype_stringptr, &deliver_domain }, - { "domain_data", vtype_stringptr, &deliver_domain_data }, -+ { "found_extension", vtype_stringptr, &found_extension }, - { "home", vtype_stringptr, &deliver_home }, - { "host", vtype_stringptr, &deliver_host }, - { "host_address", vtype_stringptr, &deliver_host_address }, -@@ -241,6 +244,7 @@ - { "local_part_suffix", vtype_stringptr, &deliver_localpart_suffix }, - { "local_scan_data", vtype_stringptr, &local_scan_data }, - { "localhost_number", vtype_int, &host_number }, -+ { "malware_name", vtype_stringptr, &malware_name }, - { "message_age", vtype_int, &message_age }, - { "message_body", vtype_msgbody, &message_body }, - { "message_body_end", vtype_msgbody_end, &message_body_end }, -@@ -275,6 +279,7 @@ - { "received_protocol", vtype_stringptr, &received_protocol }, - { "recipients", vtype_recipients, NULL }, - { "recipients_count", vtype_int, &recipients_count }, -+ { "regex_match_string", vtype_stringptr, ®ex_match_string }, - { "reply_address", vtype_reply, NULL }, - { "return_path", vtype_stringptr, &return_path }, - { "return_size_limit", vtype_int, &return_size_limit }, -@@ -302,6 +307,10 @@ - { "sn7", vtype_filter_int, &filter_sn[7] }, - { "sn8", vtype_filter_int, &filter_sn[8] }, - { "sn9", vtype_filter_int, &filter_sn[9] }, -+ { "spam_bar", vtype_stringptr, &spam_bar }, -+ { "spam_report", vtype_stringptr, &spam_report }, -+ { "spam_score", vtype_stringptr, &spam_score }, -+ { "spam_score_int", vtype_stringptr, &spam_score_int }, - { "spool_directory", vtype_stringptr, &spool_directory }, - { "thisaddress", vtype_stringptr, &filter_thisaddress }, - { "tls_certificate_verified", vtype_int, &tls_certificate_verified }, -diff -urN exim-4.20-orig/src/functions.h exim-4.20/src/functions.h ---- exim-4.20-orig/src/functions.h Mon May 12 15:39:19 2003 -+++ exim-4.20/src/functions.h Wed May 14 12:04:24 2003 -@@ -64,6 +64,7 @@ - extern void deliver_set_expansions(address_item *); - extern int deliver_split_address(address_item *); - extern void deliver_succeeded(address_item *); -+extern int demime(uschar **); - extern BOOL directory_make(uschar *, uschar *, int, BOOL); - extern dns_address *dns_address_from_rr(dns_answer *, dns_record *); - extern void dns_init(BOOL, BOOL); -@@ -114,6 +115,7 @@ - - extern void log_close_all(void); - -+extern int malware(uschar **); - extern int match_address_list(uschar *, BOOL, uschar **, unsigned int *, - int, int); - extern int match_check_list(uschar **, int, tree_node **, unsigned int **, -@@ -227,6 +229,8 @@ - extern BOOL smtp_start_session(void); - extern int smtp_ungetc(int); - extern int smtp_write_command(smtp_outblock *, BOOL, char *, ...); -+extern int spam(uschar **); -+extern FILE *spool_mbox(unsigned long long *); - extern BOOL spool_move_message(uschar *, uschar *, uschar *, uschar *); - extern BOOL spool_open_datafile(uschar *); - extern int spool_open_temp(uschar *); -@@ -277,6 +281,8 @@ - extern tree_node *tree_search(tree_node *, uschar *); - extern void tree_write(tree_node *, FILE *); - -+extern void unspool_mbox(void); -+ - extern int verify_address(address_item *, FILE *, int, int, BOOL *); - extern int verify_check_dnsbl(uschar **); - extern int verify_check_header_address(uschar **, uschar **, int); -diff -urN exim-4.20-orig/src/globals.c exim-4.20/src/globals.c ---- exim-4.20-orig/src/globals.c Mon May 12 15:39:19 2003 -+++ exim-4.20/src/globals.c Fri May 23 11:06:00 2003 -@@ -286,6 +286,7 @@ - uschar *auth_defer_msg = US"reason not recorded"; - uschar *auth_defer_user_msg = US""; - int auto_thaw = 0; -+uschar *av_scanner = US"sophie:/var/run/sophie"; - - BOOL background_daemon = TRUE; - uschar *base62_chars= -@@ -400,6 +401,8 @@ - BOOL deliver_selectstring_regex = FALSE; - uschar *deliver_selectstring_sender = NULL; - BOOL deliver_selectstring_sender_regex = FALSE; -+int demime_errorlevel = 0; -+uschar *demime_reason = NULL; - BOOL disable_logging = FALSE; - - uschar *dns_again_means_nonexist = NULL; -@@ -443,6 +446,7 @@ - uschar *filter_test = NULL; - uschar *filter_thisaddress = NULL; - int finduser_retries = 0; -+uschar *found_extension = NULL; - uschar *freeze_tell = NULL; - - uschar *gecos_name = NULL; -@@ -565,6 +569,7 @@ - uschar *lookup_value = NULL; - - macro_item *macros = NULL; -+uschar *malware_name = NULL; - int max_username_length = 0; - int message_age = 0; - uschar *message_body = NULL; -@@ -680,6 +685,7 @@ - const pcre *regex_PIPELINING = NULL; - const pcre *regex_SIZE = NULL; - const pcre *regex_ismsgid = NULL; -+uschar *regex_match_string = NULL; - int remote_delivery_count = 0; - int remote_max_parallel = 2; - uschar *remote_sort_domains = NULL; -@@ -846,6 +852,11 @@ - int smtp_rlr_threshold = INT_MAX; - BOOL smtp_use_pipelining = FALSE; - BOOL smtp_use_size = FALSE; -+uschar *spamd_address = US"127.0.0.1 783"; -+uschar *spam_bar = NULL; -+uschar *spam_report = NULL; -+uschar *spam_score = NULL; -+uschar *spam_score_int = NULL; - BOOL split_spool_directory = FALSE; - uschar *spool_directory = US SPOOL_DIRECTORY - "\0<--------------Space to patch spool_directory->"; -diff -urN exim-4.20-orig/src/globals.h exim-4.20/src/globals.h ---- exim-4.20-orig/src/globals.h Mon May 12 15:39:19 2003 -+++ exim-4.20/src/globals.h Wed May 14 12:04:24 2003 -@@ -129,6 +129,7 @@ - extern uschar *auth_defer_msg; /* Error message for log */ - extern uschar *auth_defer_user_msg; /* Error message for user */ - extern int auto_thaw; /* Auto-thaw interval */ -+extern uschar *av_scanner; /* AntiVirus scanner to use for the malware condition */ - - extern BOOL background_daemon; /* Set FALSE to keep in foreground */ - extern uschar *base62_chars; /* Table of base-62 characters */ -@@ -210,6 +211,8 @@ - extern BOOL deliver_selectstring_regex; /* String is regex */ - extern uschar *deliver_selectstring_sender; /* For selecting by sender */ - extern BOOL deliver_selectstring_sender_regex; /* String is regex */ -+extern int demime_errorlevel; /* Severity of MIME error */ -+extern uschar *demime_reason; /* Reason for broken MIME container */ - extern BOOL disable_logging; /* Disables log writing when TRUE */ - - extern uschar *dns_again_means_nonexist; /* Domains that are badly set up */ -@@ -250,6 +253,7 @@ - extern uschar *filter_test; /* Run as a filter tester on this file */ - extern uschar *filter_thisaddress; /* For address looping */ - extern int finduser_retries; /* Retry count for getpwnam() */ -+extern uschar *found_extension; /* demime acl condition: file extension found */ - extern uschar *freeze_tell; /* Message on (some) freezings */ - - extern uschar *gecos_name; /* To be expanded when pattern matches */ -@@ -322,6 +326,7 @@ - extern uschar *lookup_value; /* Value looked up from file */ - - extern macro_item *macros; /* Configuration macros */ -+extern uschar *malware_name; /* Name of virus or malware ("W32/Klez-H") */ - extern int max_username_length; /* For systems with broken getpwnam() */ - extern int message_age; /* In seconds */ - extern uschar *message_body; /* Start of message body for filter */ -@@ -421,6 +426,7 @@ - extern const pcre *regex_PIPELINING; /* For recognizing PIPELINING */ - extern const pcre *regex_SIZE; /* For recognizing SIZE settings */ - extern const pcre *regex_ismsgid; /* Compiled r.e. for message it */ -+extern uschar *regex_match_string; /* regex that matched a line (regex ACL condition) */ - extern int remote_delivery_count; /* Number of remote addresses */ - extern int remote_max_parallel; /* Maximum parallel delivery */ - extern uschar *remote_sort_domains; /* Remote domain sorting order */ -@@ -511,6 +517,11 @@ - extern BOOL smtp_use_pipelining; /* Global for passed connections */ - extern BOOL smtp_use_size; /* Global for passed connections */ - extern BOOL split_spool_directory; /* TRUE to use multiple subdirs */ -+extern uschar *spamd_address; /* address for the spamassassin daemon */ -+extern uschar *spam_bar; /* the spam "bar" (textual representation of spam_score) */ -+extern uschar *spam_report; /* the spamd report (multiline) */ -+extern uschar *spam_score; /* the spam score (float) */ -+extern uschar *spam_score_int; /* spam_score * 10 (int) */ - extern uschar *spool_directory; /* Name of spool directory */ - extern int string_datestamp_offset;/* After insertion by string_format */ - extern BOOL strip_excess_angle_brackets; /* Surrounding route-addrs */ -diff -urN exim-4.20-orig/src/malware.c exim-4.20/src/malware.c ---- exim-4.20-orig/src/malware.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/malware.c Thu Jun 5 09:33:29 2003 -@@ -0,0 +1,635 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* Code for calling virus (malware) scanners. Called from acl.c. */ -+ -+#include "exim.h" -+ -+/* SHUT_WR seems to be undefined on Unixware ? */ -+#ifndef SHUT_WR -+#define SHUT_WR 1 -+#endif -+ -+uschar malware_name_buffer[256]; -+int malware_ok = 0; -+ -+int malware(uschar **listptr) { -+ int sep = 0; -+ uschar *list = *listptr; -+ uschar *av_scanner_work = av_scanner; -+ uschar *scanner_name; -+ uschar scanner_name_buffer[16]; -+ uschar *malware_regex; -+ uschar malware_regex_buffer[64]; -+ uschar malware_regex_default[] = ".+"; -+ unsigned long long mbox_size; -+ FILE *mbox_file; -+ int roffset; -+ const pcre *re; -+ const uschar *rerror; -+ -+ /* make sure the eml mbox file is spooled up */ -+ mbox_file = spool_mbox(&mbox_size); -+ if (mbox_file == NULL) { -+ /* error while spooling */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: error while creating mbox spool file"); -+ return DEFER; -+ }; -+ /* none of our current scanners need the mbox -+ file as a stream, so we can close it right away */ -+ fclose(mbox_file); -+ -+ /* extract the malware regex to match against from the option list */ -+ if ((malware_regex = string_nextinlist(&list, &sep, -+ malware_regex_buffer, -+ sizeof(malware_regex_buffer))) != NULL) { -+ -+ /* parse 1st option */ -+ if ( (strcmpic(malware_regex,US"false") == 0) || -+ (Ustrcmp(malware_regex,"0") == 0) ) { -+ /* explicitly no matching */ -+ return FAIL; -+ }; -+ -+ /* special cases (match anything except empty) */ -+ if ( (strcmpic(malware_regex,US"true") == 0) || -+ (Ustrcmp(malware_regex,"*") == 0) || -+ (Ustrcmp(malware_regex,"1") == 0) ) { -+ malware_regex = malware_regex_default; -+ }; -+ } -+ else { -+ /* empty means "don't match anything" */ -+ return FAIL; -+ }; -+ -+ /* compile the regex, see if it works */ -+ re = pcre_compile(CS malware_regex, PCRE_COPT, (const char **)&rerror, &roffset, NULL); -+ if (re == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: regular expression error in '%s': %s at offset %d", malware_regex, rerror, roffset); -+ return DEFER; -+ }; -+ -+ /* Do not scan twice. */ -+ if (malware_ok == 0) { -+ -+ /* find the scanner type from the av_scanner option */ -+ if ((scanner_name = string_nextinlist(&av_scanner_work, &sep, -+ scanner_name_buffer, -+ sizeof(scanner_name_buffer))) == NULL) { -+ /* no scanner given */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: av_scanner configuration variable is empty"); -+ return DEFER; -+ }; -+ -+ -+ -+ /* "kavdaemon" scanner type ------------------------------------------------ */ -+ if (strcmpic(scanner_name,US"kavdaemon") == 0) { -+ uschar *kav_options; -+ uschar kav_options_buffer[1024]; -+ uschar kav_options_default[] = "/var/run/AvpCtl"; -+ struct sockaddr_un server; -+ int sock; -+ time_t t; -+ uschar tmpbuf[1024]; -+ uschar scanrequest[1024]; -+ uschar kav_match_string[128]; -+ int kav_rc; -+ unsigned long kav_reportlen, bread; -+ pcre *kav_re; -+ -+ if ((kav_options = string_nextinlist(&av_scanner_work, &sep, -+ kav_options_buffer, -+ sizeof(kav_options_buffer))) == NULL) { -+ /* no options supplied, use default options */ -+ kav_options = kav_options_default; -+ }; -+ -+ /* open the kavdaemon socket */ -+ sock = socket(AF_UNIX, SOCK_STREAM, 0); -+ if (sock < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: can't open UNIX socket."); -+ return DEFER; -+ } -+ server.sun_family = AF_UNIX; -+ Ustrcpy(server.sun_path, kav_options); -+ if (connect(sock, (struct sockaddr *) &server, sizeof(struct sockaddr_un)) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to connect to kavdaemon UNIX socket (%s). errno=%d", kav_options, errno); -+ return DEFER; -+ } -+ -+ /* get current date and time, build scan request */ -+ time(&t); -+ strftime(CS tmpbuf, sizeof(tmpbuf), "<0>%d %b %H:%M:%S:%%s/scan/%%s", localtime(&t)); -+ snprintf(CS scanrequest, 1024,CS tmpbuf, spool_directory, message_id); -+ -+ /* send scan request */ -+ if (send(sock, scanrequest, Ustrlen(scanrequest)+1, 0) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to write to kavdaemon UNIX socket (%s)", kav_options); -+ return DEFER; -+ } -+ -+ /* wait for result */ -+ if ((bread = recv(sock, tmpbuf, 2, 0) != 2)) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to read 2 bytes from kavdaemon socket."); -+ return DEFER; -+ } -+ -+ /* get errorcode from lower nibble */ -+ kav_rc = tmpbuf[0] & 0x0F; -+ -+ /* improper kavdaemon configuration */ -+ if ( (kav_rc == 5) || (kav_rc == 6) ) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: please reconfigure kavdaemon to NOT disinfect or remove infected files."); -+ return DEFER; -+ }; -+ -+ if (kav_rc == 1) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: kavdaemon reported 'scanning not completed' (code 1)."); -+ return DEFER; -+ }; -+ -+ if (kav_rc == 7) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: kavdaemon reported 'kavdaemon damaged' (code 7)."); -+ return DEFER; -+ }; -+ -+ /* code 8 is not handled, since it is ambigous. It appears mostly on -+ bounces where part of a file has been cut off */ -+ -+ /* "virus found" return codes (2-4) */ -+ if ((kav_rc > 1) && (kav_rc < 5)) { -+ -+ /* setup default virus name */ -+ Ustrcpy(malware_name_buffer,"unknown"); -+ malware_name = malware_name_buffer; -+ -+ /* read the report, if available */ -+ if( tmpbuf[1] == 1 ) { -+ /* read report size */ -+ if ((bread = recv(sock, &kav_reportlen, 4, 0)) != 4) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: cannot read report size from kavdaemon"); -+ return DEFER; -+ }; -+ -+ /* it's possible that avp returns av_buffer[1] == 1 but the -+ reportsize is 0 (!?) */ -+ if (kav_reportlen > 0) { -+ /* set up match regex, depends on retcode */ -+ if( kav_rc == 3 ) -+ Ustrcpy(kav_match_string, "suspicion:\\s*(.+?)\\s*$"); -+ else -+ Ustrcpy(kav_match_string, "infected:\\s*(.+?)\\s*$"); -+ -+ kav_re = pcre_compile( CS kav_match_string, -+ PCRE_COPT, -+ (const char **)&rerror, -+ &roffset, -+ NULL ); -+ -+ /* read report, linewise */ -+ while (kav_reportlen > 0) { -+ int result = 0; -+ int ovector[30]; -+ -+ bread = 0; -+ while ( recv(sock, &tmpbuf[bread], 1, 0) == 1 ) { -+ kav_reportlen--; -+ if ( (tmpbuf[bread] == '\n') || (bread > 1021) ) break; -+ bread++; -+ }; -+ bread++; -+ tmpbuf[bread] = '\0'; -+ -+ /* try matcher on the line, grab substring */ -+ result = pcre_exec(kav_re, NULL, CS tmpbuf, Ustrlen(tmpbuf), 0, 0, ovector, 30); -+ if (result >= 2) { -+ pcre_copy_substring(CS tmpbuf, ovector, result, 1, CS malware_name_buffer, 255); -+ break; -+ }; -+ }; -+ }; -+ }; -+ } -+ else { -+ /* no virus found */ -+ malware_name = NULL; -+ }; -+ -+ close(sock); -+ } -+ /* ----------------------------------------------------------------------- */ -+ -+ -+ /* "cmdline" scanner type ------------------------------------------------ */ -+ else if (strcmpic(scanner_name,US"cmdline") == 0) { -+ uschar *cmdline_scanner; -+ uschar cmdline_scanner_buffer[1024]; -+ uschar *cmdline_trigger; -+ uschar cmdline_trigger_buffer[1024]; -+ const pcre *cmdline_trigger_re; -+ uschar *cmdline_regex; -+ uschar cmdline_regex_buffer[1024]; -+ const pcre *cmdline_regex_re; -+ uschar file_name[1024]; -+ uschar commandline[1024]; -+ FILE *scanner_out = NULL; -+ FILE *scanner_record = NULL; -+ uschar linebuffer[32767]; -+ int trigger = 0; -+ int result; -+ int ovector[30]; -+ -+ /* find scanner command line */ -+ if ((cmdline_scanner = string_nextinlist(&av_scanner_work, &sep, -+ cmdline_scanner_buffer, -+ sizeof(cmdline_scanner_buffer))) == NULL) { -+ /* no command line supplied */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: missing commandline specification for cmdline scanner type."); -+ return DEFER; -+ }; -+ -+ /* find scanner output trigger */ -+ if ((cmdline_trigger = string_nextinlist(&av_scanner_work, &sep, -+ cmdline_trigger_buffer, -+ sizeof(cmdline_trigger_buffer))) == NULL) { -+ /* no trigger regex supplied */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: missing trigger specification for cmdline scanner type."); -+ return DEFER; -+ }; -+ -+ /* precompile trigger regex */ -+ cmdline_trigger_re = pcre_compile(CS cmdline_trigger, PCRE_COPT, (const char **)&rerror, &roffset, NULL); -+ if (cmdline_trigger_re == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: regular expression error in '%s': %s at offset %d", cmdline_trigger_re, rerror, roffset); -+ return DEFER; -+ }; -+ -+ /* find scanner name regex */ -+ if ((cmdline_regex = string_nextinlist(&av_scanner_work, &sep, -+ cmdline_regex_buffer, -+ sizeof(cmdline_regex_buffer))) == NULL) { -+ /* no name regex supplied */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: missing virus name regex specification for cmdline scanner type."); -+ return DEFER; -+ }; -+ -+ /* precompile name regex */ -+ cmdline_regex_re = pcre_compile(CS cmdline_regex, PCRE_COPT, (const char **)&rerror, &roffset, NULL); -+ if (cmdline_regex_re == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: regular expression error in '%s': %s at offset %d", cmdline_regex_re, rerror, roffset); -+ return DEFER; -+ }; -+ -+ /* prepare scanner call */ -+ snprintf(CS file_name,1024,"%s/scan/%s", spool_directory, message_id); -+ snprintf(CS commandline,1024, CS cmdline_scanner,file_name); -+ /* redirect STDERR too */ -+ Ustrcat(commandline," 2>&1"); -+ -+ scanner_out = popen(CS commandline,"r"); -+ if (scanner_out == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: calling cmdline scanner (%s) failed: %s.", commandline, strerror(errno)); -+ return DEFER; -+ }; -+ -+ snprintf(CS file_name,1024,"%s/scan/%s/%s_scanner_output", spool_directory, message_id, message_id); -+ scanner_record = fopen(CS file_name,"w"); -+ -+ if (scanner_record == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: opening scanner output file (%s) failed: %s.", file_name, strerror(errno)); -+ pclose(scanner_out); -+ return DEFER; -+ }; -+ -+ /* look for trigger while recording output */ -+ while(fgets(CS linebuffer,32767,scanner_out) != NULL) { -+ if ( Ustrlen(linebuffer) > fwrite(linebuffer, 1, Ustrlen(linebuffer), scanner_record) ) { -+ /* short write */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: short write on scanner output file (%s).", file_name); -+ pclose(scanner_out); -+ return DEFER; -+ }; -+ /* try trigger match */ -+ if (!trigger && regex_match_and_setup(cmdline_trigger_re, linebuffer, 0, -1)) -+ trigger = 1; -+ }; -+ -+ fclose(scanner_record); -+ pclose(scanner_out); -+ -+ if (trigger) { -+ /* setup default virus name */ -+ Ustrcpy(malware_name_buffer,"unknown"); -+ malware_name = malware_name_buffer; -+ -+ /* re-open the scanner output file, look for name match */ -+ scanner_record = fopen(CS file_name,"r"); -+ while(fgets(CS linebuffer,32767,scanner_record) != NULL) { -+ /* try match */ -+ result = pcre_exec(cmdline_regex_re, NULL, CS linebuffer, Ustrlen(linebuffer), 0, 0, ovector, 30); -+ if (result >= 2) { -+ pcre_copy_substring(CS linebuffer, ovector, result, 1, CS malware_name_buffer, 255); -+ }; -+ }; -+ fclose(scanner_record); -+ } -+ else { -+ /* no virus found */ -+ malware_name = NULL; -+ }; -+ } -+ /* ----------------------------------------------------------------------- */ -+ -+ -+ /* "sophie" scanner type ------------------------------------------------- */ -+ else if (strcmpic(scanner_name,US"sophie") == 0) { -+ uschar *sophie_options; -+ uschar sophie_options_buffer[1024]; -+ uschar sophie_options_default[] = "/var/run/sophie"; -+ int bread = 0; -+ struct sockaddr_un server; -+ int sock; -+ uschar file_name[1024]; -+ uschar av_buffer[1024]; -+ -+ if ((sophie_options = string_nextinlist(&av_scanner_work, &sep, -+ sophie_options_buffer, -+ sizeof(sophie_options_buffer))) == NULL) { -+ /* no options supplied, use default options */ -+ sophie_options = sophie_options_default; -+ }; -+ -+ /* open the sophie socket */ -+ sock = socket(AF_UNIX, SOCK_STREAM, 0); -+ if (sock < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: can't open UNIX socket."); -+ return DEFER; -+ } -+ server.sun_family = AF_UNIX; -+ Ustrcpy(server.sun_path, sophie_options); -+ if (connect(sock, (struct sockaddr *) &server, sizeof(struct sockaddr_un)) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to connect to sophie UNIX socket (%s). errno=%d", sophie_options, errno); -+ return DEFER; -+ } -+ -+ /* pass the scan directory to sophie */ -+ snprintf(CS file_name,1024,"%s/scan/%s", spool_directory, message_id); -+ if (write(sock, file_name, Ustrlen(file_name)) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to write to sophie UNIX socket (%s)", sophie_options); -+ return DEFER; -+ }; -+ -+ write(sock, "\n", 1); -+ -+ /* wait for result */ -+ memset(av_buffer, 0, sizeof(av_buffer)); -+ if ((!(bread = read(sock, av_buffer, sizeof(av_buffer))) > 0)) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: unable to read from sophie UNIX socket (%s)", sophie_options); -+ return DEFER; -+ }; -+ -+ close(sock); -+ -+ /* infected ? */ -+ if (av_buffer[0] == '1') { -+ if (Ustrchr(av_buffer, '\n')) *Ustrchr(av_buffer, '\n') = '\0'; -+ Ustrcpy(malware_name_buffer,&av_buffer[2]); -+ malware_name = malware_name_buffer; -+ } -+ else if (!strncmp(CS av_buffer, "-1", 2)) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: malware acl condition: sophie reported error"); -+ return DEFER; -+ } -+ else { -+ /* all ok, no virus */ -+ malware_name = NULL; -+ }; -+ } -+ /* ----------------------------------------------------------------------- */ -+ -+ -+ /* "clamd" scanner type ------------------------------------------------- */ -+ /* This code was contributed by David Saez <david@ols.es> */ -+ else if (strcmpic(scanner_name,US"clamd") == 0) { -+ uschar *clamd_options; -+ uschar clamd_options_buffer[1024]; -+ uschar clamd_options_default[] = "/tmp/clamd"; -+ uschar *p,*vname; -+ struct sockaddr_un server; -+ int sock,port,i,offset=0,bread=0; -+ uschar file_name[1024]; -+ uschar av_buffer[1024]; -+ uschar hostname[256]; -+ struct hostent *he; -+ struct in_addr in; -+ -+ if ((clamd_options = string_nextinlist(&av_scanner_work, &sep, -+ clamd_options_buffer, -+ sizeof(clamd_options_buffer))) == NULL) { -+ /* no options supplied, use default options */ -+ clamd_options = clamd_options_default; -+ } -+ -+ /* socket does not start with '/' -> network socket */ -+ if (*clamd_options != '/') { -+ -+ /* extract host and port part */ -+ if( sscanf(CS clamd_options, "%s %u", hostname, &port) != 2 ) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: invalid socket '%s'", clamd_options); -+ return DEFER; -+ }; -+ -+ /* Lookup the host */ -+ if((he = gethostbyname(CS hostname)) == 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: failed to lookup host '%s'", hostname); -+ return DEFER; -+ } -+ -+ in = *(struct in_addr *) he->h_addr_list[0]; -+ -+ /* Open the ClamAV Socket */ -+ if ( (sock = ip_socket(SOCK_STREAM, AF_INET)) < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: unable to acquire socket (%s)", -+ strerror(errno)); -+ return DEFER; -+ } -+ -+ if (ip_connect(sock, AF_INET, (uschar*)inet_ntoa(in), port, 5) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: connection to %s, port %u failed (%s)", -+ inet_ntoa(in), port, strerror(errno)); -+ return DEFER; -+ } -+ } -+ else { -+ /* open the local socket */ -+ if ((sock = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: unable to acquire socket (%s)", -+ strerror(errno)); -+ return DEFER; -+ } -+ -+ server.sun_family = AF_UNIX; -+ Ustrcpy(server.sun_path, clamd_options); -+ -+ if (connect(sock, (struct sockaddr *) &server, sizeof(struct sockaddr_un)) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: unable to connect to UNIX socket %s (%s)", -+ clamd_options, strerror(errno) ); -+ return DEFER; -+ } -+ } -+ -+ /* Pass the string to ClamAV (7 = "SCAN \n" + \0) */ -+ -+ snprintf(CS file_name,1024,"SCAN %s/scan/%s\n", spool_directory, message_id); -+ -+ if (send(sock, file_name, Ustrlen(file_name), 0) < 0) { -+ close(sock); -+ log_write(0, LOG_MAIN|LOG_PANIC,"malware acl condition: clamd: unable to write to socket (%s)", -+ strerror(errno)); -+ return DEFER; -+ } -+ -+ /* we're done sending, close socket for writing */ -+ shutdown(sock, SHUT_WR); -+ -+ /* Read the result */ -+ memset(av_buffer, 0, sizeof(av_buffer)); -+ bread = read(sock, av_buffer, sizeof(av_buffer)); -+ close(sock); -+ -+ if (!(bread > 0)) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: unable to read from socket (%s)", -+ strerror(errno)); -+ return DEFER; -+ } -+ -+ if (bread == sizeof(av_buffer)) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: buffer too small"); -+ return DEFER; -+ } -+ -+ /* Check the result. ClamAV Returns -+ infected: -> "<filename>: <virusname> FOUND" -+ not-infected: -> "<filename>: OK" -+ error: -> "<filename>: <errcode> ERROR */ -+ -+ if (!(*av_buffer)) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: ClamAV returned null"); -+ return DEFER; -+ } -+ -+ /* colon in returned output? */ -+ if((p = Ustrrchr(av_buffer,':')) == NULL) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: ClamAV returned malformed result: %s", -+ av_buffer); -+ return DEFER; -+ } -+ -+ /* strip filename strip CR at the end */ -+ vname = ++p; -+ p = vname + Ustrlen(vname) - 1; -+ if( *p == '\n' ) *p = '\0'; -+ -+ if ((p = Ustrstr(vname, "FOUND"))!=NULL) { -+ *p=0; -+ for (--p;p>vname && *p<=32;p--) *p=0; -+ Ustrcpy(malware_name_buffer,vname); -+ malware_name = malware_name_buffer; -+ } -+ else { -+ if (Ustrstr(vname, "ERROR")!=NULL) { -+ /* ClamAV reports ERROR -+ Find line start */ -+ for (;*vname!='\n' && vname>av_buffer; vname--); -+ if (*vname=='\n') vname++; -+ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware acl condition: clamd: ClamAV returned %s",vname); -+ return DEFER; -+ } -+ else { -+ /* Everything should be OK */ -+ malware_name = NULL; -+ } -+ } -+ } -+ /* ----------------------------------------------------------------------- */ -+ -+ -+ -+ /* "unknown" scanner type ------------------------------------------------- */ -+ else { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "malware condition: unknown scanner type '%s'", scanner_name); -+ return DEFER; -+ }; -+ /* ----------------------------------------------------------------------- */ -+ -+ /* set "been here, done that" marker */ -+ malware_ok = 1; -+ }; -+ -+ /* match virus name against pattern (caseless ------->----------v) */ -+ if ( (malware_name != NULL) && -+ (regex_match_and_setup(re, malware_name, 0, -1)) ) { -+ return OK; -+ } -+ else { -+ return FAIL; -+ }; -+} -diff -urN exim-4.20-orig/src/readconf.c exim-4.20/src/readconf.c ---- exim-4.20-orig/src/readconf.c Mon May 12 15:39:21 2003 -+++ exim-4.20/src/readconf.c Wed May 14 12:04:24 2003 -@@ -152,6 +152,7 @@ - { "allow_utf8_domains", opt_bool, &allow_utf8_domains }, - { "auth_advertise_hosts", opt_stringptr, &auth_advertise_hosts }, - { "auto_thaw", opt_time, &auto_thaw }, -+ { "av_scanner", opt_stringptr, &av_scanner }, - { "bi_command", opt_stringptr, &bi_command }, - { "bounce_message_file", opt_stringptr, &bounce_message_file }, - { "bounce_message_text", opt_stringptr, &bounce_message_text }, -@@ -297,6 +298,7 @@ - { "smtp_receive_timeout", opt_time, &smtp_receive_timeout }, - { "smtp_reserve_hosts", opt_stringptr, &smtp_reserve_hosts }, - { "smtp_return_error_details",opt_bool, &smtp_return_error_details }, -+ { "spamd_address", opt_stringptr, &spamd_address }, - { "split_spool_directory", opt_bool, &split_spool_directory }, - { "spool_directory", opt_stringptr, &spool_directory }, - { "strip_excess_angle_brackets", opt_bool, &strip_excess_angle_brackets }, -diff -urN exim-4.20-orig/src/regex.c exim-4.20/src/regex.c ---- exim-4.20-orig/src/regex.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/regex.c Wed May 14 12:04:24 2003 -@@ -0,0 +1,110 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* Code for matching regular expressions against headers and body. -+ Called from acl.c. */ -+ -+#include "exim.h" -+ -+/* Structure to hold a list of Regular expressions */ -+typedef struct pcre_list { -+ pcre *re; -+ uschar *pcre_text; -+ struct pcre_list *next; -+} pcre_list; -+ -+uschar regex_match_string_buffer[1024]; -+ -+int regex(uschar **listptr) { -+ int sep = 0; -+ uschar *list = *listptr; -+ uschar *regex_string; -+ uschar regex_string_buffer[1024]; -+ unsigned long long mbox_size; -+ FILE *mbox_file; -+ pcre *re; -+ pcre_list *re_list_head = NULL; -+ pcre_list *re_list_item; -+ const char *pcre_error; -+ int pcre_erroffset; -+ uschar *linebuffer; -+ -+ /* reset expansion variable */ -+ regex_match_string = NULL; -+ -+ /* make sure the eml mbox file is spooled up */ -+ mbox_file = spool_mbox(&mbox_size); -+ if (mbox_file == NULL) { -+ /* error while spooling */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "regex acl condition: error while creating mbox spool file"); -+ return DEFER; -+ }; -+ -+ /* precompile our regexes */ -+ while ((regex_string = string_nextinlist(&list, &sep, -+ regex_string_buffer, -+ sizeof(regex_string_buffer))) != NULL) { -+ -+ /* parse option */ -+ if ( (strcmpic(regex_string,US"false") == 0) || -+ (Ustrcmp(regex_string,"0") == 0) ) { -+ /* explicitly no matching */ -+ continue; -+ }; -+ -+ /* compile our regular expression */ -+ re = pcre_compile( CS regex_string, -+ 0, -+ &pcre_error, -+ &pcre_erroffset, -+ NULL ); -+ -+ if (re == NULL) { -+ log_write(0, LOG_MAIN, -+ "regex acl condition warning - error in regex '%s': %s at offset %d, skipped.", regex_string, pcre_error, pcre_erroffset); -+ continue; -+ } -+ else { -+ re_list_item = store_get(sizeof(pcre_list)); -+ re_list_item->re = re; -+ re_list_item->pcre_text = string_copy(regex_string); -+ re_list_item->next = re_list_head; -+ re_list_head = re_list_item; -+ }; -+ }; -+ -+ /* no regexes -> nothing to do */ -+ if (re_list_head == NULL) { -+ return FAIL; -+ }; -+ -+ /* match each line against all regexes */ -+ linebuffer = store_get(32767); -+ while (fgets(CS linebuffer, 32767, mbox_file) != NULL) { -+ re_list_item = re_list_head; -+ do { -+ /* try matcher on the line */ -+ if (pcre_exec(re_list_item->re, NULL, CS linebuffer, -+ (int)Ustrlen(linebuffer), 0, 0, NULL, 0) >= 0) { -+ Ustrncpy(regex_match_string_buffer, re_list_item->pcre_text, 1023); -+ regex_match_string = regex_match_string_buffer; -+ fclose(mbox_file); -+ return OK; -+ }; -+ re_list_item = re_list_item->next; -+ } while (re_list_item != NULL); -+ }; -+ -+ fclose(mbox_file); -+ -+ /* no matches ... */ -+ return FAIL; -+} -diff -urN exim-4.20-orig/src/spam.c exim-4.20/src/spam.c ---- exim-4.20-orig/src/spam.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/spam.c Wed May 21 09:35:15 2003 -@@ -0,0 +1,264 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* Code for calling spamassassin's spamd. Called from acl.c. */ -+ -+#include "exim.h" -+#include "spam.h" -+ -+uschar spam_score_buffer[16]; -+uschar spam_score_int_buffer[16]; -+uschar spam_bar_buffer[128]; -+uschar spam_report_buffer[32600]; -+uschar prev_user_name[128]; -+int spam_ok = 0; -+int spam_rc = 0; -+ -+int spam(uschar **listptr) { -+ int sep = 0; -+ uschar *list = *listptr; -+ uschar *user_name; -+ uschar user_name_buffer[128]; -+ unsigned long long mbox_size; -+ FILE *mbox_file; -+ int spamd_sock; -+ uschar tcp_addr[24]; -+ unsigned int tcp_port; -+ uschar spamd_buffer[32600]; -+ int i, j, offset; -+ uschar spamd_version[8]; -+ uschar spamd_score_char; -+ double spamd_threshold, spamd_score; -+ int spamd_report_offset; -+ uschar *p,*q; -+ int override = 0; -+ -+ /* find the username from the option list */ -+ if ((user_name = string_nextinlist(&list, &sep, -+ user_name_buffer, -+ sizeof(user_name_buffer))) == NULL) { -+ /* no username given, this means no scanning should be done */ -+ return FAIL; -+ }; -+ -+ /* if username is "0" or "false", do not scan */ -+ if ( (Ustrcmp(user_name,"0") == 0) || -+ (strcmpic(user_name,US"false") == 0) ) { -+ return FAIL; -+ }; -+ -+ /* if there is an additional option, check if it is "true" */ -+ if (strcmpic(list,US"true") == 0) { -+ /* in that case, always return true later */ -+ override = 1; -+ }; -+ -+ /* if we scanned for this username last time, just return */ -+ if ( spam_ok && ( Ustrcmp(prev_user_name, user_name) == 0 ) ) -+ if (override) -+ return OK; -+ else -+ return spam_rc; -+ -+ /* make sure the eml mbox file is spooled up */ -+ mbox_file = spool_mbox(&mbox_size); -+ -+ if (mbox_file == NULL) { -+ /* error while spooling */ -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: error while creating mbox spool file"); -+ return DEFER; -+ }; -+ -+ /* contact spamd */ -+ spamd_sock = ip_socket(SOCK_STREAM, AF_INET); -+ if (spamd_sock < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: error creating IP socket for spamd"); -+ fclose(mbox_file); -+ return DEFER; -+ }; -+ -+ if (ip_bind(spamd_sock, AF_INET, US"0.0.0.0", 0) < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: bind socket for spamd failed: %s",strerror(errno)); -+ fclose(mbox_file); -+ close(spamd_sock); -+ return DEFER; -+ }; -+ -+ /* grok spamd address and port */ -+ if( sscanf(CS spamd_address, "%s %u", tcp_addr, &tcp_port) != 2 ) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: invalid spamd address: '%s'", spamd_address); -+ fclose(mbox_file); -+ close(spamd_sock); -+ return DEFER; -+ }; -+ -+ if (ip_connect(spamd_sock, AF_INET, tcp_addr, tcp_port, 5) < 0) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: spamd connection to %s, port %u failed: %s", tcp_addr, tcp_port, strerror(errno)); -+ fclose(mbox_file); -+ close(spamd_sock); -+ return DEFER; -+ }; -+ -+ /* now we are connected to spamd on spamd_sock */ -+ snprintf(CS spamd_buffer, -+ sizeof(spamd_buffer), -+ "REPORT SPAMC/1.2\r\nUser: %s\r\nContent-length: %lld\r\n\r\n", -+ user_name, -+ mbox_size); -+ -+ /* send our request */ -+ if (send(spamd_sock, spamd_buffer, Ustrlen(spamd_buffer), 0) < 0) { -+ close(spamd_sock); -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: spamd send failed: %s", strerror(errno)); -+ fclose(mbox_file); -+ close(spamd_sock); -+ return DEFER; -+ }; -+ -+ /* now send the file */ -+ do { -+ j = fread(spamd_buffer,1,sizeof(spamd_buffer),mbox_file); -+ if (j > 0) { -+ i = send(spamd_sock,spamd_buffer,j,0); -+ if (i != j) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: error/short send to spamd"); -+ close(spamd_sock); -+ fclose(mbox_file); -+ return DEFER; -+ }; -+ }; -+ } -+ while (j > 0); -+ -+ fclose(mbox_file); -+ -+ /* we're done sending, close socket for writing */ -+ shutdown(spamd_sock,SHUT_WR); -+ -+ /* read spamd response */ -+ memset(spamd_buffer, 0, sizeof(spamd_buffer)); -+ offset = 0; -+ while((i = ip_recv(spamd_sock, -+ spamd_buffer + offset, -+ sizeof(spamd_buffer) - offset - 1, -+ SPAMD_READ_TIMEOUT)) > 0 ) { -+ offset += i; -+ } -+ -+ /* error handling */ -+ if((i <= 0) && (errno != 0)) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: error reading from spamd socket: %s", strerror(errno)); -+ close(spamd_sock); -+ return DEFER; -+ } -+ -+ /* reading done */ -+ close(spamd_sock); -+ -+ /* dig in the spamd output and put the report in a multiline header, if requested */ -+ if( sscanf(CS spamd_buffer,"SPAMD/%s 0 EX_OK\r\nContent-length: %*u\r\n\r\n%lf/%lf\r\n%n", -+ spamd_version,&spamd_score,&spamd_threshold,&spamd_report_offset) != 3 ) { -+ -+ /* try to fall back to pre-2.50 spamd output */ -+ if( sscanf(CS spamd_buffer,"SPAMD/%s 0 EX_OK\r\nSpam: %*s ; %lf / %lf\r\n\r\n%n", -+ spamd_version,&spamd_score,&spamd_threshold,&spamd_report_offset) != 3 ) { -+ log_write(0, LOG_MAIN|LOG_PANIC, -+ "spam acl condition: cannot parse spamd output"); -+ return DEFER; -+ }; -+ }; -+ -+ /* Create report. Since this is a multiline string, -+ we must hack it into shape first */ -+ p = &spamd_buffer[spamd_report_offset]; -+ q = spam_report_buffer; -+ while (*p != '\0') { -+ /* skip \r */ -+ if (*p == '\r') { -+ p++; -+ continue; -+ }; -+ *q = *p; -+ q++; -+ if (*p == '\n') { -+ *q = '\t'; -+ q++; -+ /* eat whitespace */ -+ while( (*p <= ' ') && (*p != '\0') ) { -+ p++; -+ }; -+ p--; -+ }; -+ p++; -+ }; -+ /* NULL-terminate */ -+ *q = '\0'; -+ q--; -+ /* cut off trailing leftovers */ -+ while (*q <= ' ') { -+ *q = '\0'; -+ q--; -+ }; -+ spam_report = spam_report_buffer; -+ -+ /* create spam bar */ -+ spamd_score_char = spamd_score > 0 ? '+' : '-'; -+ j = abs((int)(spamd_score)); -+ i = 0; -+ if( j != 0 ) { -+ while((i < j) && (i <= MAX_SPAM_BAR_CHARS)) -+ spam_bar_buffer[i++] = spamd_score_char; -+ } -+ else{ -+ spam_bar_buffer[0] = '/'; -+ i = 1; -+ } -+ spam_bar_buffer[i] = '\0'; -+ spam_bar = spam_bar_buffer; -+ -+ /* create "float" spam score */ -+ snprintf(CS spam_score_buffer, sizeof(spam_score_buffer),"%.1f", spamd_score); -+ spam_score = spam_score_buffer; -+ -+ /* create "int" spam score */ -+ j = (int)(spamd_score*10); -+ snprintf(CS spam_score_int_buffer, sizeof(spam_score_int_buffer), "%d", j); -+ spam_score_int = spam_score_int_buffer; -+ -+ /* compare threshold against score */ -+ if (spamd_score >= spamd_threshold) { -+ /* spam as determined by user's threshold */ -+ spam_rc = OK; -+ } -+ else { -+ /* not spam */ -+ spam_rc = FAIL; -+ }; -+ -+ /* remember user name and "been here" for it */ -+ Ustrcpy(prev_user_name, user_name); -+ spam_ok = 1; -+ -+ if (override) { -+ /* always return OK, no matter what the score */ -+ return OK; -+ } -+ else { -+ return spam_rc; -+ }; -+} -diff -urN exim-4.20-orig/src/spam.h exim-4.20/src/spam.h ---- exim-4.20-orig/src/spam.h Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/spam.h Wed May 14 12:04:24 2003 -@@ -0,0 +1,23 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* spam defines */ -+ -+/* timeout for reading from spamd */ -+#define SPAMD_READ_TIMEOUT 3600 -+ -+/* maximum length of the spam bar */ -+#define MAX_SPAM_BAR_CHARS 50 -+ -+/* SHUT_WR seems to be undefined on Unixware ? */ -+#ifndef SHUT_WR -+#define SHUT_WR 1 -+#endif -+ -diff -urN exim-4.20-orig/src/spool_in.c exim-4.20/src/spool_in.c ---- exim-4.20-orig/src/spool_in.c Mon May 12 15:39:22 2003 -+++ exim-4.20/src/spool_in.c Wed May 14 12:04:24 2003 -@@ -248,6 +248,7 @@ - interface_port = 0; - local_error_message = FALSE; - local_scan_data = NULL; -+spam_score_int = NULL; - message_linecount = 0; - received_protocol = NULL; - recipients_list = NULL; -@@ -347,6 +348,8 @@ - local_error_message = TRUE; - else if (Ustrncmp(big_buffer, "-local_scan ", 12) == 0) - local_scan_data = string_copy(big_buffer + 12); -+ else if (Ustrncmp(big_buffer, "-spam_score_int ", 16) == 0) -+ spam_score_int = string_copy(big_buffer + 16); - else if (Ustrcmp(big_buffer, "-host_lookup_failed") == 0) - host_lookup_failed = TRUE; - else if (Ustrncmp(big_buffer, "-body_linecount", 15) == 0) -diff -urN exim-4.20-orig/src/spool_mbox.c exim-4.20/src/spool_mbox.c ---- exim-4.20-orig/src/spool_mbox.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/spool_mbox.c Tue Jun 10 15:43:43 2003 -@@ -0,0 +1,157 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Copyright (c) Tom Kistner <tom@duncanthrax.net> 2003-???? */ -+/* License: GPL */ -+ -+/* Code for setting up a MBOX style spool file inside a /scan/<msgid> -+sub directory of exim's spool directory. */ -+ -+#include "exim.h" -+ -+/* externals, we must reset them on unspooling */ -+extern int demime_ok; -+extern int malware_ok; -+extern int spam_ok; -+extern struct file_extension *file_extensions; -+ -+int spool_mbox_ok = 0; -+ -+/* returns a pointer to the FILE, and puts the size in bytes into mbox_file_size */ -+ -+FILE *spool_mbox(unsigned long long *mbox_file_size) { -+ uschar mbox_path[1024]; -+ uschar message_subdir[2]; -+ uschar data_buffer[65535]; -+ FILE *mbox_file; -+ FILE *data_file; -+ header_line *my_headerlist; -+ struct stat statbuf; -+ int i,j; -+ -+ if (!spool_mbox_ok) { -+ /* create scan directory, if not present */ -+ if (!directory_make(spool_directory, US "scan", 0750, FALSE)) { -+ debug_printf("unable to create directory: %s/scan\n", spool_directory); -+ return NULL; -+ }; -+ -+ /* create temp directory inside scan dir */ -+ snprintf(CS mbox_path, 1024, "%s/scan/%s", spool_directory, message_id); -+ if (!directory_make(NULL, mbox_path, 0750, FALSE)) { -+ debug_printf("unable to create directory: %s/scan/%s\n", spool_directory, message_id); -+ return NULL; -+ }; -+ -+ /* open [message_id].eml file for writing */ -+ snprintf(CS mbox_path, 1024, "%s/scan/%s/%s.eml", spool_directory, message_id, message_id); -+ mbox_file = Ufopen(mbox_path,"w"); -+ -+ if (mbox_file == NULL) { -+ debug_printf("unable to open file for writing: %s\n", mbox_path); -+ return NULL; -+ }; -+ -+ /* write all header lines to mbox file */ -+ my_headerlist = header_list; -+ while (my_headerlist != NULL) { -+ -+ /* skip deleted headers */ -+ if (my_headerlist->type == '*') { -+ my_headerlist = my_headerlist->next; -+ continue; -+ }; -+ -+ i = fwrite(my_headerlist->text, 1, my_headerlist->slen, mbox_file); -+ if (i != my_headerlist->slen) { -+ debug_printf("error/short write on writing in: %s", mbox_path); -+ fclose(mbox_file); -+ return NULL; -+ }; -+ -+ my_headerlist = my_headerlist->next; -+ }; -+ -+ /* copy body file */ -+ message_subdir[1] = '\0'; -+ for (i = 0; i < 2; i++) { -+ message_subdir[0] = (split_spool_directory == (i == 0))? message_id[5] : 0; -+ sprintf(CS mbox_path, "%s/input/%s/%s-D", spool_directory, message_subdir, message_id); -+ data_file = Ufopen(mbox_path,"r"); -+ if (data_file != NULL) -+ break; -+ }; -+ -+ fread(data_buffer, 1, 18, data_file); -+ -+ do { -+ j = fread(data_buffer, 1, sizeof(data_buffer), data_file); -+ if (j > 0) { -+ i = fwrite(data_buffer, 1, j, mbox_file); -+ if (i != j) { -+ debug_printf("error/short write on writing in: %s", mbox_path); -+ fclose(mbox_file); -+ fclose(data_file); -+ return NULL; -+ }; -+ }; -+ } while (j > 0); -+ -+ fclose(data_file); -+ fclose(mbox_file); -+ spool_mbox_ok = 1; -+ }; -+ -+ snprintf(CS mbox_path, 1024, "%s/scan/%s/%s.eml", spool_directory, message_id, message_id); -+ -+ /* get the size of the mbox message */ -+ stat(CS mbox_path, &statbuf); -+ *mbox_file_size = statbuf.st_size; -+ -+ /* open [message_id].eml file for reading */ -+ mbox_file = Ufopen(mbox_path,"r"); -+ -+ return mbox_file; -+} -+ -+/* remove mbox spool file, demimed files and temp directory */ -+void unspool_mbox(void) { -+ -+ /* reset all exiscan state variables */ -+ demime_ok = 0; -+ file_extensions = NULL; -+ spam_ok = 0; -+ malware_ok = 0; -+ -+ if (spool_mbox_ok) { -+ uschar mbox_path[1024]; -+ uschar file_path[1024]; -+ int n; -+ struct dirent *entry; -+ DIR *tempdir; -+ -+ spool_mbox_ok = 0; -+ -+ snprintf(CS mbox_path, 1024, "%s/scan/%s", spool_directory, message_id); -+ -+ tempdir = opendir(CS mbox_path); -+ /* loop thru dir & delete entries */ -+ n = 0; -+ do { -+ entry = readdir(tempdir); -+ if (entry == NULL) break; -+ snprintf(CS file_path, 1024,"%s/scan/%s/%s", spool_directory, message_id, entry->d_name); -+ if ( (Ustrcmp(entry->d_name,"..") != 0) && (Ustrcmp(entry->d_name,".") != 0) ) -+ n = unlink(CS file_path); -+ } while (n > -1); -+ -+ closedir(tempdir); -+ -+ /* remove directory */ -+ n = rmdir(CS mbox_path); -+ }; -+} -diff -urN exim-4.20-orig/src/spool_out.c exim-4.20/src/spool_out.c ---- exim-4.20-orig/src/spool_out.c Mon May 12 15:39:22 2003 -+++ exim-4.20/src/spool_out.c Wed May 14 12:04:24 2003 -@@ -200,6 +200,7 @@ - if (sender_local) fprintf(f, "-local\n"); - if (local_error_message) fprintf(f, "-localerror\n"); - if (local_scan_data != NULL) fprintf(f, "-local_scan %s\n", local_scan_data); -+if (spam_score_int != NULL) fprintf(f,"-spam_score_int %s\n", spam_score_int); - if (deliver_manual_thaw) fprintf(f, "-manual_thaw\n"); - if (sender_set_untrusted) fprintf(f, "-sender_set_untrusted\n"); - -diff -urN exim-4.20-orig/src/tnef.c exim-4.20/src/tnef.c ---- exim-4.20-orig/src/tnef.c Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/tnef.c Wed May 14 12:04:24 2003 -@@ -0,0 +1,741 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/* Code for unpacking TNEF containers. Called from demime.c. */ -+ -+/*************************************************************************** -+ * tnef2txt -+* A program to decode application/ms-tnef MIME attachments into text -+* for those fortunate enough not to be running either a Microsoft -+* operating system or mailer. -+* -+ * 18/10/2001 -+* Brutally cropped by Paul L Daniels (pldaniels@pldaniels.com) in order -+* to accommodate the needs of ripMIME/Xamime/Inflex without carrying too -+* much excess baggage. -+* -+ * Brandon Long (blong@uiuc.edu), April 1997 -+* 1.0 Version -+* Supports most types, but doesn't decode properties. Maybe some other -+* time. -+* -+ * 1.1 Version (7/1/97) -+* Supports saving of attAttachData to a file given by attAttachTitle -+* start of property decoding support -+* -+ * 1.2 Version (7/19/97) -+* Some architectures don't like reading 16/32 bit data on unaligned -+* boundaries. Fixed, losing efficiency, but this doesn't really -+* need efficiency anyways. (Still...) -+* Also, the #pragma pack from the MSVC include file wasn't liked -+* by most Unix compilers, replaced with a GCCism. This should work -+* with GCC, but other compilers I don't know. -+* -+ * 1.3 Version (7/22/97) -+* Ok, take out the DTR over the stream, now uses read_16. -+* -+ * NOTE: THIS SOFTWARE IS FOR YOUR PERSONAL GRATIFICATION ONLY. I DON'T -+* IMPLY IN ANY LEGAL SENSE THAT THIS SOFTWARE DOES ANYTHING OR THAT IT WILL -+* BE USEFULL IN ANY WAY. But, you can send me fixes to it, I don't mind. -+***************************************************************************/ -+ -+#include <stdio.h> -+#include <sys/stat.h> -+#include <stdlib.h> -+#include <errno.h> -+#include <string.h> -+#include <netinet/in.h> -+#include "tnef.h" -+ -+ -+#define VERSION "pldtnef/0.0.1" -+ -+int _TNEF_syslogging = 0; -+int _TNEF_stderrlogging = 0; -+int _TNEF_verbose = 0; -+int _TNEF_debug = 0; -+ -+int Verbose = FALSE; -+int SaveData = FALSE; -+ -+char _TNEF_path[1024]=""; -+ -+uint8 *tnef_home; -+uint8 *tnef_limit; -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_set_path ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_set_path( char *path ) -+{ -+ snprintf(_TNEF_path,1023,"%s",path); -+ -+ return 0; -+} -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_set_verbosity ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_set_verbosity( int level ) -+{ -+ _TNEF_verbose = level; -+ return _TNEF_verbose; -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_set_debug ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_set_debug( int level ) -+{ -+ _TNEF_debug = level; -+ TNEF_set_verbosity( level ); -+ return _TNEF_debug; -+} -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_set_syslogging ID:1 -+Purpose: Turns on/off the syslog feature for TNEF error messages -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_set_syslogging( int level ) -+{ -+ _TNEF_syslogging = level; -+ return _TNEF_syslogging; -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_set_stderrlogging ID:1 -+Purpose: Turns on/off the stderr feature for TNEF error messages -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_set_stderrlogging( int level ) -+{ -+ _TNEF_stderrlogging = level; -+ return _TNEF_stderrlogging; -+} -+ -+ -+/* Some systems don't like to read unaligned data */ -+/*------------------------------------------------------------------------ -+Procedure: read_32 ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+uint32 read_32(uint8 *tsp) -+{ -+ uint8 a,b,c,d; -+ uint32 ret; -+ -+ if (tsp > tnef_limit) -+ { -+ if ((_TNEF_verbose)||(_TNEF_stderrlogging)||(_TNEF_debug)) fprintf(stderr,"TNEF read_32() Attempting to read past end\n"); -+ return -1; -+ } -+ -+ a = *tsp; -+ b = *(tsp+1); -+ c = *(tsp+2); -+ d = *(tsp+3); -+ -+ ret = long_little_endian(a<<24 | b<<16 | c<<8 | d); -+ -+ return ret; -+} -+ -+/*------------------------------------------------------------------------ -+Procedure: read_16 ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+uint16 read_16(uint8 *tsp) -+{ -+ uint8 a,b; -+ uint16 ret; -+ -+ if (tsp > tnef_limit) -+ { -+ if ((_TNEF_verbose)||(_TNEF_stderrlogging)||(_TNEF_debug)) fprintf(stderr,"TNEF read_16() Attempting to read past end\n"); -+ return -1; -+ } -+ -+ -+ a = *tsp; -+ b = *(tsp + 1); -+ -+ ret = little_endian(a<<8 | b); -+ -+ return ret; -+} -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: make_string ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+char *make_string(uint8 *tsp, int size) -+{ -+ static char s[256] = ""; -+ int len = (size>sizeof(s)-1) ? sizeof(s)-1 : size; -+ -+ strncpy(s,(char *)tsp, len); -+ s[len] = '\0'; -+ return s; -+} -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: handle_props ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int handle_props(uint8 *tsp) -+{ -+ int bytes = 0; -+ uint32 num_props = 0; -+ uint32 x = 0; -+ -+ -+ num_props = read_32(tsp); -+ bytes += sizeof(num_props); -+ -+ while (x < num_props) -+ { -+ uint32 prop_tag; -+ uint32 num; -+ char filename[256]; -+ static int file_num = 0; -+ -+ prop_tag = read_32(tsp+bytes); -+ bytes += sizeof(prop_tag); -+ -+ switch (prop_tag & PROP_TYPE_MASK) -+ { -+ case PT_BINARY: -+ num = read_32(tsp+bytes); -+ bytes += sizeof(num); -+ num = read_32(tsp+bytes); -+ bytes += sizeof(num); -+ if (prop_tag == PR_RTF_COMPRESSED) -+ { -+ sprintf (filename, "XAM_%d.rtf", file_num); -+ file_num++; -+ save_attach_data(filename, tsp+bytes, num); -+ } -+ /* num + PAD */ -+ bytes += num + ((num % 4) ? (4 - num%4) : 0); -+ break; -+ case PT_STRING8: -+ num = read_32(tsp+bytes); -+ bytes += sizeof(num); -+ num = read_32(tsp+bytes); -+ bytes += sizeof(num); -+ make_string(tsp+bytes,num); -+ bytes += num + ((num % 4) ? (4 - num%4) : 0); -+ break; -+ case PT_UNICODE: -+ case PT_OBJECT: -+ break; -+ case PT_I2: -+ bytes += 2; -+ break; -+ case PT_LONG: -+ bytes += 4; -+ break; -+ case PT_R4: -+ bytes += 4; -+ break; -+ case PT_DOUBLE: -+ bytes += 8; -+ break; -+ case PT_CURRENCY: -+ case PT_APPTIME: -+ case PT_ERROR: -+ bytes += 4; -+ break; -+ case PT_BOOLEAN: -+ bytes += 4; -+ break; -+ case PT_I8: -+ bytes += 8; -+ case PT_SYSTIME: -+ bytes += 8; -+ break; -+ } -+ x++; -+ } -+ -+ return 0; -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: save_attach_data ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int save_attach_data(char *title, uint8 *tsp, uint32 size) -+{ -+ FILE *out; -+ char filename[1024]; -+ -+ /* -+ if ((*tsp +size) > _TNEF_size) -+ { -+ return -1; -+ } -+ */ -+ snprintf(filename,1023,"%s/%s",_TNEF_path,title); -+ -+ out = fopen(filename, "w"); -+ if (!out) -+ { -+ if (_TNEF_stderrlogging > 0) fprintf(stderr, "Error openning file %s for writing\n", filename); -+ return -1; -+ } -+ -+ fwrite(tsp, sizeof(uint8), size, out); -+ fclose(out); -+ return 0; -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: default_handler ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int default_handler(uint32 attribute, uint8 *tsp, uint32 size) -+{ -+ uint16 type = ATT_TYPE(attribute); -+ -+ switch (type) { -+ case atpTriples: -+ break; -+ case atpString: -+ case atpText: -+ break; -+ case atpDate: -+ break; -+ case atpShort: -+ break; -+ case atpLong: -+ break; -+ case atpByte: -+ break; -+ case atpWord: -+ break; -+ case atpDword: -+ break; -+ default: -+ break; -+ } -+ return 0; -+ -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: read_attribute ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int read_attribute(uint8 *tsp) -+{ -+ -+ int bytes = 0, header = 0; -+ uint32 attribute; -+ uint8 component = 0; -+ uint32 size = 0; -+ uint16 checksum = 0; -+ static char attach_title[256] = { -+ 0 }; -+ static uint32 attach_size = 0; -+ static uint32 attach_loc = 0; -+ uint8 *ptr; -+ -+ /* What component are we look at? */ -+ component = *tsp; -+ -+ bytes += sizeof(uint8); -+ -+ /* Read the attributes of this component */ -+ -+ if (_TNEF_debug) fprintf(stderr,"read_attribute: Reading Attribute...\n"); -+ attribute = read_32(tsp+bytes); -+ if (attribute == -1) return -1; -+ bytes += sizeof(attribute); -+ -+ /* Read the size of the information we have to read */ -+ -+ if (_TNEF_debug) fprintf(stderr,"read_attribute: Reading Size...\n"); -+ size = read_32(tsp+bytes); -+ if (size == -1) return -1; -+ bytes += sizeof(size); -+ -+ /* The header size equals the sum of all the things we've read -+ so far. */ -+ -+ header = bytes; -+ -+ /* The is a bit of a tricky one [if you're being slow -+ it moves the number of bytes ahead by the amount of data of -+ the attribute we're about to read, so that for next -+ "read_attribute()" -+ call starts in the right place. -+ */ -+ -+ bytes += size; -+ -+ /* Read in the checksum for this component -+ -+ AMMENDMENT - 19/07/02 - 17H01 -+ Small code change to deal with strange sitations that occur with non -+ english characters. - Submitted by wtcheuk@netvigator.com @ 19/07/02 -+ */ -+ -+ if ( bytes < 0 ) return -1; -+ -+ /* --END of ammendment. */ -+ -+ if (_TNEF_debug) fprintf(stderr,"read_attribute: Reading Checksum...(offset %d, bytes=%d)\n", tsp -tnef_home, bytes); -+ checksum = read_16(tsp+bytes); -+ bytes += sizeof(checksum); -+ -+ if (_TNEF_debug) fprintf(stderr,"Decoding attribute %d\n",attribute); -+ -+ switch (attribute) { -+ case attNull: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attFrom: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attSubject: -+ break; -+ case attDateSent: -+ break; -+ case attDateRecd: -+ break; -+ case attMessageStatus: -+ break; -+ case attMessageClass: -+ break; -+ case attMessageID: -+ break; -+ case attParentID: -+ break; -+ case attConversationID: -+ break; -+ case attBody: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attPriority: -+ break; -+ case attAttachData: -+ attach_size=size; -+ attach_loc =(int)tsp+header; -+ if (SaveData && strlen(attach_title)>0 && attach_size > 0) { -+ if (!save_attach_data(attach_title, (uint8 *)attach_loc,attach_size)) -+ { -+ if (_TNEF_verbose) fprintf(stdout,"Decoding %s\n", attach_title); -+ } -+ else -+ { -+ if (_TNEF_syslogging > 0) syslog(1,"TNEF: Error saving attachment %s\n",attach_title); -+ } -+ } -+ break; -+ case attAttachTitle: -+ strncpy(attach_title, make_string(tsp+header,size),255); -+ if (SaveData && strlen(attach_title)>0 && attach_size > 0) { -+ if (!save_attach_data(attach_title, (uint8 *)attach_loc,attach_size)) -+ { -+ if (_TNEF_verbose) fprintf(stdout,"Decoding %s\n", attach_title); -+ } -+ else -+ { -+ if (_TNEF_syslogging > 0) syslog(1,"TNEF: Error saving attachment %s\n",attach_title); -+ } -+ } -+ break; -+ case attAttachMetaFile: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attAttachCreateDate: -+ break; -+ case attAttachModifyDate: -+ break; -+ case attDateModified: -+ break; -+ case attAttachTransportFilename: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attAttachRenddata: -+ attach_title[0]=0; -+ attach_size=0; -+ attach_loc=0; -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attMAPIProps: -+ handle_props(tsp+header); -+ break; -+ case attRecipTable: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attAttachment: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attTnefVersion: -+ { -+ uint32 version; -+ version = read_32(tsp+header); -+ if (version == -1) return -1; -+ } -+ break; -+ case attOemCodepage: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attOriginalMessageClass: -+ break; -+ case attOwner: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attSentFor: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attDelegate: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attDateStart: -+ break; -+ case attDateEnd: -+ break; -+ case attAidOwner: -+ default_handler(attribute, tsp+header, size); -+ break; -+ case attRequestRes: -+ default_handler(attribute, tsp+header, size); -+ break; -+ default: -+ default_handler(attribute, tsp+header, size); -+ break; -+ } -+ return bytes; -+ -+} -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: decode_tnef ID:1 -+Purpose: -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_decode_tnef(uint8 *tnef_stream, int size) -+{ -+ -+ int ra_response; -+ uint8 *tsp; -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF_decode_tnef: Start. Size = %d\n",size); -+ -+ /* TSP == TNEF Stream Pointer (well memory block actually!) -+ */ -+ tsp = tnef_stream; -+ -+ /* Read in the signature of this TNEF -+ */ -+ if (TNEF_SIGNATURE == read_32(tsp)) -+ { -+ if (_TNEF_debug) fprintf(stderr,"TNEF signature is good\n"); -+ } -+ else -+ { -+ if (_TNEF_stderrlogging > 0) fprintf(stderr,"TNEF_decode_tnef: Bad TNEF signature, expecting %lx got %lx\n",TNEF_SIGNATURE,read_32(tsp)); -+ } -+ -+ /* Move tsp pointer along -+ */ -+ tsp += sizeof(TNEF_SIGNATURE); -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF Attach Key: %x\n",read_16(tsp)); -+ /* Move tsp pointer along -+ */ -+ tsp += sizeof(uint16); -+ -+ /* While we still have more bytes to process, -+ go through entire memory block and extract -+ all the required attributes and files -+ */ -+ if (_TNEF_debug) fprintf(stderr,"TNEF - Commence reading attributes\n"); -+ while ((tsp - tnef_stream) < size) -+ { -+ if (_TNEF_debug) fprintf(stderr,"Offset = %d\n",tsp -tnef_home); -+ ra_response = read_attribute(tsp); -+ if ( ra_response > 0 ) -+ { -+ tsp += ra_response; -+ } else { -+ -+ /* Must find out /WHY/ this happens, and, how to rectify the issue. */ -+ -+ tsp++; -+ if (_TNEF_debug) fprintf(stderr,"TNEF - Attempting to read attribute at %d resulted in a sub-zero response, ending decoding to be safe\n"); -+ break; -+ } -+ } -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF - DONE.\n"); -+ -+ return 0; -+} -+ -+ -+ -+ -+ -+ -+/*------------------------------------------------------------------------ -+Procedure: TNEF_main ID:1 -+Purpose: Decodes a given TNEF encoded file -+Input: -+Output: -+Errors: -+------------------------------------------------------------------------*/ -+int TNEF_main( char *filename ) -+{ -+ FILE *fp; -+ struct stat sb; -+ uint8 *tnef_stream; -+ int size, nread; -+ int dump = 0; -+ int x; -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF_main: Start, decoding %s\n",filename); -+ -+ SaveData = TRUE; -+ -+ /* Test to see if the file actually exists -+ */ -+ if (stat(filename,&sb) == -1) -+ { -+ if (_TNEF_stderrlogging > 0) fprintf(stderr,"Error stating file %s (%s)\n", filename,strerror(errno)); -+ return -1; -+ } -+ -+ /* Get the filesize */ -+ -+ size = sb.st_size; -+ -+ /* Allocate enough memory to read in the ENTIRE file -+ FIXME - This could be a real consumer if multiple -+ instances of TNEF decoding is going on -+ */ -+ -+ tnef_home = tnef_stream = (uint8 *)malloc(size); -+ tnef_limit = tnef_home +size; -+ -+ /* If we were unable to allocate enough memory, then we -+ should report this */ -+ -+ if (tnef_stream == NULL) -+ { -+ if (_TNEF_stderrlogging > 0) fprintf(stderr,"Error allocating %d bytes for loading file (%s)\n", size,strerror(errno)); -+ return -1; -+ } -+ -+ /* Attempt to open up the TNEF encoded file... if it fails -+ then report the failed condition to syslog */ -+ -+ if ((fp = fopen(filename,"r")) == NULL) -+ { -+ if (_TNEF_stderrlogging > 0) fprintf(stderr,"Error opening file %s for reading (%s)\n", filename,strerror(errno)); -+ return -1; -+ } -+ -+ /* Attempt to read in the entire file */ -+ -+ nread = fread(tnef_stream, sizeof(uint8), size, fp); -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF: Read %d bytes\n",nread); -+ -+ /* If we did not read in all the bytes, then let syslogs know! */ -+ -+ if (nread < size) -+ { -+ return -1; -+ } -+ -+ /* Close the file */ -+ -+ fclose(fp); -+ -+ /* Proceed to decode the file */ -+ -+ TNEF_decode_tnef(tnef_stream,size); -+ -+ -+ if (_TNEF_debug) fprintf(stderr,"TNEF - finished decoding.\n"); -+ -+ return 0; -+} -+ -+ -+/* --------------------------END. */ -+ -+ -+ -diff -urN exim-4.20-orig/src/tnef.h exim-4.20/src/tnef.h ---- exim-4.20-orig/src/tnef.h Thu Jan 1 01:00:00 1970 -+++ exim-4.20/src/tnef.h Wed May 14 12:04:24 2003 -@@ -0,0 +1,1839 @@ -+/************************************************* -+* Exim - an Internet mail transport agent * -+*************************************************/ -+ -+/* This file is part of the exiscan-acl content scanner -+patch. It is NOT part of the standard exim distribution. */ -+ -+/*************************************************************************** -+ * -+ * config.h for tnef decoder by Brandon Long -+ * Based on config.h from S3MOD by Dan Marks and David Jeske -+ * -+ * (C) 1994,1995 By Daniel Marks and David Jeske -+ * -+ * While we retain the copyright to this code, this source code is FREE. -+ * You may use it in any way you wish, in any product you wish. You may -+ * NOT steal the copyright for this code from us. -+ * -+ * We respectfully ask that you email one of us, if possible, if you -+ * produce something significant with this code, or if you have any bug -+ * fixes to contribute. We also request that you give credit where -+ * credit is due if you include part of this code in a program of your own. -+ * -+ *************************************************************************** -+ * -+ * config.h - compile time configuration options and system specific defines -+ * -+ */ -+ -+/* 2003-02-03 Merged all TNEF and MAPI related headers in this file to reduce -+ clutter -+ - Tom Kistner -+*/ -+ -+#ifndef _CONFIG_H -+#define _CONFIG_H 1 -+ -+/***************************************************************************/ -+/* The following are system specific settings */ -+/***************************************************************************/ -+ -+#if defined(SUN) -+#define BIT_32 -+#define ___TNEF_BYTE_ORDER 4321 -+#undef NEAR_FAR_PTR -+ -+#elif defined (HPUX) -+#define BIT_32 -+#define ___TNEF_BYTE_ORDER 4321 -+#undef NEAR_FAR_PTR -+ -+#elif defined(DEC) -+#undef NEAR_FAR_PTR -+ -+#elif defined(__sgi) -+#define BIT_32 -+#define ___TNEF_BYTE_ORDER 4321 -+#undef NEAR_FAR_PTR -+ -+#elif defined(AIX) -+#undef NEAR_FAR_PTR -+#define ___TNEF_BYTE_ORDER 4321 -+#define BIT_32 -+ -+#elif defined(LINUX) -+#define BIT_32 -+#undef NEAR_FAR_PTR -+ -+#elif defined(MSDOS) -+#define NEAR_FAR_PTR -+#undef BIT_32 -+ -+#else -+#undef NEAR_FAR_PTR -+#define BIT_32 -+ -+ -+#endif /* OS/MACH TYPE */ -+ -+/***************************************************************************/ -+/* 16/32 Bit and Byte Order hacks */ -+/***************************************************************************/ -+ -+#ifdef BIT_32 -+typedef short int int16; -+typedef unsigned short int uint16; -+typedef int int32; -+typedef unsigned int uint32; -+typedef char int8; -+typedef unsigned char uint8; -+#else -+typedef int int16; -+typedef unsigned int uint16; -+typedef long int int32; -+typedef unsigned long int uint32; -+typedef char int8; -+typedef unsigned char uint8; -+#endif /* BIT_32 */ -+ -+#ifndef WIN32_TYPES -+#define ULONG uint32 -+#define SCODE uint32 -+#define FAR -+#define LPVOID void * -+#define WORD uint16 -+#define DWORD uint32 -+#define LONG int32 -+#define BYTE uint8 -+#endif /* !WIN32_TYPES */ -+ -+#define endian_switch(x) (((((uint16)(x)) & 0xFF00) >> 8) | \ -+ ((((uint16)(x)) & 0xFF) << 8)) -+ -+#define long_endian_switch(x) ( ((((uint32)(x)) & 0xFF00UL) << 8) | \ -+ ((((uint32)(x)) & 0xFFUL) << 24) | \ -+ ((((uint32)(x)) & 0xFF0000UL) >> 8) | \ -+ ((((uint32)(x)) & 0xFF000000UL) >> 24)) -+ -+#if ___TNEF_BYTE_ORDER == 4321 -+#define big_endian(x) (x) -+#define long_big_endian(x) (x) -+#define little_endian(x) (endian_switch(x)) -+#define long_little_endian(x) (long_endian_switch(x)) -+#else -+#define big_endian(x) (endian_switch(x)) -+#define long_big_endian(x) (long_endian_switch(x)) -+#define little_endian(x) (x) -+#define long_little_endian(x) (x) -+#endif /* ___TNEF_BYTE_ORDER */ -+ -+#ifndef TRUE -+#define TRUE 1 -+#endif -+#ifndef FALSE -+#define FALSE 0 -+#endif -+ -+ -+#endif /* _CONFIG_H */ -+/* -+ * Taken from the Win32 SDK or the MSVC4 include files, I'm not sure which. -+ * The document describing the TNEF format alludes to this document for more -+ * information. This file was stripped a bit to allow it to compile with -+ * GCC and without random other Windows header files so it could be used -+ * to decode TNEF bitstreams with tnef2txt. -+ * -+ * T N E F . H -+ * -+ * -+ * This file contains structure and function definitions for the -+ * MAPI implementation of the Transport Neutral Encapsilation Format -+ * used by MAPI providers for the neutral serialization of a MAPI -+ * message. This implementation sits on top of the IStream object as -+ * documented in the OLE 2 Specs. -+ * -+ * Copyright 1986-1996 Microsoft Corporation. All Rights Reserved. -+ */ -+ -+#ifndef TNEF_H -+#define TNEF_H -+ -+#ifdef __cplusplus -+extern "C" { -+#endif -+ -+ -+#ifndef BEGIN_INTERFACE -+#define BEGIN_INTERFACE -+#endif -+ -+#ifndef MAPI_DIM -+#define MAPI_DIM 1 -+#endif -+ -+#define TNTNoffsetof(s,m) (unsigned long)&(((s *)0)->m) -+ -+/* ------------------------------------ */ -+/* TNEF Problem and TNEF Problem Arrays */ -+/* ------------------------------------ */ -+ -+typedef struct _STnefProblem -+{ -+ ULONG ulComponent; -+ ULONG ulAttribute; -+ ULONG ulPropTag; -+ SCODE scode; -+} STnefProblem; -+ -+typedef struct _STnefProblemArray -+{ -+ ULONG cProblem; -+ STnefProblem aProblem[MAPI_DIM]; -+} STnefProblemArray, FAR * LPSTnefProblemArray; -+ -+#if 0 -+#define CbNewSTnefProblemArray(_cprob) \ -+ (TNoffsetof(STnefProblemArray,aProblem) + (_cprob)*sizeof(STnefProblem)) -+#define CbSTnefProblemArray(_lparray) \ -+ (TNoffsetof(STnefProblemArray,aProblem) + \ -+ (UINT) ((_lparray)->cProblem*sizeof(STnefProblem))) -+#endif -+ -+/* Pointers to TNEF Interface ---------------------------------------- */ -+ -+#if 0 -+DECLARE_MAPI_INTERFACE_PTR(ITnef, LPITNEF); -+#endif -+ -+/* OpenTNEFStream */ -+ -+#define TNEF_DECODE ((ULONG) 0) -+#define TNEF_ENCODE ((ULONG) 2) -+ -+#define TNEF_PURE ((ULONG) 0x00010000) -+#define TNEF_COMPATIBILITY ((ULONG) 0x00020000) -+#define TNEF_BEST_DATA ((ULONG) 0x00040000) -+#define TNEF_COMPONENT_ENCODING ((ULONG) 0x80000000) -+ -+/* AddProps, ExtractProps */ -+ -+#define TNEF_PROP_INCLUDE ((ULONG) 0x00000001) -+#define TNEF_PROP_EXCLUDE ((ULONG) 0x00000002) -+#define TNEF_PROP_CONTAINED ((ULONG) 0x00000004) -+#define TNEF_PROP_MESSAGE_ONLY ((ULONG) 0x00000008) -+#define TNEF_PROP_ATTACHMENTS_ONLY ((ULONG) 0x00000010) -+#define TNEF_PROP_CONTAINED_TNEF ((ULONG) 0x00000040) -+ -+/* FinishComponent */ -+ -+#define TNEF_COMPONENT_MESSAGE ((ULONG) 0x00001000) -+#define TNEF_COMPONENT_ATTACHMENT ((ULONG) 0x00002000) -+ -+#if 0 -+#define MAPI_ITNEF_METHODS(IPURE) \ -+ MAPIMETHOD(AddProps) \ -+ (THIS_ ULONG ulFlags, \ -+ ULONG ulElemID, \ -+ LPVOID lpvData, \ -+ LPSPropTagArray lpPropList) IPURE; \ -+ MAPIMETHOD(ExtractProps) \ -+ (THIS_ ULONG ulFlags, \ -+ LPSPropTagArray lpPropList, \ -+ LPSTnefProblemArray FAR * lpProblems) IPURE; \ -+ MAPIMETHOD(Finish) \ -+ (THIS_ ULONG ulFlags, \ -+ WORD FAR * lpKey, \ -+ LPSTnefProblemArray FAR * lpProblems) IPURE; \ -+ MAPIMETHOD(OpenTaggedBody) \ -+ (THIS_ LPMESSAGE lpMessage, \ -+ ULONG ulFlags, \ -+ LPSTREAM FAR * lppStream) IPURE; \ -+ MAPIMETHOD(SetProps) \ -+ (THIS_ ULONG ulFlags, \ -+ ULONG ulElemID, \ -+ ULONG cValues, \ -+ LPSPropValue lpProps) IPURE; \ -+ MAPIMETHOD(EncodeRecips) \ -+ (THIS_ ULONG ulFlags, \ -+ LPMAPITABLE lpRecipientTable) IPURE; \ -+ MAPIMETHOD(FinishComponent) \ -+ (THIS_ ULONG ulFlags, \ -+ ULONG ulComponentID, \ -+ LPSPropTagArray lpCustomPropList, \ -+ LPSPropValue lpCustomProps, \ -+ LPSPropTagArray lpPropList, \ -+ LPSTnefProblemArray FAR * lpProblems) IPURE; \ -+ -+#undef INTERFACE -+#define INTERFACE ITnef -+DECLARE_MAPI_INTERFACE_(ITnef, IUnknown) -+{ -+ BEGIN_INTERFACE -+ MAPI_IUNKNOWN_METHODS(PURE) -+ MAPI_ITNEF_METHODS(PURE) -+}; -+ -+STDMETHODIMP OpenTnefStream( -+ LPVOID lpvSupport, -+ LPSTREAM lpStream, -+ LPTSTR lpszStreamName, -+ ULONG ulFlags, -+ LPMESSAGE lpMessage, -+ WORD wKeyVal, -+ LPITNEF FAR * lppTNEF); -+ -+typedef HRESULT (STDMETHODCALLTYPE FAR * LPOPENTNEFSTREAM) ( -+ LPVOID lpvSupport, -+ LPSTREAM lpStream, -+ LPTSTR lpszStreamName, -+ ULONG ulFlags, -+ LPMESSAGE lpMessage, -+ WORD wKeyVal, -+ LPITNEF FAR * lppTNEF); -+ -+STDMETHODIMP OpenTnefStreamEx( -+ LPVOID lpvSupport, -+ LPSTREAM lpStream, -+ LPTSTR lpszStreamName, -+ ULONG ulFlags, -+ LPMESSAGE lpMessage, -+ WORD wKeyVal, -+ LPADRBOOK lpAdressBook, -+ LPITNEF FAR * lppTNEF); -+ -+typedef HRESULT (STDMETHODCALLTYPE FAR * LPOPENTNEFSTREAMEX) ( -+ LPVOID lpvSupport, -+ LPSTREAM lpStream, -+ LPTSTR lpszStreamName, -+ ULONG ulFlags, -+ LPMESSAGE lpMessage, -+ WORD wKeyVal, -+ LPADRBOOK lpAdressBook, -+ LPITNEF FAR * lppTNEF); -+ -+STDMETHODIMP GetTnefStreamCodepage ( -+ LPSTREAM lpStream, -+ ULONG FAR * lpulCodepage, -+ ULONG FAR * lpulSubCodepage); -+ -+typedef HRESULT (STDMETHODCALLTYPE FAR * LPGETTNEFSTREAMCODEPAGE) ( -+ LPSTREAM lpStream, -+ ULONG FAR * lpulCodepage, -+ ULONG FAR * lpulSubCodepage); -+ -+#define OPENTNEFSTREAM "OpenTnefStream" -+#define OPENTNEFSTREAMEX "OpenTnefStreamEx" -+#define GETTNEFSTREAMCODEPAGE "GetTnefStreamCodePage" -+#endif -+ -+/* -------------------------- */ -+/* TNEF Signature and Version */ -+/* -------------------------- */ -+ -+#define MAKE_TNEF_VERSION(_mj,_mn) (((ULONG)(0x0000FFFF & _mj) << 16) | (ULONG)(0x0000FFFF & _mn)) -+#define TNEF_SIGNATURE ((ULONG) 0x223E9F78) -+#define TNEF_VERSION ((ULONG) MAKE_TNEF_VERSION(1,0)) -+ -+ -+/* ------------------------------------------- */ -+/* TNEF Down-level Attachment Types/Structures */ -+/* ------------------------------------------- */ -+ -+typedef WORD ATYP; -+enum { atypNull, atypFile, atypOle, atypPicture, atypMax }; -+ -+#define MAC_BINARY ((DWORD) 0x00000001) -+ -+typedef struct _renddata -+{ -+ ATYP atyp; -+ ULONG ulPosition; -+ WORD dxWidth; -+ WORD dyHeight; -+ DWORD dwFlags; -+ -+} RENDDATA, *PRENDDATA; -+ -+/* ----------------------------------- */ -+/* TNEF Down-level Date/Time Structure */ -+/* ----------------------------------- */ -+ -+typedef struct _dtr -+{ -+ WORD wYear; -+ WORD wMonth; -+ WORD wDay; -+ WORD wHour; -+ WORD wMinute; -+ WORD wSecond; -+ WORD wDayOfWeek; -+ -+} DTR; -+ -+ -+/* ----------------------------- */ -+/* TNEF Down-level Message Flags */ -+/* ----------------------------- */ -+ -+#define fmsNull ((BYTE) 0x00) -+#define fmsModified ((BYTE) 0x01) -+#define fmsLocal ((BYTE) 0x02) -+#define fmsSubmitted ((BYTE) 0x04) -+#define fmsRead ((BYTE) 0x20) -+#define fmsHasAttach ((BYTE) 0x80) -+ -+ -+/* ----------------------------------------- */ -+/* TNEF Down-level Triple Address Structures */ -+/* ----------------------------------------- */ -+ -+#define trpidNull ((WORD) 0x0000) -+#define trpidUnresolved ((WORD) 0x0001) -+#define trpidResolvedNSID ((WORD) 0x0002) -+#define trpidResolvedAddress ((WORD) 0x0003) -+#define trpidOneOff ((WORD) 0x0004) -+#define trpidGroupNSID ((WORD) 0x0005) -+#define trpidOffline ((WORD) 0x0006) -+#define trpidIgnore ((WORD) 0x0007) -+#define trpidClassEntry ((WORD) 0x0008) -+#define trpidResolvedGroupAddress ((WORD) 0x0009) -+typedef struct _trp -+{ -+ WORD trpid; -+ WORD cbgrtrp; -+ WORD cch; -+ WORD cbRgb; -+ -+} TRP, *PTRP, *PGRTRP, FAR * LPTRP; -+#define CbOfTrp(_p) (sizeof(TRP) + (_p)->cch + (_p)->cbRgb) -+#define LpszOfTrp(_p) ((LPSTR)(((LPTRP) (_p)) + 1)) -+#define LpbOfTrp(_p) (((LPBYTE)(((LPTRP)(_p)) + 1)) + (_p)->cch) -+#define LptrpNext(_p) ((LPTRP)((LPBYTE)(_p) + CbOfTrp(_p))) -+ -+typedef DWORD XTYPE; -+#define xtypeUnknown ((XTYPE) 0) -+#define xtypeInternet ((XTYPE) 6) -+ -+#define cbDisplayName 41 -+#define cbEmailName 11 -+#define cbSeverName 12 -+typedef struct _ADDR_ALIAS -+{ -+ char rgchName[cbDisplayName]; -+ char rgchEName[cbEmailName]; -+ char rgchSrvr[cbSeverName]; -+ ULONG dibDetail; -+ WORD type; -+ -+} ADDRALIAS, FAR * LPADDRALIAS; -+#define cbALIAS sizeof(ALIAS) -+ -+#define cbTYPE 16 -+#define cbMaxIdData 200 -+typedef struct _NSID -+{ -+ DWORD dwSize; -+ unsigned char uchType[cbTYPE]; -+ XTYPE xtype; -+ LONG lTime; -+ -+ union -+ { -+ ADDRALIAS alias; -+ char rgchInterNet[1]; -+ -+ } address; -+ -+} NSID, * LPNSID; -+#define cbNSID sizeof(NSID) -+ -+ -+/* -------------------------- */ -+/* TNEF Down-level Priorities */ -+/* -------------------------- */ -+ -+#define prioLow 3 -+#define prioNorm 2 -+#define prioHigh 1 -+ -+ -+/* ------------------------------------- */ -+/* TNEF Down-level Attributes/Properties */ -+/* ------------------------------------- */ -+ -+#define atpTriples ((WORD) 0x0000) -+#define atpString ((WORD) 0x0001) -+#define atpText ((WORD) 0x0002) -+#define atpDate ((WORD) 0x0003) -+#define atpShort ((WORD) 0x0004) -+#define atpLong ((WORD) 0x0005) -+#define atpByte ((WORD) 0x0006) -+#define atpWord ((WORD) 0x0007) -+#define atpDword ((WORD) 0x0008) -+#define atpMax ((WORD) 0x0009) -+ -+#define LVL_MESSAGE ((BYTE) 0x01) -+#define LVL_ATTACHMENT ((BYTE) 0x02) -+ -+#define ATT_ID(_att) ((WORD) ((_att) & 0x0000FFFF)) -+#define ATT_TYPE(_att) ((WORD) (((_att) >> 16) & 0x0000FFFF)) -+#define ATT(_atp, _id) ((((DWORD) (_atp)) << 16) | ((WORD) (_id))) -+ -+#define attNull ATT( 0, 0x0000) -+#define attFrom ATT( atpTriples, 0x8000) /* PR_ORIGINATOR_RETURN_ADDRESS */ -+#define attSubject ATT( atpString, 0x8004) /* PR_SUBJECT */ -+#define attDateSent ATT( atpDate, 0x8005) /* PR_CLIENT_SUBMIT_TIME */ -+#define attDateRecd ATT( atpDate, 0x8006) /* PR_MESSAGE_DELIVERY_TIME */ -+#define attMessageStatus ATT( atpByte, 0x8007) /* PR_MESSAGE_FLAGS */ -+#define attMessageClass ATT( atpWord, 0x8008) /* PR_MESSAGE_CLASS */ -+#define attMessageID ATT( atpString, 0x8009) /* PR_MESSAGE_ID */ -+#define attParentID ATT( atpString, 0x800A) /* PR_PARENT_ID */ -+#define attConversationID ATT( atpString, 0x800B) /* PR_CONVERSATION_ID */ -+#define attBody ATT( atpText, 0x800C) /* PR_BODY */ -+#define attPriority ATT( atpShort, 0x800D) /* PR_IMPORTANCE */ -+#define attAttachData ATT( atpByte, 0x800F) /* PR_ATTACH_DATA_xxx */ -+#define attAttachTitle ATT( atpString, 0x8010) /* PR_ATTACH_FILENAME */ -+#define attAttachMetaFile ATT( atpByte, 0x8011) /* PR_ATTACH_RENDERING */ -+#define attAttachCreateDate ATT( atpDate, 0x8012) /* PR_CREATION_TIME */ -+#define attAttachModifyDate ATT( atpDate, 0x8013) /* PR_LAST_MODIFICATION_TIME */ -+#define attDateModified ATT( atpDate, 0x8020) /* PR_LAST_MODIFICATION_TIME */ -+#define attAttachTransportFilename ATT( atpByte, 0x9001) /* PR_ATTACH_TRANSPORT_NAME */ -+#define attAttachRenddata ATT( atpByte, 0x9002) -+#define attMAPIProps ATT( atpByte, 0x9003) -+#define attRecipTable ATT( atpByte, 0x9004) /* PR_MESSAGE_RECIPIENTS */ -+#define attAttachment ATT( atpByte, 0x9005) -+#define attTnefVersion ATT( atpDword, 0x9006) -+#define attOemCodepage ATT( atpByte, 0x9007) -+#define attOriginalMessageClass ATT( atpWord, 0x0006) /* PR_ORIG_MESSAGE_CLASS */ -+ -+#define attOwner ATT( atpByte, 0x0000) /* PR_RCVD_REPRESENTING_xxx or -+ PR_SENT_REPRESENTING_xxx */ -+#define attSentFor ATT( atpByte, 0x0001) /* PR_SENT_REPRESENTING_xxx */ -+#define attDelegate ATT( atpByte, 0x0002) /* PR_RCVD_REPRESENTING_xxx */ -+#define attDateStart ATT( atpDate, 0x0006) /* PR_DATE_START */ -+#define attDateEnd ATT( atpDate, 0x0007) /* PR_DATE_END */ -+#define attAidOwner ATT( atpLong, 0x0008) /* PR_OWNER_APPT_ID */ -+#define attRequestRes ATT( atpShort, 0x0009) /* PR_RESPONSE_REQUESTED */ -+ -+#ifdef __cplusplus -+} -+#endif -+ -+#endif /* defined TNEF_H */ -+/* -+ * M A P I D E F S . H -+ * -+ * Definitions used by MAPI clients and service providers. -+ * -+ * Copyright 1986-1996 Microsoft Corporation. All Rights Reserved. -+ */ -+ -+#ifndef MAPIDEFS_H -+#define MAPIDEFS_H -+ -+ -+/* Array dimension for structures with variable-sized arrays at the end. */ -+ -+/* Simple data types */ -+ -+ -+typedef WORD WCHAR; -+ -+#ifdef UNICODE -+typedef WCHAR TCHAR; -+#else -+typedef char TCHAR; -+#endif -+ -+typedef WCHAR * LPWSTR; -+typedef const WCHAR * LPCWSTR; -+typedef TCHAR * LPTSTR; -+typedef const TCHAR * LPCTSTR; -+typedef BYTE * LPBYTE; -+ -+typedef ULONG * LPULONG; -+ -+#ifndef __LHANDLE -+#define __LHANDLE -+typedef unsigned long LHANDLE, * LPLHANDLE; -+#endif -+ -+#if !defined(_WINBASE_) && !defined(_FILETIME_) -+#define _FILETIME_ -+typedef struct _FILETIME -+{ -+ DWORD dwLowDateTime; -+ DWORD dwHighDateTime; -+} FILETIME, * LPFILETIME; -+#endif -+ -+/* -+ * This flag is used in many different MAPI calls to signify that -+ * the object opened by the call should be modifiable (MAPI_MODIFY). -+ * If the flag MAPI_MAX_ACCESS is set, the object returned should be -+ * returned at the maximum access level allowed. An additional -+ * property available on the object (PR_ACCESS_LEVEL) uses the same -+ * MAPI_MODIFY flag to say just what this new access level is. -+ */ -+ -+#define MAPI_MODIFY ((ULONG) 0x00000001) -+ -+/* -+ * The following flags are used to indicate to the client what access -+ * level is permissible in the object. They appear in PR_ACCESS in -+ * message and folder objects as well as in contents and associated -+ * contents tables -+ */ -+ -+#define MAPI_ACCESS_MODIFY ((ULONG) 0x00000001) -+#define MAPI_ACCESS_READ ((ULONG) 0x00000002) -+#define MAPI_ACCESS_DELETE ((ULONG) 0x00000004) -+#define MAPI_ACCESS_CREATE_HIERARCHY ((ULONG) 0x00000008) -+#define MAPI_ACCESS_CREATE_CONTENTS ((ULONG) 0x00000010) -+#define MAPI_ACCESS_CREATE_ASSOCIATED ((ULONG) 0x00000020) -+ -+/* -+ * The MAPI_UNICODE flag is used in many different MAPI calls to signify -+ * that strings passed through the interface are in Unicode (a 16-bit -+ * character set). The default is an 8-bit character set. -+ * -+ * The value fMapiUnicode can be used as the 'normal' value for -+ * that bit, given the application's default character set. -+ */ -+ -+#define MAPI_UNICODE ((ULONG) 0x80000000) -+ -+#ifdef UNICODE -+#define fMapiUnicode MAPI_UNICODE -+#else -+#define fMapiUnicode 0 -+#endif -+ -+/* successful HRESULT */ -+#define hrSuccess 0 -+ -+ -+ -+/* Recipient types */ -+#ifndef MAPI_ORIG /* also defined in mapi.h */ -+#define MAPI_ORIG 0 /* Recipient is message originator */ -+#define MAPI_TO 1 /* Recipient is a primary recipient */ -+#define MAPI_CC 2 /* Recipient is a copy recipient */ -+#define MAPI_BCC 3 /* Recipient is blind copy recipient */ -+#define MAPI_P1 0x10000000 /* Recipient is a P1 resend recipient */ -+#define MAPI_SUBMITTED 0x80000000 /* Recipient is already processed */ -+/* #define MAPI_AUTHORIZE 4 recipient is a CMC authorizing user */ -+/*#define MAPI_DISCRETE 0x10000000 Recipient is a P1 resend recipient */ -+#endif -+ -+/* Bit definitions for abFlags[0] of ENTRYID */ -+#define MAPI_SHORTTERM 0x80 -+#define MAPI_NOTRECIP 0x40 -+#define MAPI_THISSESSION 0x20 -+#define MAPI_NOW 0x10 -+#define MAPI_NOTRESERVED 0x08 -+ -+/* Bit definitions for abFlags[1] of ENTRYID */ -+#define MAPI_COMPOUND 0x80 -+ -+/* ENTRYID */ -+typedef struct -+{ -+ BYTE abFlags[4]; -+ BYTE ab[MAPI_DIM]; -+} ENTRYID, *LPENTRYID; -+ -+#define CbNewENTRYID(_cb) (offsetof(ENTRYID,ab) + (_cb)) -+#define CbENTRYID(_cb) (offsetof(ENTRYID,ab) + (_cb)) -+ -+/* Byte-order-independent version of GUID (world-unique identifier) */ -+typedef struct _MAPIUID -+{ -+ BYTE ab[16]; -+} MAPIUID, * LPMAPIUID; -+ -+/* Note: need to include C run-times (memory.h) to use this macro */ -+ -+#define IsEqualMAPIUID(lpuid1, lpuid2) (!memcmp(lpuid1, lpuid2, sizeof(MAPIUID))) -+ -+/* -+ * Constants for one-off entry ID: -+ * The MAPIUID that identifies the one-off provider; -+ * the flag that defines whether the embedded strings are Unicode; -+ * the flag that specifies whether the recipient gets TNEF or not. -+ */ -+ -+#define MAPI_ONE_OFF_UID { 0x81, 0x2b, 0x1f, 0xa4, 0xbe, 0xa3, 0x10, 0x19, 0x9d, 0x6e, 0x00, 0xdd, 0x01, 0x0f, 0x54, 0x02 } -+#define MAPI_ONE_OFF_UNICODE 0x8000 -+#define MAPI_ONE_OFF_NO_RICH_INFO 0x0001 -+ -+/* Object type */ -+ -+#define MAPI_STORE ((ULONG) 0x00000001) /* Message Store */ -+#define MAPI_ADDRBOOK ((ULONG) 0x00000002) /* Address Book */ -+#define MAPI_FOLDER ((ULONG) 0x00000003) /* Folder */ -+#define MAPI_ABCONT ((ULONG) 0x00000004) /* Address Book Container */ -+#define MAPI_MESSAGE ((ULONG) 0x00000005) /* Message */ -+#define MAPI_MAILUSER ((ULONG) 0x00000006) /* Individual Recipient */ -+#define MAPI_ATTACH ((ULONG) 0x00000007) /* Attachment */ -+#define MAPI_DISTLIST ((ULONG) 0x00000008) /* Distribution List Recipient */ -+#define MAPI_PROFSECT ((ULONG) 0x00000009) /* Profile Section */ -+#define MAPI_STATUS ((ULONG) 0x0000000A) /* Status Object */ -+#define MAPI_SESSION ((ULONG) 0x0000000B) /* Session */ -+#define MAPI_FORMINFO ((ULONG) 0x0000000C) /* Form Information */ -+ -+ -+/* -+ * Maximum length of profile names and passwords, not including -+ * the null termination character. -+ */ -+#ifndef cchProfileNameMax -+#define cchProfileNameMax 64 -+#define cchProfilePassMax 64 -+#endif -+ -+ -+/* Property Types */ -+ -+#define MV_FLAG 0x1000 /* Multi-value flag */ -+ -+#define PT_UNSPECIFIED ((ULONG) 0) /* (Reserved for interface use) type doesn't matter to caller */ -+#define PT_NULL ((ULONG) 1) /* NULL property value */ -+#define PT_I2 ((ULONG) 2) /* Signed 16-bit value */ -+#define PT_LONG ((ULONG) 3) /* Signed 32-bit value */ -+#define PT_R4 ((ULONG) 4) /* 4-byte floating point */ -+#define PT_DOUBLE ((ULONG) 5) /* Floating point double */ -+#define PT_CURRENCY ((ULONG) 6) /* Signed 64-bit int (decimal w/ 4 digits right of decimal pt) */ -+#define PT_APPTIME ((ULONG) 7) /* Application time */ -+#define PT_ERROR ((ULONG) 10) /* 32-bit error value */ -+#define PT_BOOLEAN ((ULONG) 11) /* 16-bit boolean (non-zero true) */ -+#define PT_OBJECT ((ULONG) 13) /* Embedded object in a property */ -+#define PT_I8 ((ULONG) 20) /* 8-byte signed integer */ -+#define PT_STRING8 ((ULONG) 30) /* Null terminated 8-bit character string */ -+#define PT_UNICODE ((ULONG) 31) /* Null terminated Unicode string */ -+#define PT_SYSTIME ((ULONG) 64) /* FILETIME 64-bit int w/ number of 100ns periods since Jan 1,1601 */ -+#define PT_CLSID ((ULONG) 72) /* OLE GUID */ -+#define PT_BINARY ((ULONG) 258) /* Uninterpreted (counted byte array) */ -+/* Changes are likely to these numbers, and to their structures. */ -+ -+/* Alternate property type names for ease of use */ -+#define PT_SHORT PT_I2 -+#define PT_I4 PT_LONG -+#define PT_FLOAT PT_R4 -+#define PT_R8 PT_DOUBLE -+#define PT_LONGLONG PT_I8 -+ -+/* -+ * The type of a MAPI-defined string property is indirected, so -+ * that it defaults to Unicode string on a Unicode platform and to -+ * String8 on an ANSI or DBCS platform. -+ * -+ * Macros are defined here both for the property type, and for the -+ * field of the property value structure which should be -+ * dereferenced to obtain the string pointer. -+ */ -+ -+#ifdef UNICODE -+#define PT_TSTRING PT_UNICODE -+#define PT_MV_TSTRING (MV_FLAG|PT_UNICODE) -+#define LPSZ lpszW -+#define LPPSZ lppszW -+#define MVSZ MVszW -+#else -+#define PT_TSTRING PT_STRING8 -+#define PT_MV_TSTRING (MV_FLAG|PT_STRING8) -+#define LPSZ lpszA -+#define LPPSZ lppszA -+#define MVSZ MVszA -+#endif -+ -+ -+/* Property Tags -+ * -+ * By convention, MAPI never uses 0 or FFFF as a property ID. -+ * Use as null values, initializers, sentinels, or what have you. -+ */ -+ -+#define PROP_TYPE_MASK ((ULONG)0x0000FFFF) /* Mask for Property type */ -+#define PROP_TYPE(ulPropTag) (((ULONG)(ulPropTag))&PROP_TYPE_MASK) -+#define PROP_ID(ulPropTag) (((ULONG)(ulPropTag))>>16) -+#define PROP_TAG(ulPropType,ulPropID) ((((ULONG)(ulPropID))<<16)|((ULONG)(ulPropType))) -+#define PROP_ID_NULL 0 -+#define PROP_ID_INVALID 0xFFFF -+#define PR_NULL PROP_TAG( PT_NULL, PROP_ID_NULL) -+#if 0 -+#define CHANGE_PROP_TYPE(ulPropTag, ulPropType) \ -+ (((ULONG)0xFFFF0000 & ulPropTag) | ulPropType) -+#endif -+ -+ -+/* Multi-valued Property Types */ -+ -+#define PT_MV_I2 (MV_FLAG|PT_I2) -+#define PT_MV_LONG (MV_FLAG|PT_LONG) -+#define PT_MV_R4 (MV_FLAG|PT_R4) -+#define PT_MV_DOUBLE (MV_FLAG|PT_DOUBLE) -+#define PT_MV_CURRENCY (MV_FLAG|PT_CURRENCY) -+#define PT_MV_APPTIME (MV_FLAG|PT_APPTIME) -+#define PT_MV_SYSTIME (MV_FLAG|PT_SYSTIME) -+#define PT_MV_STRING8 (MV_FLAG|PT_STRING8) -+#define PT_MV_BINARY (MV_FLAG|PT_BINARY) -+#define PT_MV_UNICODE (MV_FLAG|PT_UNICODE) -+#define PT_MV_CLSID (MV_FLAG|PT_CLSID) -+#define PT_MV_I8 (MV_FLAG|PT_I8) -+ -+/* Alternate property type names for ease of use */ -+#define PT_MV_SHORT PT_MV_I2 -+#define PT_MV_I4 PT_MV_LONG -+#define PT_MV_FLOAT PT_MV_R4 -+#define PT_MV_R8 PT_MV_DOUBLE -+#define PT_MV_LONGLONG PT_MV_I8 -+ -+/* -+ * Property type reserved bits -+ * -+ * MV_INSTANCE is used as a flag in table operations to request -+ * that a multi-valued property be presented as a single-valued -+ * property appearing in multiple rows. -+ */ -+ -+#define MV_INSTANCE 0x2000 -+#define MVI_FLAG (MV_FLAG | MV_INSTANCE) -+#define MVI_PROP(tag) ((tag) | MVI_FLAG) -+ -+ -+ -+#endif /* MAPIDEFS_H */ -+/* -+ * M A P I T A G S . H -+ * -+ * Property tag definitions for standard properties of MAPI -+ * objects. -+ * -+ * The following ranges should be used for all property IDs. Note that -+ * property IDs for objects other than messages and recipients should -+ * all fall in the range 0x3000 to 0x3FFF: -+ * -+ * From To Kind of property -+ * -------------------------------- -+ * 0001 0BFF MAPI_defined envelope property -+ * 0C00 0DFF MAPI_defined per-recipient property -+ * 0E00 0FFF MAPI_defined non-transmittable property -+ * 1000 2FFF MAPI_defined message content property -+ * -+ * 3000 3FFF MAPI_defined property (usually not message or recipient) -+ * -+ * 4000 57FF Transport-defined envelope property -+ * 5800 5FFF Transport-defined per-recipient property -+ * 6000 65FF User-defined non-transmittable property -+ * 6600 67FF Provider-defined internal non-transmittable property -+ * 6800 7BFF Message class-defined content property -+ * 7C00 7FFF Message class-defined non-transmittable -+ * property -+ * -+ * 8000 FFFE User-defined Name-to-id mapped property -+ * -+ * The 3000-3FFF range is further subdivided as follows: -+ * -+ * From To Kind of property -+ * -------------------------------- -+ * 3000 33FF Common property such as display name, entry ID -+ * 3400 35FF Message store object -+ * 3600 36FF Folder or AB container -+ * 3700 38FF Attachment -+ * 3900 39FF Address book object -+ * 3A00 3BFF Mail user -+ * 3C00 3CFF Distribution list -+ * 3D00 3DFF Profile section -+ * 3E00 3FFF Status object -+ * -+ * Copyright 1986-1996 Microsoft Corporation. All Rights Reserved. -+ */ -+ -+#ifndef MAPITAGS_H -+#define MAPITAGS_H -+ -+/* Determine if a property is transmittable. */ -+ -+#define FIsTransmittable(ulPropTag) \ -+ ((PROP_ID (ulPropTag) < (ULONG)0x0E00) || \ -+ (PROP_ID (ulPropTag) >= (ULONG)0x8000) || \ -+ ((PROP_ID (ulPropTag) >= (ULONG)0x1000) && (PROP_ID (ulPropTag) < (ULONG)0x6000)) || \ -+ ((PROP_ID (ulPropTag) >= (ULONG)0x6800) && (PROP_ID (ulPropTag) < (ULONG)0x7C00))) -+ -+/* -+ * Message envelope properties -+ */ -+ -+#define PR_ACKNOWLEDGEMENT_MODE PROP_TAG( PT_LONG, 0x0001) -+#define PR_ALTERNATE_RECIPIENT_ALLOWED PROP_TAG( PT_BOOLEAN, 0x0002) -+#define PR_AUTHORIZING_USERS PROP_TAG( PT_BINARY, 0x0003) -+#define PR_AUTO_FORWARD_COMMENT PROP_TAG( PT_TSTRING, 0x0004) -+#define PR_AUTO_FORWARD_COMMENT_W PROP_TAG( PT_UNICODE, 0x0004) -+#define PR_AUTO_FORWARD_COMMENT_A PROP_TAG( PT_STRING8, 0x0004) -+#define PR_AUTO_FORWARDED PROP_TAG( PT_BOOLEAN, 0x0005) -+#define PR_CONTENT_CONFIDENTIALITY_ALGORITHM_ID PROP_TAG( PT_BINARY, 0x0006) -+#define PR_CONTENT_CORRELATOR PROP_TAG( PT_BINARY, 0x0007) -+#define PR_CONTENT_IDENTIFIER PROP_TAG( PT_TSTRING, 0x0008) -+#define PR_CONTENT_IDENTIFIER_W PROP_TAG( PT_UNICODE, 0x0008) -+#define PR_CONTENT_IDENTIFIER_A PROP_TAG( PT_STRING8, 0x0008) -+#define PR_CONTENT_LENGTH PROP_TAG( PT_LONG, 0x0009) -+#define PR_CONTENT_RETURN_REQUESTED PROP_TAG( PT_BOOLEAN, 0x000A) -+ -+ -+ -+#define PR_CONVERSATION_KEY PROP_TAG( PT_BINARY, 0x000B) -+ -+#define PR_CONVERSION_EITS PROP_TAG( PT_BINARY, 0x000C) -+#define PR_CONVERSION_WITH_LOSS_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x000D) -+#define PR_CONVERTED_EITS PROP_TAG( PT_BINARY, 0x000E) -+#define PR_DEFERRED_DELIVERY_TIME PROP_TAG( PT_SYSTIME, 0x000F) -+#define PR_DELIVER_TIME PROP_TAG( PT_SYSTIME, 0x0010) -+#define PR_DISCARD_REASON PROP_TAG( PT_LONG, 0x0011) -+#define PR_DISCLOSURE_OF_RECIPIENTS PROP_TAG( PT_BOOLEAN, 0x0012) -+#define PR_DL_EXPANSION_HISTORY PROP_TAG( PT_BINARY, 0x0013) -+#define PR_DL_EXPANSION_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x0014) -+#define PR_EXPIRY_TIME PROP_TAG( PT_SYSTIME, 0x0015) -+#define PR_IMPLICIT_CONVERSION_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x0016) -+#define PR_IMPORTANCE PROP_TAG( PT_LONG, 0x0017) -+#define PR_IPM_ID PROP_TAG( PT_BINARY, 0x0018) -+#define PR_LATEST_DELIVERY_TIME PROP_TAG( PT_SYSTIME, 0x0019) -+#define PR_MESSAGE_CLASS PROP_TAG( PT_TSTRING, 0x001A) -+#define PR_MESSAGE_CLASS_W PROP_TAG( PT_UNICODE, 0x001A) -+#define PR_MESSAGE_CLASS_A PROP_TAG( PT_STRING8, 0x001A) -+#define PR_MESSAGE_DELIVERY_ID PROP_TAG( PT_BINARY, 0x001B) -+ -+ -+ -+ -+ -+#define PR_MESSAGE_SECURITY_LABEL PROP_TAG( PT_BINARY, 0x001E) -+#define PR_OBSOLETED_IPMS PROP_TAG( PT_BINARY, 0x001F) -+#define PR_ORIGINALLY_INTENDED_RECIPIENT_NAME PROP_TAG( PT_BINARY, 0x0020) -+#define PR_ORIGINAL_EITS PROP_TAG( PT_BINARY, 0x0021) -+#define PR_ORIGINATOR_CERTIFICATE PROP_TAG( PT_BINARY, 0x0022) -+#define PR_ORIGINATOR_DELIVERY_REPORT_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0023) -+#define PR_ORIGINATOR_RETURN_ADDRESS PROP_TAG( PT_BINARY, 0x0024) -+ -+ -+ -+#define PR_PARENT_KEY PROP_TAG( PT_BINARY, 0x0025) -+#define PR_PRIORITY PROP_TAG( PT_LONG, 0x0026) -+ -+ -+ -+#define PR_ORIGIN_CHECK PROP_TAG( PT_BINARY, 0x0027) -+#define PR_PROOF_OF_SUBMISSION_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0028) -+#define PR_READ_RECEIPT_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0029) -+#define PR_RECEIPT_TIME PROP_TAG( PT_SYSTIME, 0x002A) -+#define PR_RECIPIENT_REASSIGNMENT_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x002B) -+#define PR_REDIRECTION_HISTORY PROP_TAG( PT_BINARY, 0x002C) -+#define PR_RELATED_IPMS PROP_TAG( PT_BINARY, 0x002D) -+#define PR_ORIGINAL_SENSITIVITY PROP_TAG( PT_LONG, 0x002E) -+#define PR_LANGUAGES PROP_TAG( PT_TSTRING, 0x002F) -+#define PR_LANGUAGES_W PROP_TAG( PT_UNICODE, 0x002F) -+#define PR_LANGUAGES_A PROP_TAG( PT_STRING8, 0x002F) -+#define PR_REPLY_TIME PROP_TAG( PT_SYSTIME, 0x0030) -+#define PR_REPORT_TAG PROP_TAG( PT_BINARY, 0x0031) -+#define PR_REPORT_TIME PROP_TAG( PT_SYSTIME, 0x0032) -+#define PR_RETURNED_IPM PROP_TAG( PT_BOOLEAN, 0x0033) -+#define PR_SECURITY PROP_TAG( PT_LONG, 0x0034) -+#define PR_INCOMPLETE_COPY PROP_TAG( PT_BOOLEAN, 0x0035) -+#define PR_SENSITIVITY PROP_TAG( PT_LONG, 0x0036) -+#define PR_SUBJECT PROP_TAG( PT_TSTRING, 0x0037) -+#define PR_SUBJECT_W PROP_TAG( PT_UNICODE, 0x0037) -+#define PR_SUBJECT_A PROP_TAG( PT_STRING8, 0x0037) -+#define PR_SUBJECT_IPM PROP_TAG( PT_BINARY, 0x0038) -+#define PR_CLIENT_SUBMIT_TIME PROP_TAG( PT_SYSTIME, 0x0039) -+#define PR_REPORT_NAME PROP_TAG( PT_TSTRING, 0x003A) -+#define PR_REPORT_NAME_W PROP_TAG( PT_UNICODE, 0x003A) -+#define PR_REPORT_NAME_A PROP_TAG( PT_STRING8, 0x003A) -+#define PR_SENT_REPRESENTING_SEARCH_KEY PROP_TAG( PT_BINARY, 0x003B) -+#define PR_X400_CONTENT_TYPE PROP_TAG( PT_BINARY, 0x003C) -+#define PR_SUBJECT_PREFIX PROP_TAG( PT_TSTRING, 0x003D) -+#define PR_SUBJECT_PREFIX_W PROP_TAG( PT_UNICODE, 0x003D) -+#define PR_SUBJECT_PREFIX_A PROP_TAG( PT_STRING8, 0x003D) -+#define PR_NON_RECEIPT_REASON PROP_TAG( PT_LONG, 0x003E) -+#define PR_RECEIVED_BY_ENTRYID PROP_TAG( PT_BINARY, 0x003F) -+#define PR_RECEIVED_BY_NAME PROP_TAG( PT_TSTRING, 0x0040) -+#define PR_RECEIVED_BY_NAME_W PROP_TAG( PT_UNICODE, 0x0040) -+#define PR_RECEIVED_BY_NAME_A PROP_TAG( PT_STRING8, 0x0040) -+#define PR_SENT_REPRESENTING_ENTRYID PROP_TAG( PT_BINARY, 0x0041) -+#define PR_SENT_REPRESENTING_NAME PROP_TAG( PT_TSTRING, 0x0042) -+#define PR_SENT_REPRESENTING_NAME_W PROP_TAG( PT_UNICODE, 0x0042) -+#define PR_SENT_REPRESENTING_NAME_A PROP_TAG( PT_STRING8, 0x0042) -+#define PR_RCVD_REPRESENTING_ENTRYID PROP_TAG( PT_BINARY, 0x0043) -+#define PR_RCVD_REPRESENTING_NAME PROP_TAG( PT_TSTRING, 0x0044) -+#define PR_RCVD_REPRESENTING_NAME_W PROP_TAG( PT_UNICODE, 0x0044) -+#define PR_RCVD_REPRESENTING_NAME_A PROP_TAG( PT_STRING8, 0x0044) -+#define PR_REPORT_ENTRYID PROP_TAG( PT_BINARY, 0x0045) -+#define PR_READ_RECEIPT_ENTRYID PROP_TAG( PT_BINARY, 0x0046) -+#define PR_MESSAGE_SUBMISSION_ID PROP_TAG( PT_BINARY, 0x0047) -+#define PR_PROVIDER_SUBMIT_TIME PROP_TAG( PT_SYSTIME, 0x0048) -+#define PR_ORIGINAL_SUBJECT PROP_TAG( PT_TSTRING, 0x0049) -+#define PR_ORIGINAL_SUBJECT_W PROP_TAG( PT_UNICODE, 0x0049) -+#define PR_ORIGINAL_SUBJECT_A PROP_TAG( PT_STRING8, 0x0049) -+#define PR_DISC_VAL PROP_TAG( PT_BOOLEAN, 0x004A) -+#define PR_ORIG_MESSAGE_CLASS PROP_TAG( PT_TSTRING, 0x004B) -+#define PR_ORIG_MESSAGE_CLASS_W PROP_TAG( PT_UNICODE, 0x004B) -+#define PR_ORIG_MESSAGE_CLASS_A PROP_TAG( PT_STRING8, 0x004B) -+#define PR_ORIGINAL_AUTHOR_ENTRYID PROP_TAG( PT_BINARY, 0x004C) -+#define PR_ORIGINAL_AUTHOR_NAME PROP_TAG( PT_TSTRING, 0x004D) -+#define PR_ORIGINAL_AUTHOR_NAME_W PROP_TAG( PT_UNICODE, 0x004D) -+#define PR_ORIGINAL_AUTHOR_NAME_A PROP_TAG( PT_STRING8, 0x004D) -+#define PR_ORIGINAL_SUBMIT_TIME PROP_TAG( PT_SYSTIME, 0x004E) -+#define PR_REPLY_RECIPIENT_ENTRIES PROP_TAG( PT_BINARY, 0x004F) -+#define PR_REPLY_RECIPIENT_NAMES PROP_TAG( PT_TSTRING, 0x0050) -+#define PR_REPLY_RECIPIENT_NAMES_W PROP_TAG( PT_UNICODE, 0x0050) -+#define PR_REPLY_RECIPIENT_NAMES_A PROP_TAG( PT_STRING8, 0x0050) -+ -+#define PR_RECEIVED_BY_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0051) -+#define PR_RCVD_REPRESENTING_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0052) -+#define PR_READ_RECEIPT_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0053) -+#define PR_REPORT_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0054) -+#define PR_ORIGINAL_DELIVERY_TIME PROP_TAG( PT_SYSTIME, 0x0055) -+#define PR_ORIGINAL_AUTHOR_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0056) -+ -+#define PR_MESSAGE_TO_ME PROP_TAG( PT_BOOLEAN, 0x0057) -+#define PR_MESSAGE_CC_ME PROP_TAG( PT_BOOLEAN, 0x0058) -+#define PR_MESSAGE_RECIP_ME PROP_TAG( PT_BOOLEAN, 0x0059) -+ -+#define PR_ORIGINAL_SENDER_NAME PROP_TAG( PT_TSTRING, 0x005A) -+#define PR_ORIGINAL_SENDER_NAME_W PROP_TAG( PT_UNICODE, 0x005A) -+#define PR_ORIGINAL_SENDER_NAME_A PROP_TAG( PT_STRING8, 0x005A) -+#define PR_ORIGINAL_SENDER_ENTRYID PROP_TAG( PT_BINARY, 0x005B) -+#define PR_ORIGINAL_SENDER_SEARCH_KEY PROP_TAG( PT_BINARY, 0x005C) -+#define PR_ORIGINAL_SENT_REPRESENTING_NAME PROP_TAG( PT_TSTRING, 0x005D) -+#define PR_ORIGINAL_SENT_REPRESENTING_NAME_W PROP_TAG( PT_UNICODE, 0x005D) -+#define PR_ORIGINAL_SENT_REPRESENTING_NAME_A PROP_TAG( PT_STRING8, 0x005D) -+#define PR_ORIGINAL_SENT_REPRESENTING_ENTRYID PROP_TAG( PT_BINARY, 0x005E) -+#define PR_ORIGINAL_SENT_REPRESENTING_SEARCH_KEY PROP_TAG( PT_BINARY, 0x005F) -+ -+#define PR_START_DATE PROP_TAG( PT_SYSTIME, 0x0060) -+#define PR_END_DATE PROP_TAG( PT_SYSTIME, 0x0061) -+#define PR_OWNER_APPT_ID PROP_TAG( PT_LONG, 0x0062) -+#define PR_RESPONSE_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0063) -+ -+#define PR_SENT_REPRESENTING_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0064) -+#define PR_SENT_REPRESENTING_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0064) -+#define PR_SENT_REPRESENTING_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0064) -+#define PR_SENT_REPRESENTING_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0065) -+#define PR_SENT_REPRESENTING_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0065) -+#define PR_SENT_REPRESENTING_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0065) -+ -+#define PR_ORIGINAL_SENDER_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0066) -+#define PR_ORIGINAL_SENDER_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0066) -+#define PR_ORIGINAL_SENDER_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0066) -+#define PR_ORIGINAL_SENDER_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0067) -+#define PR_ORIGINAL_SENDER_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0067) -+#define PR_ORIGINAL_SENDER_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0067) -+ -+#define PR_ORIGINAL_SENT_REPRESENTING_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0068) -+#define PR_ORIGINAL_SENT_REPRESENTING_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0068) -+#define PR_ORIGINAL_SENT_REPRESENTING_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0068) -+#define PR_ORIGINAL_SENT_REPRESENTING_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0069) -+#define PR_ORIGINAL_SENT_REPRESENTING_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0069) -+#define PR_ORIGINAL_SENT_REPRESENTING_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0069) -+ -+#define PR_CONVERSATION_TOPIC PROP_TAG( PT_TSTRING, 0x0070) -+#define PR_CONVERSATION_TOPIC_W PROP_TAG( PT_UNICODE, 0x0070) -+#define PR_CONVERSATION_TOPIC_A PROP_TAG( PT_STRING8, 0x0070) -+#define PR_CONVERSATION_INDEX PROP_TAG( PT_BINARY, 0x0071) -+ -+#define PR_ORIGINAL_DISPLAY_BCC PROP_TAG( PT_TSTRING, 0x0072) -+#define PR_ORIGINAL_DISPLAY_BCC_W PROP_TAG( PT_UNICODE, 0x0072) -+#define PR_ORIGINAL_DISPLAY_BCC_A PROP_TAG( PT_STRING8, 0x0072) -+#define PR_ORIGINAL_DISPLAY_CC PROP_TAG( PT_TSTRING, 0x0073) -+#define PR_ORIGINAL_DISPLAY_CC_W PROP_TAG( PT_UNICODE, 0x0073) -+#define PR_ORIGINAL_DISPLAY_CC_A PROP_TAG( PT_STRING8, 0x0073) -+#define PR_ORIGINAL_DISPLAY_TO PROP_TAG( PT_TSTRING, 0x0074) -+#define PR_ORIGINAL_DISPLAY_TO_W PROP_TAG( PT_UNICODE, 0x0074) -+#define PR_ORIGINAL_DISPLAY_TO_A PROP_TAG( PT_STRING8, 0x0074) -+ -+#define PR_RECEIVED_BY_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0075) -+#define PR_RECEIVED_BY_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0075) -+#define PR_RECEIVED_BY_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0075) -+#define PR_RECEIVED_BY_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0076) -+#define PR_RECEIVED_BY_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0076) -+#define PR_RECEIVED_BY_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0076) -+ -+#define PR_RCVD_REPRESENTING_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0077) -+#define PR_RCVD_REPRESENTING_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0077) -+#define PR_RCVD_REPRESENTING_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0077) -+#define PR_RCVD_REPRESENTING_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0078) -+#define PR_RCVD_REPRESENTING_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0078) -+#define PR_RCVD_REPRESENTING_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0078) -+ -+#define PR_ORIGINAL_AUTHOR_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0079) -+#define PR_ORIGINAL_AUTHOR_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0079) -+#define PR_ORIGINAL_AUTHOR_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0079) -+#define PR_ORIGINAL_AUTHOR_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x007A) -+#define PR_ORIGINAL_AUTHOR_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x007A) -+#define PR_ORIGINAL_AUTHOR_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x007A) -+ -+#define PR_ORIGINALLY_INTENDED_RECIP_ADDRTYPE PROP_TAG( PT_TSTRING, 0x007B) -+#define PR_ORIGINALLY_INTENDED_RECIP_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x007B) -+#define PR_ORIGINALLY_INTENDED_RECIP_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x007B) -+#define PR_ORIGINALLY_INTENDED_RECIP_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x007C) -+#define PR_ORIGINALLY_INTENDED_RECIP_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x007C) -+#define PR_ORIGINALLY_INTENDED_RECIP_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x007C) -+ -+#define PR_TRANSPORT_MESSAGE_HEADERS PROP_TAG(PT_TSTRING, 0x007D) -+#define PR_TRANSPORT_MESSAGE_HEADERS_W PROP_TAG(PT_UNICODE, 0x007D) -+#define PR_TRANSPORT_MESSAGE_HEADERS_A PROP_TAG(PT_STRING8, 0x007D) -+ -+#define PR_DELEGATION PROP_TAG(PT_BINARY, 0x007E) -+ -+#define PR_TNEF_CORRELATION_KEY PROP_TAG(PT_BINARY, 0x007F) -+ -+ -+ -+/* -+ * Message content properties -+ */ -+ -+#define PR_BODY PROP_TAG( PT_TSTRING, 0x1000) -+#define PR_BODY_W PROP_TAG( PT_UNICODE, 0x1000) -+#define PR_BODY_A PROP_TAG( PT_STRING8, 0x1000) -+#define PR_REPORT_TEXT PROP_TAG( PT_TSTRING, 0x1001) -+#define PR_REPORT_TEXT_W PROP_TAG( PT_UNICODE, 0x1001) -+#define PR_REPORT_TEXT_A PROP_TAG( PT_STRING8, 0x1001) -+#define PR_ORIGINATOR_AND_DL_EXPANSION_HISTORY PROP_TAG( PT_BINARY, 0x1002) -+#define PR_REPORTING_DL_NAME PROP_TAG( PT_BINARY, 0x1003) -+#define PR_REPORTING_MTA_CERTIFICATE PROP_TAG( PT_BINARY, 0x1004) -+ -+/* Removed PR_REPORT_ORIGIN_AUTHENTICATION_CHECK with DCR 3865, use PR_ORIGIN_CHECK */ -+ -+#define PR_RTF_SYNC_BODY_CRC PROP_TAG( PT_LONG, 0x1006) -+#define PR_RTF_SYNC_BODY_COUNT PROP_TAG( PT_LONG, 0x1007) -+#define PR_RTF_SYNC_BODY_TAG PROP_TAG( PT_TSTRING, 0x1008) -+#define PR_RTF_SYNC_BODY_TAG_W PROP_TAG( PT_UNICODE, 0x1008) -+#define PR_RTF_SYNC_BODY_TAG_A PROP_TAG( PT_STRING8, 0x1008) -+#define PR_RTF_COMPRESSED PROP_TAG( PT_BINARY, 0x1009) -+#define PR_RTF_SYNC_PREFIX_COUNT PROP_TAG( PT_LONG, 0x1010) -+#define PR_RTF_SYNC_TRAILING_COUNT PROP_TAG( PT_LONG, 0x1011) -+#define PR_ORIGINALLY_INTENDED_RECIP_ENTRYID PROP_TAG( PT_BINARY, 0x1012) -+ -+/* -+ * Reserved 0x1100-0x1200 -+ */ -+ -+ -+/* -+ * Message recipient properties -+ */ -+ -+#define PR_CONTENT_INTEGRITY_CHECK PROP_TAG( PT_BINARY, 0x0C00) -+#define PR_EXPLICIT_CONVERSION PROP_TAG( PT_LONG, 0x0C01) -+#define PR_IPM_RETURN_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C02) -+#define PR_MESSAGE_TOKEN PROP_TAG( PT_BINARY, 0x0C03) -+#define PR_NDR_REASON_CODE PROP_TAG( PT_LONG, 0x0C04) -+#define PR_NDR_DIAG_CODE PROP_TAG( PT_LONG, 0x0C05) -+#define PR_NON_RECEIPT_NOTIFICATION_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C06) -+#define PR_DELIVERY_POINT PROP_TAG( PT_LONG, 0x0C07) -+ -+#define PR_ORIGINATOR_NON_DELIVERY_REPORT_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C08) -+#define PR_ORIGINATOR_REQUESTED_ALTERNATE_RECIPIENT PROP_TAG( PT_BINARY, 0x0C09) -+#define PR_PHYSICAL_DELIVERY_BUREAU_FAX_DELIVERY PROP_TAG( PT_BOOLEAN, 0x0C0A) -+#define PR_PHYSICAL_DELIVERY_MODE PROP_TAG( PT_LONG, 0x0C0B) -+#define PR_PHYSICAL_DELIVERY_REPORT_REQUEST PROP_TAG( PT_LONG, 0x0C0C) -+#define PR_PHYSICAL_FORWARDING_ADDRESS PROP_TAG( PT_BINARY, 0x0C0D) -+#define PR_PHYSICAL_FORWARDING_ADDRESS_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C0E) -+#define PR_PHYSICAL_FORWARDING_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x0C0F) -+#define PR_PHYSICAL_RENDITION_ATTRIBUTES PROP_TAG( PT_BINARY, 0x0C10) -+#define PR_PROOF_OF_DELIVERY PROP_TAG( PT_BINARY, 0x0C11) -+#define PR_PROOF_OF_DELIVERY_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C12) -+#define PR_RECIPIENT_CERTIFICATE PROP_TAG( PT_BINARY, 0x0C13) -+#define PR_RECIPIENT_NUMBER_FOR_ADVICE PROP_TAG( PT_TSTRING, 0x0C14) -+#define PR_RECIPIENT_NUMBER_FOR_ADVICE_W PROP_TAG( PT_UNICODE, 0x0C14) -+#define PR_RECIPIENT_NUMBER_FOR_ADVICE_A PROP_TAG( PT_STRING8, 0x0C14) -+#define PR_RECIPIENT_TYPE PROP_TAG( PT_LONG, 0x0C15) -+#define PR_REGISTERED_MAIL_TYPE PROP_TAG( PT_LONG, 0x0C16) -+#define PR_REPLY_REQUESTED PROP_TAG( PT_BOOLEAN, 0x0C17) -+#define PR_REQUESTED_DELIVERY_METHOD PROP_TAG( PT_LONG, 0x0C18) -+#define PR_SENDER_ENTRYID PROP_TAG( PT_BINARY, 0x0C19) -+#define PR_SENDER_NAME PROP_TAG( PT_TSTRING, 0x0C1A) -+#define PR_SENDER_NAME_W PROP_TAG( PT_UNICODE, 0x0C1A) -+#define PR_SENDER_NAME_A PROP_TAG( PT_STRING8, 0x0C1A) -+#define PR_SUPPLEMENTARY_INFO PROP_TAG( PT_TSTRING, 0x0C1B) -+#define PR_SUPPLEMENTARY_INFO_W PROP_TAG( PT_UNICODE, 0x0C1B) -+#define PR_SUPPLEMENTARY_INFO_A PROP_TAG( PT_STRING8, 0x0C1B) -+#define PR_TYPE_OF_MTS_USER PROP_TAG( PT_LONG, 0x0C1C) -+#define PR_SENDER_SEARCH_KEY PROP_TAG( PT_BINARY, 0x0C1D) -+#define PR_SENDER_ADDRTYPE PROP_TAG( PT_TSTRING, 0x0C1E) -+#define PR_SENDER_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x0C1E) -+#define PR_SENDER_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x0C1E) -+#define PR_SENDER_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x0C1F) -+#define PR_SENDER_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x0C1F) -+#define PR_SENDER_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x0C1F) -+ -+/* -+ * Message non-transmittable properties -+ */ -+ -+/* -+ * The two tags, PR_MESSAGE_RECIPIENTS and PR_MESSAGE_ATTACHMENTS, -+ * are to be used in the exclude list passed to -+ * IMessage::CopyTo when the caller wants either the recipients or attachments -+ * of the message to not get copied. It is also used in the ProblemArray -+ * return from IMessage::CopyTo when an error is encountered copying them -+ */ -+ -+#define PR_CURRENT_VERSION PROP_TAG( PT_I8, 0x0E00) -+#define PR_DELETE_AFTER_SUBMIT PROP_TAG( PT_BOOLEAN, 0x0E01) -+#define PR_DISPLAY_BCC PROP_TAG( PT_TSTRING, 0x0E02) -+#define PR_DISPLAY_BCC_W PROP_TAG( PT_UNICODE, 0x0E02) -+#define PR_DISPLAY_BCC_A PROP_TAG( PT_STRING8, 0x0E02) -+#define PR_DISPLAY_CC PROP_TAG( PT_TSTRING, 0x0E03) -+#define PR_DISPLAY_CC_W PROP_TAG( PT_UNICODE, 0x0E03) -+#define PR_DISPLAY_CC_A PROP_TAG( PT_STRING8, 0x0E03) -+#define PR_DISPLAY_TO PROP_TAG( PT_TSTRING, 0x0E04) -+#define PR_DISPLAY_TO_W PROP_TAG( PT_UNICODE, 0x0E04) -+#define PR_DISPLAY_TO_A PROP_TAG( PT_STRING8, 0x0E04) -+#define PR_PARENT_DISPLAY PROP_TAG( PT_TSTRING, 0x0E05) -+#define PR_PARENT_DISPLAY_W PROP_TAG( PT_UNICODE, 0x0E05) -+#define PR_PARENT_DISPLAY_A PROP_TAG( PT_STRING8, 0x0E05) -+#define PR_MESSAGE_DELIVERY_TIME PROP_TAG( PT_SYSTIME, 0x0E06) -+#define PR_MESSAGE_FLAGS PROP_TAG( PT_LONG, 0x0E07) -+#define PR_MESSAGE_SIZE PROP_TAG( PT_LONG, 0x0E08) -+#define PR_PARENT_ENTRYID PROP_TAG( PT_BINARY, 0x0E09) -+#define PR_SENTMAIL_ENTRYID PROP_TAG( PT_BINARY, 0x0E0A) -+#define PR_CORRELATE PROP_TAG( PT_BOOLEAN, 0x0E0C) -+#define PR_CORRELATE_MTSID PROP_TAG( PT_BINARY, 0x0E0D) -+#define PR_DISCRETE_VALUES PROP_TAG( PT_BOOLEAN, 0x0E0E) -+#define PR_RESPONSIBILITY PROP_TAG( PT_BOOLEAN, 0x0E0F) -+#define PR_SPOOLER_STATUS PROP_TAG( PT_LONG, 0x0E10) -+#define PR_TRANSPORT_STATUS PROP_TAG( PT_LONG, 0x0E11) -+#define PR_MESSAGE_RECIPIENTS PROP_TAG( PT_OBJECT, 0x0E12) -+#define PR_MESSAGE_ATTACHMENTS PROP_TAG( PT_OBJECT, 0x0E13) -+#define PR_SUBMIT_FLAGS PROP_TAG( PT_LONG, 0x0E14) -+#define PR_RECIPIENT_STATUS PROP_TAG( PT_LONG, 0x0E15) -+#define PR_TRANSPORT_KEY PROP_TAG( PT_LONG, 0x0E16) -+#define PR_MSG_STATUS PROP_TAG( PT_LONG, 0x0E17) -+#define PR_MESSAGE_DOWNLOAD_TIME PROP_TAG( PT_LONG, 0x0E18) -+#define PR_CREATION_VERSION PROP_TAG( PT_I8, 0x0E19) -+#define PR_MODIFY_VERSION PROP_TAG( PT_I8, 0x0E1A) -+#define PR_HASATTACH PROP_TAG( PT_BOOLEAN, 0x0E1B) -+#define PR_BODY_CRC PROP_TAG( PT_LONG, 0x0E1C) -+#define PR_NORMALIZED_SUBJECT PROP_TAG( PT_TSTRING, 0x0E1D) -+#define PR_NORMALIZED_SUBJECT_W PROP_TAG( PT_UNICODE, 0x0E1D) -+#define PR_NORMALIZED_SUBJECT_A PROP_TAG( PT_STRING8, 0x0E1D) -+#define PR_RTF_IN_SYNC PROP_TAG( PT_BOOLEAN, 0x0E1F) -+#define PR_ATTACH_SIZE PROP_TAG( PT_LONG, 0x0E20) -+#define PR_ATTACH_NUM PROP_TAG( PT_LONG, 0x0E21) -+#define PR_PREPROCESS PROP_TAG( PT_BOOLEAN, 0x0E22) -+ -+/* PR_ORIGINAL_DISPLAY_TO, _CC, and _BCC moved to transmittible range 03/09/95 */ -+ -+#define PR_ORIGINATING_MTA_CERTIFICATE PROP_TAG( PT_BINARY, 0x0E25) -+#define PR_PROOF_OF_SUBMISSION PROP_TAG( PT_BINARY, 0x0E26) -+ -+ -+/* -+ * The range of non-message and non-recipient property IDs (0x3000 - 0x3FFF) is -+ * further broken down into ranges to make assigning new property IDs easier. -+ * -+ * From To Kind of property -+ * -------------------------------- -+ * 3000 32FF MAPI_defined common property -+ * 3200 33FF MAPI_defined form property -+ * 3400 35FF MAPI_defined message store property -+ * 3600 36FF MAPI_defined Folder or AB Container property -+ * 3700 38FF MAPI_defined attachment property -+ * 3900 39FF MAPI_defined address book property -+ * 3A00 3BFF MAPI_defined mailuser property -+ * 3C00 3CFF MAPI_defined DistList property -+ * 3D00 3DFF MAPI_defined Profile Section property -+ * 3E00 3EFF MAPI_defined Status property -+ * 3F00 3FFF MAPI_defined display table property -+ */ -+ -+/* -+ * Properties common to numerous MAPI objects. -+ * -+ * Those properties that can appear on messages are in the -+ * non-transmittable range for messages. They start at the high -+ * end of that range and work down. -+ * -+ * Properties that never appear on messages are defined in the common -+ * property range (see above). -+ */ -+ -+/* -+ * properties that are common to multiple objects (including message objects) -+ * -- these ids are in the non-transmittable range -+ */ -+ -+#define PR_ENTRYID PROP_TAG( PT_BINARY, 0x0FFF) -+#define PR_OBJECT_TYPE PROP_TAG( PT_LONG, 0x0FFE) -+#define PR_ICON PROP_TAG( PT_BINARY, 0x0FFD) -+#define PR_MINI_ICON PROP_TAG( PT_BINARY, 0x0FFC) -+#define PR_STORE_ENTRYID PROP_TAG( PT_BINARY, 0x0FFB) -+#define PR_STORE_RECORD_KEY PROP_TAG( PT_BINARY, 0x0FFA) -+#define PR_RECORD_KEY PROP_TAG( PT_BINARY, 0x0FF9) -+#define PR_MAPPING_SIGNATURE PROP_TAG( PT_BINARY, 0x0FF8) -+#define PR_ACCESS_LEVEL PROP_TAG( PT_LONG, 0x0FF7) -+#define PR_INSTANCE_KEY PROP_TAG( PT_BINARY, 0x0FF6) -+#define PR_ROW_TYPE PROP_TAG( PT_LONG, 0x0FF5) -+#define PR_ACCESS PROP_TAG( PT_LONG, 0x0FF4) -+ -+/* -+ * properties that are common to multiple objects (usually not including message objects) -+ * -- these ids are in the transmittable range -+ */ -+ -+#define PR_ROWID PROP_TAG( PT_LONG, 0x3000) -+#define PR_DISPLAY_NAME PROP_TAG( PT_TSTRING, 0x3001) -+#define PR_DISPLAY_NAME_W PROP_TAG( PT_UNICODE, 0x3001) -+#define PR_DISPLAY_NAME_A PROP_TAG( PT_STRING8, 0x3001) -+#define PR_ADDRTYPE PROP_TAG( PT_TSTRING, 0x3002) -+#define PR_ADDRTYPE_W PROP_TAG( PT_UNICODE, 0x3002) -+#define PR_ADDRTYPE_A PROP_TAG( PT_STRING8, 0x3002) -+#define PR_EMAIL_ADDRESS PROP_TAG( PT_TSTRING, 0x3003) -+#define PR_EMAIL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x3003) -+#define PR_EMAIL_ADDRESS_A PROP_TAG( PT_STRING8, 0x3003) -+#define PR_COMMENT PROP_TAG( PT_TSTRING, 0x3004) -+#define PR_COMMENT_W PROP_TAG( PT_UNICODE, 0x3004) -+#define PR_COMMENT_A PROP_TAG( PT_STRING8, 0x3004) -+#define PR_DEPTH PROP_TAG( PT_LONG, 0x3005) -+#define PR_PROVIDER_DISPLAY PROP_TAG( PT_TSTRING, 0x3006) -+#define PR_PROVIDER_DISPLAY_W PROP_TAG( PT_UNICODE, 0x3006) -+#define PR_PROVIDER_DISPLAY_A PROP_TAG( PT_STRING8, 0x3006) -+#define PR_CREATION_TIME PROP_TAG( PT_SYSTIME, 0x3007) -+#define PR_LAST_MODIFICATION_TIME PROP_TAG( PT_SYSTIME, 0x3008) -+#define PR_RESOURCE_FLAGS PROP_TAG( PT_LONG, 0x3009) -+#define PR_PROVIDER_DLL_NAME PROP_TAG( PT_TSTRING, 0x300A) -+#define PR_PROVIDER_DLL_NAME_W PROP_TAG( PT_UNICODE, 0x300A) -+#define PR_PROVIDER_DLL_NAME_A PROP_TAG( PT_STRING8, 0x300A) -+#define PR_SEARCH_KEY PROP_TAG( PT_BINARY, 0x300B) -+#define PR_PROVIDER_UID PROP_TAG( PT_BINARY, 0x300C) -+#define PR_PROVIDER_ORDINAL PROP_TAG( PT_LONG, 0x300D) -+ -+/* -+ * MAPI Form properties -+ */ -+#define PR_FORM_VERSION PROP_TAG(PT_TSTRING, 0x3301) -+#define PR_FORM_VERSION_W PROP_TAG(PT_UNICODE, 0x3301) -+#define PR_FORM_VERSION_A PROP_TAG(PT_STRING8, 0x3301) -+#define PR_FORM_CLSID PROP_TAG(PT_CLSID, 0x3302) -+#define PR_FORM_CONTACT_NAME PROP_TAG(PT_TSTRING, 0x3303) -+#define PR_FORM_CONTACT_NAME_W PROP_TAG(PT_UNICODE, 0x3303) -+#define PR_FORM_CONTACT_NAME_A PROP_TAG(PT_STRING8, 0x3303) -+#define PR_FORM_CATEGORY PROP_TAG(PT_TSTRING, 0x3304) -+#define PR_FORM_CATEGORY_W PROP_TAG(PT_UNICODE, 0x3304) -+#define PR_FORM_CATEGORY_A PROP_TAG(PT_STRING8, 0x3304) -+#define PR_FORM_CATEGORY_SUB PROP_TAG(PT_TSTRING, 0x3305) -+#define PR_FORM_CATEGORY_SUB_W PROP_TAG(PT_UNICODE, 0x3305) -+#define PR_FORM_CATEGORY_SUB_A PROP_TAG(PT_STRING8, 0x3305) -+#define PR_FORM_HOST_MAP PROP_TAG(PT_MV_LONG, 0x3306) -+#define PR_FORM_HIDDEN PROP_TAG(PT_BOOLEAN, 0x3307) -+#define PR_FORM_DESIGNER_NAME PROP_TAG(PT_TSTRING, 0x3308) -+#define PR_FORM_DESIGNER_NAME_W PROP_TAG(PT_UNICODE, 0x3308) -+#define PR_FORM_DESIGNER_NAME_A PROP_TAG(PT_STRING8, 0x3308) -+#define PR_FORM_DESIGNER_GUID PROP_TAG(PT_CLSID, 0x3309) -+#define PR_FORM_MESSAGE_BEHAVIOR PROP_TAG(PT_LONG, 0x330A) -+ -+/* -+ * Message store properties -+ */ -+ -+#define PR_DEFAULT_STORE PROP_TAG( PT_BOOLEAN, 0x3400) -+#define PR_STORE_SUPPORT_MASK PROP_TAG( PT_LONG, 0x340D) -+#define PR_STORE_STATE PROP_TAG( PT_LONG, 0x340E) -+ -+#define PR_IPM_SUBTREE_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3410) -+#define PR_IPM_OUTBOX_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3411) -+#define PR_IPM_WASTEBASKET_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3412) -+#define PR_IPM_SENTMAIL_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3413) -+#define PR_MDB_PROVIDER PROP_TAG( PT_BINARY, 0x3414) -+#define PR_RECEIVE_FOLDER_SETTINGS PROP_TAG( PT_OBJECT, 0x3415) -+ -+#define PR_VALID_FOLDER_MASK PROP_TAG( PT_LONG, 0x35DF) -+#define PR_IPM_SUBTREE_ENTRYID PROP_TAG( PT_BINARY, 0x35E0) -+ -+#define PR_IPM_OUTBOX_ENTRYID PROP_TAG( PT_BINARY, 0x35E2) -+#define PR_IPM_WASTEBASKET_ENTRYID PROP_TAG( PT_BINARY, 0x35E3) -+#define PR_IPM_SENTMAIL_ENTRYID PROP_TAG( PT_BINARY, 0x35E4) -+#define PR_VIEWS_ENTRYID PROP_TAG( PT_BINARY, 0x35E5) -+#define PR_COMMON_VIEWS_ENTRYID PROP_TAG( PT_BINARY, 0x35E6) -+#define PR_FINDER_ENTRYID PROP_TAG( PT_BINARY, 0x35E7) -+ -+/* Proptags 0x35E8-0x35FF reserved for folders "guaranteed" by PR_VALID_FOLDER_MASK */ -+ -+ -+/* -+ * Folder and AB Container properties -+ */ -+ -+#define PR_CONTAINER_FLAGS PROP_TAG( PT_LONG, 0x3600) -+#define PR_FOLDER_TYPE PROP_TAG( PT_LONG, 0x3601) -+#define PR_CONTENT_COUNT PROP_TAG( PT_LONG, 0x3602) -+#define PR_CONTENT_UNREAD PROP_TAG( PT_LONG, 0x3603) -+#define PR_CREATE_TEMPLATES PROP_TAG( PT_OBJECT, 0x3604) -+#define PR_DETAILS_TABLE PROP_TAG( PT_OBJECT, 0x3605) -+#define PR_SEARCH PROP_TAG( PT_OBJECT, 0x3607) -+#define PR_SELECTABLE PROP_TAG( PT_BOOLEAN, 0x3609) -+#define PR_SUBFOLDERS PROP_TAG( PT_BOOLEAN, 0x360A) -+#define PR_STATUS PROP_TAG( PT_LONG, 0x360B) -+#define PR_ANR PROP_TAG( PT_TSTRING, 0x360C) -+#define PR_ANR_W PROP_TAG( PT_UNICODE, 0x360C) -+#define PR_ANR_A PROP_TAG( PT_STRING8, 0x360C) -+#define PR_CONTENTS_SORT_ORDER PROP_TAG( PT_MV_LONG, 0x360D) -+#define PR_CONTAINER_HIERARCHY PROP_TAG( PT_OBJECT, 0x360E) -+#define PR_CONTAINER_CONTENTS PROP_TAG( PT_OBJECT, 0x360F) -+#define PR_FOLDER_ASSOCIATED_CONTENTS PROP_TAG( PT_OBJECT, 0x3610) -+#define PR_DEF_CREATE_DL PROP_TAG( PT_BINARY, 0x3611) -+#define PR_DEF_CREATE_MAILUSER PROP_TAG( PT_BINARY, 0x3612) -+#define PR_CONTAINER_CLASS PROP_TAG( PT_TSTRING, 0x3613) -+#define PR_CONTAINER_CLASS_W PROP_TAG( PT_UNICODE, 0x3613) -+#define PR_CONTAINER_CLASS_A PROP_TAG( PT_STRING8, 0x3613) -+#define PR_CONTAINER_MODIFY_VERSION PROP_TAG( PT_I8, 0x3614) -+#define PR_AB_PROVIDER_ID PROP_TAG( PT_BINARY, 0x3615) -+#define PR_DEFAULT_VIEW_ENTRYID PROP_TAG( PT_BINARY, 0x3616) -+#define PR_ASSOC_CONTENT_COUNT PROP_TAG( PT_LONG, 0x3617) -+ -+/* Reserved 0x36C0-0x36FF */ -+ -+/* -+ * Attachment properties -+ */ -+ -+#define PR_ATTACHMENT_X400_PARAMETERS PROP_TAG( PT_BINARY, 0x3700) -+#define PR_ATTACH_DATA_OBJ PROP_TAG( PT_OBJECT, 0x3701) -+#define PR_ATTACH_DATA_BIN PROP_TAG( PT_BINARY, 0x3701) -+#define PR_ATTACH_ENCODING PROP_TAG( PT_BINARY, 0x3702) -+#define PR_ATTACH_EXTENSION PROP_TAG( PT_TSTRING, 0x3703) -+#define PR_ATTACH_EXTENSION_W PROP_TAG( PT_UNICODE, 0x3703) -+#define PR_ATTACH_EXTENSION_A PROP_TAG( PT_STRING8, 0x3703) -+#define PR_ATTACH_FILENAME PROP_TAG( PT_TSTRING, 0x3704) -+#define PR_ATTACH_FILENAME_W PROP_TAG( PT_UNICODE, 0x3704) -+#define PR_ATTACH_FILENAME_A PROP_TAG( PT_STRING8, 0x3704) -+#define PR_ATTACH_METHOD PROP_TAG( PT_LONG, 0x3705) -+#define PR_ATTACH_LONG_FILENAME PROP_TAG( PT_TSTRING, 0x3707) -+#define PR_ATTACH_LONG_FILENAME_W PROP_TAG( PT_UNICODE, 0x3707) -+#define PR_ATTACH_LONG_FILENAME_A PROP_TAG( PT_STRING8, 0x3707) -+#define PR_ATTACH_PATHNAME PROP_TAG( PT_TSTRING, 0x3708) -+#define PR_ATTACH_PATHNAME_W PROP_TAG( PT_UNICODE, 0x3708) -+#define PR_ATTACH_PATHNAME_A PROP_TAG( PT_STRING8, 0x3708) -+#define PR_ATTACH_RENDERING PROP_TAG( PT_BINARY, 0x3709) -+#define PR_ATTACH_TAG PROP_TAG( PT_BINARY, 0x370A) -+#define PR_RENDERING_POSITION PROP_TAG( PT_LONG, 0x370B) -+#define PR_ATTACH_TRANSPORT_NAME PROP_TAG( PT_TSTRING, 0x370C) -+#define PR_ATTACH_TRANSPORT_NAME_W PROP_TAG( PT_UNICODE, 0x370C) -+#define PR_ATTACH_TRANSPORT_NAME_A PROP_TAG( PT_STRING8, 0x370C) -+#define PR_ATTACH_LONG_PATHNAME PROP_TAG( PT_TSTRING, 0x370D) -+#define PR_ATTACH_LONG_PATHNAME_W PROP_TAG( PT_UNICODE, 0x370D) -+#define PR_ATTACH_LONG_PATHNAME_A PROP_TAG( PT_STRING8, 0x370D) -+#define PR_ATTACH_MIME_TAG PROP_TAG( PT_TSTRING, 0x370E) -+#define PR_ATTACH_MIME_TAG_W PROP_TAG( PT_UNICODE, 0x370E) -+#define PR_ATTACH_MIME_TAG_A PROP_TAG( PT_STRING8, 0x370E) -+#define PR_ATTACH_ADDITIONAL_INFO PROP_TAG( PT_BINARY, 0x370F) -+ -+/* -+ * AB Object properties -+ */ -+ -+#define PR_DISPLAY_TYPE PROP_TAG( PT_LONG, 0x3900) -+#define PR_TEMPLATEID PROP_TAG( PT_BINARY, 0x3902) -+#define PR_PRIMARY_CAPABILITY PROP_TAG( PT_BINARY, 0x3904) -+ -+ -+/* -+ * Mail user properties -+ */ -+#define PR_7BIT_DISPLAY_NAME PROP_TAG( PT_STRING8, 0x39FF) -+#define PR_ACCOUNT PROP_TAG( PT_TSTRING, 0x3A00) -+#define PR_ACCOUNT_W PROP_TAG( PT_UNICODE, 0x3A00) -+#define PR_ACCOUNT_A PROP_TAG( PT_STRING8, 0x3A00) -+#define PR_ALTERNATE_RECIPIENT PROP_TAG( PT_BINARY, 0x3A01) -+#define PR_CALLBACK_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A02) -+#define PR_CALLBACK_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A02) -+#define PR_CALLBACK_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A02) -+#define PR_CONVERSION_PROHIBITED PROP_TAG( PT_BOOLEAN, 0x3A03) -+#define PR_DISCLOSE_RECIPIENTS PROP_TAG( PT_BOOLEAN, 0x3A04) -+#define PR_GENERATION PROP_TAG( PT_TSTRING, 0x3A05) -+#define PR_GENERATION_W PROP_TAG( PT_UNICODE, 0x3A05) -+#define PR_GENERATION_A PROP_TAG( PT_STRING8, 0x3A05) -+#define PR_GIVEN_NAME PROP_TAG( PT_TSTRING, 0x3A06) -+#define PR_GIVEN_NAME_W PROP_TAG( PT_UNICODE, 0x3A06) -+#define PR_GIVEN_NAME_A PROP_TAG( PT_STRING8, 0x3A06) -+#define PR_GOVERNMENT_ID_NUMBER PROP_TAG( PT_TSTRING, 0x3A07) -+#define PR_GOVERNMENT_ID_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A07) -+#define PR_GOVERNMENT_ID_NUMBER_A PROP_TAG( PT_STRING8, 0x3A07) -+#define PR_BUSINESS_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A08) -+#define PR_BUSINESS_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A08) -+#define PR_BUSINESS_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A08) -+#define PR_OFFICE_TELEPHONE_NUMBER PR_BUSINESS_TELEPHONE_NUMBER -+#define PR_OFFICE_TELEPHONE_NUMBER_W PR_BUSINESS_TELEPHONE_NUMBER_W -+#define PR_OFFICE_TELEPHONE_NUMBER_A PR_BUSINESS_TELEPHONE_NUMBER_A -+#define PR_HOME_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A09) -+#define PR_HOME_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A09) -+#define PR_HOME_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A09) -+#define PR_INITIALS PROP_TAG( PT_TSTRING, 0x3A0A) -+#define PR_INITIALS_W PROP_TAG( PT_UNICODE, 0x3A0A) -+#define PR_INITIALS_A PROP_TAG( PT_STRING8, 0x3A0A) -+#define PR_KEYWORD PROP_TAG( PT_TSTRING, 0x3A0B) -+#define PR_KEYWORD_W PROP_TAG( PT_UNICODE, 0x3A0B) -+#define PR_KEYWORD_A PROP_TAG( PT_STRING8, 0x3A0B) -+#define PR_LANGUAGE PROP_TAG( PT_TSTRING, 0x3A0C) -+#define PR_LANGUAGE_W PROP_TAG( PT_UNICODE, 0x3A0C) -+#define PR_LANGUAGE_A PROP_TAG( PT_STRING8, 0x3A0C) -+#define PR_LOCATION PROP_TAG( PT_TSTRING, 0x3A0D) -+#define PR_LOCATION_W PROP_TAG( PT_UNICODE, 0x3A0D) -+#define PR_LOCATION_A PROP_TAG( PT_STRING8, 0x3A0D) -+#define PR_MAIL_PERMISSION PROP_TAG( PT_BOOLEAN, 0x3A0E) -+#define PR_MHS_COMMON_NAME PROP_TAG( PT_TSTRING, 0x3A0F) -+#define PR_MHS_COMMON_NAME_W PROP_TAG( PT_UNICODE, 0x3A0F) -+#define PR_MHS_COMMON_NAME_A PROP_TAG( PT_STRING8, 0x3A0F) -+#define PR_ORGANIZATIONAL_ID_NUMBER PROP_TAG( PT_TSTRING, 0x3A10) -+#define PR_ORGANIZATIONAL_ID_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A10) -+#define PR_ORGANIZATIONAL_ID_NUMBER_A PROP_TAG( PT_STRING8, 0x3A10) -+#define PR_SURNAME PROP_TAG( PT_TSTRING, 0x3A11) -+#define PR_SURNAME_W PROP_TAG( PT_UNICODE, 0x3A11) -+#define PR_SURNAME_A PROP_TAG( PT_STRING8, 0x3A11) -+#define PR_ORIGINAL_ENTRYID PROP_TAG( PT_BINARY, 0x3A12) -+#define PR_ORIGINAL_DISPLAY_NAME PROP_TAG( PT_TSTRING, 0x3A13) -+#define PR_ORIGINAL_DISPLAY_NAME_W PROP_TAG( PT_UNICODE, 0x3A13) -+#define PR_ORIGINAL_DISPLAY_NAME_A PROP_TAG( PT_STRING8, 0x3A13) -+#define PR_ORIGINAL_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3A14) -+#define PR_POSTAL_ADDRESS PROP_TAG( PT_TSTRING, 0x3A15) -+#define PR_POSTAL_ADDRESS_W PROP_TAG( PT_UNICODE, 0x3A15) -+#define PR_POSTAL_ADDRESS_A PROP_TAG( PT_STRING8, 0x3A15) -+#define PR_COMPANY_NAME PROP_TAG( PT_TSTRING, 0x3A16) -+#define PR_COMPANY_NAME_W PROP_TAG( PT_UNICODE, 0x3A16) -+#define PR_COMPANY_NAME_A PROP_TAG( PT_STRING8, 0x3A16) -+#define PR_TITLE PROP_TAG( PT_TSTRING, 0x3A17) -+#define PR_TITLE_W PROP_TAG( PT_UNICODE, 0x3A17) -+#define PR_TITLE_A PROP_TAG( PT_STRING8, 0x3A17) -+#define PR_DEPARTMENT_NAME PROP_TAG( PT_TSTRING, 0x3A18) -+#define PR_DEPARTMENT_NAME_W PROP_TAG( PT_UNICODE, 0x3A18) -+#define PR_DEPARTMENT_NAME_A PROP_TAG( PT_STRING8, 0x3A18) -+#define PR_OFFICE_LOCATION PROP_TAG( PT_TSTRING, 0x3A19) -+#define PR_OFFICE_LOCATION_W PROP_TAG( PT_UNICODE, 0x3A19) -+#define PR_OFFICE_LOCATION_A PROP_TAG( PT_STRING8, 0x3A19) -+#define PR_PRIMARY_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1A) -+#define PR_PRIMARY_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1A) -+#define PR_PRIMARY_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1A) -+#define PR_BUSINESS2_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1B) -+#define PR_BUSINESS2_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1B) -+#define PR_BUSINESS2_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1B) -+#define PR_OFFICE2_TELEPHONE_NUMBER PR_BUSINESS2_TELEPHONE_NUMBER -+#define PR_OFFICE2_TELEPHONE_NUMBER_W PR_BUSINESS2_TELEPHONE_NUMBER_W -+#define PR_OFFICE2_TELEPHONE_NUMBER_A PR_BUSINESS2_TELEPHONE_NUMBER_A -+#define PR_MOBILE_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1C) -+#define PR_MOBILE_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1C) -+#define PR_MOBILE_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1C) -+#define PR_CELLULAR_TELEPHONE_NUMBER PR_MOBILE_TELEPHONE_NUMBER -+#define PR_CELLULAR_TELEPHONE_NUMBER_W PR_MOBILE_TELEPHONE_NUMBER_W -+#define PR_CELLULAR_TELEPHONE_NUMBER_A PR_MOBILE_TELEPHONE_NUMBER_A -+#define PR_RADIO_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1D) -+#define PR_RADIO_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1D) -+#define PR_RADIO_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1D) -+#define PR_CAR_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1E) -+#define PR_CAR_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1E) -+#define PR_CAR_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1E) -+#define PR_OTHER_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A1F) -+#define PR_OTHER_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A1F) -+#define PR_OTHER_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A1F) -+#define PR_TRANSMITABLE_DISPLAY_NAME PROP_TAG( PT_TSTRING, 0x3A20) -+#define PR_TRANSMITABLE_DISPLAY_NAME_W PROP_TAG( PT_UNICODE, 0x3A20) -+#define PR_TRANSMITABLE_DISPLAY_NAME_A PROP_TAG( PT_STRING8, 0x3A20) -+#define PR_PAGER_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A21) -+#define PR_PAGER_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A21) -+#define PR_PAGER_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A21) -+#define PR_BEEPER_TELEPHONE_NUMBER PR_PAGER_TELEPHONE_NUMBER -+#define PR_BEEPER_TELEPHONE_NUMBER_W PR_PAGER_TELEPHONE_NUMBER_W -+#define PR_BEEPER_TELEPHONE_NUMBER_A PR_PAGER_TELEPHONE_NUMBER_A -+#define PR_USER_CERTIFICATE PROP_TAG( PT_BINARY, 0x3A22) -+#define PR_PRIMARY_FAX_NUMBER PROP_TAG( PT_TSTRING, 0x3A23) -+#define PR_PRIMARY_FAX_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A23) -+#define PR_PRIMARY_FAX_NUMBER_A PROP_TAG( PT_STRING8, 0x3A23) -+#define PR_BUSINESS_FAX_NUMBER PROP_TAG( PT_TSTRING, 0x3A24) -+#define PR_BUSINESS_FAX_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A24) -+#define PR_BUSINESS_FAX_NUMBER_A PROP_TAG( PT_STRING8, 0x3A24) -+#define PR_HOME_FAX_NUMBER PROP_TAG( PT_TSTRING, 0x3A25) -+#define PR_HOME_FAX_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A25) -+#define PR_HOME_FAX_NUMBER_A PROP_TAG( PT_STRING8, 0x3A25) -+#define PR_COUNTRY PROP_TAG( PT_TSTRING, 0x3A26) -+#define PR_COUNTRY_W PROP_TAG( PT_UNICODE, 0x3A26) -+#define PR_COUNTRY_A PROP_TAG( PT_STRING8, 0x3A26) -+#define PR_BUSINESS_ADDRESS_COUNTRY PR_COUNTRY -+#define PR_BUSINESS_ADDRESS_COUNTRY_W PR_COUNTRY_W -+#define PR_BUSINESS_ADDRESS_COUNTRY_A PR_COUNTRY_A -+ -+#define PR_LOCALITY PROP_TAG( PT_TSTRING, 0x3A27) -+#define PR_LOCALITY_W PROP_TAG( PT_UNICODE, 0x3A27) -+#define PR_LOCALITY_A PROP_TAG( PT_STRING8, 0x3A27) -+#define PR_BUSINESS_ADDRESS_CITY PR_LOCALITY -+#define PR_BUSINESS_ADDRESS_CITY_W PR_LOCALITY_W -+#define PR_BUSINESS_ADDRESS_CITY_A PR_LOCALITY_A -+ -+#define PR_STATE_OR_PROVINCE PROP_TAG( PT_TSTRING, 0x3A28) -+#define PR_STATE_OR_PROVINCE_W PROP_TAG( PT_UNICODE, 0x3A28) -+#define PR_STATE_OR_PROVINCE_A PROP_TAG( PT_STRING8, 0x3A28) -+#define PR_BUSINESS_ADDRESS_STATE_OR_PROVINCE PR_STATE_OR_PROVINCE -+#define PR_BUSINESS_ADDRESS_STATE_OR_PROVINCE_W PR_STATE_OR_PROVINCE_W -+#define PR_BUSINESS_ADDRESS_STATE_OR_PROVINCE_A PR_STATE_OR_PROVINCE_A -+ -+#define PR_STREET_ADDRESS PROP_TAG( PT_TSTRING, 0x3A29) -+#define PR_STREET_ADDRESS_W PROP_TAG( PT_UNICODE, 0x3A29) -+#define PR_STREET_ADDRESS_A PROP_TAG( PT_STRING8, 0x3A29) -+#define PR_BUSINESS_ADDRESS_STREET PR_STREET_ADDRESS -+#define PR_BUSINESS_ADDRESS_STREET_W PR_STREET_ADDRESS_W -+#define PR_BUSINESS_ADDRESS_STREET_A PR_STREET_ADDRESS_A -+ -+#define PR_POSTAL_CODE PROP_TAG( PT_TSTRING, 0x3A2A) -+#define PR_POSTAL_CODE_W PROP_TAG( PT_UNICODE, 0x3A2A) -+#define PR_POSTAL_CODE_A PROP_TAG( PT_STRING8, 0x3A2A) -+#define PR_BUSINESS_ADDRESS_POSTAL_CODE PR_POSTAL_CODE -+#define PR_BUSINESS_ADDRESS_POSTAL_CODE_W PR_POSTAL_CODE_W -+#define PR_BUSINESS_ADDRESS_POSTAL_CODE_A PR_POSTAL_CODE_A -+ -+ -+#define PR_POST_OFFICE_BOX PROP_TAG( PT_TSTRING, 0x3A2B) -+#define PR_POST_OFFICE_BOX_W PROP_TAG( PT_UNICODE, 0x3A2B) -+#define PR_POST_OFFICE_BOX_A PROP_TAG( PT_STRING8, 0x3A2B) -+#define PR_BUSINESS_ADDRESS_POST_OFFICE_BOX PR_POST_OFFICE_BOX -+#define PR_BUSINESS_ADDRESS_POST_OFFICE_BOX_W PR_POST_OFFICE_BOX_W -+#define PR_BUSINESS_ADDRESS_POST_OFFICE_BOX_A PR_POST_OFFICE_BOX_A -+ -+ -+#define PR_TELEX_NUMBER PROP_TAG( PT_TSTRING, 0x3A2C) -+#define PR_TELEX_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A2C) -+#define PR_TELEX_NUMBER_A PROP_TAG( PT_STRING8, 0x3A2C) -+#define PR_ISDN_NUMBER PROP_TAG( PT_TSTRING, 0x3A2D) -+#define PR_ISDN_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A2D) -+#define PR_ISDN_NUMBER_A PROP_TAG( PT_STRING8, 0x3A2D) -+#define PR_ASSISTANT_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A2E) -+#define PR_ASSISTANT_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A2E) -+#define PR_ASSISTANT_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A2E) -+#define PR_HOME2_TELEPHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A2F) -+#define PR_HOME2_TELEPHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A2F) -+#define PR_HOME2_TELEPHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A2F) -+#define PR_ASSISTANT PROP_TAG( PT_TSTRING, 0x3A30) -+#define PR_ASSISTANT_W PROP_TAG( PT_UNICODE, 0x3A30) -+#define PR_ASSISTANT_A PROP_TAG( PT_STRING8, 0x3A30) -+#define PR_SEND_RICH_INFO PROP_TAG( PT_BOOLEAN, 0x3A40) -+ -+#define PR_WEDDING_ANNIVERSARY PROP_TAG( PT_SYSTIME, 0x3A41) -+#define PR_BIRTHDAY PROP_TAG( PT_SYSTIME, 0x3A42) -+ -+ -+#define PR_HOBBIES PROP_TAG( PT_TSTRING, 0x3A43) -+#define PR_HOBBIES_W PROP_TAG( PT_UNICODE, 0x3A43) -+#define PR_HOBBIES_A PROP_TAG( PT_STRING8, 0x3A43) -+ -+#define PR_MIDDLE_NAME PROP_TAG( PT_TSTRING, 0x3A44) -+#define PR_MIDDLE_NAME_W PROP_TAG( PT_UNICODE, 0x3A44) -+#define PR_MIDDLE_NAME_A PROP_TAG( PT_STRING8, 0x3A44) -+ -+#define PR_DISPLAY_NAME_PREFIX PROP_TAG( PT_TSTRING, 0x3A45) -+#define PR_DISPLAY_NAME_PREFIX_W PROP_TAG( PT_UNICODE, 0x3A45) -+#define PR_DISPLAY_NAME_PREFIX_A PROP_TAG( PT_STRING8, 0x3A45) -+ -+#define PR_PROFESSION PROP_TAG( PT_TSTRING, 0x3A46) -+#define PR_PROFESSION_W PROP_TAG( PT_UNICODE, 0x3A46) -+#define PR_PROFESSION_A PROP_TAG( PT_STRING8, 0x3A46) -+ -+#define PR_PREFERRED_BY_NAME PROP_TAG( PT_TSTRING, 0x3A47) -+#define PR_PREFERRED_BY_NAME_W PROP_TAG( PT_UNICODE, 0x3A47) -+#define PR_PREFERRED_BY_NAME_A PROP_TAG( PT_STRING8, 0x3A47) -+ -+#define PR_SPOUSE_NAME PROP_TAG( PT_TSTRING, 0x3A48) -+#define PR_SPOUSE_NAME_W PROP_TAG( PT_UNICODE, 0x3A48) -+#define PR_SPOUSE_NAME_A PROP_TAG( PT_STRING8, 0x3A48) -+ -+#define PR_COMPUTER_NETWORK_NAME PROP_TAG( PT_TSTRING, 0x3A49) -+#define PR_COMPUTER_NETWORK_NAME_W PROP_TAG( PT_UNICODE, 0x3A49) -+#define PR_COMPUTER_NETWORK_NAME_A PROP_TAG( PT_STRING8, 0x3A49) -+ -+#define PR_CUSTOMER_ID PROP_TAG( PT_TSTRING, 0x3A4A) -+#define PR_CUSTOMER_ID_W PROP_TAG( PT_UNICODE, 0x3A4A) -+#define PR_CUSTOMER_ID_A PROP_TAG( PT_STRING8, 0x3A4A) -+ -+#define PR_TTYTDD_PHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A4B) -+#define PR_TTYTDD_PHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A4B) -+#define PR_TTYTDD_PHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A4B) -+ -+#define PR_FTP_SITE PROP_TAG( PT_TSTRING, 0x3A4C) -+#define PR_FTP_SITE_W PROP_TAG( PT_UNICODE, 0x3A4C) -+#define PR_FTP_SITE_A PROP_TAG( PT_STRING8, 0x3A4C) -+ -+#define PR_GENDER PROP_TAG( PT_SHORT, 0x3A4D) -+ -+#define PR_MANAGER_NAME PROP_TAG( PT_TSTRING, 0x3A4E) -+#define PR_MANAGER_NAME_W PROP_TAG( PT_UNICODE, 0x3A4E) -+#define PR_MANAGER_NAME_A PROP_TAG( PT_STRING8, 0x3A4E) -+ -+#define PR_NICKNAME PROP_TAG( PT_TSTRING, 0x3A4F) -+#define PR_NICKNAME_W PROP_TAG( PT_UNICODE, 0x3A4F) -+#define PR_NICKNAME_A PROP_TAG( PT_STRING8, 0x3A4F) -+ -+#define PR_PERSONAL_HOME_PAGE PROP_TAG( PT_TSTRING, 0x3A50) -+#define PR_PERSONAL_HOME_PAGE_W PROP_TAG( PT_UNICODE, 0x3A50) -+#define PR_PERSONAL_HOME_PAGE_A PROP_TAG( PT_STRING8, 0x3A50) -+ -+ -+#define PR_BUSINESS_HOME_PAGE PROP_TAG( PT_TSTRING, 0x3A51) -+#define PR_BUSINESS_HOME_PAGE_W PROP_TAG( PT_UNICODE, 0x3A51) -+#define PR_BUSINESS_HOME_PAGE_A PROP_TAG( PT_STRING8, 0x3A51) -+ -+#define PR_CONTACT_VERSION PROP_TAG( PT_CLSID, 0x3A52) -+#define PR_CONTACT_ENTRYIDS PROP_TAG( PT_MV_BINARY, 0x3A53) -+ -+#define PR_CONTACT_ADDRTYPES PROP_TAG( PT_MV_TSTRING, 0x3A54) -+#define PR_CONTACT_ADDRTYPES_W PROP_TAG( PT_MV_UNICODE, 0x3A54) -+#define PR_CONTACT_ADDRTYPES_A PROP_TAG( PT_MV_STRING8, 0x3A54) -+ -+#define PR_CONTACT_DEFAULT_ADDRESS_INDEX PROP_TAG( PT_LONG, 0x3A55) -+ -+#define PR_CONTACT_EMAIL_ADDRESSES PROP_TAG( PT_MV_TSTRING, 0x3A56) -+#define PR_CONTACT_EMAIL_ADDRESSES_W PROP_TAG( PT_MV_UNICODE, 0x3A56) -+#define PR_CONTACT_EMAIL_ADDRESSES_A PROP_TAG( PT_MV_STRING8, 0x3A56) -+ -+ -+#define PR_COMPANY_MAIN_PHONE_NUMBER PROP_TAG( PT_TSTRING, 0x3A57) -+#define PR_COMPANY_MAIN_PHONE_NUMBER_W PROP_TAG( PT_UNICODE, 0x3A57) -+#define PR_COMPANY_MAIN_PHONE_NUMBER_A PROP_TAG( PT_STRING8, 0x3A57) -+ -+#define PR_CHILDRENS_NAMES PROP_TAG( PT_MV_TSTRING, 0x3A58) -+#define PR_CHILDRENS_NAMES_W PROP_TAG( PT_MV_UNICODE, 0x3A58) -+#define PR_CHILDRENS_NAMES_A PROP_TAG( PT_MV_STRING8, 0x3A58) -+ -+ -+ -+#define PR_HOME_ADDRESS_CITY PROP_TAG( PT_TSTRING, 0x3A59) -+#define PR_HOME_ADDRESS_CITY_W PROP_TAG( PT_UNICODE, 0x3A59) -+#define PR_HOME_ADDRESS_CITY_A PROP_TAG( PT_STRING8, 0x3A59) -+ -+#define PR_HOME_ADDRESS_COUNTRY PROP_TAG( PT_TSTRING, 0x3A5A) -+#define PR_HOME_ADDRESS_COUNTRY_W PROP_TAG( PT_UNICODE, 0x3A5A) -+#define PR_HOME_ADDRESS_COUNTRY_A PROP_TAG( PT_STRING8, 0x3A5A) -+ -+#define PR_HOME_ADDRESS_POSTAL_CODE PROP_TAG( PT_TSTRING, 0x3A5B) -+#define PR_HOME_ADDRESS_POSTAL_CODE_W PROP_TAG( PT_UNICODE, 0x3A5B) -+#define PR_HOME_ADDRESS_POSTAL_CODE_A PROP_TAG( PT_STRING8, 0x3A5B) -+ -+#define PR_HOME_ADDRESS_STATE_OR_PROVINCE PROP_TAG( PT_TSTRING, 0x3A5C) -+#define PR_HOME_ADDRESS_STATE_OR_PROVINCE_W PROP_TAG( PT_UNICODE, 0x3A5C) -+#define PR_HOME_ADDRESS_STATE_OR_PROVINCE_A PROP_TAG( PT_STRING8, 0x3A5C) -+ -+#define PR_HOME_ADDRESS_STREET PROP_TAG( PT_TSTRING, 0x3A5D) -+#define PR_HOME_ADDRESS_STREET_W PROP_TAG( PT_UNICODE, 0x3A5D) -+#define PR_HOME_ADDRESS_STREET_A PROP_TAG( PT_STRING8, 0x3A5D) -+ -+#define PR_HOME_ADDRESS_POST_OFFICE_BOX PROP_TAG( PT_TSTRING, 0x3A5E) -+#define PR_HOME_ADDRESS_POST_OFFICE_BOX_W PROP_TAG( PT_UNICODE, 0x3A5E) -+#define PR_HOME_ADDRESS_POST_OFFICE_BOX_A PROP_TAG( PT_STRING8, 0x3A5E) -+ -+#define PR_OTHER_ADDRESS_CITY PROP_TAG( PT_TSTRING, 0x3A5F) -+#define PR_OTHER_ADDRESS_CITY_W PROP_TAG( PT_UNICODE, 0x3A5F) -+#define PR_OTHER_ADDRESS_CITY_A PROP_TAG( PT_STRING8, 0x3A5F) -+ -+#define PR_OTHER_ADDRESS_COUNTRY PROP_TAG( PT_TSTRING, 0x3A60) -+#define PR_OTHER_ADDRESS_COUNTRY_W PROP_TAG( PT_UNICODE, 0x3A60) -+#define PR_OTHER_ADDRESS_COUNTRY_A PROP_TAG( PT_STRING8, 0x3A60) -+ -+#define PR_OTHER_ADDRESS_POSTAL_CODE PROP_TAG( PT_TSTRING, 0x3A61) -+#define PR_OTHER_ADDRESS_POSTAL_CODE_W PROP_TAG( PT_UNICODE, 0x3A61) -+#define PR_OTHER_ADDRESS_POSTAL_CODE_A PROP_TAG( PT_STRING8, 0x3A61) -+ -+#define PR_OTHER_ADDRESS_STATE_OR_PROVINCE PROP_TAG( PT_TSTRING, 0x3A62) -+#define PR_OTHER_ADDRESS_STATE_OR_PROVINCE_W PROP_TAG( PT_UNICODE, 0x3A62) -+#define PR_OTHER_ADDRESS_STATE_OR_PROVINCE_A PROP_TAG( PT_STRING8, 0x3A62) -+ -+#define PR_OTHER_ADDRESS_STREET PROP_TAG( PT_TSTRING, 0x3A63) -+#define PR_OTHER_ADDRESS_STREET_W PROP_TAG( PT_UNICODE, 0x3A63) -+#define PR_OTHER_ADDRESS_STREET_A PROP_TAG( PT_STRING8, 0x3A63) -+ -+#define PR_OTHER_ADDRESS_POST_OFFICE_BOX PROP_TAG( PT_TSTRING, 0x3A64) -+#define PR_OTHER_ADDRESS_POST_OFFICE_BOX_W PROP_TAG( PT_UNICODE, 0x3A64) -+#define PR_OTHER_ADDRESS_POST_OFFICE_BOX_A PROP_TAG( PT_STRING8, 0x3A64) -+ -+ -+/* -+ * Profile section properties -+ */ -+ -+#define PR_STORE_PROVIDERS PROP_TAG( PT_BINARY, 0x3D00) -+#define PR_AB_PROVIDERS PROP_TAG( PT_BINARY, 0x3D01) -+#define PR_TRANSPORT_PROVIDERS PROP_TAG( PT_BINARY, 0x3D02) -+ -+#define PR_DEFAULT_PROFILE PROP_TAG( PT_BOOLEAN, 0x3D04) -+#define PR_AB_SEARCH_PATH PROP_TAG( PT_MV_BINARY, 0x3D05) -+#define PR_AB_DEFAULT_DIR PROP_TAG( PT_BINARY, 0x3D06) -+#define PR_AB_DEFAULT_PAB PROP_TAG( PT_BINARY, 0x3D07) -+ -+#define PR_FILTERING_HOOKS PROP_TAG( PT_BINARY, 0x3D08) -+#define PR_SERVICE_NAME PROP_TAG( PT_TSTRING, 0x3D09) -+#define PR_SERVICE_NAME_W PROP_TAG( PT_UNICODE, 0x3D09) -+#define PR_SERVICE_NAME_A PROP_TAG( PT_STRING8, 0x3D09) -+#define PR_SERVICE_DLL_NAME PROP_TAG( PT_TSTRING, 0x3D0A) -+#define PR_SERVICE_DLL_NAME_W PROP_TAG( PT_UNICODE, 0x3D0A) -+#define PR_SERVICE_DLL_NAME_A PROP_TAG( PT_STRING8, 0x3D0A) -+#define PR_SERVICE_ENTRY_NAME PROP_TAG( PT_STRING8, 0x3D0B) -+#define PR_SERVICE_UID PROP_TAG( PT_BINARY, 0x3D0C) -+#define PR_SERVICE_EXTRA_UIDS PROP_TAG( PT_BINARY, 0x3D0D) -+#define PR_SERVICES PROP_TAG( PT_BINARY, 0x3D0E) -+#define PR_SERVICE_SUPPORT_FILES PROP_TAG( PT_MV_TSTRING, 0x3D0F) -+#define PR_SERVICE_SUPPORT_FILES_W PROP_TAG( PT_MV_UNICODE, 0x3D0F) -+#define PR_SERVICE_SUPPORT_FILES_A PROP_TAG( PT_MV_STRING8, 0x3D0F) -+#define PR_SERVICE_DELETE_FILES PROP_TAG( PT_MV_TSTRING, 0x3D10) -+#define PR_SERVICE_DELETE_FILES_W PROP_TAG( PT_MV_UNICODE, 0x3D10) -+#define PR_SERVICE_DELETE_FILES_A PROP_TAG( PT_MV_STRING8, 0x3D10) -+#define PR_AB_SEARCH_PATH_UPDATE PROP_TAG( PT_BINARY, 0x3D11) -+#define PR_PROFILE_NAME PROP_TAG( PT_TSTRING, 0x3D12) -+#define PR_PROFILE_NAME_A PROP_TAG( PT_STRING8, 0x3D12) -+#define PR_PROFILE_NAME_W PROP_TAG( PT_UNICODE, 0x3D12) -+ -+/* -+ * Status object properties -+ */ -+ -+#define PR_IDENTITY_DISPLAY PROP_TAG( PT_TSTRING, 0x3E00) -+#define PR_IDENTITY_DISPLAY_W PROP_TAG( PT_UNICODE, 0x3E00) -+#define PR_IDENTITY_DISPLAY_A PROP_TAG( PT_STRING8, 0x3E00) -+#define PR_IDENTITY_ENTRYID PROP_TAG( PT_BINARY, 0x3E01) -+#define PR_RESOURCE_METHODS PROP_TAG( PT_LONG, 0x3E02) -+#define PR_RESOURCE_TYPE PROP_TAG( PT_LONG, 0x3E03) -+#define PR_STATUS_CODE PROP_TAG( PT_LONG, 0x3E04) -+#define PR_IDENTITY_SEARCH_KEY PROP_TAG( PT_BINARY, 0x3E05) -+#define PR_OWN_STORE_ENTRYID PROP_TAG( PT_BINARY, 0x3E06) -+#define PR_RESOURCE_PATH PROP_TAG( PT_TSTRING, 0x3E07) -+#define PR_RESOURCE_PATH_W PROP_TAG( PT_UNICODE, 0x3E07) -+#define PR_RESOURCE_PATH_A PROP_TAG( PT_STRING8, 0x3E07) -+#define PR_STATUS_STRING PROP_TAG( PT_TSTRING, 0x3E08) -+#define PR_STATUS_STRING_W PROP_TAG( PT_UNICODE, 0x3E08) -+#define PR_STATUS_STRING_A PROP_TAG( PT_STRING8, 0x3E08) -+#define PR_X400_DEFERRED_DELIVERY_CANCEL PROP_TAG( PT_BOOLEAN, 0x3E09) -+#define PR_HEADER_FOLDER_ENTRYID PROP_TAG( PT_BINARY, 0x3E0A) -+#define PR_REMOTE_PROGRESS PROP_TAG( PT_LONG, 0x3E0B) -+#define PR_REMOTE_PROGRESS_TEXT PROP_TAG( PT_TSTRING, 0x3E0C) -+#define PR_REMOTE_PROGRESS_TEXT_W PROP_TAG( PT_UNICODE, 0x3E0C) -+#define PR_REMOTE_PROGRESS_TEXT_A PROP_TAG( PT_STRING8, 0x3E0C) -+#define PR_REMOTE_VALIDATE_OK PROP_TAG( PT_BOOLEAN, 0x3E0D) -+ -+/* -+ * Display table properties -+ */ -+ -+#define PR_CONTROL_FLAGS PROP_TAG( PT_LONG, 0x3F00) -+#define PR_CONTROL_STRUCTURE PROP_TAG( PT_BINARY, 0x3F01) -+#define PR_CONTROL_TYPE PROP_TAG( PT_LONG, 0x3F02) -+#define PR_DELTAX PROP_TAG( PT_LONG, 0x3F03) -+#define PR_DELTAY PROP_TAG( PT_LONG, 0x3F04) -+#define PR_XPOS PROP_TAG( PT_LONG, 0x3F05) -+#define PR_YPOS PROP_TAG( PT_LONG, 0x3F06) -+#define PR_CONTROL_ID PROP_TAG( PT_BINARY, 0x3F07) -+#define PR_INITIAL_DETAILS_PANE PROP_TAG( PT_LONG, 0x3F08) -+ -+/* -+ * Secure property id range -+ */ -+ -+#define PROP_ID_SECURE_MIN 0x67F0 -+#define PROP_ID_SECURE_MAX 0x67FF -+ -+ -+#endif /* MAPITAGS_H */ |