<?php /* $Id: m_mem.php,v 1.19 2006/01/12 08:04:43 anarcat Exp $ ---------------------------------------------------------------------- LICENSE This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License (GPL) 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. To read the license please visit http://www.gnu.org/copyleft/gpl.html ---------------------------------------------------------------------- Original Author of file: Benjamin Sonntag Purpose of file: Manage Login session on the virtual desktop and member parameters ---------------------------------------------------------------------- */ /** * This class manage user sessions in the web desktop. * * This class manage user sessions and administration in AlternC. * @copyright AlternC-Team 2002-2005 http://alternc.org/ * */ class m_mem { /** Original uid for the temporary uid swapping (for administrators) */ var $olduid=0; /** This array contains the Tableau contenant les champs de la table "membres" du membre courant * Ce tableau est utilisable globalement par toutes les classes filles. */ var $user; /** Tableau contenant les champs de la table "local" du membre courant * Ce tableau est utilisable globalement par toutes les classes filles. * Note : les champs de "local" sont sp�cifiques � l'h�bergeur. */ var $local; /* ----------------------------------------------------------------- */ /** * Constructeur */ function m_mem() { } /* ----------------------------------------------------------------- */ /** Check that the current user is an admnistrator. * @return boolean TRUE if we are super user, or FALSE if we are not. */ function checkright() { return ($this->user["su"]=="1"); } /* ----------------------------------------------------------------- */ /** Start a session in the web desktop. Check username and password. * <b>Note : </b>If the user entered a bas password, the failure will be logged * and told to the corresponding user on next successfull login. * @param $username string Username that want to get connected. * @param $password string User Password. * @return boolean TRUE if the user has been successfully connected, or FALSE if an error occured. */ function login($username,$password,$restrictip=0) { global $db,$session,$err,$cuid; $err->log("mem","login",$username); // $username=addslashes($username); // $password=addslashes($password); $db->query("select * from membres where login='$username';"); if ($db->num_rows()==0) { $err->raise("mem",1); return false; } $db->next_record(); if (_md5cr($password,$db->f("pass"))!=$db->f("pass")) { $db->query("UPDATE membres SET lastfail=lastfail+1 WHERE uid='".$db->f("uid")."';"); $err->raise("mem",1); return false; } if (!$db->f("enabled")) { $err->raise("mem",2); return false; } $this->user=$db->Record; $cuid=$db->f("uid"); if ($restrictip) { $ip="INET_ATON('".getenv("REMOTE_ADDR")."')"; } else $ip="0"; /* Close sessions that are more than 2 days old. */ $db->query("DELETE FROM sessions WHERE DATE_ADD(ts,INTERVAL 2 DAY)<NOW();"); /* Open the session : */ $session=md5(uniqid(mt_rand())); $db->query("insert into sessions (sid,ip,uid) values ('$session',$ip,'$cuid');"); setcookie("session",$session,0,"/"); $err->error=0; /* Fill in $local */ $db->query("SELECT * FROM local WHERE uid='$cuid';"); if ($db->num_rows()) { $db->next_record(); $this->local=$db->Record; } return true; } /* ----------------------------------------------------------------- */ /** Start a session as another user from an administrator account. * This function is not the same as su. setid connect the current user in the destination * account (for good), and su allow any user to become another account for some commands only. * (del_user, add_user ...) and allow to bring back admin rights with unsu * * @param $id integer User id where we will connect to. * @return boolean TRUE if the user has been successfully connected, FALSE else. */ function setid($id) { global $db,$session,$err,$cuid; $err->log("mem","setid",$username); $db->query("select * from membres where uid='$id';"); if ($db->num_rows()==0) { $err->raise("mem",1); return false; } $db->next_record(); $this->user=$db->Record; $cuid=$db->f("uid"); $ip=getenv("REMOTE_ADDR"); $session=md5(uniqid(mt_rand())); $db->query("insert into sessions (sid,ip,uid) values ('$session',INET_ATON('$ip'),'$cuid');"); setcookie("session",$session,0,"/"); $err->error=0; /* Fill in $local */ $db->query("SELECT * FROM local WHERE uid='$cuid';"); if ($db->num_rows()) { $db->next_record(); $this->local=$db->Record; } return true; } /* ----------------------------------------------------------------- */ /** Suite � la connexion de l'utilisateur, r�initialise ses param�tres de derni�re connexion */ function resetlast() { global $db,$cuid; $ip=addslashes(getenv("REMOTE_HOST")); if (!$ip) $ip=addslashes(getenv("REMOTE_ADDR")); $db->query("UPDATE membres SET lastlogin=NOW(), lastfail=0, lastip='$ip' WHERE uid='$cuid';"); } /* ----------------------------------------------------------------- */ /** V�rifie que la session courante est correcte (cookie ok et ip valide). * Si besoin, et si r�ception des champs username & password, cr�e une nouvelle * session pour l'utilisateur annonc�. * Cette fonction doit �tre appell�e � chaque page devant �tre authentifi�e. * et AVANT d'�mettre des donn�es. (un cookie peut �tre envoy�) * @global string $session Le cookie de session eventuel * @global string $username/password le login/pass de l'utilisateur * @return TRUE si la session est correcte, FALSE sinon. */ function checkid() { global $db,$err,$session,$username,$password,$cuid,$restrictip; if ($username && $password) { return $this->login($username,$password,$restrictip); } $session=addslashes($session); if (strlen($session)!=32) { $err->raise("mem",3); return false; } $ip=getenv("REMOTE_ADDR"); $db->query("select uid,INET_ATON('$ip') as me,ip from sessions where sid='$session'"); if ($db->num_rows()==0) { $err->raise("mem",4); return false; } $db->next_record(); if ($db->f("ip")) { if ($db->f("me")!=$db->f("ip")) { $err->raise("mem",5); return false; } } $cuid=$db->f("uid"); $db->query("select * from membres where uid='$cuid';"); $db->next_record(); $this->user=$db->Record; $err->error=0; /* Remplissage de $local */ $db->query("SELECT * FROM local WHERE uid='$cuid';"); if ($db->num_rows()) { $db->next_record(); $this->local=$db->Record; } return true; } /* ----------------------------------------------------------------- */ /** Change l'identit� d'un utilisateur temporairement. * @global string $uid Utilisateur dont on prends l'identit� * @return TRUE si la session est correcte, FALSE sinon. */ function su($uid) { global $cuid,$db,$err; if (!$this->olduid) $this->olduid=$cuid; $db->query("select * from membres where uid='$uid';"); if ($db->num_rows()==0) { $err->raise("mem",1); return false; } $db->next_record(); $this->user=$db->Record; $cuid=$db->f("uid"); return true; } /* ----------------------------------------------------------------- */ /** Retourne a l'identite d'origine de l'utilisateur apres su. * @return TRUE si la session est correcte, FALSE sinon. */ function unsu() { global $cuid; if (!$this->olduid) return false; $this->su($this->olduid); $this->olduid=0; return true; } /* ----------------------------------------------------------------- */ /** Termine une session du bureau virtuel (logout) * @return boolean TRUE si la session a bien �t� d�truite, FALSE sinon. */ function del_session() { global $db,$session,$user,$err,$cuid,$classes; $err->log("mem","del_session"); $session=addslashes($session); setcookie("session","",0,"/"); if ($session=="") { $err->error=0; return true; } if (strlen($session)!=32) { $err->raise("mem",3); return false; } $ip=getenv("REMOTE_ADDR"); $db->query("select uid,INET_ATON('$ip') as me,ip from sessions where sid='$session'"); if ($db->num_rows()==0) { $err->raise("mem",4); return false; } $db->next_record(); if ($db->f("me")!=$db->f("ip")) { $err->raise("mem",5); return false; } $cuid=$db->f("uid"); $db->query("delete from sessions where sid='$session';"); $err->error=0; # Invoker le logout dans toutes les autres classes foreach($classes as $c) { if (method_exists($GLOBALS[$c],"alternc_del_session")) { $GLOBALS[$c]->alternc_del_session($dom); } } return true; } /* ----------------------------------------------------------------- */ /** Change le mot de passe de l'utilisateur courant. * @param string $oldpass Ancien mot de passe. * @param string $newpass Nouveau mot de passe * @param string $newpass2 Nouveau mot de passe (� nouveau) * @return boolean TRUE si le mot de passe a �t� chang�, FALSE sinon. */ function passwd($oldpass,$newpass,$newpass2) { global $db,$err,$cuid; $err->log("mem","passwd"); $oldpass=stripslashes($oldpass); $newpass=stripslashes($newpass); $newpass2=stripslashes($newpass2); if (!$this->user["canpass"]) { $err->raise("mem",11); return false; } if ($this->user["pass"]!=_md5cr($oldpass,$this->user["pass"])) { $err->raise("mem",6); return false; } if ($newpass!=$newpass2) { $err->raise("mem",7); return false; } if (strlen($newpass)<3) { $err->raise("mem",8); return false; } $newpass=_md5cr($newpass); $db->query("UPDATE membres SET pass='$newpass' WHERE uid='$cuid';"); $err->error=0; return true; } /* ----------------------------------------------------------------- */ /** Change les pr�f�rences administrateur d'un compte * @param integer $admlist Mode de visualisation des membres (0=large 1=courte) * @return boolean TRUE si les pr�f�rences ont �t� chang�es, FALSE sinon. */ function adminpref($admlist) { global $db,$err,$cuid; $err->log("mem","admlist"); if (!$this->user["su"]) { $err->raise("mem",12); return false; } $db->query("UPDATE membres SET admlist='$admlist' WHERE uid='$cuid';"); $err->error=0; return true; } /* ----------------------------------------------------------------- */ /** Envoie en mail le mot de passe d'un compte. * <b>Note : </b>On ne peut demander le mot de passe qu'une seule fois par jour. * TODO : Translate this mail into the localization program. * TODO : Check this function's ! * @return boolean TRUE si le mot de passe a �t� envoy� avec succ�s, FALSE sinon. */ function send_pass($login) { global $err,$db,$L_HOSTING,$L_FQDN; $err->log("mem","send_pass"); $db->query("SELECT * FROM membres WHERE login='$login';"); if (!$db->num_rows()) { $err->raise("mem",2); return false; } $db->next_record(); if (time()-$db->f("lastaskpass")<86400) { $err->raise("mem",7); return false; } $txt="Bonjour, Il semblerait que vous ayez demand� � recevoir le mot de passe du compte ".$login." sur $L_HOSTING Voici donc le nom d'utilisateur et le mot de passe qui vous permettront de rentrer sur le bureau virtuel : -------------------------------------- Nom d'utilisateur : ".$db->f("login")." Mot de passe : ".$db->f("pass")." -------------------------------------- Note : si vous n'avez pas fait cette demande, cela signifie que quelqu'un l'a faite pour vous. Vous pouvez donc ignorer ce message. Si cela se reproduit, n'h�sitez pas � contacter l'administrateur de votre serveur. Cordialement. "; mail($db->f("mail"),"Votre mot de passe sur $L_HOSTING",$txt,"From: postmaster@$L_FQDN\nReply-to: postmaster@$L_FQDN"); $db->query("UPDATE membres SET lastaskpass=".time()." WHERE login='$login';"); return true; } /* ----------------------------------------------------------------- */ /** Change le mail d'un membre (premi�re etape, envoi du CookiE) * TODO : insert this mail string into the localization system * @param string $newmail Nouveau mail souhait� pour le membre. * @return string le cookie si le mail a bien �t� envoy�, FALSE sinon */ function ChangeMail1($newmail) { global $err,$db,$L_HOSTING,$L_FQDN,$cuid; $err->log("mem","changemail1",$newmail); $db->query("SELECT * FROM membres WHERE uid='$cuid';"); if (!$db->num_rows()) { $err->raise("mem",2); return false; } $db->next_record(); // un cookie de 20 caract�res pour le mail $COOKIE=substr(md5(uniqid(rand(),1)),0,20); // et de 6 pour la cl� � entrer. ca me semble suffisant... $KEY=substr(md5(uniqid(rand(),1)),0,6); // TODO : Translate this and insert this in alternc.po $txt="Bonjour, Quelqu'un (peut-etre vous) a demand� le changement de l'email du compte ".$db->f("login")." sur $L_HOSTING Afin de confirmer que cet email est valide, merci de vous rendre � l'adresse ci-dessous : https://$L_FQDN/admin/mem_cm.php?usr=$cuid&cookie=$COOKIE (attention : si cette adresse est coup�e sur 2 lignes, ne pas oublier de reconstituer sur une seule ligne). Le bureau vous demandera la cl� qui vous a �t� donn�e lors de la demande de changement d'email. Note : si vous n'avez pas fait cette demande, cela signifie que quelqu'un l'a faite pour vous. Vous pouvez donc ignorer ce message. Si cela se reproduit, n'h�sitez pas � contacter l'administrateur de votre serveur. Cordialement. "; mail($newmail,"Changement d'email sur $L_HOSTING",$txt,"From: postmaster@$L_FQDN\nReply-to: postmaster@$L_FQDN"); // Supprime les demandes pr�c�dentes de ce compte ! $db->query("DELETE FROM chgmail WHERE uid='$cuid';"); $db->query("INSERT INTO chgmail (cookie,ckey,uid,mail,ts) VALUES ('$COOKIE','$KEY','$cuid','$newmail',".time().");"); // Supprime les cookies de la veille :) $lts=time()-86400; $db->query("DELETE FROM chgmail WHERE ts<'$lts';"); return $KEY; } /* ----------------------------------------------------------------- */ /** Change le mail d'un membre (seconde etape, CookiE+cl� = application) * @param string $COOKIE Cookie envoy� par mail * @param string $KEY cl� affich�e � l'�cran * @param integer $uid Utilisateur concern� (on est hors session) * @return TRUE si le mail a bien �t� modifi�, FALSE sinon */ function ChangeMail2($COOKIE,$KEY,$uid) { global $err,$db,$L_HOSTING,$L_FQDN; $err->log("mem","changemail2",$uid); $db->query("SELECT * FROM chgmail WHERE cookie='$COOKIE' and ckey='$KEY' and uid='$uid';"); if (!$db->num_rows()) { $err->raise("mem",9); return false; } $db->next_record(); // met � jour le compte : $db->query("UPDATE membres SET mail='".$db->f("mail")."' WHERE uid='$uid';"); $db->query("DELETE FROM chgmail WHERE uid='$uid';"); // Supprime les cookies de la veille :) $lts=time()-86400; $db->query("DELETE FROM chgmail WHERE ts<'$lts';"); return true; } /* ----------------------------------------------------------------- */ /** Modifie le param�tre d'aide en ligne (1/0) * @param integer $show Faut-il (1) ou non (0) afficher l'aide en ligne */ function set_help_param($show) { global $db,$err,$cuid; $err->log("mem","set_help_param",$show); $db->query("UPDATE membres SET show_help='$show' WHERE uid='$cuid';"); } /* ----------------------------------------------------------------- */ /** Dit si l'aide en ligne est demand�e * @return boolean TRUE si l'aide en ligne est demand�e, FALSE sinon. */ function get_help_param() { return $this->user["show_help"]; } /* ----------------------------------------------------------------- */ /** Affiche (echo) l'aide contextuelle * @param integer $file Num�ro de fichier d'aide � afficher. * @return TRUE si l'aide contextuelle a �t� trouv�e, FALSE sinon */ function show_help($file) { global $err; $err->log("mem","show_help",$show); if ($this->user["show_help"]) { $hlp=_("hlp_$file"); if ($hlp!="hlp_$file") { $hlp=ereg_replace( "HELPID_([0-9]*)", "<a href=\"javascript:help(\\1);\"><img src=\"/admin/aide/help.png\" width=\"17\" height=\"17\" style=\"vertical-align: middle;\" alt=\""._("Help")."\" /></a>",$hlp); echo "<p class=\"hlp\">".$hlp."</p>"; return true; } return false; } else { return true; } } /* ----------------------------------------------------------------- */ /** * Exports all the personnal user related information for an account. * @access private * EXPERIMENTAL 'sid' function ;) */ function alternc_export($tmpdir) { global $db,$err; $err->log("mem","export"); $str="<mem>\n"; foreach ($this->user as $k=>$v) { $str.=" <$k>".xml_entities($v)."</$k>\n"; } $str.="</mem>\n"; return $str; } } /* Classe Membre */ ?>