Accéder au contenu.
Menu Sympa

fr - create_list.conf -> default hidden ?

Objet : Pour les administrateurs de serveurs de listes utilisant le logiciel Sympa

Archives de la liste

Chronologique Discussions  
  • From: adresse@cachée
  • To: adresse@cachée
  • Subject: create_list.conf -> default hidden ?
  • Date: Fri, 29 Jun 2001 16:41:15 +0200

Bonjour,

Une humble remarque qui, je l'espère, pour être utile à d'autre :

Je n'ai pas réussi à obtenir le comportement suivant :
1- tous les create_list_templates sont HIDDEN
2- j'en autorise certains

ce qui donnerait par exemple :

* hidden
#default hidden ?
discussion_list READ
private_working_group READ


Dans tools.pl, on a :

foreach my $template ( sort grep (!/^\./,readdir(DIR))) {

next if ($list_conf->{$template} eq 'hidden') ;
next if ($list_conf->{'default'} eq 'hidden') ;


Or il faudrait plutôt :

foreach my $template ( sort grep (!/^\./,readdir(DIR))) {

next if ($list_conf->{$template} eq 'hidden') ;
next if (($list_conf->{'default'} eq 'hidden')
&& ($list_conf->{$template} ne 'read')) ;

Ça fonctionne.

qq remarques :
- Préciser dans le fichier ou la doc que create_list.conf est case-sensitive
OU upcaser les values ?
- "* read" n'a pas d'incidence (ignoré), alors autant le supprimer de la
distrib.

Ci-joint tools.pl avec modif.

Cordialement

Foudil BRETEL
## This module is part of ML and provides some tools

package tools;

use POSIX;
use Mail::Internet;
use Mail::Header;
use Conf;
use Language;
use Log;

## RCS identification.
#my $id = '@(#)$Id: tools.pl,v 1.18 2001/04/04 11:18:39 salaun Exp $';

## global var to store a CipherSaber object
my $cipher;

## Sorts the list of adresses by domain name
## Input : users hash
## Sort by domain.
sub sortbydomain {
my($x, $y) = @_;
$x = join('.', reverse(split(/[@\.]/, $x)));
$y = join('.', reverse(split(/[@\.]/, $y)));
#print "$x $y\n";
$x cmp $y;
}

## Safefork does several tries before it gives up.
## Do 3 trials and wait 10 seconds between each.
## Exit with a fatal error is fork failed after all
## tests have been exhausted.
sub safefork {
my($i, $pid);

for ($i = 1; $i < 360; $i++) {
my($pid) = fork;
return $pid if (defined($pid));
do_log ('warning', "Can't create new process in safefork: %m");
## should send a mail to the listmaster
sleep(10 * $i);
}
fatal_err("Can't create new process in safefork: %m");
## No return.
}

@avoid_hdr = (
'help',
'ind(ex)?',
'lists?',

'(please\s+)?(add|unsub?(scribe)?|remove|del|sub\s|sub?s?cribe|sign?o?f?f?)',
'rev(iew)?\s+\S+',
'stats\s+\S+',
'get\s+\S+\s+\S+',
'set\s+\S+\s+(no)?(mail|conceal|digest)',
'purge\s+\S+\s+\S+',
'exp(ire)?\s+\S+\s+\S+\s+\S+',
'exp(ire)?del\s+\S+',
'exp(ire)?ind(ex)?\s+\S+',
'ind(ex)?exp(ire)?\s+\S+',
'mod(eration)?ind(ex)?\s+\S+',
'ind(ex)?mod(eration)?\s+\S+',
'rev(iew)?\s+\S+',
'dis(tribute)?\s+\S+\s+\S+',
'rej(ect)?\s+\S+\s+\S+',
'(re)?con(firm)?\s+\S+',
'rev(iew)?\s+\S+',
);

## Check for commands in the body of the message. Returns true
## if there are some commands in it.
sub checkcommand {
my($msg, $sender) = @_;
do_log('debug2', 'tools::checkcommand(msg->head->get(subject):
%s,%s)',$msg->head->get('Subject'), $sender);

my($avoid, $i);

return 0 if ($#{$msg->body} >= 15); ## More than 15 lines in the text.
my $hdr = $msg->head;

## Check for commands in the subject.
my $subject = $msg->head->get('Subject');
if ($subject) {
foreach $avoid (@avoid_hdr) {
if ($subject =~ /^\s*(quiet)?($avoid)(\s+|$)/im) {
&rejectMessage($msg, $sender);
return 1;
}
}
}
foreach $i (@{$msg->body}) {
foreach $avoid (@avoid_hdr) {
if ($i =~ /^\s*(quiet)?($avoid)(\s+|$)/im) { ## Suspicious line
&rejectMessage($msg, $sender);
return 1;
}
}
## Control is only applied to first non-blank line
last unless $i =~ /^\s*$/;

}
return 0;
}

sub rejectMessage {
my($msg, $sender) = @_;
do_log('debug2', 'tools::rejectMessage(%s)', $sender);

*REJ = smtp::smtpto($Conf{'request'}, \$sender);
print REJ "To: $sender\n";
print REJ "Subject: [sympa] " . Msg(5, 2, "Misadressed message ?") . "\n";
printf REJ "MIME-Version: %s\n", Msg(12, 1, '1.0');
printf REJ "Content-Type: text/plain; charset=%s\n", Msg(12, 2,
'us-ascii');
printf REJ "Content-Transfer-Encoding: %s\n", Msg(12, 3, '7bit');
print REJ "\n";
printf REJ Msg(5, 3, "\
Your message has been sent to a list but it seems it contains commands like
subscribe, signoff, help, index, get, ...

If your message did really contain a command, please note that such messages
must be sent to %s only.

If it happens that your message was by mistake considered as containing
commands, then please contact the manager of this service %s
so that he can take care of your problem.

Thank you for your attention.

------ Beginning of the suspect message --------
"), "$Conf{'sympa'}", $Conf{'request'};
$msg->print(\*REJ);
print REJ Msg(5, 4, "------- Fin message suspect ---------\n");
close(REJ);
}

## return a hash from the edit_list_conf file
sub load_edit_list_conf {
my $file;
my $conf ;

if (-r "$Conf{'etc'}/edit_list.conf") {
$file = "$Conf{'etc'}/edit_list.conf";
}elsif (-r "/local/sympa/bin/etc/edit_list.conf") {
$file = "/local/sympa/bin/etc/edit_list.conf";
}else {
&do_log('info','Cannot find edit_list.conf');
return undef;
}

unless (open (FILE, $file)) {
&do_log('info','Unable to open config file %s', $file);
return undef;
}

while (<FILE>) {
next if /^\s*(\#.*|\s*)$/;

if
(/^\s*(\S+)\s+(listmaster|privileged_owner|owner|editor|subscriber|default)\s+(read|write|hidden)\s*$/i)
{
$conf->{$1}{$2} = $3;
}else{
&do_log ('info', 'unknown parameter in %s (Ignored) %s',
"$Conf{'etc'}/edit_list.conf",$_ );
next;
}
}

close FILE;
return $conf;
}


## return a hash from the edit_list_conf file
sub load_create_list_conf {

my $file;
my $conf ;

if (-r "$Conf{'etc'}/create_list.conf") {
$file = "$Conf{'etc'}/create_list.conf";
}elsif (-r "/local/sympa/bin/etc/create_list.conf") {
$file = "/local/sympa/bin/etc/create_list.conf";
}else {
&do_log('info','unable to read
/local/sympa/bin/etc/create_list.conf');
return undef;
}

unless (open (FILE, $file)) {
&do_log('info','Unable to open config file %s', $file);
return undef;
}

while (<FILE>) {
next if /^\s*(\#.*|\s*)$/;

if (/^\s*(\S+)\s+(read|hidden)\s*$/i) {
$conf->{$1} = uc $2;
}else{
&do_log ('info', 'unknown parameter in %s (Ignored) %s',
"$Conf{'etc'}/create_list.conf",$_ );
next;
}
}

close FILE;
return $conf;
}

## Loads the list of topics
sub load_topics_conf {
do_log('debug2', 'tools::load_topics_conf');

my $conf_file = "$Conf{'etc'}/topics.conf";
my $topics = {};

unless (-r $conf_file) {
&do_log('info',"Unable to read $conf_file");
return undef;
}

unless (open (FILE, $conf_file)) {
&do_log('info',"Unable to open config file $conf_file");
return undef;
}

my $index;
while (<FILE>) {
if (/^([\w\/]+)\s+(.+)\s*$/) {
my @tree = split '/', $1;
$index++;

if ($#tree == 0) {
$topics->{$tree[0]}{'title'} = $2;
$topics->{$tree[0]}{'order'} = $index;
}else {
my $subtopic = join ('/', @tree[1..$#tree]);
$topics->{$tree[0]}{'sub'}{$subtopic} =
&_add_topic($subtopic,$2);
}
}
}
close FILE;

return $topics;
}

sub _add_topic {
my ($name, $title) = @_;
my $topic = {};

my @tree = split '/', $name;
if ($#tree == 0) {
return {'title' => $title};
}else {
$topic->{'sub'}{$name} = &_add_topic(join ('/', @tree[1..$#tree]),
$title);
return $topic;
}
}

sub get_list_list_tpl {
my $list_conf;
my $list_templates ;
unless ($list_conf = &load_create_list_conf()) {
return undef;
}

foreach my $dir ('/local/sympa/bin/etc/create_list_templates',
"$Conf{'etc'}/create_list_templates") {
if (opendir(DIR, $dir)) {
foreach my $template ( sort grep (!/^\./,readdir(DIR))) {

next if ($list_conf->{$template} eq 'HIDDEN') ;
next if (($list_conf->{'default'} eq 'HIDDEN')
&& ($list_conf->{$template} ne 'READ')) ;

$list_templates->{$template}{'path'} = $dir;

if (-r $dir.'/'.$template.'/comment') {
$list_templates->{$template}{'comment'} =
$dir.'/'.$template.'/comment';
}
}
closedir(DIR);
}
}

return ($list_templates);
}

# input object msg and listname, output signed message object
sub smime_sign {
my $in_msg = shift;
my $list = shift;

do_log('debug2', 'tools::smime_sign (%s,%s)',$in_msg,$list);

my $cert = "$Conf{'home'}/$list/cert.pem";
my $key = "$Conf{'home'}/$list/private_key";
my $temporary_file = $Conf{'tmpdir'}."/".$list.".".$$ ;

my $signed_msg,$pass_option ;
$pass_option = "-passin file:$Conf{'tmpdir'}/pass.$$" if
($Conf{'key_passwd'} ne '') ;

## dump the incomming message.
if (!open(MSGDUMP,"> $temporary_file")) {
&do_log('info', 'Can\'t store message in file %s', $temporary_file);
return undef;
}
$in_msg->print(\*MSGDUMP);
close(MSGDUMP);

unless (open (NEWMSG,"$Conf{'openssl'} smime -sign -signer $cert
$pass_option -inkey $key -in $temporary_file 2>&1 |")) {
&do_log('notice', 'Cannot sign message');
}
if ($Conf{'key_passwd'} ne '') {
unless ( &POSIX::mkfifo("$Conf{'tmpdir'}/pass.$$",0600)) {
do_log('notice', 'Unable to make fifo for
%s/pass.%s',$Conf{'tmpdir'},$$);
}
unless (open (FIFO,"> $Conf{'tmpdir'}/pass.$$")) {
do_log('notice', 'Unable to open fifo for
%s/pass.%s',$Conf{'tmpdir'},$$);
}

print FIFO $Conf{'key_passwd'};
close FIFO;
unlink ("$Conf{'tmpdir'}/pass.$$");
}

my $parser = new MIME::Parser;

$parser->output_to_core(1);
unless ($signed_msg = $parser->read(\*NEWMSG)) {
do_log('notice', 'Unable to parse message');
return undef;
}
close NEWMSG ;

unlink ($temporary_file) unless ($main::options{'debug'} ||
$main::options{'debug2'}) ;

## foreach header defined in the incomming message but undefined in the
## crypted message, add this header in the crypted form.
my $predefined_headers ;
foreach my $header ($signed_msg->head->tags) {
$predefined_headers->{$header} = 1 if
($signed_msg->head->get($header)) ;
}
foreach my $header ($in_msg->head->tags) {
$signed_msg->head->add($header,$in_msg->head->get($header)) unless
$predefined_headers->{$header} ;
}
return $signed_msg;
}


sub smime_sign_check {

my $msg = shift;
my $sender = shift;
my $file = shift;
$sender= lc($sender);

do_log('debug2', 'tools::smime_sign_check (message, %s, %s)', $sender,
$file);

my $is_signed = {};
$is_signed->{'body'} = undef;
$is_signed->{'subject'} = undef;

my $verify ;

## first step is the msg signing OK ; /tmp/sympa-smime.$$ is created
## to store the signer certificat for step two. I known, that's durty.



my $temporary_file = "/tmp/smime-sender.".$$ ;
do_log('debug2', "xxx $Conf{'openssl'} smime -verify
$Conf{'trusted_ca_options'} -signer $temporary_file");
unless (open (MSGDUMP, "| $Conf{'openssl'} smime -verify
$Conf{'trusted_ca_options'} -signer $temporary_file > /dev/null")) {

do_log('err', "unable to verify smime signature from $sender
$verify");
return undef ;
}

unless (open MSG, $file) {
do_log('err', 'Unable to open file %s: %s', $file, $!);
return undef;
}

print MSGDUMP <MSG>;
close MSGDUMP; close MSG;

## second step is the message signer match the sender
## a better analyse should be performed to extract the signer email.
my $signer = `cat $temporary_file | $Conf{'openssl'} x509 -subject
-noout`;


if ($signer =~ /email=$sender/i) {
do_log('debug', "S/MIME signed message, signature checked and sender
match signer(%s)",$signer);
## store the signer certificat
unless (-d $Conf{'ssl_cert_dir'}) {
if ( mkdir ($Conf{'ssl_cert_dir'}, 0775)) {
do_log('info', "creating spool $Conf{'ssl_cert_dir'}");
}else{
do_log('err', "Unable to create user certificat directory
$Conf{'ssl_cert_dir'}");
}
}
my $filename = "$Conf{'ssl_cert_dir'}/".&escape_chars($sender);

open (CERTIF,$temporary_file);
if (open (USERCERTIF, "> $filename")) {
print USERCERTIF <CERTIF> ;
close USERCERTIF ;
}else{
&do_log('err','Unable to rename %s %s',$temporary_file,$filename);
}
close CERTIF;
unlink($temporary_file) unless ($main::options{'debug'} ||
$main::options{'debug2'}) ;

$is_signed->{'body'} = 'smime';

# futur version should check if the subject was part of the SMIME
signature.
$is_signed->{'subject'} = undef;
return $is_signed;
}else{
unlink($temporary_file) unless ($main::options{'debug'} ||
$main::options{'debug2'}) ;
do_log('notice', "S/MIME signed message, sender($sender) do NOT match
signer($signer)",$sender,$signer);
return undef;
}
return undef ;
}

# input : msg object, return a new message object encrypted
sub smime_encrypt {
my $msg_header = shift;
my $msg_body = shift;
my $email = shift ;
my $list = shift ;

my $usercert;


&do_log('debug2', 'tools::smime_encrypt message msg from %s for %s
%s',$msg->head->get('from'),$list, $email);
if ($list eq 'list') {
$usercert = "$Conf{'home'}/$email/cert.pem";
}else{
$usercert = "$Conf{'ssl_cert_dir'}/".&tools::escape_chars($email);
}
if (-r $usercert) {
unless (open (MSGDUMP , "> $Conf{'tmpdir'}/MSG.$$")) {
&do_log('err', 'unable to open %s/MSG.%s',$Conf{'tmpdir'},$$);
return undef;
}
my $temporary_file = $Conf{'tmpdir'}."/".$email.".".$$ ;

## encrypt the incomming message parse it.
do_log ('debug2', "xxxx $Conf{'openssl'} smime -encrypt -out
$temporary_file -des3 $usercert");
if (!open(MSGDUMP, "| $Conf{'openssl'} smime -encrypt -out
$temporary_file -des3 $usercert")) {
&do_log('info', 'Can\'t encrypt message for recipient %s',
$email);
}
$msg->print(\*MSGDUMP);
close(MSGDUMP);

my $cryptedmsg;

## Get as MIME object
open (NEWMSG, $temporary_file);
my $parser = new MIME::Parser;
$parser->output_to_core(1);
unless ($cryptedmsg = $parser->read(\*NEWMSG)) {
do_log('notice', 'Unable to parse message');
return undef;
}
close NEWMSG ;

## Get body
open (NEWMSG, $temporary_file);
my $encypted_body;
my $in_header = 1 ;
while (<NEWMSG>) {
if ( !$in_header) {
$encrypted_body .= $_;
}else {
$in_header = 0 if (/^$/);
}
}
close NEWMSG;

unlink ($temporary_file) unless ($main::options{'debug'} ||
$main::options{'debug2'}) ;

## foreach header defined in the incomming message but undefined in
the
## crypted message, add this header in the crypted form.
my $predefined_headers ;
foreach my $header ($cryptedmsg->head->tags) {
$predefined_headers->{$header} = 1
if ($cryptedmsg->head->get($header)) ;
}
foreach my $header ($msg_header->tags) {
$cryptedmsg->head->add($header,$msg_header->get($header))
unless $predefined_headers->{$header} ;
}

}else{
do_log ('notice','unable to encrypt message to %s (missing certificat
%s)',$email,$usercert);
return undef;
}

return $cryptedmsg->head->as_string . "\n" . $encrypted_body;
}

# input : msg object for a list, return a new message object decrypted
sub smime_decrypt {
my $msg = shift;
my $list = shift ; ## the recipient of the msg


&do_log('debug2', 'tools::smime_decrypt message msg from
%s,%s',$msg->head->get('from'),$list);

my $certfile = "$Conf{'home'}/$list/cert.pem" ;
unless (-r $certfile){
do_log('err', "unable to decrypt message : cert missing $certfile");
return undef;
}
my $keyfile = "$Conf{'home'}/$list/private_key";

unless (open (MSGDUMP , "> $Conf{'tmpdir'}/MSG.$$")) {
&do_log('err', 'unable to open %s/MSG.%s',$Conf{'tmpdir'},$$);
return undef;
}
my $temporary_file = $Conf{'tmpdir'}."/".$list.".".$$ ;

## dump the incomming message.
if (!open(MSGDUMP,"> $temporary_file")) {
&do_log('info', 'Can\'t store message in file %s',$temporary_file);
}
$msg->print(\*MSGDUMP);
close(MSGDUMP);


my $decryptedmsg,$pass_option;
if ($Conf{'key_passwd'} ne '') {
# if password is define in sympa.conf pass the password to OpenSSL
using
$pass_option = "-passin file:$Conf{'tmpdir'}/pass.$$";
}
open (NEWMSG, "$Conf{'openssl'} smime -decrypt -in $temporary_file -recip
$certfile -inkey $keyfile $pass_option 2>&1 |");
if ($Conf{'key_passwd'} ne '') {
unless (&POSIX::mkfifo("$Conf{'tmpdir'}/pass.$$",0600)) {
do_log('notice', 'Unable to make fifo for
%s/pass.%s',$Conf{'tmpdir'},$$);
}
unless (open (FIFO,"> $Conf{'tmpdir'}/pass.$$")) {
do_log('notice', 'Unable to open fifo for
%s/pass.%s',$Conf{'tmpdir'},$$);
}
print FIFO $Conf{'key_passwd'};
close FIFO;
unlink ("$Conf{'tmpdir'}/pass.$$");
}

my $parser = new MIME::Parser;
$parser->output_to_core(1);
unless ($decryptedmsg = $parser->read(\*NEWMSG)) {
do_log('notice', 'Unable to parse message');
return undef;
}
close NEWMSG ;
unlink ($temporary_file) unless ($main::options{'debug'} ||
$main::options{'debug2'}) ;

## foreach header defined in the incomming message but undefined in the
## decrypted message, add this header in the decrypted form.
my $predefined_headers ;
foreach my $header ($decryptedmsg->head->tags) {
$predefined_headers->{$header} = 1 if
($decryptedmsg->head->get($header)) ;
}

foreach my $header ($msg->head->tags) {
$decryptedmsg->head->add($header,$msg->head->get($header)) unless
$predefined_headers->{$header} ;
}
## Some headers from the initial message should not be restored
## Content-Disposition and Content-Transfer-Encoding if the result is
multipart
$decryptedmsg->head->delete('Content-Disposition') if
($msg->head->get('Content-Disposition'));
if ($decryptedmsg->head->get('Content-Type') =~ /multipart/) {
$decryptedmsg->head->delete('Content-Transfer-Encoding') if
($msg->head->get('Content-Transfer-Encoding'));
}


return $decryptedmsg;


}


## Make a multipart/alternative, a singlepart
sub as_singlepart {
my ($msg, $preferred_type) = @_;
my $done = 0;

my @parts = $msg->parts();
foreach my $index (0..$#parts) {
if ($parts[$index]->effective_type() eq $preferred_type) {
## Only keep the first matching part
$msg->parts([$parts[$index]]);
$msg->make_singlepart();
$done = 1;
last;
}
}

return $done;
}


## Escape weird characters
sub escape_chars {
my $s = shift;

$s =~ s/\%/\%25/g;
$s =~ s/\"/\%22/g;
$s =~ s/\s/\%20/g;
$s =~ s/\xa5/\%a5/g;
$s =~ s/\//\%a5/g; ## Special traetment for '/'
$s =~ s/\:/\%3a/g;
$s =~ s/\#/\%23/g;

return $s;
}

## Unescape weird characters
sub unescape_chars {
my $s = shift;

$s =~ s/\%25/\%/g;
$s =~ s/\%22/\"/g;
$s =~ s/\%20/ /g;
$s =~ s/\%a5/\//g; ## Special traetment for '/'
$s =~ s/\%3a/\:/g;
$s =~ s/\%23/\#/g;

return $s;
}

sub escape_html {
my $s = shift;

$s =~ s/\"/\&quot\;/g;

return $s;
}

sub tmp_passwd {
my $email = shift;

return ('init'.substr(MD5->hexhash(join('/', $Conf{'cookie'}, $email)),
-8)) ;
}

# Check sum used to authenticate communication from wwsympa to sympa
sub sympa_checksum {
my $rcpt = shift;
return (substr(MD5->hexhash(join('/', $Conf{'cookie'}, $rcpt)), -10)) ;
}

# create a cipher
sub ciphersaber_installed {

my $is_installed;
foreach my $dir (@INC) {
if (-f "$dir/Crypt/CipherSaber.pm") {
$is_installed = 1;
last;
}
}

if ($is_installed) {
require Crypt::CipherSaber;
$cipher = Crypt::CipherSaber->new($Conf{'cookie'});
}else{
$cipher = 'no_cipher';
}
}

## encrypt a password
sub crypt_password {
my $inpasswd = shift ;

unless (defined($cipher)){
$cipher = ciphersaber_installed();
}
return $inpasswd if ($cipher eq 'no_cipher') ;
return ("crypt.".&MIME::Base64::encode($cipher->encrypt ($inpasswd))) ;
}

## decrypt a password
sub decrypt_password {
my $inpasswd = shift ;
do_log('debug2', 'tools::decrypt_password (%s)', $inpasswd);

return $inpasswd unless ($inpasswd =~ /^crypt\.(.*)$/) ;
$inpasswd = $1;

unless (defined($cipher)){
$cipher = ciphersaber_installed();
}
if ($cipher eq 'no_cipher') {
do_log('info','password seems crypted while CipherSaber is not
installed !');
return $inpasswd ;
}
return ($cipher->decrypt(&MIME::Base64::decode($inpasswd)));
}



1;


  • create_list.conf -> default hidden ?, Foudil . Bretel, 29/06/2001

Archives gérées par MHonArc 2.6.19+.

Haut de le page