Current Path : C:/xampp/htdocs/moodle/lib/classes/dataformat/ |
Current File : C:/xampp/htdocs/moodle/lib/classes/dataformat/base.php |
<?php // This file is part of Moodle - http://moodle.org/ // // Moodle is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Moodle 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. // // You should have received a copy of the GNU General Public License // along with Moodle. If not, see <http://www.gnu.org/licenses/>. /** * Base class for dataformat. * * @package core * @subpackage dataformat * @copyright 2016 Brendan Heywood (brendan@catalyst-au.net) * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace core\dataformat; use coding_exception; /** * Base class for dataformat. * * @package core * @subpackage dataformat * @copyright 2016 Brendan Heywood (brendan@catalyst-au.net) * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ abstract class base { /** @var $mimetype */ protected $mimetype = "text/plain"; /** @var $extension */ protected $extension = ".txt"; /** @var $filename */ protected $filename = ''; /** @var string The location to store the output content */ protected $filepath = ''; /** * Get the file extension * * @return string file extension */ public function get_extension() { return $this->extension; } /** * Set download filename base * * @param string $filename */ public function set_filename($filename) { $this->filename = $filename; } /** * Set file path when writing to file * * @param string $filepath * @throws coding_exception */ public function set_filepath(string $filepath): void { $filedir = dirname($filepath); if (!is_writable($filedir)) { throw new coding_exception('File path is not writable'); } $this->filepath = $filepath; // Some dataformat writers may expect filename to be set too. $this->set_filename(pathinfo($this->filepath, PATHINFO_FILENAME)); } /** * Set the title of the worksheet inside a spreadsheet * * For some formats this will be ignored. * * @param string $title */ public function set_sheettitle($title) { } /** * Output file headers to initialise the download of the file. */ public function send_http_headers() { if (defined('BEHAT_SITE_RUNNING') || PHPUNIT_TEST) { // For text based formats - we cannot test the output with behat if we force a file download. return; } if (is_https()) { // HTTPS sites - watch out for IE! KB812935 and KB316431. header('Cache-Control: max-age=10'); header('Pragma: '); } else { // Normal http - prevent caching at all cost. header('Cache-Control: private, must-revalidate, pre-check=0, post-check=0, max-age=0'); header('Pragma: no-cache'); } header('Expires: '. gmdate('D, d M Y H:i:s', 0) .' GMT'); header("Content-Type: $this->mimetype\n"); $filename = $this->filename . $this->get_extension(); header("Content-Disposition: attachment; filename=\"$filename\""); } /** * Set the dataformat to be output to current file. Calling code must call {@see base::close_output_to_file()} when finished */ public function start_output_to_file(): void { // Raise memory limit to ensure we can store the entire content. Start collecting output. raise_memory_limit(MEMORY_EXTRA); ob_start(); $this->start_output(); } /** * Write the start of the file. */ public function start_output() { // Override me if needed. } /** * Write the start of the sheet we will be adding data to. * * @param array $columns */ public function start_sheet($columns) { // Override me if needed. } /** * Method to define whether the dataformat supports export of HTML * * @return bool */ public function supports_html(): bool { return false; } /** * Apply formatting to the cells of a given record * * @param array|\stdClass $record * @return array */ protected function format_record($record): array { $record = (array)$record; // If the dataformat supports export of HTML, we need to allow them to manage embedded images. if ($this->supports_html()) { $record = array_map([$this, 'replace_pluginfile_images'], $record); } return $record; } /** * Given a stored_file, return a suitable source attribute for an img element in the export (or null to use the original) * * @param \stored_file $file * @return string|null */ protected function export_html_image_source(\stored_file $file): ?string { return null; } /** * We need to locate all img tags within a given cell that match pluginfile URL's. Partly so the exported file will show * the image without requiring the user is logged in; and also to prevent some of the dataformats requesting the file * themselves, which is likely to fail due to them not having an active session * * @param string|null $content * @return string */ protected function replace_pluginfile_images(?string $content): string { $content = (string)$content; // Examine content to see if it contains any HTML image tags. return preg_replace_callback('/(?<pre><img[^>]+src=")(?<source>[^"]*)(?<post>".*>)/i', function(array $matches) { $source = $matches['source']; // Now check if the image source looks like a pluginfile URL. if (preg_match('/pluginfile.php\/(?<context>\d+)\/(?<component>[^\/]+)\/(?<filearea>[^\/]+)\/(?:(?<itemid>\d+)\/)?' . '(?<path>.*)/u', $source, $args)) { $context = $args['context']; $component = clean_param($args['component'], PARAM_COMPONENT); $filearea = clean_param($args['filearea'], PARAM_AREA); $itemid = $args['itemid'] ?: 0; $path = clean_param(urldecode($args['path']), PARAM_PATH); // Try and get the matching file from storage, allow the dataformat to define the replacement source. $fullpath = "/{$context}/{$component}/{$filearea}/{$itemid}/{$path}"; if ($file = get_file_storage()->get_file_by_hash(sha1($fullpath))) { $exportsource = $this->export_html_image_source($file); if ($exportsource) { $source = $exportsource; } } } return $matches['pre'] . $source . $matches['post']; }, $content); } /** * Write a single record * * @param array $record * @param int $rownum */ abstract public function write_record($record, $rownum); /** * Write the end of the sheet containing the data. * * @param array $columns */ public function close_sheet($columns) { // Override me if needed. } /** * Write the end of the file. */ public function close_output() { // Override me if needed. } /** * Write the data to disk. Calling code should have previously called {@see base::start_output_to_file()} * * @return bool Whether the write succeeded */ public function close_output_to_file(): bool { $this->close_output(); $filecontent = ob_get_contents(); ob_end_clean(); return file_put_contents($this->filepath, $filecontent) !== false; } }