#!/usr/bin/perl -wT
#
# TWiki WikiClone (see wiki.pm for $wikiversion and other info)
#
# Copyright (C) 1999-2000 Peter Thoeny, peter@thoeny.com
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details, published at 
# http://www.gnu.org/copyleft/gpl.html

use lib ( '.' );
use lib ( '../lib' );
use TWiki;
use TWiki::Net;

my $debug = ! ( @ARGV && $ARGV[0] eq "-q" );

&main();

sub main
{
    $debug && print "TWiki mail notification\n";
    $debug && print "- to suppress all normal output: mailnotify -q\n";

    my $dataDir = &TWiki::getDataDir();
    opendir( DIR, "$dataDir" ) or die "could not open $dataDir";
    @weblist = grep !/^\.\.?$/, readdir DIR;
    closedir DIR;
    foreach $web ( @weblist ) {
        if( -d "$dataDir/$web" ) {
             processWeb( $web );

             # remove obsolete .lock files
             &TWiki::Store::removeObsoleteTopicLocks( $web );
        }
    }
    $debug && print "End TWiki mail notification\n";
}

sub processWeb
{
    my( $web) = @_;

    my ( $topic, $webName, $dummy, $userName, $dataDir) = 
	&TWiki::initialize( "/$web", "nobody" );
    $dummy = "";  # to suppress warning

    $debug && print "Checking TWiki.$webName\n";

    if( ! &TWiki::Store::webExists( $webName ) ) {
	print STDERR "* ERROR: TWiki mailnotify does not find web $webName\n";
	return;
    }

    
    my %subscriptions = &TWiki::getEmailNotifySubscriptionList($webName);
    my %notifylist;

    my $emailbody = "";
    my $topiclist = "";

    my $text = &TWiki::Store::readTemplate( "changes" );
    my $changes= &TWiki::Store::readFile( "$dataDir/$webName/.changes" );

    my %exclude;

    $text = &TWiki::handleCommonTags($text, $topic);
    $text =~ s/<img src=.*?[^>]>/[IMG]/goi;  # remove all images
    my $before = "";
    my $after = "";
    ( $before, $text, $after) = split( /%REPEAT%/, $text );
    $emailbody = &TWiki::getRenderedVersion( $before );
    $after = &TWiki::getRenderedVersion( $after );

    my $prevLastmodify = &TWiki::Store::readFile( "$dataDir/$webName/.mailnotify" ) || "0";
    my $currLastmodify = "";
    my $scriptSuffix = $TWiki::scriptSuffix;
    my $scriptUrlPath = $TWiki::scriptUrlPath;
    my $scriptUrl = "$TWiki::urlHost$scriptUrlPath";
    my $frev = "";

    foreach( reverse split( /\n/, $changes ) ) {
	@bar = split( /\t/);
	$foo = $text;
        my $topicName = $bar[0];
	if( ( ! %exclude ) || ( ! $exclude{ $topicName } ) ) {
            next unless TWiki::Store::topicExists( $webName, $topicName );
	    
            if( ! $currLastmodify ) {
	        # newest entry
	        $time = &TWiki::formatGmTime( $prevLastmodify );
	        if( $prevLastmodify eq $bar[2] ) {
	            # newest entry is same as at time of previous notification
	            $debug && print "- Note: No topics changed since $time\n";
	            return;
	        }
	        $currLastmodify = $bar[2];
	        $debug && print "- Changed topics since $time: ";
	    }

	    if( $prevLastmodify >= $bar[2] ) {
	        #print "Date: found item of last notification\n";
	        # found item of last notification
	        last;
	    }

            # add one for every global subscriber
            for (keys %{ $subscriptions{'*'} })
            {
                $notifylist{$_} += 1;
                $debug && print "$_ = $notifylist{$_}\n";
            }   

            if (exists $subscriptions{$topicName})
            {
                my %topiclist = %{ $subscriptions{$topicName} };
                for (keys %topiclist)
                {
                    if ($topiclist{$_} eq '-')
                    {
                        --$notifylist{$_};
                    }
                    else
                    {
                        ++$notifylist{$_};
                    }
                }
            }

            $frev = "";
            if( $bar[3] ) {
                if( $bar[3] > 1 ) {
                    $frev = "r1.$bar[3]";
                } else {
                    $frev = "<b>NEW</b>";
                }
            }

	    #create entry in HTML attachment
	    $foo = $text;
	    $foo =~ s/%TOPICNAME%/$topicName/go;
	    $wikiuser = &TWiki::userToWikiName( $bar[1] );
	    $foo =~ s/%AUTHOR%/$wikiuser/go;
            $foo =~ s/%LOCKED%//go;
	    $time = &TWiki::formatGmTime( $bar[2] );
            $foo =~ s/%TIME%/$time/go;
            $foo =~ s/%REVISION%/$frev/go;
	    $foo = &TWiki::getRenderedVersion( $foo );

            $head = &TWiki::Store::readFileHead( "$dataDir\/$webName\/$topicName.txt", 16 );
            $head = &TWiki::makeTopicSummary( $head, $topicName, $webName );
            $foo =~ s/%TEXTHEAD%/$head/go;

	    $emailbody .= $foo;
	    $exclude{ $topicName } = "1";

	    $debug && print "$topicName, ";

	    #add new item to topic list in email body
	    $foo = "- $topicName  ($wikiuser)\n  $scriptUrl/view$scriptSuffix/$webName/$topicName\n";
            $foo =~ s/Main\.//go;
	    $topiclist = "$topiclist$foo";
	}
    }

    if( $topiclist eq "" ) {
       $debug && print "- Note: Topic list is empty\n";
       return;
    }
    $debug && print "\n";

    $emailbody .= $after;

    my $from = &TWiki::Prefs::getPreferencesValue("WIKIWEBMASTER");

    foreach (keys %notifylist)
    {
        delete $notifylist{$_} if ($notifylist{$_} < 1);
    }
    my @notifylist = sort keys %notifylist;
    my $error = 0;

    if (scalar @notifylist)
    {
        my $notifylist = join ', ', @notifylist;

        $text = &TWiki::Store::readTemplate( "mailnotify" );
        $text =~ s/%EMAILFROM%/$from/go;
        $text =~ s/%EMAILTO%/$notifylist/go;
        $text =~ s/%EMAILBODY%/$emailbody/go;
        $text =~ s/%TOPICLIST%/$topiclist/go;
        $text =~ s/%LASTDATE%/&TWiki::formatGmTime($prevLastmodify)/geo;
        $text = &TWiki::handleCommonTags( $text, $topic );

        # change absolute addresses to relative ones & do some cleanup
        $text =~ s/(href=\")$scriptUrlPath/$1..\/../goi;
        $text =~ s/(action=\")$scriptUrlPath/$1..\/../goi;
        $text =~ s|( ?) *</*nop/*>\n?|$1|gois;
 
        $debug && print "- Sending mail notification to: $notifylist\n";
 
        $error = &TWiki::Net::sendEmail( $text );
    }

    if( $error ) {
        print STDERR "* $error\n";
        $debug && print "- End Twiki.$webName\n";

    } else {
        &TWiki::Store::saveFile( "$dataDir/$webName/.mailnotify", $currLastmodify );
        $debug && print "- End Twiki.$webName, mail notification sent\n";
    }
}
