#! @im_path_perl@ ################################################################ ### ### imjoin ### ### Author: Internet Message Group ### Created: May 5, 1997 ### Revised: Apr 23, 2007 ### BEGIN { @im_my_siteperl@ @im_src_siteperl@ }; $Prog = 'imjoin'; my $VERSION_DATE = "20161010"; my $VERSION_NUMBER = "153"; my $VERSION = "${Prog} version ${VERSION_DATE}(IM${VERSION_NUMBER})"; my $VERSION_INFORMATION = "${Prog} (IM ${VERSION_NUMBER}) ${VERSION_DATE} Copyright (C) 1999 IM developing team This is free software; see the source for copying conditions. There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. "; ## ## Require packages ## use IM::MsgStore qw(store_message); use IM::Config; use IM::Util; use integer; use strict; use vars qw($Prog $EXPLANATION @EnvConfig @OptConfig $opt_noscan $opt_src $opt_dst $opt_verbose $opt_debug $opt_help $opt_version); ## ## Environments ## $EXPLANATION = "$VERSION join Message/partial messages Usage: imjoin [OPTIONS] MSGS "; @OptConfig = ( 'src;f;;' => "Source folder", 'dst;s;+inbox;' => "Destination folder", 'verbose;b;;' => 'With verbose messages', 'debug;d;;' => "With debug message", 'help;b;;' => "Display this help and exit", 'version,V;b;;' => "Output version information and exit", ); ## ## Profile and option processing ## init_opt(\@OptConfig); read_env(\@EnvConfig); read_cfg(); read_opt(\@ARGV); # help? print("${VERSION_INFORMATION}") && exit $EXIT_SUCCESS if $opt_version; help($EXPLANATION) && exit $EXIT_SUCCESS if $opt_help; debug_option($opt_debug) if $opt_debug; ## ## Main ## my @msgs = @ARGV; my $msg = $msgs[0]; my @Message = join_msg(@msgs); if ($opt_dst eq "stdout") { print join('', @Message); } else { store_message(\@Message, $opt_dst, 1); } exit $EXIT_SUCCESS; sub join_msg($) { my @msgs = @_; my @paths; my @index; my @Message; local $_; if ($#msgs == 0) { # search partial messages using history_db @paths = get_paths("$opt_src/$msgs[0]"); } else { # all partial message is specified by command line require IM::Folder && import IM::Folder qw(message_name); my $msg; foreach $msg (@msgs) { if ($msg =~ /^\//) { push(@paths, $msg); } elsif ($msg =~ /(.*)\/(\d+)/) { push(@paths, message_name($1, $2)); } else { push(@paths, message_name($opt_src, $msg)); } } } # sort each part number on each part my $total = 0; my($path, $header); foreach $path (@paths) { if (im_open(\*MSG, "<$path")) { $/ = "\n\n"; $header = ; $/ = "\n"; close(MSG); } $header =~ s/\n\s+//g; $header =~ s/[ \t]+//g; $header =~ s/\n/;\n/g; $header = "\n$header;\n"; my $number = 0; my $this_total = 0; if ($header =~ /\nContent-Type:Message\/partial(;[^\n]+)\n/i) { my $rest = $1; if ($rest =~ /;number=(\d+);/i) { $number = $1; } if ($rest =~ /;total=(\d+);/i) { $this_total = $1; } } if ($number == 0 || $this_total == 0) { im_warn("$_: not a partial message, skipping.\n"); next; } if ($total) { if ($total != $this_total) { im_warn("$_: total of partial messages mismatch, skipping.\n"); next; } } else { $total = $this_total; } $index[$number] = $path; im_notice("$path is part $number.\n"); } # check existance of all partial messages my $missing = 0; my $i; for ($i = 1; $i <= $#index; $i++) { if ($index[$i] eq '') { im_err("part $i is missing.\n"); exit $EXIT_SUCCESS; } } # show in sequence for ($i = 1; $i <= $#index; $i++) { if (im_open(\*MSG, "<$index[$i]")) { $/ = "\n\n"; if ($i == 1) { # first partial message my $header = ; # header of enclosing message my $skip = 0; foreach (split("\n", $header)) { next if (/^[ \t]/ && $skip); $skip = 0; if (/^(Content|Message-ID|Subject|Encrypted|MIME-Version)/i) { $skip = 1; next; } last if (/^$/); push(@Message, "$_\n"); } $header = ; $skip = 0; foreach (split("\n", $header)) { next if (/^[ \t]/ && $skip); $skip = 0; unless (/^(Content-|Subject|Message-ID|Encrypted|MIME-Version)/i || /^[ \t]/) { $skip = 1; next; } last if (/^$/); push(@Message, "$_\n"); } push(@Message, "\n"); } else { # skip header part ; } $/ = "\n"; while () { push(@Message, $_); } close(MSG); } } return @Message; } sub get_paths($) { my $msg = shift; my $path; local $_; unless (msgdbfile()) { im_die("need history database to join by one message.\n"); exit $EXIT_ERROR; } require IM::History; import IM::History qw(history_open history_lookup history_close); # get master Message-ID my $header = ''; if ($msg =~ /^\+/) { $path = &expand_path($msg); } else { $path = $msg; } if (im_open(\*MSG, "<$path")) { $/ = "\n\n"; $header = ; $/ = "\n"; close(MSG); } if ($header eq '') { im_err("specified message is not found at $path.\n"); exit $EXIT_ERROR; } $header =~ s/\n\s+//g; $header =~ s/[ \t]+//g; $header =~ s/\n/;\n/g; $header = "\n$header"; my $master = ''; if ($header =~ m|\nContent-Type:Message/partial;(.*;)?id=([^;]+);|i) { $master = $2; $master =~ s/^"(.*)"$/$1/; } else { im_err("specified message is not a partial.\n"); exit $EXIT_ERROR; } im_notice("Master Message-ID: $master.\n"); # get Message-IDs of partial if (history_open(0) < 0) { im_err("cannot open history.\n"); exit $EXIT_ERROR; } my $ids = history_lookup("partial:$master", 'LookUpMsg'); if ($ids eq '') { im_err("information on partial messages is not found in history.\n"); exit $EXIT_ERROR; } im_notice("partial Message-IDs: $ids.\n"); # get path and part number on each part my @paths; foreach (split(',', $ids)) { my $locate = history_lookup($_, 'LookUpMsg'); if ($locate eq '') { im_warn("message $_ not found, skipping.\n"); next; } my $path = &expand_path($locate); if ($path eq '') { im_warn("no path for message $locate, skipping.\n"); next; } push(@paths, $path); } history_close(); return @paths; } __END__ =head1 NAME imjoin - join Message/partial messages =head1 SYNOPSIS B [OPTIONS] MSGS =head1 DESCRIPTION The I command joins Message/partial messages. This command is provided by IM (Internet Message). =head1 OPTIONS =over 5 =item I<-s, --src=FOLDER> Source folder. =item I<-d, --dst=FOLDER> Destination folder. Default value is "+inbox". =item I<-v, --verbose={on,off}> Print verbose messages when running. =item I<--debug=DEBUG_OPTION> Print debug messages when running. =item I<-h, --help> Display help message and exit. =item I<--version> Output version information and exit. =back =head1 COPYRIGHT IM (Internet Message) is copyrighted by IM developing team. You can redistribute it and/or modify it under the modified BSD license. See the copyright file for more details. =cut ### Copyright (C) 1997, 1998, 1999 IM developing team ### All rights reserved. ### ### Redistribution and use in source and binary forms, with or without ### modification, are permitted provided that the following conditions ### are met: ### ### 1. Redistributions of source code must retain the above copyright ### notice, this list of conditions and the following disclaimer. ### 2. Redistributions in binary form must reproduce the above copyright ### notice, this list of conditions and the following disclaimer in the ### documentation and/or other materials provided with the distribution. ### 3. Neither the name of the team nor the names of its contributors ### may be used to endorse or promote products derived from this software ### without specific prior written permission. ### ### THIS SOFTWARE IS PROVIDED BY THE TEAM AND CONTRIBUTORS ``AS IS'' AND ### ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE ### IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR ### PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE TEAM OR CONTRIBUTORS BE ### LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR ### CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF ### SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR ### BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, ### WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE ### OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN ### IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ### Local Variables: ### mode: perl ### End: