Le téléchargement de fichiers du client vers le serveur (upload en bon français) est une opération de plus en plus courante et extrêmement simple avec PHP.
Pour que l'utilisateur puisse choisir le fichier à télécharger sur son ordinateur nous utilisons un formulaire et une zone de saisie <input> de type file, spécialement concue pour cet usage.
<form method="post" action="page.php" enctype="multipart/form-data"> Choisissez un fichier à télécharger : <input type="file" name="uplFichier"> <input type="submit" name="btnUpload" value="Envoyer"> </form>
Le tag <form> doit contenir
l'attribut enctype avec la valeur multipart/form-data.
Par défaut,
le type d'encodage (enctype) d'un
formulaire est application/x-www-forme-urlencoded
qui indique que les données saisies dans le formulaire seront
transmises au serveur sous la forme de l'encodage des URLs (Zone1=Valeur1&Zone2=Valeur2&...). Ce
type est inefficace pour envoyer de grandes quantités de données
binaires ou des textes qui contiennent des caractères non-ASCII.
La RFC2388 (http://www.faqs.org/rfcs/rfc2388.html) définit
donc la valeur multipart/form-data pour
l'attribut enctype. Cette valeur
prévient le serveur que le formulaire va envoyer éventuellement
autre chose que des données contenues dans des commandes HTTP, et
qu'il peut s'attendre à recevoir des données non-ASCII et des
données binaires (des fichiers par exemple).
Ensuite, il faut prendre garde à la taille des fichiers téléchargés.
L'option de configuration upload_max_filesize
dans le fichier php.ini détermine la taille maximum d'un fichier
uploadé (par défaut : 2 MB). Cette taille peut être réduite
ponctuellement si nous ajoutons une zone cachée dans le formulaire :
<input type="hidden"
name="MAX_FILE_SIZE" value="10240">
Si vous modifiez l'option de configuration upload_max_filesize il faut aussi modifier post_max_size en conséquence.
Cette zone cachée doit se trouver directement aprés le tag <form>, et doit s'appeler MAX_FILE_SIZE. Sa valeur détermine la taille maximum en KB du fichier qui pourra être téléchargé (dans la ligne d'exemple 10 KB). Si cette valeur est supérieure à la valeur de l'option de configuration upload_max_filesize, elle sera ignorée.
Le code de notre formulaire devient :
<form enctype="multipart/form-data" method="post" action="page.php"> <input type="hidden" name="MAX_FILE_SIZE" value="102400"> Sélectionnez le fichier à télécharger : <input type="file" name="uplFichier"> <input type="submit" name="btnUpload" value="Envoyer"> </form>
Le code de traitement pour le côté client est terminé. Nous pouvons maintenant examiner ce qu'il y a à faire du côté serveur pour correctement récupérer et traiter les informations passées par le formulaire.
Sur le serveur, les informations sur le fichier transmises par le formulaire seront accessible dans le tableau superglobal $_FILES. Chaque élément du tableau représentant un fichier télécharge est lui même un tableau associatif de 4 éléments :
| name | le nom du fichier envoyé par le navigateur. |
| type | le type MIME du fichier uploadé, tel que déterminé par le navigateur (exemples : application/zip, image/png, video/mp4, application/pdf). |
| size | la taille en bytes du fichier uploadé. Si l'utilisateur essaye de télécharger un fichier dont la taille est supérieure à la limite autorisée, size sera égal à 0. |
| tmp_name | le nom du fichier temporaire créé sur le serveur. Si l'utilisateur essaye de télécharger un fichier dont la taille est supérieure à la limite autorisée, tmp_name est egal à "none". |
| error | code d'erreur 0 = pas d'erreur / 1 = la taille du fichier dépasse la taille de upload_max_filesize dans php.ini / 2 = la taille du fichier dépasse la taille spécifiée dans le formulaire HTML / 3 = fichier partiellement uploadé / 4 = fichier introuvable. |
La récupération du fichier sur le serveur doit se faire avec 2 fonctions dédiées :
La fonction is_uploaded_file() prévient des erreurs de transfert et garantit que l'on travaille bien avec un fichier téléchargé, et non un fichier "natif" du serveur (prévention contre les risques d'attaque pirates).
La fonction move_uploaded_file() nous permet de déplacer le fichier téléchargé du répertoire d'upload vers son répertoire final sur le serveur. En effet, un fichier téléchargé sur le serveur est toujours stocké dans un répertoire temporaire, déterminé par l'option de configuration upload_tmp_dir dans le fichier php.ini. Je vous recommande pour des raisons de sécurité de ne jamais utiliser une fonction de copie de fichier comme copy(), mais de toujours utiliser move_uploaded_file() qui vérifie que le fichier copié est bien un fichier téléchargé.
Il faut bien entendu que le répertoire de fichiers temporaire et le répertoire dans lequel vous voulez déposer le fichier téléchargé permettent l'accès en lecture / écriture à l'utilisateur système associé au serveur Web (par exemple IUSR_NomMachine sous NT, l'utilisateur Apache ou nobody sous Linux).
Vous pouvez remarquer l'utilisation du caractère @ devant certaines
fonctions :
if (! @is_uploaded_file($f['tmp_name']))
{
et
if (@move_uploaded_file($f['tmp_name'],
$place)) {
Le caractère @ placé devant une fonction empêche tout
message d'erreur éventuel d'être affiché par le navigateur. Dans
notre cas, si une erreur se produit, aucune information sur des
répertoires ne sera affichée, rendant la page plus sécurisée.
Voici quelques problèmes typiques que l'on peut rencontrer lors de l'upload de fichier.
Pas de fichier uploadé, pas de message d'erreur.
Le
plus souvent c'est dû à la taille du fichier qui dépasse le maximum
autorisé par PHP. Testez si un fichier de petite taille (quelque ko)
est correctement uploadé.
Pas de fichier copié, erreur de permission.
Le
script doit avoir les droits d'écriture et de lecture sur le
répertoire final ou le fichier est uploadé.
Pas de fichier uploadé ou copié, erreurs telles que
"open_basedir restriction", "Safe Mode Restriction", "function has
been disabled"
PHP fonctionne en mode sécurisé (Safe
Mode), ce qui permet à l'administrateur de contrôler les fonctions
qu'un utilisateur a le droit d'exécuter, et d'interdire l'exécution
de certaines fonctions (comme la copie ou le transfert de fichiers)