Sophie

Sophie

distrib > Mandriva > 2010.1 > x86_64 > by-pkgid > 30a817b1a5cbc3b9a11942bda9a91aed > files > 2152

gosa-2.5.14-6mdv2010.1.noarch.rpm

#!/usr/bin/perl -w -I.
#
# This code is part of GOsa (https://gosa.gonicus.de)
# Copyright (C) 2007 Frank Moeller
#
# 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.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA

use strict;
use IMAP::Sieve;
use XML::Simple;
use Data::Dumper;
use Net::LDAP;
use utf8;
use Getopt::Std;
use vars qw/ %opt /;

#
# Definitions
#
my $gosa_config = "/etc/gosa/gosa.conf";
my $opt_string = 'l:h';
my $location = "";
my $today = time ();
# default mailMethod = kolab
my $server_attribute = "kolabHomeServer";
my $alternate_address_attribute = "alias";
my $gosa_sieve_script_name = "gosa";
my $simple_bind_dn = "";
my $simple_bind_dn_pwd = "";
my $gosa_sieve_script_status = "FALSE";
my $sieve_vacation = "";
my $gosa_sieve_spam_header = "Sort mails with higher spam level";

#
# Templates
#
my $gosa_sieve_header = "\#\#\#GOSA\nrequire\ \[\"fileinto\",\ \"reject\",\ \"vacation\"\]\;\n\n";
my $vacation_header_template = "\# Begin vacation message";
my $vacation_footer_template = "\# End vacation message";

#
# Usage
#
sub usage {
	die "Usage:\nperl $0 [option]\n\n\tOptions:\n\t\t-l <\"location name\">\tuse special location\n\t\t-h\t\t\tthis help \n";
}

#
# Config import
#
sub read_config {
	my $input = shift || die "need config file: $!";
	my $stream = "";
	open ( FILE, "< $input" ) or die "Error opening file $input: $! \n";
	{
	        local $/ = undef;
	        $stream = <FILE>;
	}
	close ( FILE );
	return $stream;
}

#
# XML parser
#
sub parseconfig {
	my $c_location = shift;
	my $xmldata = shift;
	chomp $c_location;
	chomp $xmldata;
	my $data = $xmldata;
	my $xml = new XML::Simple ();
	my $c_data = $xml -> XMLin( $xmldata);
	my $config = {};
	my $server;
	my $config_base;
	my $ldap_admin;
	my $ldap_admin_pwd;
	my $mailMethod;
	if ( ! ( ref($c_data->{main}->{location}->{server}) ) ) {
		$server = $c_data->{main}->{location}->{server};
		$config_base = $c_data->{main}->{location}->{config};
		$ldap_admin = $c_data->{main}->{location}->{referral}->{admin};
		$ldap_admin_pwd = $c_data->{main}->{location}->{referral}->{password};
		$mailMethod = $c_data->{main}->{location}->{mailMethod};
	} else {
		$server = $c_data->{main}->{location}->{$c_location}->{server};
		$config_base = $c_data->{main}->{location}->{$c_location}->{config};
		$ldap_admin = $c_data->{main}->{location}->{$c_location}->{referral}->{admin};
		$ldap_admin_pwd = $c_data->{main}->{location}->{$c_location}->{referral}->{password};
		$mailMethod = $c_data->{main}->{location}->{$c_location}->{mailMethod};
	}
	$config->{server} = $server;
	$config->{config_base} = $config_base;
	$config->{mailMethod} = $mailMethod;
	$config->{ldap_admin} = $ldap_admin;
	$config->{ldap_admin_pwd} = $ldap_admin_pwd;

	return $config;
}

#
# Get default location
#
sub get_default_location {
	my $xmldata = shift;
	my $xml = new XML::Simple ( RootName=>'conf' );
	my $c_data = $xml -> XMLin( $xmldata );
	my $default = $c_data->{main}->{default};

	return $default;
}

#
# LDAP error handling
#
sub ldap_error {
	my ($from, $mesg) = @_;
	print "Return code: ", $mesg->code;
	print "\tMessage: ", $mesg->error_name;
	print " :",          $mesg->error_text;
	print "MessageID: ", $mesg->mesg_id;
	print "\tDN: ", $mesg->dn;
}


#
# LDAP search
#
sub ldap_search {
	my $url = shift;
	my $searchString = shift;
	my $scope = shift;
	my $base = shift;
	my $attrs = shift;
	my $bind_dn = shift;
	my $bind_dn_pwd = shift;
	
	if ( $base eq "NULL" ) {
		$base = "";
	}
	my $ldap = Net::LDAP->new( $url ) or die "$@";
	if ( ( ! ( $bind_dn ) ) || ( ! ( $bind_dn_pwd ) ) ) {
		$ldap->bind;
	} else {
		$ldap->bind ( $bind_dn, password => $bind_dn_pwd );
	}

	my $result = $ldap->search (    base    => "$base",
					scope   => "$scope",
					filter  => "$searchString",
					attrs   =>  $attrs
					);
	if ( $result->code ) {
		ldap_error ( "Searching", $result );
	}

	$ldap->unbind;
	
	return $result;
}

#
# Retrieve LDAP base
#
sub get_ldap_base {
	my $url = shift;
	my $config_base = shift;
	my $bind_dn = shift;
	my $bind_dn_pwd = shift;
	my $filter = "(objectClass=*)";
	my $init_base = "NULL";
	my $scope = "base";
	my $attributes = [ 'namingcontexts' ];
	my $entry = {};
	my $base = "";

	$config_base =~ s/\,\ +/\,/g;
	#print $url."\n";
	#print $config_base."\n";
	my $result = ldap_search ( $url, $filter, $scope, $init_base, $attributes, $bind_dn, $bind_dn_pwd );
	my @entries = $result->entries;
	my $noe = @entries;
	#print $noe."\n";
	foreach $entry ( @entries ) {
		my $tmp = $entry->get_value ( 'namingcontexts' );
		#print $tmp."\n";
		$tmp =~ s/\,\ +/\,/g;
		if ( $config_base =~ m/$tmp/ ) {
			$base = $entry->get_value ( 'namingcontexts' );
		}
	}

	return $base;
}

#
# SIEVE functions
#
sub opensieve {
	my $admin = shift;
	my $pass = shift;
	my $user = shift;
	my $server = shift;
	my $port = shift;

	#print ( "##### Proxy => $user, Server => $server, Login => $admin, Password => $pass, Port => $port ####\n" );

	my $sieve = IMAP::Sieve->new ( 'Proxy' => $user, 'Server' => $server, 'Login' => $admin, 'Password' => $pass, 'Port' => $port );
	return $sieve;
}

sub closesieve {
	my $sieve = shift;

	if ($sieve) {$sieve->close};
}

sub listscripts {
	my $sieve = shift;

	my @scripts = $sieve->listscripts;
	my $script_list = join("\n",@scripts)."\n";
	return $script_list;
}

sub getscript {
	my $sieve = shift;
	my $script = shift;
	my $scriptfile;
	chomp $script;
	#print "$sieve\n";
	#print "$script\n";

	$scriptfile = $sieve->getscript($script);
	return $scriptfile;
}

sub putscript {
	my $sieve = shift;
	my $scriptname = shift;
	my $script = shift;
	#print "$sieve\n";
	#print "$scriptname\n";
	#print "$script\n";

	my $res=$sieve->putscript($scriptname,$script);
	if ($res) {print $sieve->{'Error'}}
	return;
}

sub setactive {
	my $sieve = shift;
	my $script = shift;

	my $res=$sieve->setactive($script);
	if ($res) { print $sieve->{'Error'};}
	return;
}

#
# main ()
#
# read options
getopts( "$opt_string", \%opt );

# read GOsa config
my $input_stream = read_config ( $gosa_config );

# get location
if ( $opt{l} ) {
	$location = $opt{l};
} elsif ( $opt{h} ) {
	usage ();
	exit (0);
} else {
	$location = get_default_location ( $input_stream );
}
#print "$location\n";

# parse config
my $config = parseconfig ( $location, $input_stream );
my $ldap_url = $config->{server};
my $gosa_config_base = $config->{config_base};
my $bind_dn = $config->{ldap_admin};
my $bind_dn_pwd = $config->{ldap_admin_pwd};
my $mailMethod = $config->{mailMethod};
utf8::encode($ldap_url);
utf8::encode($gosa_config_base);
utf8::encode($mailMethod);

if ( $mailMethod =~ m/cyrus/i ) {
	my $server_attribute = "gosaMailServer";
	my $alternate_address_attribute = "gosaMailAlternateAddress";
}

# determine LDAP base
my $ldap_base = get_ldap_base ( $ldap_url, $gosa_config_base, $simple_bind_dn, $simple_bind_dn_pwd );

# retrieve user informations with activated vacation feature
my $filter = "(&(objectClass=gosaMailAccount)(gosaMailDeliveryMode=*V*)(!(gosaMailDeliveryMode=*C*)))";
my $list_of_attributes = [ 'uid', 'mail', $alternate_address_attribute, 'gosaVacationMessage', 'gosaVacationStart', 'gosaVacationStop', $server_attribute ];
my $search_scope = "sub";
my $result = ldap_search ( $ldap_url, $filter, $search_scope, $ldap_base, $list_of_attributes, $simple_bind_dn, $simple_bind_dn_pwd );

my @entries = $result->entries;
my $entry = {};
foreach $entry ( @entries ) {
	my $uid_v = $entry->get_value ( 'uid' );
	my $mail_v = $entry->get_value ( 'mail' );
	my @mailalternate = $entry->get_value ( $alternate_address_attribute );
	my $vacation = $entry->get_value ( 'gosaVacationMessage' );
	my $start_v = $entry->get_value ( 'gosaVacationStart' );
	my $stop_v = $entry->get_value ( 'gosaVacationStop' );
	my $server_v = $entry->get_value ( $server_attribute );
	if ( ! ( $uid_v ) ) {
		$uid_v = "";
	}
	if ( ! ( $mail_v ) ) {
		$mail_v = "";
	}
	my @mailAddress = ($mail_v);
	my $alias = "";
	foreach $alias ( @mailalternate ) {
		push @mailAddress, $alias;
	}
	my $addresses = "";
	foreach $alias ( @mailAddress ) {
		$addresses .= "\"" . $alias . "\", ";
	}
	$addresses =~ s/\ *$//;
	$addresses =~ s/\,$//;
	if ( ! ( $vacation ) ) {
		$vacation = "";
	}
	if ( ! ( $start_v ) ) {
		$start_v = 0;
	}
	if ( ! ( $stop_v ) ) {
		$stop_v = 0;
	}
	if ( ! ( $server_v ) ) {
		$server_v = "";
		next;
	}
	#print $uid_v . " | " .
	#	$addresses . " | " .
	#	"\n";

	my ($sieve_user, $tmp) = split ( /\@/, $mail_v );

	if ( ( $today >= $start_v ) && ( $today < $stop_v ) ) {
		print "activating vacation for user $uid_v\n";

		my $srv_filter = "(&(goImapName=$server_v)(objectClass=goImapServer))";
		my $srv_list_of_attributes = [ 'goImapSieveServer', 'goImapSievePort', 'goImapAdmin', 'goImapPassword' ];
		my $srv_result = ldap_search ( $ldap_url, $srv_filter, $search_scope, $ldap_base, $srv_list_of_attributes, $bind_dn, $bind_dn_pwd );
		my @srv_entries = $srv_result->entries;
		my $srv_entry = {};
		my $noe = @srv_entries;
		if ( $noe == 0 ) {
			printf STDERR "Error: no $server_attribute defined! Aboarting...";
		} elsif ( $noe > 1 ) {
			printf STDERR "Error: multiple $server_attribute defined! Aboarting...";
		} else {
			my $goImapSieveServer = $srv_entries[0]->get_value ( 'goImapSieveServer' );
			my $goImapSievePort = $srv_entries[0]->get_value ( 'goImapSievePort' );
			my $goImapAdmin = $srv_entries[0]->get_value ( 'goImapAdmin' );
			my $goImapPassword = $srv_entries[0]->get_value ( 'goImapPassword' );
			if ( ( $goImapSieveServer ) && ( $goImapSievePort ) && ( $goImapAdmin ) && ( $goImapPassword ) ) {
				my $sieve = opensieve ( $goImapAdmin, $goImapPassword, $sieve_user, $goImapSieveServer, $goImapSievePort);
				my @sieve_scripts = listscripts ( $sieve );
				my $script_name = "";
				if ( @sieve_scripts ) {
					foreach $script_name ( @sieve_scripts ) {
						if ( $script_name =~ m/$gosa_sieve_script_name/ ) {
							$gosa_sieve_script_status = "TRUE";
						}
					}
					if ( $gosa_sieve_script_status eq "TRUE" ) {
						print "retrieving and modifying gosa sieve script\n";
						# requirements
						my $sieve_script = getscript( $sieve, $gosa_sieve_script_name );
						#print "$sieve_script\n";
						if ( ! ( $sieve_script ) ) {
							print "No Sieve Script! Creating New One!\n";
							$sieve_script = $gosa_sieve_header;
						}
						if ( $sieve_script =~ m/require.*\[.*["|'] *vacation *["|'].*\]/ ) {
							print "require vacation ok\n";
						} else {
							print "require vacation not ok\n";
							print "modifying require statement\n";
							$sieve_script =~ s/require(.*\[.*)\]/require$1\, "vacation"\]/;
						}
						if ( ! ( $sieve_script =~ m/$vacation_header_template/ ) ) {
							print "no match header template\n";
							$sieve_vacation = $vacation_header_template .
										"\n" .
										"vacation :addresses [$addresses]\n" .
										"\"" .
										$vacation . 
										"\n\"\;" .
										"\n" .
										$vacation_footer_template .
										"\n\n";
						}
						#print ( "$sieve_vacation\n" );
						#print ( "$sieve_script\n" );
						# including vacation message
						if ( $sieve_script =~ m/$gosa_sieve_spam_header/ ) {
							#print "MATCH\n";
							$sieve_script =~ s/($gosa_sieve_spam_header[^{}]*{[^{}]*})/$1\n\n$sieve_vacation/;
						} else {
							$sieve_script =~ s/require(.*\[.*\])/require$1\n\n$sieve_vacation/;
						}
						#print ( "START SIEVE $sieve_script\nSTOP SIEVE" );
						# uploading new sieve script
						putscript( $sieve, $gosa_sieve_script_name, $sieve_script );
						# activating new sieve script
						setactive( $sieve, $gosa_sieve_script_name );
					} else {
						print "no gosa script available, creating new one";
						my $sieve_script = $gosa_sieve_header . "\n\n" . $sieve_vacation;
						# uploading new sieve script
						putscript( $sieve, $gosa_sieve_script_name, $sieve_script );
						# activating new sieve script
						setactive( $sieve, $gosa_sieve_script_name );
					}
				}
			}
		}
	} elsif ( $today >= $stop_v ) {
		print "deactivating vacation for user $uid_v\n";

		my $srv_filter = "(&(goImapName=$server_v)(objectClass=goImapServer))";
		my $srv_list_of_attributes = [ 'goImapSieveServer', 'goImapSievePort', 'goImapAdmin', 'goImapPassword' ];
		my $srv_result = ldap_search ( $ldap_url, $srv_filter, $search_scope, $ldap_base, $srv_list_of_attributes, $bind_dn, $bind_dn_pwd );
		my @srv_entries = $srv_result->entries;
		my $srv_entry = {};
		my $noe = @srv_entries;
		if ( $noe == 0 ) {
			printf STDERR "Error: no $server_attribute defined! Aboarting...";
		} elsif ( $noe > 1 ) {
			printf STDERR "Error: multiple $server_attribute defined! Aboarting...";
		} else {
			my $goImapSieveServer = $srv_entries[0]->get_value ( 'goImapSieveServer' );
			my $goImapSievePort = $srv_entries[0]->get_value ( 'goImapSievePort' );
			my $goImapAdmin = $srv_entries[0]->get_value ( 'goImapAdmin' );
			my $goImapPassword = $srv_entries[0]->get_value ( 'goImapPassword' );
			if ( ( $goImapSieveServer ) && ( $goImapSievePort ) && ( $goImapAdmin ) && ( $goImapPassword ) ) {
				my $sieve = opensieve ( $goImapAdmin, $goImapPassword, $sieve_user, $goImapSieveServer, $goImapSievePort);
				my @sieve_scripts = listscripts ( $sieve );
				my $script_name = "";
				if ( @sieve_scripts ) {
					foreach $script_name ( @sieve_scripts ) {
						if ( $script_name =~ m/$gosa_sieve_script_name/ ) {
							$gosa_sieve_script_status = "TRUE";
						}
					}
					if ( $gosa_sieve_script_status eq "TRUE" ) {
						# removing vacation part
						my $sieve_script = getscript( $sieve, $gosa_sieve_script_name );
						if ( $sieve_script ) {
							#print "OLD SIEVE SCRIPT:\n$sieve_script\n\n";
							$sieve_script =~ s/$vacation_header_template[^#]*$vacation_footer_template//;
							#print "NEW SIEVE SCRIPT:\n$sieve_script\n\n";
							# uploading new sieve script
							putscript( $sieve, $gosa_sieve_script_name, $sieve_script );
							# activating new sieve script
							setactive( $sieve, $gosa_sieve_script_name );
						}
					}
				}
			}
		}
	} else {
		print "no vacation process necessary\n";
	}
}