2 // This file is part of Moodle - http://moodle.org/
4 // Moodle is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
9 // Moodle is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
18 * Tests for antivirus manager.
20 * @package core_antivirus
22 * @copyright 2016 Ruslan Kabalin, Lancaster University.
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
26 defined('MOODLE_INTERNAL') ||
die();
27 require_once(__DIR__
. '/fixtures/testable_antivirus.php');
29 class core_antivirus_testcase
extends advanced_testcase
{
32 protected function setUp(): void
{
34 // Use our special testable fixture plugin.
35 $CFG->antiviruses
= 'testable';
37 $this->resetAfterTest();
40 $tempfolder = make_request_directory(false);
41 $this->tempfile
= $tempfolder . '/' . rand();
42 touch($this->tempfile
);
45 protected function tearDown(): void
{
46 @unlink
($this->tempfile
);
49 public function test_manager_get_antivirus() {
50 // We are using clamav plugin in the test,
51 // as the only plugin we know exists for sure.
52 $antivirusviaget = \core\antivirus\manager
::get_antivirus('clamav');
53 $antivirusdirect = new \antivirus_clamav\
scanner();
54 $this->assertEquals($antivirusdirect, $antivirusviaget);
57 public function test_manager_scan_file_no_virus() {
59 $this->assertFileExists($this->tempfile
);
60 $this->assertEmpty(\core\antivirus\manager
::scan_file($this->tempfile
, 'OK', true));
61 // File expected to remain in place.
62 $this->assertFileExists($this->tempfile
);
65 public function test_manager_scan_file_error() {
67 $this->assertFileExists($this->tempfile
);
68 $this->assertEmpty(\core\antivirus\manager
::scan_file($this->tempfile
, 'ERROR', true));
69 // File expected to remain in place.
70 $this->assertFileExists($this->tempfile
);
73 public function test_manager_scan_file_virus() {
74 // Run mock scanning without deleting infected file.
75 $this->assertFileExists($this->tempfile
);
76 $this->expectException(\core\antivirus\scanner_exception
::class);
77 $this->assertEmpty(\core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', false));
78 // File expected to remain in place.
79 $this->assertFileExists($this->tempfile
);
81 // Run mock scanning with deleting infected file.
82 $this->expectException(\core\antivirus\scanner_exception
::class);
83 $this->assertEmpty(\core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', true));
84 // File expected to be deleted.
85 $this->assertFileDoesNotExist($this->tempfile
);
88 public function test_manager_send_message_to_user_email_scan_file_virus() {
89 $sink = $this->redirectEmails();
92 set_config('notifyemail', 'fake@example.com', 'antivirus');
93 \core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', true);
94 } catch (\core\antivirus\scanner_exception
$ex) {
97 $this->assertNotEmpty($exception);
98 $result = $sink->get_messages();
99 $this->assertCount(1, $result);
100 $this->assertStringContainsString('fake@example.com', $result[0]->to
);
104 public function test_manager_send_message_to_admin_email_scan_file_virus() {
105 $sink = $this->redirectMessages();
108 \core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', true);
109 } catch (\core\antivirus\scanner_exception
$ex) {
112 $this->assertNotEmpty($exception);
113 $result = $sink->get_messages();
114 $admins = array_keys(get_admins());
115 $this->assertCount(1, $admins);
116 $this->assertCount(1, $result);
117 $this->assertEquals($result[0]->useridto
, reset($admins));
121 public function test_manager_quarantine_file_virus() {
123 set_config('enablequarantine', true, 'antivirus');
124 \core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', true);
125 } catch (\core\antivirus\scanner_exception
$ex) {
128 $this->assertNotEmpty($exception);
129 // Quarantined files.
130 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
131 $this->assertEquals(1, count($quarantinedfiles));
133 \core\antivirus\quarantine
::clean_up_quarantine_folder(time());
134 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
135 $this->assertEquals(0, count($quarantinedfiles));
138 public function test_manager_none_quarantine_file_virus() {
140 \core\antivirus\manager
::scan_file($this->tempfile
, 'FOUND', true);
141 } catch (\core\antivirus\scanner_exception
$ex) {
144 $this->assertNotEmpty($exception);
145 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
146 $this->assertEquals(0, count($quarantinedfiles));
149 public function test_manager_scan_data_no_virus() {
150 // Run mock scanning.
151 $this->assertEmpty(\core\antivirus\manager
::scan_data('OK'));
154 public function test_manager_scan_data_error() {
155 // Run mock scanning.
156 $this->assertEmpty(\core\antivirus\manager
::scan_data('ERROR'));
159 public function test_manager_scan_data_virus() {
160 // Run mock scanning.
161 $this->expectException(\core\antivirus\scanner_exception
::class);
162 $this->assertEmpty(\core\antivirus\manager
::scan_data('FOUND'));
165 public function test_manager_send_message_to_user_email_scan_data_virus() {
166 $sink = $this->redirectEmails();
167 set_config('notifyemail', 'fake@example.com', 'antivirus');
170 \core\antivirus\manager
::scan_data('FOUND');
171 } catch (\core\antivirus\scanner_exception
$ex) {
174 $this->assertNotEmpty($exception);
175 $result = $sink->get_messages();
176 $this->assertCount(1, $result);
177 $this->assertStringContainsString('fake@example.com', $result[0]->to
);
181 public function test_manager_send_message_to_admin_email_scan_data_virus() {
182 $sink = $this->redirectMessages();
185 \core\antivirus\manager
::scan_data('FOUND');
186 } catch (\core\antivirus\scanner_exception
$ex) {
189 $this->assertNotEmpty($exception);
190 $result = $sink->get_messages();
191 $admins = array_keys(get_admins());
192 $this->assertCount(1, $admins);
193 $this->assertCount(1, $result);
194 $this->assertEquals($result[0]->useridto
, reset($admins));
198 public function test_manager_quarantine_data_virus() {
199 set_config('enablequarantine', true, 'antivirus');
202 \core\antivirus\manager
::scan_data('FOUND');
203 } catch (\core\antivirus\scanner_exception
$ex) {
206 $this->assertNotEmpty($exception);
207 // Quarantined files.
208 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
209 $this->assertEquals(1, count($quarantinedfiles));
211 \core\antivirus\quarantine
::clean_up_quarantine_folder(time());
212 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
213 $this->assertEquals(0, count($quarantinedfiles));
217 public function test_manager_none_quarantine_data_virus() {
220 \core\antivirus\manager
::scan_data('FOUND');
221 } catch (\core\antivirus\scanner_exception
$ex) {
224 $this->assertNotEmpty($exception);
225 // No Quarantined files.
226 $quarantinedfiles = \core\antivirus\quarantine
::get_quarantined_files();
227 $this->assertEquals(0, count($quarantinedfiles));