. /** * This file contains the moodle format implementation of the content writer. * * @package core_privacy * @copyright 2018 Andrew Nicols * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace core_privacy\tests\request; defined('MOODLE_INTERNAL') || die(); /** * An implementation of the content_writer for use in unit tests. * * This implementation does not export any data but instead stores it in * structures within the instance which can be easily queried for use * during unit tests. * * @copyright 2018 Andrew Nicols * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class content_writer implements \core_privacy\local\request\content_writer { /** * @var \context The context currently being exported. */ protected $context; /** * @var \stdClass The collection of metadata which has been exported. */ protected $metadata; /** * @var \stdClass The data which has been exported. */ protected $data; /** * @var \stdClass The related data which has been exported. */ protected $relateddata; /** * @var \stdClass The list of stored files which have been exported. */ protected $files; /** * @var \stdClass The custom files which have been exported. */ protected $customfiles; /** * @var \stdClass The user preferences which have been exported. */ protected $userprefs; /** * Whether any data has been exported at all within the current context. * * @param array $subcontext The location within the current context that this data belongs - * in this method it can be partial subcontext path (or none at all to check presence of any data anywhere). * User preferences never have subcontext, if $subcontext is specified, user preferences are not checked. * @return bool */ public function has_any_data($subcontext = []) { if (empty($subcontext)) { // When subcontext is not specified check presence of user preferences in this context and in system context. $hasuserprefs = !empty($this->userprefs->{$this->context->id}); $systemcontext = \context_system::instance(); $hasglobaluserprefs = !empty($this->userprefs->{$systemcontext->id}); if ($hasuserprefs || $hasglobaluserprefs) { return true; } } foreach (['data', 'relateddata', 'metadata', 'files', 'customfiles'] as $datatype) { if (!property_exists($this->$datatype, $this->context->id)) { // No data of this type for this context at all. Continue to the next data type. continue; } $basepath = $this->$datatype->{$this->context->id}; foreach ($subcontext as $subpath) { if (!isset($basepath->children->$subpath)) { // No data of this type is present for this path. Continue to the next data type. continue 2; } $basepath = $basepath->children->$subpath; } if (!empty($basepath)) { // Some data found for this type for this subcontext. return true; } } return false; } /** * Whether any data has been exported for any context. * * @return bool */ public function has_any_data_in_any_context() { $checkfordata = function($location) { foreach ($location as $context => $data) { if (!empty($data)) { return true; } } return false; }; $hasanydata = $checkfordata($this->data); $hasanydata = $hasanydata || $checkfordata($this->relateddata); $hasanydata = $hasanydata || $checkfordata($this->metadata); $hasanydata = $hasanydata || $checkfordata($this->files); $hasanydata = $hasanydata || $checkfordata($this->customfiles); $hasanydata = $hasanydata || $checkfordata($this->userprefs); return $hasanydata; } /** * Constructor for the content writer. * * Note: The writer_factory must be passed. * @param \core_privacy\local\request\writer $writer The writer factory. */ public function __construct(\core_privacy\local\request\writer $writer) { $this->data = (object) []; $this->relateddata = (object) []; $this->metadata = (object) []; $this->files = (object) []; $this->customfiles = (object) []; $this->userprefs = (object) []; } /** * Set the context for the current item being processed. * * @param \context $context The context to use */ public function set_context(\context $context) : \core_privacy\local\request\content_writer { $this->context = $context; if (isset($this->data->{$this->context->id}) && empty((array) $this->data->{$this->context->id})) { $this->data->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } if (isset($this->relateddata->{$this->context->id}) && empty((array) $this->relateddata->{$this->context->id})) { $this->relateddata->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } if (isset($this->metadata->{$this->context->id}) && empty((array) $this->metadata->{$this->context->id})) { $this->metadata->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } if (isset($this->files->{$this->context->id}) && empty((array) $this->files->{$this->context->id})) { $this->files->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } if (isset($this->customfiles->{$this->context->id}) && empty((array) $this->customfiles->{$this->context->id})) { $this->customfiles->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } if (isset($this->userprefs->{$this->context->id}) && empty((array) $this->userprefs->{$this->context->id})) { $this->userprefs->{$this->context->id} = (object) [ 'children' => (object) [], 'data' => [], ]; } return $this; } /** * Return the current context. * * @return \context */ public function get_current_context() : \context { return $this->context; } /** * Export the supplied data within the current context, at the supplied subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @param \stdClass $data The data to be exported */ public function export_data(array $subcontext, \stdClass $data) : \core_privacy\local\request\content_writer { $current = $this->fetch_root($this->data, $subcontext); $current->data = $data; return $this; } /** * Get all data within the subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @return array The metadata as a series of keys to value + descrition objects. */ public function get_data(array $subcontext = []) { return $this->fetch_data_root($this->data, $subcontext); } /** * Export metadata about the supplied subcontext. * * Metadata consists of a key/value pair and a description of the value. * * @param array $subcontext The location within the current context that this data belongs. * @param string $key The metadata name. * @param string $value The metadata value. * @param string $description The description of the value. * @return $this */ public function export_metadata(array $subcontext, string $key, $value, string $description) : \core_privacy\local\request\content_writer { $current = $this->fetch_root($this->metadata, $subcontext); $current->data[$key] = (object) [ 'value' => $value, 'description' => $description, ]; return $this; } /** * Get all metadata within the subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @return array The metadata as a series of keys to value + descrition objects. */ public function get_all_metadata(array $subcontext = []) { return $this->fetch_data_root($this->metadata, $subcontext); } /** * Get the specified metadata within the subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @param string $key The metadata to be fetched within the context + subcontext. * @param boolean $valueonly Whether to fetch only the value, rather than the value + description. * @return array The metadata as a series of keys to value + descrition objects. */ public function get_metadata(array $subcontext = [], $key, $valueonly = true) { $keys = $this->get_all_metadata($subcontext); if (isset($keys[$key])) { $metadata = $keys[$key]; } else { return null; } if ($valueonly) { return $metadata->value; } else { return $metadata; } } /** * Export a piece of related data. * * @param array $subcontext The location within the current context that this data belongs. * @param string $name The name of the file to be exported. * @param \stdClass $data The related data to export. */ public function export_related_data(array $subcontext, $name, $data) : \core_privacy\local\request\content_writer { $current = $this->fetch_root($this->relateddata, $subcontext); $current->data[$name] = $data; return $this; } /** * Get all data within the subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @param string $filename The name of the intended filename. * @return array The metadata as a series of keys to value + descrition objects. */ public function get_related_data(array $subcontext = [], $filename = null) { $current = $this->fetch_data_root($this->relateddata, $subcontext); if (null === $filename) { return $current; } if (isset($current[$filename])) { return $current[$filename]; } return []; } /** * Export a piece of data in a custom format. * * @param array $subcontext The location within the current context that this data belongs. * @param string $filename The name of the file to be exported. * @param string $filecontent The content to be exported. */ public function export_custom_file(array $subcontext, $filename, $filecontent) : \core_privacy\local\request\content_writer { $filename = clean_param($filename, PARAM_FILE); $current = $this->fetch_root($this->customfiles, $subcontext); $current->data[$filename] = $filecontent; return $this; } /** * Get the specified custom file within the subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @param string $filename The name of the file to be fetched within the context + subcontext. * @return string The content of the file. */ public function get_custom_file(array $subcontext = [], $filename = null) { $current = $this->fetch_data_root($this->customfiles, $subcontext); if (null === $filename) { return $current; } if (isset($current[$filename])) { return $current[$filename]; } return null; } /** * Prepare a text area by processing pluginfile URLs within it. * * Note that this method does not implement the pluginfile URL rewriting. Such a job tightly depends on how the * actual writer exports files so it can be reliably tested only in real writers such as * {@link core_privacy\local\request\moodle_content_writer}. * * However we have to remove @@PLUGINFILE@@ since otherwise {@link format_text()} shows debugging messages * * @param array $subcontext The location within the current context that this data belongs. * @param string $component The name of the component that the files belong to. * @param string $filearea The filearea within that component. * @param string $itemid Which item those files belong to. * @param string $text The text to be processed * @return string The processed string */ public function rewrite_pluginfile_urls(array $subcontext, $component, $filearea, $itemid, $text) : string { return str_replace('@@PLUGINFILE@@/', 'files/', $text); } /** * Export all files within the specified component, filearea, itemid combination. * * @param array $subcontext The location within the current context that this data belongs. * @param string $component The name of the component that the files belong to. * @param string $filearea The filearea within that component. * @param string $itemid Which item those files belong to. */ public function export_area_files(array $subcontext, $component, $filearea, $itemid) : \core_privacy\local\request\content_writer { $fs = get_file_storage(); $files = $fs->get_area_files($this->context->id, $component, $filearea, $itemid); foreach ($files as $file) { $this->export_file($subcontext, $file); } return $this; } /** * Export the specified file in the target location. * * @param array $subcontext The location within the current context that this data belongs. * @param \stored_file $file The file to be exported. */ public function export_file(array $subcontext, \stored_file $file) : \core_privacy\local\request\content_writer { if (!$file->is_directory()) { $filepath = $file->get_filepath(); // Directory separator in the stored_file class should always be '/'. The following line is just a fail safe. $filepath = str_replace(DIRECTORY_SEPARATOR, '/', $filepath); $filepath = explode('/', $filepath); $filepath[] = $file->get_filename(); $filepath = array_filter($filepath); $filepath = implode('/', $filepath); $current = $this->fetch_root($this->files, $subcontext); $current->data[$filepath] = $file; } return $this; } /** * Get all files in the specfied subcontext. * * @param array $subcontext The location within the current context that this data belongs. * @return \stored_file[] The list of stored_files in this context + subcontext. */ public function get_files(array $subcontext = []) { return $this->fetch_data_root($this->files, $subcontext); } /** * Export the specified user preference. * * @param string $component The name of the component. * @param string $key The name of th key to be exported. * @param string $value The value of the preference * @param string $description A description of the value * @return \core_privacy\local\request\content_writer */ public function export_user_preference( string $component, string $key, string $value, string $description ) : \core_privacy\local\request\content_writer { $prefs = $this->fetch_root($this->userprefs, []); if (!isset($prefs->{$component})) { $prefs->{$component} = (object) []; } $prefs->{$component}->$key = (object) [ 'value' => $value, 'description' => $description, ]; return $this; } /** * Get all user preferences for the specified component. * * @param string $component The name of the component. * @return \stdClass */ public function get_user_preferences(string $component) { $context = \context_system::instance(); $prefs = $this->fetch_root($this->userprefs, [], $context->id); if (isset($prefs->{$component})) { return $prefs->{$component}; } else { return (object) []; } } /** * Get all user preferences for the specified component. * * @param string $component The name of the component. * @return \stdClass */ public function get_user_context_preferences(string $component) { $prefs = $this->fetch_root($this->userprefs, []); if (isset($prefs->{$component})) { return $prefs->{$component}; } else { return (object) []; } } /** * Perform any required finalisation steps and return the location of the finalised export. * * @return string */ public function finalise_content() : string { return 'mock_path'; } /** * Fetch the entire root record at the specified location type, creating it if required. * * @param \stdClass $base The base to use - e.g. $this->data * @param array $subcontext The subcontext to fetch * @param int $temporarycontextid A temporary context ID to use for the fetch. * @return array */ protected function fetch_root($base, $subcontext, $temporarycontextid = null) { $contextid = !empty($temporarycontextid) ? $temporarycontextid : $this->context->id; if (!isset($base->{$contextid})) { $base->{$contextid} = (object) [ 'children' => (object) [], 'data' => [], ]; } $current = $base->{$contextid}; foreach ($subcontext as $node) { if (!isset($current->children->{$node})) { $current->children->{$node} = (object) [ 'children' => (object) [], 'data' => [], ]; } $current = $current->children->{$node}; } return $current; } /** * Fetch the data region of the specified root. * * @param \stdClass $base The base to use - e.g. $this->data * @param array $subcontext The subcontext to fetch * @return array */ protected function fetch_data_root($base, $subcontext) { $root = $this->fetch_root($base, $subcontext); return $root->data; } }