. /** * This file contains tests for the repository_nextcloud class. * * @package repository_nextcloud * @copyright 2017 Project seminar (Learnweb, University of Münster) * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ defined('MOODLE_INTERNAL') || die(); global $CFG; require_once($CFG->dirroot . '/repository/lib.php'); require_once($CFG->libdir . '/webdavlib.php'); /** * Class repository_nextcloud_lib_testcase * @group repository_nextcloud * @copyright 2017 Project seminar (Learnweb, University of Münster) * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class repository_nextcloud_lib_testcase extends advanced_testcase { /** @var null|\repository_nextcloud the repository_nextcloud object, which the tests are run on. */ private $repo = null; /** @var null|\core\oauth2\issuer which belongs to the repository_nextcloud object.*/ private $issuer = null; /** * SetUp to create an repository instance. */ protected function setUp() { $this->resetAfterTest(true); // Admin is neccessary to create api and issuer objects. $this->setAdminUser(); /** @var repository_nextcloud_generator $generator */ $generator = $this->getDataGenerator()->get_plugin_generator('repository_nextcloud'); $this->issuer = $generator->test_create_issuer(); // Create Endpoints for issuer. $generator->test_create_endpoints($this->issuer->get('id')); // Params for the config form. $reptype = $generator->create_type([ 'visible' => 1, 'enableuserinstances' => 0, 'enablecourseinstances' => 0, ]); $instance = $generator->create_instance([ 'issuerid' => $this->issuer->get('id'), 'pluginname' => 'Nextcloud', 'controlledlinkfoldername' => 'Moodlefiles', 'supportedreturntypes' => 'both', 'defaultreturntype' => FILE_INTERNAL, ]); // At last, create a repository_nextcloud object from the instance id. $this->repo = new repository_nextcloud($instance->id); $this->repo->options['typeid'] = $reptype->id; $this->repo->options['sortorder'] = 1; $this->resetAfterTest(true); } /** * Checks the is_visible method in case the repository is set to hidden in the database. */ public function test_is_visible_parent_false() { global $DB; $id = $this->repo->options['typeid']; // Check, if the method returns false, when the repository is set to visible in the database // and the client configuration data is complete. $DB->update_record('repository', (object) array('id' => $id, 'visible' => 0)); $this->assertFalse($this->repo->is_visible()); } /** * Test whether the repo is disabled. */ public function test_repo_creation() { $issuerid = $this->repo->get_option('issuerid'); // Config saves the right id. $this->assertEquals($this->issuer->get('id'), $issuerid); // Function that is used in construct method returns the right id. $constructissuer = \core\oauth2\api::get_issuer($issuerid); $this->assertEquals($this->issuer->get('id'), $constructissuer->get('id')); $this->assertEquals(true, $constructissuer->get('enabled')); $this->assertFalse($this->repo->disabled); } /** * Returns an array of endpoints or null. * @param string $endpointname * @return array|null */ private function get_endpoint_id($endpointname) { $endpoints = \core\oauth2\api::get_endpoints($this->issuer); $id = array(); foreach ($endpoints as $endpoint) { $name = $endpoint->get('name'); if ($name === $endpointname) { $id[$endpoint->get('id')] = $endpoint->get('id'); } } if (empty($id)) { return null; } return $id; } /** * Test if repository is disabled when webdav_endpoint is deleted. */ public function test_issuer_webdav() { $idwebdav = $this->get_endpoint_id('webdav_endpoint'); if (!empty($idwebdav)) { foreach ($idwebdav as $id) { \core\oauth2\api::delete_endpoint($id); } } $this->assertFalse(\repository_nextcloud\issuer_management::is_valid_issuer($this->issuer)); } /** * Test if repository is disabled when ocs_endpoint is deleted. */ public function test_issuer_ocs() { $idocs = $this->get_endpoint_id('ocs_endpoint'); if (!empty($idocs)) { foreach ($idocs as $id) { \core\oauth2\api::delete_endpoint($id); } } $this->assertFalse(\repository_nextcloud\issuer_management::is_valid_issuer($this->issuer)); } /** * Test if repository is disabled when userinfo_endpoint is deleted. */ public function test_issuer_userinfo() { $idtoken = $this->get_endpoint_id('userinfo_endpoint'); if (!empty($idtoken)) { foreach ($idtoken as $id) { \core\oauth2\api::delete_endpoint($id); } } $this->assertFalse(\repository_nextcloud\issuer_management::is_valid_issuer($this->issuer)); } /** * Test if repository is disabled when token_endpoint is deleted. */ public function test_issuer_token() { $idtoken = $this->get_endpoint_id('token_endpoint'); if (!empty($idtoken)) { foreach ($idtoken as $id) { \core\oauth2\api::delete_endpoint($id); } } $this->assertFalse(\repository_nextcloud\issuer_management::is_valid_issuer($this->issuer)); } /** * Test if repository is disabled when auth_endpoint is deleted. */ public function test_issuer_authorization() { $idauth = $this->get_endpoint_id('authorization_endpoint'); if (!empty($idauth)) { foreach ($idauth as $id) { \core\oauth2\api::delete_endpoint($id); } } $this->assertFalse(\repository_nextcloud\issuer_management::is_valid_issuer($this->issuer)); } /** * Test if repository throws an error when endpoint does not exist. */ public function test_parse_endpoint_url_error() { $this->expectException(\repository_nextcloud\configuration_exception::class); \repository_nextcloud\issuer_management::parse_endpoint_url('notexisting', $this->issuer); } /** * Test get_listing method with an example directory. Tests error cases. */ public function test_get_listing_error() { $ret = $this->get_initialised_return_array(); $this->setUser(); // WebDAV socket is not opened. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(false)); $private = $this->set_private_property($mock, 'dav'); $this->assertEquals($ret, $this->repo->get_listing('/')); // Response is not an array. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(true)); $mock->expects($this->once())->method('ls')->will($this->returnValue('notanarray')); $private->setValue($this->repo, $mock); $this->assertEquals($ret, $this->repo->get_listing('/')); } /** * Test get_listing method with an example directory. Tests the root directory. */ public function test_get_listing_root() { $this->setUser(); $ret = $this->get_initialised_return_array(); // This is the expected response from the ls method. $response = array( array( 'href' => 'remote.php/webdav/', 'lastmodified' => 'Thu, 08 Dec 2016 16:06:26 GMT', 'resourcetype' => 'collection', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '' ), array( 'href' => 'remote.php/webdav/Documents/', 'lastmodified' => 'Thu, 08 Dec 2016 16:06:26 GMT', 'resourcetype' => 'collection', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '' ), array( 'href' => 'remote.php/webdav/welcome.txt', 'lastmodified' => 'Thu, 08 Dec 2016 16:06:26 GMT', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '163' ) ); // The expected result from the get_listing method in the repository_nextcloud class. $ret['list'] = array( 'DOCUMENTS/' => array( 'title' => 'Documents', 'thumbnail' => null, 'children' => array(), 'datemodified' => 1481213186, 'path' => '/Documents/' ), 'WELCOME.TXT' => array( 'title' => 'welcome.txt', 'thumbnail' => null, 'size' => '163', 'datemodified' => 1481213186, 'source' => '/welcome.txt' ) ); // Valid response from the client. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(true)); $mock->expects($this->once())->method('ls')->will($this->returnValue($response)); $this->set_private_property($mock, 'dav'); $ls = $this->repo->get_listing('/'); // Those attributes can not be tested properly. $ls['list']['DOCUMENTS/']['thumbnail'] = null; $ls['list']['WELCOME.TXT']['thumbnail'] = null; $this->assertEquals($ret, $ls); } /** * Test get_listing method with an example directory. Tests a different directory than the root * directory. */ public function test_get_listing_directory() { $ret = $this->get_initialised_return_array(); $this->setUser(); // An additional directory path has to be added to the 'path' field within the returned array. $ret['path'][1] = array( 'name' => 'dir', 'path' => '/dir/' ); // This is the expected response from the get_listing method in the Nextcloud client. $response = array( array( 'href' => 'remote.php/webdav/dir/', 'lastmodified' => 'Thu, 08 Dec 2016 16:06:26 GMT', 'resourcetype' => 'collection', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '' ), array( 'href' => 'remote.php/webdav/dir/Documents/', 'lastmodified' => null, 'resourcetype' => 'collection', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '' ), array( 'href' => 'remote.php/webdav/dir/welcome.txt', 'lastmodified' => 'Thu, 08 Dec 2016 16:06:26 GMT', 'status' => 'HTTP/1.1 200 OK', 'getcontentlength' => '163' ) ); // The expected result from the get_listing method in the repository_nextcloud class. $ret['list'] = array( 'DOCUMENTS/' => array( 'title' => 'Documents', 'thumbnail' => null, 'children' => array(), 'datemodified' => null, 'path' => '/dir/Documents/' ), 'WELCOME.TXT' => array( 'title' => 'welcome.txt', 'thumbnail' => null, 'size' => '163', 'datemodified' => 1481213186, 'source' => '/dir/welcome.txt' ) ); // Valid response from the client. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(true)); $mock->expects($this->once())->method('ls')->will($this->returnValue($response)); $this->set_private_property($mock, 'dav'); $ls = $this->repo->get_listing('/dir/'); // Can not be tested properly. $ls['list']['DOCUMENTS/']['thumbnail'] = null; $ls['list']['WELCOME.TXT']['thumbnail'] = null; $this->assertEquals($ret, $ls); } /** * Test the get_link method. */ public function test_get_link_success() { $mock = $this->getMockBuilder(\repository_nextcloud\ocs_client::class)->disableOriginalConstructor()->disableOriginalClone( )->getMock(); $file = '/datei'; $expectedresponse = << ok 100 2 3 admin admin 1 1502883721 QXbqrJj8DcMaXen admin admin /somefile file application/pdf home::admin 1 6 6 4 /somefile https://www.default.test/somefile 0 XML; // Expected Parameters. $ocsquery = [ 'path' => $file, 'shareType' => \repository_nextcloud\ocs_client::SHARE_TYPE_PUBLIC, 'publicUpload' => false, 'permissions' => \repository_nextcloud\ocs_client::SHARE_PERMISSION_READ ]; // With test whether mock is called with right parameters. $mock->expects($this->once())->method('call')->with('create_share', $ocsquery)->will($this->returnValue($expectedresponse)); $this->set_private_property($mock, 'ocsclient'); // Method does extract the link from the xml format. $this->assertEquals('https://www.default.test/somefile/download', $this->repo->get_link($file)); } /** * get_link can get OCS failure responses. Test that this is handled appropriately. */ public function test_get_link_failure() { $mock = $this->getMockBuilder(\repository_nextcloud\ocs_client::class)->disableOriginalConstructor()->disableOriginalClone( )->getMock(); $file = '/datei'; $expectedresponse = << failure 404 Msg XML; // Expected Parameters. $ocsquery = [ 'path' => $file, 'shareType' => \repository_nextcloud\ocs_client::SHARE_TYPE_PUBLIC, 'publicUpload' => false, 'permissions' => \repository_nextcloud\ocs_client::SHARE_PERMISSION_READ ]; // With test whether mock is called with right parameters. $mock->expects($this->once())->method('call')->with('create_share', $ocsquery)->will($this->returnValue($expectedresponse)); $this->set_private_property($mock, 'ocsclient'); // Suppress (expected) XML parse error... Nextcloud sometimes returns JSON on extremely bad errors. libxml_use_internal_errors(true); // Method get_link correctly raises an exception that contains error code and message. $this->expectException(\repository_nextcloud\request_exception::class); $params = array('instance' => $this->repo->get_name(), 'errormessage' => sprintf('(%s) %s', '404', 'Msg')); $this->expectExceptionMessage(get_string('request_exception', 'repository_nextcloud', $params)); $this->repo->get_link($file); } /** * get_link can get OCS responses that are not actually XML. Test that this is handled appropriately. */ public function test_get_link_problem() { $mock = $this->getMockBuilder(\repository_nextcloud\ocs_client::class)->disableOriginalConstructor()->disableOriginalClone( )->getMock(); $file = '/datei'; $expectedresponse = << $file, 'shareType' => \repository_nextcloud\ocs_client::SHARE_TYPE_PUBLIC, 'publicUpload' => false, 'permissions' => \repository_nextcloud\ocs_client::SHARE_PERMISSION_READ ]; // With test whether mock is called with right parameters. $mock->expects($this->once())->method('call')->with('create_share', $ocsquery)->will($this->returnValue($expectedresponse)); $this->set_private_property($mock, 'ocsclient'); // Suppress (expected) XML parse error... Nextcloud sometimes returns JSON on extremely bad errors. libxml_use_internal_errors(true); // Method get_link correctly raises an exception. $this->expectException(\repository_nextcloud\request_exception::class); $this->repo->get_link($file); } /** * Test get_file reference, merely returns the input if no optional_param is set. */ public function test_get_file_reference_withoutoptionalparam() { $this->assertEquals('/somefile', $this->repo->get_file_reference('/somefile')); } /** * Test logout. */ public function test_logout() { $mock = $this->createMock(\core\oauth2\client::class); $mock->expects($this->exactly(2))->method('log_out'); $this->set_private_property($mock, 'client'); $this->repo->options['ajax'] = false; $this->expectOutputString('Log in to your account' . 'Log in to your account'); $this->assertEquals($this->repo->print_login(), $this->repo->logout()); $mock->expects($this->exactly(2))->method('get_login_url')->will($this->returnValue(new moodle_url('url'))); $this->repo->options['ajax'] = true; $this->assertEquals($this->repo->print_login(), $this->repo->logout()); } /** * Test for the get_file method from the repository_nextcloud class. */ public function test_get_file() { // WebDAV socket is not open. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(false)); $private = $this->set_private_property($mock, 'dav'); $this->assertFalse($this->repo->get_file('path')); // WebDAV socket is open and the request successful. $mock = $this->createMock(\webdav_client::class); $mock->expects($this->once())->method('open')->will($this->returnValue(true)); $mock->expects($this->once())->method('get_file')->will($this->returnValue(true)); $private->setValue($this->repo, $mock); $result = $this->repo->get_file('path', 'file'); $this->assertNotNull($result['path']); } /** * Test callback. */ public function test_callback() { $mock = $this->createMock(\core\oauth2\client::class); // Should call check_login exactly once. $mock->expects($this->once())->method('log_out'); $mock->expects($this->once())->method('is_logged_in'); $this->set_private_property($mock, 'client'); $this->repo->callback(); } /** * Test check_login. */ public function test_check_login() { $mock = $this->createMock(\core\oauth2\client::class); $mock->expects($this->once())->method('is_logged_in')->will($this->returnValue(true)); $this->set_private_property($mock, 'client'); $this->assertTrue($this->repo->check_login()); } /** * Test print_login. */ public function test_print_login() { $mock = $this->createMock(\core\oauth2\client::class); $mock->expects($this->exactly(2))->method('get_login_url')->will($this->returnValue(new moodle_url('url'))); $this->set_private_property($mock, 'client'); // Test with ajax activated. $this->repo->options['ajax'] = true; $url = new moodle_url('url'); $ret = array(); $btn = new \stdClass(); $btn->type = 'popup'; $btn->url = $url->out(false); $ret['login'] = array($btn); $this->assertEquals($ret, $this->repo->print_login()); // Test without ajax. $this->repo->options['ajax'] = false; $output = html_writer::link($url, get_string('login', 'repository'), array('target' => '_blank', 'rel' => 'noopener noreferrer')); $this->expectOutputString($output); $this->repo->print_login(); } /** * Test the initiate_webdavclient function. */ public function test_initiate_webdavclient() { global $CFG; $idwebdav = $this->get_endpoint_id('webdav_endpoint'); if (!empty($idwebdav)) { foreach ($idwebdav as $id) { \core\oauth2\api::delete_endpoint($id); } } $generator = $this->getDataGenerator()->get_plugin_generator('repository_nextcloud'); $generator->test_create_single_endpoint($this->issuer->get('id'), "webdav_endpoint", "https://www.default.test:8080/webdav/index.php"); $fakeaccesstoken = new stdClass(); $fakeaccesstoken->token = "fake access token"; $oauthmock = $this->createMock(\core\oauth2\client::class); $oauthmock->expects($this->once())->method('get_accesstoken')->will($this->returnValue($fakeaccesstoken)); $this->set_private_property($oauthmock, 'client'); $dav = phpunit_util::call_internal_method($this->repo, "initiate_webdavclient", [], 'repository_nextcloud'); // Verify that port is set correctly (private property). $refclient = new ReflectionClass($dav); $property = $refclient->getProperty('_port'); $property->setAccessible(true); $port = $property->getValue($dav); $this->assertEquals('8080', $port); } /** * Test supported_returntypes. * FILE_INTERNAL when no system account is connected. * FILE_INTERNAL | FILE_CONTROLLED_LINK when a system account is connected. */ public function test_supported_returntypes() { global $DB; $this->assertEquals(FILE_INTERNAL, $this->repo->supported_returntypes()); $dataobject = new stdClass(); $dataobject->timecreated = time(); $dataobject->timemodified = time(); $dataobject->usermodified = 2; $dataobject->issuerid = $this->issuer->get('id'); $dataobject->refreshtoken = 'sometokenthatwillnotbeused'; $dataobject->grantedscopes = 'openid profile email'; $dataobject->email = 'some.email@some.de'; $dataobject->username = 'someusername'; $DB->insert_record('oauth2_system_account', $dataobject); // When a system account is registered the file_type FILE_CONTROLLED_LINK is supported. $this->assertEquals(FILE_INTERNAL | FILE_CONTROLLED_LINK, $this->repo->supported_returntypes()); } /** * The reference_file_selected() methode is called every time a FILE_CONTROLLED_LINK is chosen for upload. * Since the function is very long the private function are tested separately, and merely the abortion of the * function are tested. * */ public function test_reference_file_selected_error() { $this->repo->disabled = true; $this->expectException(\repository_exception::class); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->disabled = false; $this->expectException(\repository_exception::class); $this->expectExceptionMessage('Cannot connect as system user'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $mock = $this->createMock(\core\oauth2\client::class); $mock->expects($this->once())->method('get_system_oauth_client')->with($this->issuer)->willReturn(true); $this->expectException(\repository_exception::class); $this->expectExceptionMessage('Cannot connect as current user'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->expects($this->once())->method('get_user_oauth_client')->willReturn(true); $this->expectException(\repository_exception::class); $this->expectExceptionMessage('cannotdownload'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->expects($this->once())->method('get_user_oauth_client')->willReturn(true); $this->expectException(\repository_exception::class); $this->expectExceptionMessage('cannotdownload'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->expects($this->once())->method('get_user_oauth_client')->willReturn(true); $this->repo->expects($this->once())->method('copy_file_to_path')->willReturn(array('statuscode' => array('success' => 400))); $this->expectException(\repository_exception::class); $this->expectExceptionMessage('Could not copy file'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->expects($this->once())->method('get_user_oauth_client')->willReturn(true); $this->repo->expects($this->once())->method('copy_file_to_path')->willReturn(array('statuscode' => array('success' => 201))); $this->repo->expects($this->once())->method('delete_share_dataowner_sysaccount')->willReturn( array('statuscode' => array('success' => 400))); $this->expectException(\repository_exception::class); $this->expectExceptionMessage('Share is still present'); $this->repo->reference_file_selected('', context_system::instance(), '', '', ''); $this->repo->expects($this->once())->method('get_user_oauth_client')->willReturn(true); $this->repo->expects($this->once())->method('copy_file_to_path')->willReturn(array('statuscode' => array('success' => 201))); $this->repo->expects($this->once())->method('delete_share_dataowner_sysaccount')->willReturn( array('statuscode' => array('success' => 100))); $filereturn = array(); $filereturn->link = 'some/fullpath' . 'some/target/path'; $filereturn->name = 'mysource'; $filereturn->usesystem = true; $filereturn = json_encode($filereturn); $return = $this->repo->reference_file_selected('mysource', context_system::instance(), '', '', ''); $this->assertEquals($filereturn, $return); } /** * Test the send_file function for access controlled links. */ public function test_send_file_errors() { $fs = get_file_storage(); $storedfile = $fs->create_file_from_reference([ 'contextid' => context_system::instance()->id, 'component' => 'core', 'filearea' => 'unittest', 'itemid' => 0, 'filepath' => '/', 'filename' => 'testfile.txt', ], $this->repo->id, json_encode([ 'type' => 'FILE_CONTROLLED_LINK', 'link' => 'https://test.local/fakelink/', 'usesystem' => true, ])); $this->set_private_property('', 'client'); $this->expectException(repository_nextcloud\request_exception::class); $this->expectExceptionMessage(get_string('contactadminwith', 'repository_nextcloud', 'The OAuth clients could not be connected.')); $this->repo->send_file($storedfile, '', '', ''); // Testing whether the mock up appears is topic to behat. $mock = $this->createMock(\core\oauth2\client::class); $mock->expects($this->once())->method('is_logged_in')->willReturn(true); $this->repo->send_file($storedfile, '', '', ''); // Checks that setting for foldername are used. $mock->expects($this->once())->method('is_dir')->with('Moodlefiles')->willReturn(false); // In case of false as return value mkcol is called to create the folder. $parsedwebdavurl = parse_url($this->issuer->get_endpoint_url('webdav')); $webdavprefix = $parsedwebdavurl['path']; $mock->expects($this->once())->method('mkcol')->with( $webdavprefix . 'Moodlefiles')->willReturn(400); $this->expectException(\repository_nextcloud\request_exception::class); $this->expectExceptionMessage(get_string('requestnotexecuted', 'repository_nextcloud')); $this->repo->send_file($storedfile, '', '', ''); $expectedresponse = << ok 100 6 0 tech tech 19 1511877999 tech tech /System/Category Miscellaneous/Course Example Course/File morefiles/mod_resource/content/0/merge.txt file text/plain home::tech 4 824 824 823 /merge (3).txt user2 user1 0 5 0 tech tech 19 1511877999 tech tech /System/Category Miscellaneous/Course Example Course/File morefiles/mod_resource/content/0/merge.txt file text/plain home::tech 4 824 824 823 /merged (3).txt user1 user1 0 XML; // Checks that setting for foldername are used. $mock->expects($this->once())->method('is_dir')->with('Moodlefiles')->willReturn(true); // In case of true as return value mkcol is not called to create the folder. $shareid = 5; $mockocsclient = $this->getMockBuilder( \repository_nextcloud\ocs_client::class)->disableOriginalConstructor()->disableOriginalClone()->getMock(); $mockocsclient->expects($this->exactly(2))->method('call')->with('get_information_of_share', array('share_id' => $shareid))->will($this->returnValue($expectedresponse)); $this->set_private_property($mock, 'ocsclient'); $this->repo->expects($this->once())->method('move_file_to_folder')->with('/merged (3).txt', 'Moodlefiles', $mock)->willReturn(array('success' => 201)); $this->repo->send_file('', '', '', ''); // Create test for statuscode 403. // Checks that setting for foldername are used. $mock->expects($this->once())->method('is_dir')->with('Moodlefiles')->willReturn(true); // In case of true as return value mkcol is not called to create the folder. $shareid = 5; $mockocsclient = $this->getMockBuilder(\repository_nextcloud\ocs_client::class )->disableOriginalConstructor()->disableOriginalClone()->getMock(); $mockocsclient->expects($this->exactly(1))->method('call')->with('get_shares', array('path' => '/merged (3).txt', 'reshares' => true))->will($this->returnValue($expectedresponse)); $mockocsclient->expects($this->exactly(1))->method('call')->with('get_information_of_share', array('share_id' => $shareid))->will($this->returnValue($expectedresponse)); $this->set_private_property($mock, 'ocsclient'); $this->repo->expects($this->once())->method('move_file_to_folder')->with('/merged (3).txt', 'Moodlefiles', $mock)->willReturn(array('success' => 201)); $this->repo->send_file('', '', '', ''); } /** * Helper method, which inserts a given mock value into the repository_nextcloud object. * * @param mixed $value mock value that will be inserted. * @param string $propertyname name of the private property. * @return ReflectionProperty the resulting reflection property. */ protected function set_private_property($value, $propertyname) { $refclient = new ReflectionClass($this->repo); $private = $refclient->getProperty($propertyname); $private->setAccessible(true); $private->setValue($this->repo, $value); return $private; } /** * Helper method to set required return parameters for get_listing. * * @return array array, which contains the parameters. */ protected function get_initialised_return_array() { $ret = array(); $ret['dynload'] = true; $ret['nosearch'] = true; $ret['nologin'] = false; $ret['path'] = [ [ 'name' => $this->repo->get_meta()->name, 'path' => '', ] ]; $ret['manage'] = ''; $ret['defaultreturntype'] = FILE_INTERNAL; $ret['list'] = array(); return $ret; } }