On à tous à un moment ou un autre eu besoin de faire lancer un téléchargement à php. Et bien souvent c'est le même problème , on se souvient jamais des header à utiliser pour que cela marche correctement.
Je vous propose ici une petite classe simplifiant grandement l'envoi de fichier au client.
Utilisation :
Code :
1 2 3 4 5 6 7 8 9 10
|
$fichier = "D:/dossier/fichier.exe"; //Chemin vers le fichier à télécharger
$resumeDownload = true; //Accepte de reprendre un téléchargement suspendu.
$dl = new Net_HTTPDownload($fichier,$resumeDownload);
/*On peut définir un nom pour le fichier envoyé. Pratique dans le cas ou les fichier sur le serveur ont des noms complexes. L'utilisateur aura alors un nom simplifié*/
$dl->setName(' Mon fichier.exe'); // Facultatif
/* La classe cherche parmis les mime les plus courant et utilise application/force-download si rien trouvé mais on peut spécifier directement le bon mime si on le connait*/
$dl->setMimeType('application/mimespecial'); //facultatif
$dl->download(); //Lance l'envoi du fichier vers le client |
Classe :
Code :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
|
<?php
/**
* Téléchargement de fichier via protocole HTTP
* Pour des fichiers de taille importante préférer la solution X-SendFile header
* @see http://www.php.net/manual/en/function.fread.php#84115
* @version 1.0.1
* @author http://www.developpez.net/forums/u23146/grunk/
*
*/
class Net_HTTPDownload
{
/**
* Chemin vers le fichier
* @var string
*/
protected $path;
/**
* Nom du fichier envoyé au navigateur.
* Permet de renommer un fichier en l'envoyant.
* @var string
*/
protected $name;
/**
* Extension du fichier
* @var string
*/
protected $extension;
/**
* Type Mime du fichier
* @var string
*/
protected $mime;
/**
* Taille du fichier en octet
* @var int
*/
protected $size;
/**
* Active ou non la reprise de téléchargement
* @var boolean
*/
protected $resume;
/**
* Position de début du pointeur
* @var int
*/
protected $seekStart;
/**
* Position de fin du pointeur
* @var int
*/
protected $seekEnd;
public function __construct($path,$resume = false)
{
if(file_exists($path))
{
$this->path = $path;
$this->extension = strtolower(substr(strrchr($path,'.'),1));
$this->getMime();
$this->size = filesize($path);
$this->seekStart = 0;
$this->seekEnd = -1;
$this->resume = $resume;
$this->name = basename($path);
}
else
{
throw new Exception('Fichier innexistant');
}
}
/**
* Lance le téléchargement ou la reprise du téléchargement
*/
public function download()
{
header("Cache-Control: cache, must-revalidate");
header("Pragma: public");
header('Expires: 0');
if($this->resume)
{
$this->getRange();
header('HTTP/1.0 206 Partial Content');
header('Status: 206 Partial Content');
header('Accept-Ranges: bytes');
header('Content-Range: bytes '.$this->seekStart.'-'.$this->seekEnd.'/'.$this->size);
header('Content-Length:' . ($this->seekEnd - $this->seekStart + 1));
}
else
{
header('Content-Length:' . $this->size);
}
header('Content-Description: File Transfer');
header('Content-Type: '.$this->mime);
header('Content-Disposition: attachment;filename="'.$this->name.'"');
header('Content-Transfer-Encoding: binary');
$handle = fopen($this->path,'rb');
if(!$handle)
{
throw new Exception('Erreur pendant le téléchargement');
}
else
{
fseek($handle,$this->seekStart);
while(!feof($handle))
{
set_time_limit(0);
echo fread($handle, 1024*8); //Paquet de 8ko
flush();
ob_flush();
}
fclose($handle);
}
}
/**
* Défini le type mime du fichier
* Par défaut le type octet-stream est utilisé
* @param string $mime
*/
public function setMimeType($mime)
{
$this->mime = $mime;
}
/**
* Renomme le fichier à télécharger
* @param string $name
*/
public function setName($name)
{
$this->name = $name;
}
/**
* Récupère le range
* @see http://tools.ietf.org/id/draft-ietf-http-range-retrieval-00.txt
* @return void
*/
private function getRange()
{
//Peut être de la forme Range: bytes=0-99,500-1499,4000-
if(isset($_SERVER['HTTP_RANGE']))
{
list($unit,$ranges) = explode('=',$_SERVER['HTTP_RANGE'],2);
if($unit == 'bytes')
{
list($range,$extraRange) = explode(',',$ranges,2);
//Seul le premier range est utilisé pour des raisons de simplicité
list($seekStart,$seekEnd) = explode('-',$range,2);
if(!empty($seekStart) && $seekStart > 0)
$this->seekStart = intval($seekStart);
if(!empty($seekEnd))
$this->seekEnd = min(abs(intval($seekEnd)), ($this->size) -1);
}
}
}
/**
* Tente de déterminer le type mime du fichier
*/
private function getMime()
{
switch($this->extension)
{
case 'txt': $this->mime='text/plain'; break;
case 'pdf': $this->mime='application/pdf'; break;
case 'rtf': $this->mime='application/rtf'; break;
case 'jpg': $this->mime='image/jpeg'; break;
case 'xls': $this->mime='application/vnd.ms-excel'; break;
case 'pps': $this->mime='application/vnd.ms-powerpoint'; break;
case 'doc': $this->mime='application/msword'; break;
case 'exe': $this->mime='application/octet-stream'; break;
case 'zip': $this->mime='application/zip'; break;
case 'mp3': $this->mime='audio/mpeg'; break;
case 'mpg': $this->mime='video/mpeg'; break;
case 'avi': $this->mime='video/x-msvideo'; break;
default: $this->mime='application/force-download';
}
}
}
?> |
La classe à été testée sur les 3 principaux navigateurs. J'ai pu suspendre et reprendre des téléchargement via les gestionnaire de chrome et firefox. je n'ai pas testé d'autre gestionnaire de téléchargement type flashget.
Note 1 : Pour l'envoi de très gros fichier il existe d'autres solutions notamment avec des extensions apache et l'entête X-Send qui sont bien mieux adapté que le passage via php.
Note 2 : Il est assez simple d'intégrer une limite de débit en faisant un sleep dans la boucle de lecture. Ma problématique ne le nécessitait pas mais si vous voulez l'ajouter n'hésitez pas