ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
Delivery.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
29 
39 final class Delivery
40 {
44  public const DIRECT_PHP_OUTPUT = 'php://output';
48  public const DISP_ATTACHMENT = 'attachment';
52  public const DISP_INLINE = 'inline';
56  public const EXPIRES_IN = '+5 days';
57  private static ?string $delivery_type_static = null;
59  private string $mime_type = '';
60  private string $path_to_file = '';
61  private string $download_file_name = '';
62  private string $disposition = self::DISP_ATTACHMENT;
63  private bool $send_mime_type = true;
64  private bool $exit_after = true;
65  private bool $convert_file_name_to_asci = true;
66  private string $etag = '';
67  private bool $show_last_modified = true;
68  private bool $has_context = true;
69  private bool $cache = false;
70  private bool $hash_filename = false;
71  private bool $delete_file = false;
72  private static bool $DEBUG = false;
75 
76 
81  public function __construct($input, private Services $http)
82  {
83  if ($input instanceof FileStream) {
84  $this->resource = $input;
85  $this->path_to_file = $input->getMetadata('uri');
86  } elseif ($input === self::DIRECT_PHP_OUTPUT) {
87  $this->setPathToFile(self::DIRECT_PHP_OUTPUT);
88  } else {
89  $this->setPathToFile($input);
90  }
91 
92  $this->detemineDeliveryType();
93  $this->determineMimeType();
95 
96  $this->setHasContext(\ilContext::getType() !== null);
97  $this->factory = new FileDeliveryTypeFactory($this->http);
98  }
99 
100 
101  public function stream(): void
102  {
103  if (!$this->delivery()->supportsStreaming()) {
105  }
106  $this->deliver();
107  }
108 
109 
110  private function delivery(): ilFileDeliveryType
111  {
112  return $this->factory->getInstance($this->getDeliveryType());
113  }
114 
115 
116  public function deliver(): void
117  {
118  $response = $this->http->response()->withHeader('X-ILIAS-FileDelivery-Method', $this->getDeliveryType());
119  if (
120  !$this->delivery()->doesFileExists($this->path_to_file)
121  && $this->path_to_file !== self::DIRECT_PHP_OUTPUT
122  ) {
123  $response = $this->http->response()->withStatus(404);
124  $this->http->saveResponse($response);
125  $this->http->sendResponse();
126  $this->close();
127  }
128  $this->http->saveResponse($response);
129 
130  $this->clearBuffer();
131  $this->checkCache();
132  $this->setGeneralHeaders();
133  $this->delivery()->prepare($this->getPathToFile(), $this->resource);
134  $this->delivery()->deliver($this->getPathToFile(), $this->isDeleteFile());
135  if ($this->isDeleteFile()) {
136  $this->delivery()->handleFileDeletion($this->getPathToFile());
137  }
138  if ($this->isExitAfter()) {
139  $this->close();
140  }
141  }
142 
143 
144  public function setGeneralHeaders(): void
145  {
146  $this->checkExisting();
147  if ($this->isSendMimeType()) {
148  $response = $this->http->response()->withHeader(ResponseHeader::CONTENT_TYPE, $this->getMimeType());
149  $this->http->saveResponse($response);
150  }
151  if ($this->isConvertFileNameToAsci()) {
152  $this->cleanDownloadFileName();
153  }
154  if ($this->hasHashFilename()) {
155  $this->setDownloadFileName(md5($this->getDownloadFileName()));
156  }
157  $this->setDispositionHeaders();
158  $response = $this->http->response()->withHeader(ResponseHeader::ACCEPT_RANGES, 'bytes');
159  $this->http->saveResponse($response);
160  if ($this->getDeliveryType() === DeliveryMethod::PHP
161  && $this->getPathToFile() !== self::DIRECT_PHP_OUTPUT
162  ) {
163  $response = $this->http->response()->withHeader(ResponseHeader::CONTENT_LENGTH, (string) filesize($this->getPathToFile()));
164  $this->http->saveResponse($response);
165  }
166  $response = $this->http->response()->withHeader(ResponseHeader::CONNECTION, "close");
167  $this->http->saveResponse($response);
168  }
169 
170 
171  public function setCachingHeaders(): void
172  {
173  $response = $this->http->response()->withHeader(ResponseHeader::CACHE_CONTROL, 'must-revalidate, post-check=0, pre-check=0')->withHeader(ResponseHeader::PRAGMA, 'public');
174 
175  $this->http->saveResponse($response->withHeader(ResponseHeader::EXPIRES, date("D, j M Y H:i:s", strtotime(self::EXPIRES_IN)) . " GMT"));
176  $this->sendEtagHeader();
177  $this->sendLastModified();
178  }
179 
180 
181  public function generateEtag(): void
182  {
183  $this->setEtag(md5(filemtime($this->getPathToFile()) . filesize($this->getPathToFile())));
184  }
185 
186 
187  public function close(): void
188  {
189  $this->http->close();
190  }
191 
192 
193  private function determineMimeType(): void
194  {
196  if ($info !== '' && $info !== '0') {
197  $this->setMimeType($info);
198 
199  return;
200  }
201  $finfo = finfo_open(FILEINFO_MIME_TYPE);
202  $info = finfo_file($finfo, $this->getPathToFile());
203  finfo_close($finfo);
204  if ($info) {
205  $this->setMimeType($info);
206 
207  return;
208  }
209  }
210 
211 
212  private function determineDownloadFileName(): void
213  {
214  if ($this->getDownloadFileName() === '' || $this->getDownloadFileName() === '0') {
215  $download_file_name = basename($this->getPathToFile());
216  $this->setDownloadFileName($download_file_name);
217  }
218  }
219 
220 
221  private function detemineDeliveryType(): void
222  {
223  if (self::$delivery_type_static) {
224  $this->setDeliveryType(self::$delivery_type_static);
225 
226  return;
227  }
228 
229  if (function_exists('apache_get_modules')
230  && in_array('mod_xsendfile', apache_get_modules(), true)
231  ) {
233  }
234 
235  // if (function_exists('apache_get_version') && strpos(apache_get_version(), '2.4.') !== false) {
236  // $this->setDeliveryType(DeliveryMethod::XSENDFILE);
237  // }
238 
239  if (is_file('./components/ILIAS/FileDelivery/classes/override.php')) {
240  $override_delivery_type = false;
242  require_once('./components/ILIAS/FileDelivery/classes/override.php');
243  if ($override_delivery_type) {
244  $this->setDeliveryType($override_delivery_type);
245  }
246  }
247  $ilRuntime = \ilRuntime::getInstance();
248  if ((!$ilRuntime->isFPM() && !$ilRuntime->isHHVM())
250  ) {
252  }
253 
254  if ($this->getDeliveryType() === DeliveryMethod::XACCEL
255  && !str_starts_with($this->getPathToFile(), './public/data')
256  ) {
258  }
259 
260  self::$delivery_type_static = $this->getDeliveryType();
261  }
262 
263 
264  public function getDeliveryType(): string
265  {
266  return $this->delivery_type;
267  }
268 
269 
270  public function setDeliveryType(string $delivery_type): void
271  {
272  $this->delivery_type = $delivery_type;
273  }
274 
275 
276  public function getMimeType(): string
277  {
278  return $this->mime_type;
279  }
280 
281 
282  public function setMimeType(string $mime_type): void
283  {
284  $this->mime_type = $mime_type;
285  }
286 
287 
288  public function getPathToFile(): string
289  {
290  return $this->path_to_file;
291  }
292 
293 
294  public function setPathToFile(string $path_to_file): void
295  {
296  $this->path_to_file = $path_to_file;
297  }
298 
299 
300  public function getDownloadFileName(): string
301  {
303  }
304 
305 
306  public function setDownloadFileName(string $download_file_name): void
307  {
308  $this->download_file_name = $download_file_name;
309  }
310 
311 
312  public function getDisposition(): string
313  {
314  return $this->disposition;
315  }
316 
317 
318  public function setDisposition(string $disposition): void
319  {
320  $this->disposition = $disposition;
321  }
322 
323 
324  public function isSendMimeType(): bool
325  {
326  return $this->send_mime_type;
327  }
328 
329 
330  public function setSendMimeType(bool $send_mime_type): void
331  {
332  $this->send_mime_type = $send_mime_type;
333  }
334 
335 
336  public function isExitAfter(): bool
337  {
338  return $this->exit_after;
339  }
340 
341 
342  public function setExitAfter(bool $exit_after): void
343  {
344  $this->exit_after = $exit_after;
345  }
346 
347 
348  public function isConvertFileNameToAsci(): bool
349  {
351  }
352 
353 
354  public function setConvertFileNameToAsci(bool $convert_file_name_to_asci): void
355  {
356  $this->convert_file_name_to_asci = $convert_file_name_to_asci;
357  }
358 
359 
360  public function getEtag(): string
361  {
362  return $this->etag;
363  }
364 
365 
366  public function setEtag(string $etag): void
367  {
368  $this->etag = $etag;
369  }
370 
371 
372  public function getShowLastModified(): bool
373  {
375  }
376 
377 
378  public function setShowLastModified(bool $show_last_modified): void
379  {
380  $this->show_last_modified = $show_last_modified;
381  }
382 
383 
384  public function isHasContext(): bool
385  {
386  return $this->has_context;
387  }
388 
389 
390  public function setHasContext(bool $has_context): void
391  {
392  $this->has_context = $has_context;
393  }
394 
395 
396  public function hasCache(): bool
397  {
398  return $this->cache;
399  }
400 
401 
402  public function setCache(bool $cache): void
403  {
404  $this->cache = $cache;
405  }
406 
407 
408  public function hasHashFilename(): bool
409  {
410  return $this->hash_filename;
411  }
412 
413 
414  public function setHashFilename(bool $hash_filename): void
415  {
416  $this->hash_filename = $hash_filename;
417  }
418 
419 
420  private function sendEtagHeader(): void
421  {
422  if ($this->getEtag() !== '' && $this->getEtag() !== '0') {
423  $response = $this->http->response()->withHeader('ETag', $this->getEtag());
424  $this->http->saveResponse($response);
425  }
426  }
427 
428 
429  private function sendLastModified(): void
430  {
431  if ($this->getShowLastModified()) {
432  $response = $this->http->response()->withHeader(
433  'Last-Modified',
434  date("D, j M Y H:i:s", filemtime($this->getPathToFile()))
435  . " GMT"
436  );
437  $this->http->saveResponse($response);
438  }
439  }
440 
441  public static function isDEBUG(): bool
442  {
443  return self::$DEBUG;
444  }
445 
446 
447  public static function setDEBUG(bool $DEBUG): void
448  {
449  self::$DEBUG = $DEBUG;
450  }
451 
452 
453  public function checkCache(): void
454  {
455  if ($this->hasCache()) {
456  $this->generateEtag();
457  $this->sendEtagHeader();
458  $this->setShowLastModified(true);
459  $this->setCachingHeaders();
460  }
461  }
462 
463 
467  public function clearBuffer(): bool
468  {
469  try {
470  $ob_get_contents = ob_get_contents();
471  if ($ob_get_contents) {
472  // \ilWACLog::getInstance()->write(__CLASS__ . ' had output before file delivery: '
473  // . $ob_get_contents);
474  }
475  ob_end_clean(); // fixed 0016469, 0016467, 0016468
476  return true;
477  } catch (\Throwable) {
478  return false;
479  }
480  }
481 
482 
483  private function checkExisting(): void
484  {
485  if ($this->getPathToFile() !== self::DIRECT_PHP_OUTPUT
486  && !file_exists($this->getPathToFile())
487  ) {
488  $this->close();
489  }
490  }
491 
492 
496  private function cleanDownloadFileName(): void
497  {
498  $download_file_name = self::returnASCIIFileName($this->getDownloadFileName());
499  $this->setDownloadFileName($download_file_name);
500  }
501 
502 
510  public static function returnASCIIFileName(string $original_filename): string
511  {
512  $umlaut_mapping = [
513  "Ä" => "Ae",
514  "Ö" => "Oe",
515  "Ü" => "Ue",
516  "ä" => "ae",
517  "ö" => "oe",
518  "ü" => "ue",
519  "ß" => "ss"
520  ];
521  foreach ($umlaut_mapping as $src => $tgt) {
522  $original_filename = str_replace($src, $tgt, $original_filename);
523  }
524 
525  $ascii_filename = htmlentities($original_filename, ENT_NOQUOTES, 'UTF-8');
526  $ascii_filename = preg_replace('/\&(.)[^;]*;/', '\\1', $ascii_filename);
527  $ascii_filename = preg_replace('/[\x7f-\xff]/', '_', (string) $ascii_filename);
528 
529  // OS do not allow the following characters in filenames: \/:*?"<>|
530  $ascii_filename = preg_replace(
531  '/[:\x5c\/\*\?\"<>\|]/',
532  '_',
533  (string) $ascii_filename
534  );
535  return $ascii_filename;
536  }
537 
538 
539  public function isDeleteFile(): bool
540  {
541  return $this->delete_file;
542  }
543 
544 
545  public function setDeleteFile(bool $delete_file): void
546  {
547  $this->delete_file = $delete_file;
548  }
549 
550 
551  private function setDispositionHeaders(): void
552  {
553  $response = $this->http->response();
554  $response = $response->withHeader(
556  $this->getDisposition()
557  . '; filename="'
558  . $this->getDownloadFileName()
559  . '"'
560  );
561  $response = $response->withHeader('Content-Description', $this->getDownloadFileName());
562  $this->http->saveResponse($response);
563  }
564 }
static returnASCIIFileName(string $original_filename)
Converts a UTF-8 filename to ASCII.
Definition: Delivery.php:510
setConvertFileNameToAsci(bool $convert_file_name_to_asci)
Definition: Delivery.php:354
setDeleteFile(bool $delete_file)
Definition: Delivery.php:545
setMimeType(string $mime_type)
Definition: Delivery.php:282
FileDeliveryTypeFactory $factory
Definition: Delivery.php:73
static string $delivery_type_static
Definition: Delivery.php:57
static getInstance()
setHasContext(bool $has_context)
Definition: Delivery.php:390
factory()
$response
Definition: xapitoken.php:93
$http
Definition: deliver.php:30
static lookupMimeType(string $path_to_file, string $fallback=self::APPLICATION__OCTET_STREAM, bool $a_external=false)
Definition: MimeType.php:544
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
setDisposition(string $disposition)
Definition: Delivery.php:318
static http()
Fetches the global http state from ILIAS.
static setDEBUG(bool $DEBUG)
Definition: Delivery.php:447
cleanDownloadFileName()
Converts the filename to ASCII.
Definition: Delivery.php:496
setExitAfter(bool $exit_after)
Definition: Delivery.php:342
setDownloadFileName(string $download_file_name)
Definition: Delivery.php:306
setDeliveryType(string $delivery_type)
Definition: Delivery.php:270
Class Delivery.
Definition: Delivery.php:39
setHashFilename(bool $hash_filename)
Definition: Delivery.php:414
static getType()
Get context type.
__construct($input, private Services $http)
Definition: Delivery.php:81
setShowLastModified(bool $show_last_modified)
Definition: Delivery.php:378
The base interface for all filesystem streams.
Definition: FileStream.php:31
setPathToFile(string $path_to_file)
Definition: Delivery.php:294
setSendMimeType(bool $send_mime_type)
Definition: Delivery.php:330