Compare commits
	
		
			13 Commits
		
	
	
		
			0.7.12a
			...
			b4d3034c8c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b4d3034c8c | |||
| 9515236244 | |||
| a790494133 | |||
| ec8062309d | |||
| 631998a849 | |||
| bd64536e89 | |||
| b30f2f2362 | |||
| 1a63374e73 | |||
| efc6f92cdb | |||
| 191b24433f | |||
| 40b6f8799f | |||
|   | 13bf23b9a2 | ||
| c732d4f949 | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -17,3 +17,7 @@ compile | |||||||
| depcomp | depcomp | ||||||
| install-sh | install-sh | ||||||
| missing | missing | ||||||
|  | src/mboxgrep.exe | ||||||
|  | configure~ | ||||||
|  | .gitignore | ||||||
|  | .vscode/* | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -1,5 +0,0 @@ | |||||||
| -*- text -*- |  | ||||||
|  |  | ||||||
| mboxgrep is written and maintained by Daniel Spiljar. |  | ||||||
|  |  | ||||||
| $Id: AUTHORS,v 1.3 2005-11-16 01:37:18 dspiljar Exp $ |  | ||||||
							
								
								
									
										1
									
								
								AUTHORS.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								AUTHORS.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | mboxgrep is written and maintained by Daniel Spiljar with contributors. | ||||||
							
								
								
									
										44
									
								
								NEWS.md
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								NEWS.md
									
									
									
									
									
								
							| @@ -1,21 +1,21 @@ | |||||||
| # Changes of mboxgrep | # Changes of mboxgrep | ||||||
|  |  | ||||||
| ## Changes since 0.7.12 | ## Changes in 0.7.12a | ||||||
|  |  | ||||||
| - Fix Automake macros to include the license file and others. | - Fix Automake macros to include the license file and others. | ||||||
|  |  | ||||||
| ## Changes since 0.7.11 | ## Changes in 0.7.12 | ||||||
|  |  | ||||||
| - Port to the pcre2 library (pcre1 is no longer supported). | - Port to the pcre2 library (pcre1 is no longer supported). | ||||||
| - Check command-line options for conflicting matchers and actions. | - Check command-line options for conflicting matchers and actions. | ||||||
| - Various minor code cleanups. | - Various minor code cleanups. | ||||||
|  |  | ||||||
| ## Changes since 0.7.10 | ## Changes in 0.7.11 | ||||||
|  |  | ||||||
| - GNU Automake is now utilized instead of manually written Makefile.in files. | - GNU Automake is now utilized instead of manually written Makefile.in files. | ||||||
| - Code indentation has been corrected and documented (GNU indent is used for this purpose). | - Code indentation has been corrected and documented (GNU indent is used for this purpose). | ||||||
|  |  | ||||||
| ## Changes since 0.7.9 | ## Changes in 0.7.10 | ||||||
|  |  | ||||||
| - development has been restarted after a longer hiatus | - development has been restarted after a longer hiatus | ||||||
| - moved the source tree to git | - moved the source tree to git | ||||||
| @@ -24,7 +24,7 @@ | |||||||
| - fixed multiple compilation errors and warnings on Linux and FreeBSD | - fixed multiple compilation errors and warnings on Linux and FreeBSD | ||||||
| - started implementing a debug mode | - started implementing a debug mode | ||||||
|  |  | ||||||
| ## Changes since 0.7.8 | ## Changes in 0.7.9 | ||||||
|  |  | ||||||
| - implemented support for mbox folders compressed with bzip2 algorithm | - implemented support for mbox folders compressed with bzip2 algorithm | ||||||
| - improved detection of PCRE library; this also fixes a compilation bug | - improved detection of PCRE library; this also fixes a compilation bug | ||||||
| @@ -33,7 +33,7 @@ | |||||||
|   folder and temporary directory were on different filesystems; |   folder and temporary directory were on different filesystems; | ||||||
|   thanks to James P. Dugal <jpd AT louisiana.edu> for providing a patch |   thanks to James P. Dugal <jpd AT louisiana.edu> for providing a patch | ||||||
|  |  | ||||||
| ## Changes since 0.7.7 | ## Changes in 0.7.8 | ||||||
|  |  | ||||||
| - fixed crashes on MH folders, as well as a potential problem with mbox | - fixed crashes on MH folders, as well as a potential problem with mbox | ||||||
|   and maildir folders with the same cause |   and maildir folders with the same cause | ||||||
| @@ -42,46 +42,46 @@ | |||||||
| - mboxgrep can now read gzip compressed mbox folders from standard input | - mboxgrep can now read gzip compressed mbox folders from standard input | ||||||
|   (which is a side effect of a code cleanup!) |   (which is a side effect of a code cleanup!) | ||||||
|  |  | ||||||
| ## Changes since 0.7.6a | ## Changes in 0.7.7 | ||||||
|  |  | ||||||
| - mboxgrep can now read mbox folders from standard input | - mboxgrep can now read mbox folders from standard input | ||||||
|  |  | ||||||
| ## Changes since 0.7.6 | ## Changes in 0.7.6a | ||||||
|  |  | ||||||
| - fixed a bug which prevented mboxgrep from being compiled | - fixed a bug which prevented mboxgrep from being compiled | ||||||
|  |  | ||||||
| ## Changes since 0.7.5 | ## Changes in 0.7.6 | ||||||
|  |  | ||||||
| - fixed a bug which caused mboxgrep to coredump on recursive search of  | - fixed a bug which caused mboxgrep to coredump on recursive search of  | ||||||
|   maildirs |   maildirs | ||||||
| - code cleanups | - code cleanups | ||||||
|  |  | ||||||
| ## Changes since 0.7.4 | ## Changes in 0.7.5 | ||||||
|  |  | ||||||
| - recursive search through directories has been reimplemented | - recursive search through directories has been reimplemented | ||||||
|  |  | ||||||
| ## Changes since 0.7.3 | ## Changes in 0.7.4 | ||||||
|  |  | ||||||
| - mboxgrep can now ignore messages with identical bodies | - mboxgrep can now ignore messages with identical bodies | ||||||
|  |  | ||||||
| ## Changes since 0.7.2 | ## Changes in 0.7.3 | ||||||
|  |  | ||||||
| - file locking method can be selected at runtime | - file locking method can be selected at runtime | ||||||
| - mboxgrep used to coredump on messages with empty headers; the problem is  | - mboxgrep used to coredump on messages with empty headers; the problem is  | ||||||
|   now hopefully fixed |   now hopefully fixed | ||||||
| - mboxgrep now compiles on FreeBSD | - mboxgrep now compiles on FreeBSD | ||||||
|  |  | ||||||
| ## Changes since 0.7.1 | ## Changes in 0.7.2 | ||||||
|  |  | ||||||
| - file locking with `flock()` instead of `fcntl()` is now supported | - file locking with `flock()` instead of `fcntl()` is now supported | ||||||
| - mboxgrep should now compile on systems which don't have `getopt_long()` | - mboxgrep should now compile on systems which don't have `getopt_long()` | ||||||
| - other minor bugfixes | - other minor bugfixes | ||||||
|  |  | ||||||
| ## Changes since 0.7.0 | ## Changes in 0.7.1 | ||||||
|  |  | ||||||
| - bugfixes | - bugfixes | ||||||
|  |  | ||||||
| ## Changes since 0.5.3 | ## Changes in 0.7.0 | ||||||
|  |  | ||||||
| - NOTE: there was no stable version between 0.5.x and 0.7.x | - NOTE: there was no stable version between 0.5.x and 0.7.x | ||||||
| - mboxgrep is almost completely rewritten | - mboxgrep is almost completely rewritten | ||||||
| @@ -91,7 +91,7 @@ | |||||||
|   command) |   command) | ||||||
| - added support for compressed mbox folders | - added support for compressed mbox folders | ||||||
|  |  | ||||||
| ## Changes since 0.5.2 | ## Changes in 0.5.3 | ||||||
|  |  | ||||||
| - implemented ability to write found messages to another folder (instead | - implemented ability to write found messages to another folder (instead | ||||||
|   of standard output) |   of standard output) | ||||||
| @@ -99,7 +99,7 @@ | |||||||
|   "NEWS") |   "NEWS") | ||||||
| - wrote a manual in Texinfo format | - wrote a manual in Texinfo format | ||||||
|  |  | ||||||
| ## Changes since 0.5.1 | ## Changes in 0.5.2 | ||||||
|  |  | ||||||
| - implemented message counting (`-c, --count`) | - implemented message counting (`-c, --count`) | ||||||
| - fixed manual page installation target in src/Makefile.in -- manual page  | - fixed manual page installation target in src/Makefile.in -- manual page  | ||||||
| @@ -112,17 +112,17 @@ | |||||||
| - implemented recursive search through directories (`-r, --recursive`) | - implemented recursive search through directories (`-r, --recursive`) | ||||||
| - config.h[.in] is now wrapped (`#ifndef CONFIG_H` and stuff) | - config.h[.in] is now wrapped (`#ifndef CONFIG_H` and stuff) | ||||||
|  |  | ||||||
| ## Changes since 0.5.0 | ## Changes in 0.5.1 | ||||||
|  |  | ||||||
| - added support for qmail-style maildir folders | - added support for qmail-style maildir folders | ||||||
|  |  | ||||||
| ## Changes since 0.4.0 | ## Changes in 0.5.0 | ||||||
|  |  | ||||||
| - the source of `getopt_long()` is now included, so long options are also available | - the source of `getopt_long()` is now included, so long options are also available | ||||||
|   on platforms that don't use GNU libc |   on platforms that don't use GNU libc | ||||||
| - file locking is now performed on mbox folders only | - file locking is now performed on mbox folders only | ||||||
|  |  | ||||||
| ## Changes since 0.2.0 | ## Changes in 0.4.0 | ||||||
|  |  | ||||||
| - added support for MH mailboxes | - added support for MH mailboxes | ||||||
| - added scope selection switches `-B` (or `--body`) and `-H` (or `--headers`) | - added scope selection switches `-B` (or `--body`) and `-H` (or `--headers`) | ||||||
| @@ -133,12 +133,12 @@ | |||||||
| - the feature of reading from standard input is gone | - the feature of reading from standard input is gone | ||||||
| - added `-v` (or `--version`) switch | - added `-v` (or `--version`) switch | ||||||
|  |  | ||||||
| ## Changes since 0.1.1 | ## Changes in 0.2.0 | ||||||
|  |  | ||||||
| - mboxgrep now puts a shared lock on a mailbox before reading it | - mboxgrep now puts a shared lock on a mailbox before reading it | ||||||
| - various code cleanups | - various code cleanups | ||||||
|  |  | ||||||
| ## Changes since 0.1.0 | ## Changes in 0.1.1 | ||||||
|  |  | ||||||
| - fixed install target in Makefile (`install` is now used instead of `cp`) | - fixed install target in Makefile (`install` is now used instead of `cp`) | ||||||
| - added support for mailboxes generated by Gnus | - added support for mailboxes generated by Gnus | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								TODO.md
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								TODO.md
									
									
									
									
									
								
							| @@ -15,6 +15,7 @@ | |||||||
| - [x] reading messages from standard input | - [x] reading messages from standard input | ||||||
| - [x] run-time selection of file locking method | - [x] run-time selection of file locking method | ||||||
| - [x] add a debug function | - [x] add a debug function | ||||||
|  | - [ ] Remove the option to recursively traverse directories and instruct the users to run mboxgrep in conjuction with find(1) instead. | ||||||
|  |  | ||||||
| ## File formats, encodings and standards | ## File formats, encodings and standards | ||||||
|  |  | ||||||
| @@ -28,6 +29,7 @@ | |||||||
| - [ ] support for mail folder conversion | - [ ] support for mail folder conversion | ||||||
| - [ ] improve error detection when a directory is not a Maildir or MH folder | - [ ] improve error detection when a directory is not a Maildir or MH folder | ||||||
| - [ ] document criteria for folder format detection | - [ ] document criteria for folder format detection | ||||||
|  | - [ ] Maildir: check if an atomic rename() fails. This could be caused if "new" and "cur" subdirectories are not on the same filesystem, for example. | ||||||
|  |  | ||||||
| ## Miscellaneous | ## Miscellaneous | ||||||
|  |  | ||||||
|   | |||||||
| @@ -93,7 +93,5 @@ AC_CHECK_FUNCS([strptime]) | |||||||
| AC_CHECK_FUNCS(ftw) | AC_CHECK_FUNCS(ftw) | ||||||
| AC_CHECK_FUNCS(fts_open) | AC_CHECK_FUNCS(fts_open) | ||||||
|  |  | ||||||
| AC_CONFIG_FILES([Makefile | AC_CONFIG_FILES([Makefile doc/Makefile src/Makefile]) | ||||||
|                  doc/Makefile |  | ||||||
|                  src/Makefile]) |  | ||||||
| AC_OUTPUT | AC_OUTPUT | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| .TH MBOXGREP 1 "21 May 2023" | .TH MBOXGREP 1 "23 September 2024" | ||||||
| .SH NAME | .SH NAME | ||||||
| mboxgrep \- displays email messages matching a pattern | mboxgrep \- displays email messages matching a pattern | ||||||
| .SH SYNOPSIS | .SH SYNOPSIS | ||||||
| @@ -72,8 +72,8 @@ Do not lock files.  This option is meaningful only if a mbox folder (see below) | |||||||
| is scanned. | is scanned. | ||||||
| .IP "-c, --count" | .IP "-c, --count" | ||||||
| Suppress normal output and print a count of matching messages. | Suppress normal output and print a count of matching messages. | ||||||
| .IP "-o, --output=FOLDER" | .IP "-o, --output=MAILBOX" | ||||||
| Suppress normal output and write messages to destination folder FOLDER instead. | Suppress normal output and write messages to destination MAILBOX instead. | ||||||
| .IP "-p, --pipe=COMMAND" | .IP "-p, --pipe=COMMAND" | ||||||
| Pipe each found message to COMMAND | Pipe each found message to COMMAND | ||||||
| .IP "-d, --delete" | .IP "-d, --delete" | ||||||
| @@ -97,10 +97,26 @@ Search $MAIL for messages from Dirty Harry: | |||||||
| mboxgrep '^From:.*callahan@sanfranciscopolice\\.org' $MAIL | mboxgrep '^From:.*callahan@sanfranciscopolice\\.org' $MAIL | ||||||
| .TP | .TP | ||||||
| \(bu | \(bu | ||||||
|  | Re-mail to George messages that mention his name: | ||||||
|  | .PP | ||||||
|  | mboxgrep --pipe="/usr/lib/sendmail george" --ignore-case george ~/Mail/* | ||||||
|  | .TP | ||||||
|  | \(bu | ||||||
| Display all messages contained in folder ~/Mail/incoming, except those | Display all messages contained in folder ~/Mail/incoming, except those | ||||||
| that appear to originate from AOL: | that appear to originate from AOL: | ||||||
| .PP | .PP | ||||||
| mboxgrep -v 'Received:.*aol\\.com' ~/Mail/incoming | mboxgrep -v 'Received:.*aol\\.com' ~/Mail/incoming | ||||||
|  | .TP | ||||||
|  | \(bu | ||||||
|  | Do a case-insensitive scan of ~/Mail/incoming for messages with subject | ||||||
|  | ``Weekly News'' and write them to folder ~/Mail/archive: | ||||||
|  | .PP | ||||||
|  | mboxgrep -o ~/Mail/archive -H -i '^Subject: Weekly News' ~/Mail/incoming | ||||||
|  | .TP | ||||||
|  | \(bu | ||||||
|  | Count all messages stored in folder spam, ignoring duplicates: | ||||||
|  | .PP | ||||||
|  | mboxgrep -nd -c . spam | ||||||
| .SH BUGS | .SH BUGS | ||||||
| Report them to address below. | Report them to address below. | ||||||
| .SH SEE ALSO | .SH SEE ALSO | ||||||
| @@ -108,7 +124,8 @@ grep(1), | |||||||
| regex(7), | regex(7), | ||||||
| perlre(1), | perlre(1), | ||||||
| mbox(5), | mbox(5), | ||||||
| RFC 2822 | RFC 2822, | ||||||
|  | RFC 4155 | ||||||
| .SH DEDICATION | .SH DEDICATION | ||||||
| Mboxgrep is dedicated in loving memory of Vicky, my cat who died of | Mboxgrep is dedicated in loving memory of Vicky, my cat who died of | ||||||
| tumor on Sep 12, 2002.   | tumor on Sep 12, 2002.   | ||||||
| @@ -116,7 +133,11 @@ tumor on Sep 12, 2002. | |||||||
| You haven't been long with us, but you gave us a lot of joy and all your | You haven't been long with us, but you gave us a lot of joy and all your | ||||||
| big heart that stopped ticking too early.  I will never forget you.  Sleep | big heart that stopped ticking too early.  I will never forget you.  Sleep | ||||||
| well, little friend. | well, little friend. | ||||||
| .SH URL | .SH HOMEPAGE | ||||||
| http://www.mboxgrep.org/ | https://www.mboxgrep.org/ | ||||||
|  | .SH GIT | ||||||
|  | https://git.datatipp.se/dspiljar/mboxgrep | ||||||
|  | .PP | ||||||
|  | https://github.com/dspiljar/mboxgrep | ||||||
| .SH AUTHOR | .SH AUTHOR | ||||||
| Daniel Spiljar <dspiljar AT datatipp.se> | Daniel Spiljar <dspiljar@datatipp.se> | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| This is mboxgrep.info, produced by makeinfo version 6.7 from | This is mboxgrep.info, produced by makeinfo version 6.8 from | ||||||
| mboxgrep.texi. | mboxgrep.texi. | ||||||
|  |  | ||||||
| INFO-DIR-SECTION Mail | INFO-DIR-SECTION Mail | ||||||
| @@ -151,10 +151,10 @@ options below change such behavior. | |||||||
| '-nd' | '-nd' | ||||||
|      Ignore duplicate messages. |      Ignore duplicate messages. | ||||||
|  |  | ||||||
| '--output=FOLDER' | '--output=MAILBOX' | ||||||
| '-o FOLDER' | '-o MAILBOX' | ||||||
|      Suppress normal output and write messages to destination folder |      Suppress normal output and write messages to destination MAILBOX | ||||||
|      FOLDER instead. |      instead. | ||||||
|  |  | ||||||
|      Note that 'mboxgrep' assumes the output folder is of the same |      Note that 'mboxgrep' assumes the output folder is of the same | ||||||
|      format as the input folder.  Currently there is no possibility to |      format as the input folder.  Currently there is no possibility to | ||||||
| @@ -256,11 +256,11 @@ Node: Miscellaneous1828 | |||||||
| Node: File locking2113 | Node: File locking2113 | ||||||
| Node: Regexp selection2750 | Node: Regexp selection2750 | ||||||
| Node: Output control3302 | Node: Output control3302 | ||||||
| Node: Search scope selection4346 | Node: Search scope selection4342 | ||||||
| Node: Mailbox type selection4587 | Node: Mailbox type selection4583 | ||||||
| Node: Examples4908 | Node: Examples4904 | ||||||
| Node: Bugs5732 | Node: Bugs5728 | ||||||
| Node: To Vicky6150 | Node: To Vicky6146 | ||||||
|  |  | ||||||
| End Tag Table | End Tag Table | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  |  | ||||||
| @set EDITION 0.7 | @set EDITION 0.7 | ||||||
| @set VERSION 0.7.12a | @set VERSION 0.7.12a | ||||||
| @set UPDATED 21 May 2023 | @set UPDATED 23 September 2024 | ||||||
|  |  | ||||||
| @dircategory Mail | @dircategory Mail | ||||||
| @direntry | @direntry | ||||||
| @@ -19,7 +19,7 @@ | |||||||
| This file documents @code{mboxgrep} (version @value{VERSION}), a | This file documents @code{mboxgrep} (version @value{VERSION}), a | ||||||
| mailbox scanning utility. | mailbox scanning utility. | ||||||
|  |  | ||||||
| Copyright (C) 2000, 2001, 2002, 2003 Daniel Spiljar | Copyright (C) 2000 - 2003, 2024 Daniel Spiljar | ||||||
| @end ifinfo | @end ifinfo | ||||||
|  |  | ||||||
| @menu | @menu | ||||||
| @@ -171,10 +171,9 @@ Use with extreme caution. | |||||||
| @itemx -nd | @itemx -nd | ||||||
| Ignore duplicate messages. | Ignore duplicate messages. | ||||||
|  |  | ||||||
| @item --output=FOLDER | @item --output=MAILBOX | ||||||
| @itemx -o FOLDER | @itemx -o MAILBOX | ||||||
| Suppress normal output and write messages to destination folder FOLDER | Suppress normal output and write messages to destination MAILBOX instead. | ||||||
| instead.  |  | ||||||
|  |  | ||||||
| Note that @code{mboxgrep} assumes the output folder is of the same format  | Note that @code{mboxgrep} assumes the output folder is of the same format  | ||||||
| as the input folder.  Currently there is no possibility to convert folders. | as the input folder.  Currently there is no possibility to convert folders. | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| /* | /* | ||||||
|   mboxgrep - scan mailbox for messages matching a regular expression |   mboxgrep - scan mailbox for messages matching a regular expression | ||||||
|   Copyright (C) 2000 - 2004, 2006, 2023  Daniel Spiljar |   Copyright (C) 2000 - 2004, 2006, 2023 - 2024  Daniel Spiljar | ||||||
|  |  | ||||||
|   Mboxgrep is free software; you can redistribute it and/or modify it  |   Mboxgrep is free software; you can redistribute it and/or modify it  | ||||||
|   under the terms of the GNU General Public License as published by |   under the terms of the GNU General Public License as published by | ||||||
| @@ -117,7 +117,7 @@ main (int argc, char **argv) | |||||||
|       if (config.action == ACTION_DELETE) |       if (config.action == ACTION_DELETE) | ||||||
|         { |         { | ||||||
|           tmpmbox_create (argv[optind]); |           tmpmbox_create (argv[optind]); | ||||||
|           runtime.tmp_mbox = (mbox_t *) mbox_open (config.tmpfilename, "w"); |           runtime.tmp_mbox = (mbox_t *) mbox_open (config.tmpfilename, w); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|       config.boxname = xstrdup (argv[optind]); |       config.boxname = xstrdup (argv[optind]); | ||||||
|   | |||||||
							
								
								
									
										269
									
								
								src/mbox.c
									
									
									
									
									
								
							
							
						
						
									
										269
									
								
								src/mbox.c
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| /* | /* | ||||||
|    mboxgrep - scan mailbox for messages matching a regular expression |    mboxgrep - scan mailbox for messages matching a regular expression | ||||||
|    Copyright (C) 2000, 2001, 2002, 2003, 2004, 2006, 2023  Daniel Spiljar |    Copyright (C) 2000 - 2004, 2006, 2023 - 2024  Daniel Spiljar | ||||||
|  |  | ||||||
|    Mboxgrep is free software; you can redistribute it and/or modify it  |    Mboxgrep is free software; you can redistribute it and/or modify it  | ||||||
|    under the terms of the GNU General Public License as published by |    under the terms of the GNU General Public License as published by | ||||||
| @@ -50,166 +50,47 @@ | |||||||
| #endif /* HAVE_LIBDMALLOC */ | #endif /* HAVE_LIBDMALLOC */ | ||||||
|  |  | ||||||
| mbox_t * | mbox_t * | ||||||
| mbox_open (const char *path, const char *mode) | mbox_open (const char *path, const mbox_mode_t mbox_mode) | ||||||
| { | { | ||||||
|   mbox_t *mp; |   mbox_t *mp; | ||||||
|   static int fd; |   static int fd; | ||||||
| #ifndef HAVE_FLOCK | #ifndef HAVE_FLOCK | ||||||
|   struct flock lck; |   struct flock lck; | ||||||
| #endif /* HAVE_FLOCK */ | #endif /* HAVE_FLOCK */ | ||||||
|   char buffer[BUFSIZ]; |  | ||||||
|  |  | ||||||
|   mp = (mbox_t *) xmalloc (sizeof (mbox_t)); |   mp = (mbox_t *) xmalloc (sizeof (mbox_t)); | ||||||
|   mp->postmark_cache = (char *) xmalloc (BUFSIZ * sizeof (char)); |   mp->postmark_cache = NULL; | ||||||
|  |  | ||||||
|   if (0 == strcmp ("-", path)) |   if (0 == strcmp ("-", path)) | ||||||
|     mp->fp = stdin; |     mp->fp = stdin; | ||||||
|   else |   else | ||||||
|     { |     { | ||||||
|       if (mode[0] == 'r') |       if (mbox_mode == w) | ||||||
|         fd = m_open (path, O_RDONLY, 0); |  | ||||||
|       else if (mode[0] == 'w') |  | ||||||
|         fd = m_open (path, (O_WRONLY | O_CREAT | O_APPEND), |         fd = m_open (path, (O_WRONLY | O_CREAT | O_APPEND), | ||||||
|                      (S_IWUSR | S_IRUSR)); |                      (S_IWUSR | S_IRUSR)); | ||||||
|       else |       else | ||||||
|         { |         fd = m_open (path, O_RDONLY, 0); | ||||||
|           fprintf (stderr, "%s: mbox.c: Unknown mode %c.  You shouldn't " |  | ||||||
|                    "get this error...", APPNAME, mode[0]); |  | ||||||
|           exit (2); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|       if (fd == -1) |       if (fd == -1) | ||||||
|         { |         { | ||||||
|           if (config.merr) |  | ||||||
|             { |  | ||||||
|               fprintf (stderr, "%s: %s: ", APPNAME, path); |  | ||||||
|               perror (NULL); |  | ||||||
|             } |  | ||||||
|           errno = 0; |           errno = 0; | ||||||
|           return NULL; |           return NULL; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|       if (config.lock > LOCK_NONE) |       if (config.lock > LOCK_NONE) | ||||||
|         { |         mbox_lock (fd, path, mbox_mode); | ||||||
| #ifdef HAVE_FLOCK |  | ||||||
|           int op; |  | ||||||
|  |  | ||||||
|           if (mode[0] == 'r') |       mp->fp = mbox_fdopen (fd, path, mbox_mode); | ||||||
|             op = LOCK_SH; |  | ||||||
|           else |  | ||||||
|             op = LOCK_EX; |  | ||||||
|           if (-1 == flock (fd, op)) |  | ||||||
| #else |  | ||||||
|           memset (&lck, 0, sizeof (struct flock)); |  | ||||||
|           lck.l_whence = SEEK_SET; |  | ||||||
|           if (mode[0] == 'r') |  | ||||||
|             lck.l_type = F_RDLCK; |  | ||||||
|           else |  | ||||||
|             lck.l_type = F_WRLCK; |  | ||||||
|  |  | ||||||
|           if (-1 == fcntl (fd, F_SETLK, &lck)) |  | ||||||
| #endif /* HAVE_FLOCK */ |  | ||||||
|             { |  | ||||||
|               if (config.merr) |  | ||||||
|                 { |  | ||||||
|                   fprintf (stderr, "%s: %s: ", APPNAME, path); |  | ||||||
|                   perror (NULL); |  | ||||||
|     } |     } | ||||||
|               errno = 0; |  | ||||||
|               close (fd); |   if (mbox_mode == r) | ||||||
|  |     { | ||||||
|  |       mp->postmark_cache = mbox_check_postmark (mp, path); | ||||||
|  |  | ||||||
|  |       if (! mp->postmark_cache) | ||||||
|         return NULL; |         return NULL; | ||||||
|     } |     } | ||||||
|         } |  | ||||||
|  |  | ||||||
|       if (mode[0] == 'r') |  | ||||||
|         { |  | ||||||
|           if (config.format == FORMAT_MBOX) |  | ||||||
|             mp->fp = (FILE *) m_fdopen (fd, "r"); |  | ||||||
| #ifdef HAVE_LIBZ |  | ||||||
|           else if (config.format == FORMAT_ZMBOX) |  | ||||||
|             mp->fp = (gzFile *) m_gzdopen (fd, "rb"); |  | ||||||
| #endif /* HAVE_LIBZ */ |  | ||||||
| #ifdef HAVE_LIBBZ2 |  | ||||||
|           else if (config.format == FORMAT_BZ2MBOX) |  | ||||||
|             mp->fp = (BZFILE *) BZ2_bzdopen (fd, "rb"); |  | ||||||
| #endif /* HAVE_LIBBZ2 */ |  | ||||||
|         } |  | ||||||
|       else if (mode[0] == 'w') |  | ||||||
|         { |  | ||||||
|           if (config.format == FORMAT_MBOX) |  | ||||||
|             mp->fp = (FILE *) m_fdopen (fd, "w"); |  | ||||||
| #ifdef HAVE_LIBZ |  | ||||||
|           else if (config.format == FORMAT_ZMBOX) |  | ||||||
|             mp->fp = (gzFile *) m_gzdopen (fd, "wb"); |  | ||||||
| #endif /* HAVE_LIBZ */ |  | ||||||
| #ifdef HAVE_LIBBZ2 |  | ||||||
|           else if (config.format == FORMAT_BZ2MBOX) |  | ||||||
|             mp->fp = (BZFILE *) BZ2_bzdopen (fd, "wb"); |  | ||||||
| #endif /* HAVE_LIBBZ2 */ |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|       if (mp->fp == NULL) |  | ||||||
|         { |  | ||||||
|           if (config.merr) |  | ||||||
|             { |  | ||||||
|               fprintf (stderr, "%s: %s: ", APPNAME, path); |  | ||||||
|               perror (NULL); |  | ||||||
|             } |  | ||||||
|           errno = 0; |  | ||||||
|           close (fd); |  | ||||||
|           return NULL; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|   if (mode[0] == 'r') |  | ||||||
|     { |  | ||||||
|       if (config.format == FORMAT_MBOX) |  | ||||||
|         fgets (buffer, BUFSIZ, mp->fp); |  | ||||||
| #ifdef HAVE_LIBZ |  | ||||||
|       else if (config.format == FORMAT_ZMBOX) |  | ||||||
|         gzgets (mp->fp, buffer, BUFSIZ); |  | ||||||
| #endif /* HAVE_LIBZ */ |  | ||||||
| #ifdef HAVE_LIBBZ2 |  | ||||||
|       else if (config.format == FORMAT_BZ2MBOX) |  | ||||||
|         { |  | ||||||
|           char c[1] = "\0"; |  | ||||||
|           int n = 0; |  | ||||||
|  |  | ||||||
|           while (c[0] != '\n' && n < BUFSIZ) |  | ||||||
|             { |  | ||||||
|               BZ2_bzread (mp->fp, c, 1); |  | ||||||
|               buffer[n] = c[0]; |  | ||||||
|               n++; |  | ||||||
|             } |  | ||||||
|           buffer[n] = '\0'; |  | ||||||
|         } |  | ||||||
| #endif /* HAVE_LIBBZ2 */ |  | ||||||
|  |  | ||||||
|       if (0 != strncmp ("From ", buffer, 5)) |  | ||||||
|         { |  | ||||||
|           if (config.merr) |  | ||||||
|             { |  | ||||||
|               if (0 == strcmp ("-", path)) |  | ||||||
|                 fprintf (stderr, "%s: (standard input): Not a mbox folder\n", |  | ||||||
|                          APPNAME); |  | ||||||
|               else |  | ||||||
|                 fprintf (stderr, "%s: %s: Not a mbox folder\n", APPNAME, |  | ||||||
|                          path); |  | ||||||
|             } |  | ||||||
|           if (config.format == FORMAT_MBOX) |  | ||||||
|             fclose (mp->fp); |  | ||||||
| #ifdef HAVE_LIBZ |  | ||||||
|           else if (config.format == FORMAT_ZMBOX) |  | ||||||
|             gzclose (mp->fp); |  | ||||||
| #endif /* HAVE_LIBZ */ |  | ||||||
| #ifdef HAVE_LIBBZ2 |  | ||||||
|           else if (config.format == FORMAT_BZ2MBOX) |  | ||||||
|             BZ2_bzclose (mp->fp); |  | ||||||
| #endif /* HAVE_LIBBZ2 */ |  | ||||||
|           return NULL; |  | ||||||
|         } |  | ||||||
|       strcpy (mp->postmark_cache, buffer); |  | ||||||
|     } |  | ||||||
|   return mp; |   return mp; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -431,3 +312,127 @@ tmpfile_create (void) | |||||||
|     } |     } | ||||||
|   return fd; |   return fd; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void | ||||||
|  | mbox_lock (int fd, const char *path, const mbox_mode_t mbox_mode) | ||||||
|  | { | ||||||
|  | #ifdef HAVE_FLOCK | ||||||
|  |   int op; | ||||||
|  |  | ||||||
|  |   if (mbox_mode == r) | ||||||
|  |     op = LOCK_SH; | ||||||
|  |   else | ||||||
|  |     op = LOCK_EX; | ||||||
|  |   if (-1 == flock (fd, op)) | ||||||
|  | #else | ||||||
|  |   memset (&lck, 0, sizeof (struct flock)); | ||||||
|  |   lck.l_whence = SEEK_SET; | ||||||
|  |   if (mode[0] == 'r') | ||||||
|  |     lck.l_type = F_RDLCK; | ||||||
|  |   else | ||||||
|  |     lck.l_type = F_WRLCK; | ||||||
|  |  | ||||||
|  |   if (-1 == fcntl (fd, F_SETLK, &lck)) | ||||||
|  | #endif /* HAVE_FLOCK */ | ||||||
|  |     { | ||||||
|  |       if (config.merr) | ||||||
|  |         { | ||||||
|  |           fprintf (stderr, "%s: %s: ", APPNAME, path); | ||||||
|  |           perror (NULL); | ||||||
|  |           exit (2); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void * | ||||||
|  | mbox_fdopen (int fd, const char *path, const mbox_mode_t mbox_mode) | ||||||
|  | { | ||||||
|  |   void *f; | ||||||
|  |   char *file_mode; | ||||||
|  |  | ||||||
|  |   if (mbox_mode == w) | ||||||
|  |     file_mode = xstrdup("wb"); | ||||||
|  |   else | ||||||
|  |     file_mode = xstrdup("rb"); | ||||||
|  |  | ||||||
|  |   if (config.format == FORMAT_MBOX) | ||||||
|  |     f = (FILE *) m_fdopen (fd, file_mode); | ||||||
|  | #ifdef HAVE_LIBZ | ||||||
|  |   else if (config.format == FORMAT_ZMBOX) | ||||||
|  |     f = (gzFile *) m_gzdopen (fd, file_mode); | ||||||
|  | #endif /* HAVE_LIBZ */ | ||||||
|  | #ifdef HAVE_LIBBZ2 | ||||||
|  |   else if (config.format == FORMAT_BZ2MBOX) | ||||||
|  |     f = (BZFILE *) BZ2_bzdopen (fd, file_mode); | ||||||
|  | #endif /* HAVE_LIBBZ2 */ | ||||||
|  |  | ||||||
|  |   if (f == NULL) | ||||||
|  |     { | ||||||
|  |       if (config.merr) | ||||||
|  |         { | ||||||
|  |           fprintf (stderr, "%s: %s: ", APPNAME, path); | ||||||
|  |           perror (NULL); | ||||||
|  |         } | ||||||
|  |       errno = 0; | ||||||
|  |       close (fd); | ||||||
|  |       return NULL; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |   return f; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | char * | ||||||
|  | mbox_check_postmark(mbox_t *mp, const char *path) | ||||||
|  | { | ||||||
|  |   char *buffer; | ||||||
|  |  | ||||||
|  |   buffer = (char *) xmalloc (BUFSIZ * sizeof (char)); | ||||||
|  |   memset (buffer, 0, BUFSIZ); | ||||||
|  |  | ||||||
|  |   if (config.format == FORMAT_MBOX) | ||||||
|  |     fgets (buffer, BUFSIZ, mp->fp); | ||||||
|  | #ifdef HAVE_LIBZ | ||||||
|  |   else if (config.format == FORMAT_ZMBOX) | ||||||
|  |     gzgets (mp->fp, buffer, BUFSIZ); | ||||||
|  | #endif /* HAVE_LIBZ */ | ||||||
|  | #ifdef HAVE_LIBBZ2 | ||||||
|  |   else if (config.format == FORMAT_BZ2MBOX) | ||||||
|  |     { | ||||||
|  |       char c[1] = "\0"; | ||||||
|  |       int n = 0; | ||||||
|  |  | ||||||
|  |       while (c[0] != '\n' && n < BUFSIZ) | ||||||
|  |         { | ||||||
|  |           BZ2_bzread (mp->fp, c, 1); | ||||||
|  |           buffer[n] = c[0]; | ||||||
|  |           n++; | ||||||
|  |         } | ||||||
|  |       buffer[n] = '\0'; | ||||||
|  |     } | ||||||
|  | #endif /* HAVE_LIBBZ2 */ | ||||||
|  |  | ||||||
|  |   if (0 != strncmp ("From ", buffer, 5)) | ||||||
|  |     { | ||||||
|  |       if ((config.merr) && (buffer[0] != '\0')) | ||||||
|  |         { | ||||||
|  |           if (0 == strcmp ("-", path)) | ||||||
|  |             fprintf (stderr, "%s: (standard input): Not an mbox folder\n", | ||||||
|  |                      APPNAME); | ||||||
|  |           else | ||||||
|  |             fprintf (stderr, "%s: %s: Not an mbox folder\n", APPNAME, | ||||||
|  |                      path); | ||||||
|  |         } | ||||||
|  |       if (config.format == FORMAT_MBOX) | ||||||
|  |         fclose (mp->fp); | ||||||
|  | #ifdef HAVE_LIBZ | ||||||
|  |       else if (config.format == FORMAT_ZMBOX) | ||||||
|  |         gzclose (mp->fp); | ||||||
|  | #endif /* HAVE_LIBZ */ | ||||||
|  | #ifdef HAVE_LIBBZ2 | ||||||
|  |       else if (config.format == FORMAT_BZ2MBOX) | ||||||
|  |         BZ2_bzclose (mp->fp); | ||||||
|  | #endif /* HAVE_LIBBZ2 */ | ||||||
|  |       return NULL; | ||||||
|  |     } | ||||||
|  |   return buffer; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								src/mbox.h
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								src/mbox.h
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| /* | /* | ||||||
|    mboxgrep - scan mailbox for messages matching a regular expression |    mboxgrep - scan mailbox for messages matching a regular expression | ||||||
|    Copyright (C) 2000 - 2004, 2023  Daniel Spiljar |    Copyright (C) 2000 - 2004, 2023 - 2024 Daniel Spiljar | ||||||
|     |     | ||||||
|    Mboxgrep is free software; you can redistribute it and/or modify it  |    Mboxgrep is free software; you can redistribute it and/or modify it  | ||||||
|    under the terms of the GNU General Public License as published by |    under the terms of the GNU General Public License as published by | ||||||
| @@ -19,9 +19,7 @@ | |||||||
|  |  | ||||||
| #ifndef MBOX_H | #ifndef MBOX_H | ||||||
| #  define MBOX_H 1 | #  define MBOX_H 1 | ||||||
|  |  | ||||||
| #  include <config.h> | #  include <config.h> | ||||||
|  |  | ||||||
| #  include "message.h" | #  include "message.h" | ||||||
|  |  | ||||||
| typedef struct | typedef struct | ||||||
| @@ -31,8 +29,14 @@ typedef struct | |||||||
|   char *postmark_cache; |   char *postmark_cache; | ||||||
| } mbox_t; | } mbox_t; | ||||||
|  |  | ||||||
|  | typedef enum | ||||||
|  | { | ||||||
|  |   r, | ||||||
|  |   w, | ||||||
|  | } mbox_mode_t; | ||||||
|  |  | ||||||
| mbox_t *mbox_open (const char *path, const char *mode); |  | ||||||
|  | mbox_t *mbox_open (const char *path, const mbox_mode_t mbox_mode); | ||||||
| void tmpmbox_create (const char *path); | void tmpmbox_create (const char *path); | ||||||
| void tmpfile_name (const char *path); | void tmpfile_name (const char *path); | ||||||
| void tmpfile_mod_own (const int fd, const char *path); | void tmpfile_mod_own (const int fd, const char *path); | ||||||
| @@ -40,5 +44,8 @@ int tmpfile_create (void); | |||||||
| void mbox_close (mbox_t * mbp); | void mbox_close (mbox_t * mbp); | ||||||
| message_t *mbox_read_message (mbox_t * mp); | message_t *mbox_read_message (mbox_t * mp); | ||||||
| void mbox_write_message (message_t * m, mbox_t * mbox); | void mbox_write_message (message_t * m, mbox_t * mbox); | ||||||
|  | void mbox_lock (int fd, const char *path, const mbox_mode_t mbox_mode); | ||||||
|  | void *mbox_fdopen (int fd, const char *path, const mbox_mode_t mbox_mode); | ||||||
|  | char *mbox_check_postmark(mbox_t *mp, const char *path); | ||||||
|  |  | ||||||
| #endif /* MBOX_H */ | #endif /* MBOX_H */ | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| /* | /* | ||||||
|    mboxgrep - scan mailbox for messages matching a regular expression |    mboxgrep - scan mailbox for messages matching a regular expression | ||||||
|    Copyright (C) 2000 - 2004, 2006, 2023  Daniel Spiljar |    Copyright (C) 2000 - 2004, 2006, 2023 - 2024  Daniel Spiljar | ||||||
|     |     | ||||||
|    Mboxgrep is free software; you can redistribute it and/or modify it  |    Mboxgrep is free software; you can redistribute it and/or modify it  | ||||||
|    under the terms of the GNU General Public License as published by |    under the terms of the GNU General Public License as published by | ||||||
| @@ -107,7 +107,7 @@ scan_mailbox (char path[]) | |||||||
|   if ((config.format == FORMAT_MBOX) || (config.format == FORMAT_ZMBOX) |   if ((config.format == FORMAT_MBOX) || (config.format == FORMAT_ZMBOX) | ||||||
|       || (config.format == FORMAT_BZ2MBOX)) |       || (config.format == FORMAT_BZ2MBOX)) | ||||||
|     { |     { | ||||||
|       mbox = (mbox_t *) mbox_open (path, "r"); |       mbox = (mbox_t *) mbox_open (path, r); | ||||||
|       if (mbox == NULL) |       if (mbox == NULL) | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| @@ -179,7 +179,7 @@ scan_mailbox (char path[]) | |||||||
|               else if ((config.format == FORMAT_MBOX) || (config.format == FORMAT_ZMBOX) |               else if ((config.format == FORMAT_MBOX) || (config.format == FORMAT_ZMBOX) | ||||||
|                        || (config.format == FORMAT_BZ2MBOX)) |                        || (config.format == FORMAT_BZ2MBOX)) | ||||||
|                 { |                 { | ||||||
|                   out = mbox_open (config.outboxname, "w"); |                   out = mbox_open (config.outboxname, w); | ||||||
|                   /* fprintf (out->fp, "%s\n%s", msg->headers, msg->body); */ |                   /* fprintf (out->fp, "%s\n%s", msg->headers, msg->body); */ | ||||||
|                   mbox_write_message (msg, out); |                   mbox_write_message (msg, out); | ||||||
|                   mbox_close (out); |                   mbox_close (out); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user