Your IP : 192.168.165.1


Current Path : C:/xampp/htdocs/moodle/lib/classes/dataformat/
Upload File :
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;
    }
}