See Release Notes
Long Term Support Release
<?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/>. namespace core_h5p; use core_h5p\local\library\autoloader; /** * Test class covering the h5p data generator class. * * @package core_h5p * @category test * @copyright 2019 Mihail Geshoski <mihail@moodle.com> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @runTestsInSeparateProcesses> * @covers \core_h5p_generator*/< class generator_testcase extends \advanced_testcase {> class generator_test extends \advanced_testcase {/** * Tests set up. */< protected function setUp() {> protected function setUp(): void {parent::setUp(); autoloader::register(); } /** * Test the returned data of generate_h5p_data() when the method is called without requesting * creation of library files. */ public function test_generate_h5p_data_no_files_created_return_data() { global $DB; $this->resetAfterTest(); $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); $data = $generator->generate_h5p_data(); $mainlib = $DB->get_record('h5p_libraries', ['machinename' => 'MainLibrary']); $lib1 = $DB->get_record('h5p_libraries', ['machinename' => 'Library1']); $lib2 = $DB->get_record('h5p_libraries', ['machinename' => 'Library2']); $lib3 = $DB->get_record('h5p_libraries', ['machinename' => 'Library3']); $lib4 = $DB->get_record('h5p_libraries', ['machinename' => 'Library4']); $lib5 = $DB->get_record('h5p_libraries', ['machinename' => 'Library5']); $h5p = $DB->get_record('h5p', ['mainlibraryid' => $mainlib->id]); $expected = (object) [ 'h5pcontent' => (object) array( 'h5pid' => $h5p->id, 'contentdependencies' => array($mainlib, $lib1, $lib2, $lib3, $lib4) ), 'mainlib' => (object) array( 'data' => $mainlib, 'dependencies' => array($lib1, $lib2, $lib3) ), 'lib1' => (object) array( 'data' => $lib1, 'dependencies' => array($lib2, $lib3, $lib4) ), 'lib2' => (object) array( 'data' => $lib2, 'dependencies' => array() ), 'lib3' => (object) array( 'data' => $lib3, 'dependencies' => array($lib5) ), 'lib4' => (object) array( 'data' => $lib4, 'dependencies' => array() ), 'lib5' => (object) array( 'data' => $lib5, 'dependencies' => array() ), ]; $this->assertEquals($expected, $data); } /** * Test the returned data of generate_h5p_data() when the method requests * creation of library files. */ public function test_generate_h5p_data_files_created_return_data() { global $DB; $this->resetAfterTest(); $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); $data = $generator->generate_h5p_data(true); $mainlib = $DB->get_record('h5p_libraries', ['machinename' => 'MainLibrary']); $lib1 = $DB->get_record('h5p_libraries', ['machinename' => 'Library1']); $lib2 = $DB->get_record('h5p_libraries', ['machinename' => 'Library2']); $lib3 = $DB->get_record('h5p_libraries', ['machinename' => 'Library3']); $lib4 = $DB->get_record('h5p_libraries', ['machinename' => 'Library4']); $lib5 = $DB->get_record('h5p_libraries', ['machinename' => 'Library5']); $h5p = $DB->get_record('h5p', ['mainlibraryid' => $mainlib->id]); $expected = (object) [ 'h5pcontent' => (object) array( 'h5pid' => $h5p->id, 'contentdependencies' => array($mainlib, $lib1, $lib2, $lib3, $lib4) ), 'mainlib' => (object) array( 'data' => $mainlib, 'dependencies' => array($lib1, $lib2, $lib3) ), 'lib1' => (object) array( 'data' => $lib1, 'dependencies' => array($lib2, $lib3, $lib4) ), 'lib2' => (object) array( 'data' => $lib2, 'dependencies' => array() ), 'lib3' => (object) array( 'data' => $lib3, 'dependencies' => array($lib5) ), 'lib4' => (object) array( 'data' => $lib4, 'dependencies' => array() ), 'lib5' => (object) array( 'data' => $lib5, 'dependencies' => array() ), ]; $this->assertEquals($expected, $data); } /** * Test the behaviour of generate_h5p_data(). Test whether library files are created or not * on filesystem depending what the method defines. * * @dataProvider generate_h5p_data_files_creation_provider * @param bool $createlibraryfiles Whether to create library files on the filesystem * @param bool $expected The expectation whether the files have been created or not **/ public function test_generate_h5p_data_files_creation(bool $createlibraryfiles, bool $expected) { global $DB; $this->resetAfterTest(); $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); $generator->generate_h5p_data($createlibraryfiles); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'MainLibrary']); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'Library1']); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'Library2']); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'Library3']); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'Library4']); $libraries[] = $DB->get_record('h5p_libraries', ['machinename' => 'Library5']); foreach($libraries as $lib) { // Return the created library files. $libraryfiles = $DB->get_records('files', array( 'component' => \core_h5p\file_storage::COMPONENT, 'filearea' => \core_h5p\file_storage::LIBRARY_FILEAREA, 'itemid' => $lib->id ) ); $haslibraryfiles = !empty($libraryfiles); $this->assertEquals($expected, $haslibraryfiles); } } /** * Data provider for test_generate_h5p_data_files_creation(). * * @return array */ public function generate_h5p_data_files_creation_provider(): array { return [ 'Do not create library related files on the filesystem' => [ false, false ], 'Create library related files on the filesystem' => [ true, true ] ]; } /**> * Test the returned data of generate_h5p_data() when the method requests * Test the behaviour of create_library_record(). Test whether the library data is properly > * creation of H5P file and xAPI states. * saved in the database. > * */ > * @dataProvider generate_h5p_data_xapistates_provider public function test_create_library_record() { > * @param array|null $filerecord $this->resetAfterTest(); > */ > public function test_generate_h5p_data_xapistates(?array $filerecord) { $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); > global $DB; > $data = $generator->create_library_record('Library', 'Lib', 1, 2, 3, 'Semantics example', '/regex11/'); > $this->resetAfterTest(); unset($data->id); > > /** @var \core_h5p_generator $generator */ $expected = (object) [ > $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); 'machinename' => 'Library', > $course = $this->getDataGenerator()->create_course(); 'title' => 'Lib', > $user = $this->getDataGenerator()->create_and_enrol($course, 'student'); 'majorversion' => '1', > $this->setUser($user); 'minorversion' => '2', > $activity = $this->getDataGenerator()->create_module('h5pactivity', ['course' => $course]); 'patchversion' => '3', > $activitycontext = \context_module::instance($activity->cmid); 'runnable' => '1', > if ($filerecord) { 'fullscreen' => '1', > $filerecord['contextid'] = $activitycontext->id; 'embedtypes' => '', > $filerecord['component'] = 'mod_h5pactivity'; 'preloadedjs' => 'js/example.js', > $filerecord['filearea'] = 'package'; 'preloadedcss' => 'css/example.css', > $filerecord['itemid'] = 0; 'droplibrarycss' => '', > $filerecord['filepath'] = '/'; 'semantics' => 'Semantics example', > $filerecord['filepath'] = '/'; 'addto' => '/regex11/', > $filerecord['filename'] = 'dummy.h5p'; 'coremajor' => null, > } 'coreminor' => null, > 'metadatasettings' => null, > $data = $generator->generate_h5p_data(false, $filerecord); ]; > > $mainlib = $DB->get_record('h5p_libraries', ['machinename' => 'MainLibrary']); $this->assertEquals($expected, $data); > $lib1 = $DB->get_record('h5p_libraries', ['machinename' => 'Library1']); } > $lib2 = $DB->get_record('h5p_libraries', ['machinename' => 'Library2']); > $lib3 = $DB->get_record('h5p_libraries', ['machinename' => 'Library3']); /** > $lib4 = $DB->get_record('h5p_libraries', ['machinename' => 'Library4']); * Test the behaviour of create_h5p_record(). Test whather the h5p content data is > $lib5 = $DB->get_record('h5p_libraries', ['machinename' => 'Library5']); * properly saved in the database. > * > $h5p = $DB->get_record('h5p', ['mainlibraryid' => $mainlib->id]); * @dataProvider create_h5p_record_provider > * @param array $h5pdata The h5p content data > $expected = (object) [ * @param \stdClass $expected The expected saved data > 'h5pcontent' => (object) [ **/ > 'h5pid' => $h5p->id, public function test_create_h5p_record(array $h5pdata, \stdClass $expected) { > 'contentdependencies' => [$mainlib, $lib1, $lib2, $lib3, $lib4], global $DB; > ], > 'mainlib' => (object) [ $this->resetAfterTest(); > 'data' => $mainlib, > 'dependencies' => [$lib1, $lib2, $lib3], $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); > ], > 'lib1' => (object) [ $h5pid = call_user_func_array([$generator, 'create_h5p_record'], $h5pdata); > 'data' => $lib1, > 'dependencies' => [$lib2, $lib3, $lib4], $data = $DB->get_record('h5p', ['id' => $h5pid]); > ], unset($data->id); > 'lib2' => (object) [ unset($data->timecreated); > 'data' => $lib2, unset($data->timemodified); > 'dependencies' => [], > ], $this->assertEquals($data, $expected); > 'lib3' => (object) [ } > 'data' => $lib3, > 'dependencies' => [$lib5], /** > ], * Data provider for test_create_h5p_record(). > 'lib4' => (object) [ * > 'data' => $lib4, * @return array > 'dependencies' => [], */ > ], public function create_h5p_record_provider(): array { > 'lib5' => (object) [ $createdjsoncontent = json_encode( > 'data' => $lib5, array( > 'dependencies' => [], 'text' => '<p>Created dummy text<\/p>\n', > ], 'questions' => '<p>Test created question<\/p>\n' > ]; ) > ); > $this->assertEquals($expected, $data); > if ($filerecord) { $defaultjsoncontent = json_encode( > // Confirm the H5P file has been created (when $filerecord is not empty). array( > $fs = get_file_storage(); 'text' => '<p>Dummy text<\/p>\n', > $this->assertNotFalse($fs->get_file_by_hash($h5p->pathnamehash)); 'questions' => '<p>Test question<\/p>\n' > // Confirm xAPI state has been created when $filerecord['addxapistate'] is given. ) > if (array_key_exists('addxapistate', $filerecord) && $filerecord['addxapistate']) { ); > $this->assertEquals(1, $DB->count_records('xapi_states')); > } else { $createdfilteredcontent = json_encode( > $this->assertEquals(0, $DB->count_records('xapi_states')); array( > } 'text' => 'Created dummy text', > } else { 'questions' => 'Test created question' > // Confirm the H5P file doesn't exist when $filerecord is null. ) > $fs = get_file_storage(); ); > $this->assertFalse($fs->get_file_by_hash($h5p->pathnamehash)); > // Confirm xAPI state hasn't been created when $filerecord is null. $defaultfilteredcontent = json_encode( > $this->assertEquals(0, $DB->count_records('xapi_states')); array( > } 'text' => 'Dummy text', > } 'questions' => 'Test question' > ) > /** ); > * Data provider for test_generate_h5p_data_xapistates(). > * return [ > * @return array 'Create h5p content record with set json content and set filtered content' => [ > */ [ > public function generate_h5p_data_xapistates_provider(): array { 1, > return [ $createdjsoncontent, > 'Do not create the file nor xAPI states' => [ $createdfilteredcontent > 'filerecord' => null, ], > ], (object) array( > 'Create the H5P file but not create any xAPI state' => [ 'jsoncontent' => $createdjsoncontent, > 'filerecord' => [ 'mainlibraryid' => '1', > 'addxapistate' => false, 'displayoptions' => '8', > ], 'pathnamehash' => sha1('pathname'), > ], 'contenthash' => sha1('content'), > 'Create the H5P file and the xAPI state' => [ 'filtered' => $createdfilteredcontent, > 'filerecord' => [ ) > 'addxapistate' => true, ], > ], 'Create h5p content record with set json content and default filtered content' => [ > ], [ > ]; 1, > } $createdjsoncontent, > null > /**< $data = $generator->create_library_record('Library', 'Lib', 1, 2, 3, 'Semantics example', '/regex11/');> $data = $generator->create_library_record( > 'Library', 'Lib', 1, 2, 3, 'Semantics example', '/regex11/', 'http://tutorial.org/', 'http://example.org/' > );(object) array(> 'tutorial' => 'http://tutorial.org/', 'jsoncontent' => $createdjsoncontent, > 'example' => 'http://example.org/','mainlibraryid' => '1',> 'enabled' => 1,'displayoptions' => '8', 'pathnamehash' => sha1('pathname'), 'contenthash' => sha1('content'), 'filtered' => $defaultfilteredcontent, ) ], 'Create h5p content record with default json content and set filtered content' => [ [ 1, null, $createdfilteredcontent ], (object) array( 'jsoncontent' => $defaultjsoncontent, 'mainlibraryid' => '1', 'displayoptions' => '8', 'pathnamehash' => sha1('pathname'), 'contenthash' => sha1('content'), 'filtered' => $createdfilteredcontent, ) ], 'Create h5p content record with default json content and default filtered content' => [ [ 1, null, null ], (object) array( 'jsoncontent' => $defaultjsoncontent, 'mainlibraryid' => '1', 'displayoptions' => '8', 'pathnamehash' => sha1('pathname'), 'contenthash' => sha1('content'), 'filtered' => $defaultfilteredcontent, ) ] ]; } /** * Test the behaviour of create_contents_libraries_record(). Test whether the contents libraries * are properly saved in the database. * * @dataProvider create_contents_libraries_record_provider * @param array $contentslibrariestdata The h5p contents libraries data. * @param \stdClass $expected The expected saved data. **/ public function test_create_contents_libraries_record(array $contentslibrariestdata, \stdClass $expected) { global $DB; $this->resetAfterTest(); $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); $contentlibid = call_user_func_array([$generator, 'create_contents_libraries_record'], $contentslibrariestdata); $data = $DB->get_record('h5p_contents_libraries', ['id' => $contentlibid]); unset($data->id); $this->assertEquals($data, $expected); } /** * Data provider for test_create_contents_libraries_record(). * * @return array */ public function create_contents_libraries_record_provider(): array { return [ 'Create h5p content library with set dependency type' => [ [ 1, 1, 'dynamic' ], (object) array( 'h5pid' => '1', 'libraryid' => '1', 'dependencytype' => 'dynamic', 'dropcss' => '0', 'weight' => '1' ) ], 'Create h5p content library with a default dependency type' => [ [ 1, 1 ], (object) array( 'h5pid' => '1', 'libraryid' => '1', 'dependencytype' => 'preloaded', 'dropcss' => '0', 'weight' => '1' ) ] ]; } /** * Test the behaviour of create_library_dependency_record(). Test whether the contents libraries * are properly saved in the database. * * @dataProvider create_library_dependency_record_provider * @param array $librarydependencydata The library dependency data. * @param \stdClass $expected The expected saved data. **/ public function test_create_library_dependency_record(array $librarydependencydata, \stdClass $expected) { global $DB; $this->resetAfterTest(); $generator = $this->getDataGenerator()->get_plugin_generator('core_h5p'); $contentlibid = call_user_func_array([$generator, 'create_library_dependency_record'], $librarydependencydata); $data = $DB->get_record('h5p_library_dependencies', ['id' => $contentlibid]); unset($data->id); $this->assertEquals($data, $expected); } /** * Data provider for test_create_library_dependency_record(). * * @return array */ public function create_library_dependency_record_provider(): array { return [ 'Create h5p library dependency with set dependency type' => [ [ 1, 1, 'dynamic' ], (object) array( 'libraryid' => '1', 'requiredlibraryid' => '1', 'dependencytype' => 'dynamic' ) ], 'Create h5p library dependency with default dependency type' => [ [ 1, 1 ], (object) array( 'libraryid' => '1', 'requiredlibraryid' => '1', 'dependencytype' => 'preloaded' ) ] ]; } /** * Test the behaviour of create_content_file(). Test whether a file belonging to a content is created. * * @dataProvider create_content_file_provider * @param array $filedata Data from the file to be created. * @param array $expecteddata Data expected.Data from the file to be created. */ public function test_create_content_file($filedata, $expecteddata): void { $this->resetAfterTest(); $generator = self::getDataGenerator()->get_plugin_generator('core_h5p'); if ($expecteddata[1] === 'exception') { $this->expectException('coding_exception'); } call_user_func_array([$generator, 'create_content_file'], $filedata); $systemcontext = \context_system::instance(); $filearea = $filedata[1]; $filepath = '/'. dirname($filedata[0]). '/'; $filename = basename($filedata[0]); $itemid = $expecteddata[0]; $fs = new \file_storage(); $exists = $fs->file_exists($systemcontext->id, file_storage::COMPONENT, $filearea, $itemid, $filepath, $filename); if ($expecteddata[1] === true) { $this->assertTrue($exists); } else if ($expecteddata[1] === false) { $this->assertFalse($exists); } } /** * Data provider for test_create_content_file(). Data from different files to be created. * * @return array **/ public function create_content_file_provider(): array { return [ 'Create file in content with id 4' => [ [ 'images/img1.png', 'content', 4 ], [ 4, true ] ], 'Create file in the editor' => [ [ 'images/img1.png', 'editor' ], [ 0, true ] ], 'Create file in content without id' => [ [ 'images/img1.png', 'content' ], [ 0, 'exception' ] ] ]; } }