AutorÃa | Ultima modificación | Ver Log |
<?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 search_solr;/*** Solr search engine unit tests that can operate using a mock http_client and without creating a* search manager instance.** These tests can run without the solr PHP extension.** All 'realistic' tests of searching (e.g. index something then see if it is found by search)* require a real Solr instance for testing and should be placed in {@see engine_test}.* Tests that don't rely heavily on the real search functionality, or where we need to simulate* multiple different ways of configuring the search infrastructure, or unusual failures in* communication, may be better suited for this mock test approach.** @package search_solr* @category test* @copyright 2024 The Open University* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later* @covers \search_solr\engine*/final class mock_engine_test extends \advanced_testcase {protected function setUp(): void {parent::setUp();$this->resetAfterTest();// Minimal configuration.set_config('server_hostname', 'host.invalid', 'search_solr');set_config('indexname', 'myindex', 'search_solr');// This is not necessary on my setup but in GitHub Actions, the server_port is set to ''// instead of default 8983.set_config('server_port', '8983', 'search_solr');}/*** Tests {@see engine::get_server_url}.*/public function test_get_server_url(): void {// Basic URL.$engine = new engine();$this->assertEquals('http://host.invalid:8983/solr/',$engine->get_server_url('')->out(false),);// Same but with specified path.$this->assertEquals('http://host.invalid:8983/solr/twiddle',$engine->get_server_url('twiddle')->out(false),);// Slash at start of path will be stripped.$this->assertEquals('http://host.invalid:8983/solr/twiddle',$engine->get_server_url('/twiddle')->out(false),);// Turn on https. Due to the way the port setting works, which is bad, this will still have// the default not-secure port (even though the 'default' on the setting page will now be// shown as 8443, hmm). User has to change it manually.set_config('secure', '1', 'search_solr');$engine = new engine();$this->assertEquals('https://host.invalid:8983/solr/',$engine->get_server_url('')->out(false),);// Change port from default. User has to do this manually when enabling secure.set_config('server_port', '8443', 'search_solr');$engine = new engine();$this->assertEquals('https://host.invalid:8443/solr/',$engine->get_server_url('')->out(false),);}/*** Tests {@see engine::get_connection_url}.*/public function test_get_connection_url(): void {// Basic URL.$engine = new engine();$this->assertEquals('http://host.invalid:8983/solr/myindex/',$engine->get_connection_url('')->out(false),);}/*** Tests {@see engine::raw_get_request()} with no auth settings.*/public function test_raw_get_request_no_auth(): void {$engine = new engine();$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// When there is no auth, there aren't many options, just timeout.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));// Timeout can be changed in config.set_config('server_timeout', '10', 'search_solr');$engine = new engine();$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/frog',['connect_timeout' => 10,'read_timeout' => 10,],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));}/*** Tests {@see engine::raw_get_request()} with basic auth settings.*/public function test_raw_get_request_basic_auth(): void {set_config('server_username', 'u', 'search_solr');set_config('server_password', 'p', 'search_solr');$engine = new engine();$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// Basic auth works with an 'auth' option.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'auth' => ['u', 'p'],],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));}/*** Tests {@see engine::raw_get_request()} with a supplied user certificate.*/public function test_raw_get_request_user_cert(): void {set_config('secure', '1', 'search_solr');set_config('ssl_cert', '/tmp/cert.pem', 'search_solr');$engine = new engine();$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// User cert auth uses the 'cert' parameter, with or without a key.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'cert' => '/tmp/cert.pem',],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));}/*** Tests {@see engine::raw_get_request()} with a user key (with or without password).*/public function test_raw_get_request_user_key(): void {set_config('secure', '1', 'search_solr');set_config('ssl_key', '/tmp/key.pem', 'search_solr');$engine = new engine();$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// User cert auth uses the 'cert' parameter, with or without a key.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'ssl_key' => '/tmp/key.pem',],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));set_config('ssl_keypassword', 'frog', 'search_solr');$engine = new engine();$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'ssl_key' => ['/tmp/key.pem', 'frog'],],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));}/*** Tests {@see engine::raw_get_request()} with a certificate bundle for verifying the server.*/public function test_raw_get_request_certificate_bundle(): void {set_config('secure', '1', 'search_solr');set_config('ssl_cainfo', '/tmp/allthecerts.pem', 'search_solr');$engine = new engine();$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// User cert auth uses the 'cert' parameter, with or without a key.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'verify' => '/tmp/allthecerts.pem',],)->willReturn($response);$this->assertEquals($response, $engine->raw_get_request('frog'));}/*** Tests {@see engine::raw_get_request()} with a certificate folder for verifying the server.* Guzzle doesn't support a certificate folder (curl does) so this code makes a bundle in the* localcache area.*/public function test_raw_get_request_certificate_folder(): void {global $CFG;// Make a directory full of fake .pem files.$temp = make_request_directory();file_put_contents($temp . '/0.pem', "PEM0\n");file_put_contents($temp . '/1.pem', "PEM1\n");file_put_contents($temp . '/2.txt', "TXT2\n");set_config('secure', '1', 'search_solr');set_config('ssl_capath', $temp, 'search_solr');$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);// Party like it's 13 February 2009.$time = 1234567890;$this->mock_clock_with_frozen($time);// User cert auth uses the 'cert' parameter, with or without a key.$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);// The filename is the hash of the capath setting plus current time.$combinedfile = $CFG->dataroot .'/localcache/search_solr/capath.' .sha1($temp) .'.1234567890';$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'verify' => $combinedfile,],)->willReturn($response);$engine = new engine();$this->assertEquals($response, $engine->raw_get_request('frog'));// Check the file actually is the .pem files concatenated.$this->assertEquals("PEM0\n\n\nPEM1\n\n\n", file_get_contents($combinedfile));// Let's add another .pem file.file_put_contents($temp . '/3.pem', "PEM3\n");// 9 minutes 59 seconds later, it will still use the cached version (same file).$time += 599;$this->mock_clock_with_frozen($time);$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'verify' => $combinedfile,],)->willReturn($response);$engine = new engine();$this->assertEquals($response, $engine->raw_get_request('frog'));$this->assertEquals("PEM0\n\n\nPEM1\n\n\n", file_get_contents($combinedfile));// 10 minutes later, it will make a new cached version.$time += 1;$this->mock_clock_with_frozen($time);$combinedfile2 = $CFG->dataroot .'/localcache/search_solr/capath.' .sha1($temp) .'.1234568490';$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'verify' => $combinedfile2,],)->willReturn($response);$engine = new engine();$this->assertEquals($response, $engine->raw_get_request('frog'));$this->assertEquals("PEM0\n\n\nPEM1\n\n\nPEM3\n\n\n", file_get_contents($combinedfile2));// The old file is still there.$this->assertEquals("PEM0\n\n\nPEM1\n\n\n", file_get_contents($combinedfile));// Go another minute. We're still using the same combined file...$time += 60;$this->mock_clock_with_frozen($time);$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('https://host.invalid:8983/solr/frog',['connect_timeout' => 30,'read_timeout' => 30,'verify' => $combinedfile2,],)->willReturn($response);$engine = new engine();$this->assertEquals($response, $engine->raw_get_request('frog'));$this->assertEquals("PEM0\n\n\nPEM1\n\n\nPEM3\n\n\n", file_get_contents($combinedfile2));// But now it will delete the old one.$this->assertFalse(file_exists($combinedfile));}/*** Tests the {@see engine::get_status()} function when there is an exception connecting.*/public function test_get_status_exception_connecting(): void {$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willThrowException(new \coding_exception('ex'));$engine = new engine();$status = $engine->get_status();$this->assertFalse($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Exception occurred: Coding error detected, it must be fixed by a programmer: ex',$status['error'],);$this->assertInstanceOf(\coding_exception::class, $status['exception']);}/*** Tests the {@see engine::get_status()} function when the server returns 404.*/public function test_get_status_bad_http_status(): void {$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);$response->method('getStatusCode')->willReturn(404);$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertFalse($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Unsuccessful status code: 404', $status['error']);}/*** Creates a mock ResponseInterface with a body containing the specified string.** @param string $body Body content* @return \Psr\Http\Message\ResponseInterface Interface*/protected function get_fake_response(string $body): \Psr\Http\Message\ResponseInterface {$response = $this->createStub(\Psr\Http\Message\ResponseInterface::class);$response->method('getStatusCode')->willReturn(200);$stream = $this->createStub(\Psr\Http\Message\StreamInterface::class);$response->method('getBody')->willReturn($stream);$stream->method('getContents')->willReturn($body);return $response;}/*** Tests the {@see engine::get_status()} function when the server returns invalid JSON.* In real life this would only be likely to happen if the server is down and a load balancer* in front of it for some crazy reason interposes a page with status 200.*/public function test_get_status_not_json(): void {$response = $this->get_fake_response('notjson');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertFalse($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Invalid JSON', $status['error']);}/*** Tests the {@see engine::get_status()} function when the server returns an empty response.** This could maybe happen if the server has been configured, but not fully initialised.*/public function test_get_status_no_cores(): void {$response = $this->get_fake_response('{}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Unexpected JSON: no core status', $status['error']);}/*** Tests the {@see engine::get_status()} function when the server returns a core without a name* we can read.** In real usage this should only happen if the Solr REST interface changes unexpectedly.*/public function test_get_status_core_no_name(): void {// A core with no name (in its 'name' field, the 'frog' key is ignored).$response = $this->get_fake_response('{"status":{"frog":{}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Unexpected JSON: core has no name', $status['error']);}/*** Tests the {@see engine::get_status()} function when the server doesn't return status for a* core that matches the index name in Moodle config.** In real usage this could happen if the index got wiped from search or something.*/public function test_get_status_no_matching_core(): void {// Core is not the one we're looking for.$response = $this->get_fake_response('{"status":{"frog":{"name":"frog"}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertFalse($status['foundcore']);$this->assertEquals('Could not find core matching myindex', $status['error']);}/*** Tests the {@see engine::get_status()} function when the server returns a core without index* information.** In real usage this should only happen if the Solr REST interface changes unexpectedly. There* is a parameter to not receive index information, but we don't use it.*/public function test_get_status_core_no_index(): void {// Core exists but has no index object.$response = $this->get_fake_response('{"status":{"myindex":{"name":"myindex"}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertTrue($status['foundcore']);$this->assertEquals('Unexpected JSON: core has no index', $status['error']);}/*** Tests the {@see engine::get_status()} function when the server returns index information* without size.** In real usage this should only happen if the Solr REST interface changes unexpectedly.*/public function test_get_status_core_index_no_size(): void {// Core index objects doesn't have a size.$response = $this->get_fake_response('{"status":{"myindex":{"name":"myindex","index":{}}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertTrue($status['foundcore']);$this->assertEquals('Unexpected JSON: core index has no sizeInBytes', $status['error']);}/*** Tests the {@see engine::get_status()} function when all desired data is present, using a* single-instance Solr configuration.*/public function test_get_status_success_single_server(): void {// Core index complete with size.$response = $this->get_fake_response('{"status":{"myindex":{"name":"myindex",' .'"index":{"sizeInBytes":123}}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertTrue($status['foundcore']);$this->assertEquals(123, $status['indexsize']);}/*** Tests the {@see engine::get_status()} function when all desired data is present, using a* multiple-instance (SolrCloud) configuration.*/public function test_get_status_success_solr_cloud(): void {// Index with size, in cloud replica. These have a different name for each node but a// 'collection' field with the original index name.$response = $this->get_fake_response('{"status":{"replica1":{"name":"replica1",' .'"cloud":{"collection":"myindex"},"index":{"sizeInBytes":123}}}}');$mockedclient = $this->createMock(\core\http_client::class);\core\di::set(\core\http_client::class, $mockedclient);$mockedclient->expects($this->once())->method('get')->with('http://host.invalid:8983/solr/admin/cores',['connect_timeout' => 30,'read_timeout' => 30,],)->willReturn($response);$engine = new engine();$status = $engine->get_status();$this->assertTrue($status['connected']);$this->assertTrue($status['foundcore']);$this->assertEquals(123, $status['indexsize']);}}