Merge branch 'MDL-77468-master' of https://github.com/rmady/moodle
[moodle.git] / comment / tests / externallib_test.php
blob4ac3746dc2ffc1b75d5cf403e9bb7a0064b1efdd
1 <?php
2 // This file is part of Moodle - http://moodle.org/
3 //
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.
8 //
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/>.
17 namespace core_comment;
19 use comment_exception;
20 use core_comment_external;
21 use core_external\external_api;
22 use externallib_advanced_testcase;
24 defined('MOODLE_INTERNAL') || die();
26 global $CFG;
28 require_once($CFG->dirroot . '/webservice/tests/helpers.php');
30 /**
31 * External comment functions unit tests
33 * @package core_comment
34 * @category external
35 * @copyright 2015 Juan Leyva <juan@moodle.com>
36 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
37 * @since Moodle 2.9
39 class externallib_test extends externallib_advanced_testcase {
41 /**
42 * Tests set up
44 protected function setUp(): void {
45 $this->resetAfterTest();
48 /**
49 * Helper used to set up a course, with a module, a teacher and two students.
51 * @return array the array of records corresponding to the course, teacher, and students.
53 protected function setup_course_and_users_basic() {
54 global $CFG, $DB;
56 require_once($CFG->dirroot . '/comment/lib.php');
58 $CFG->usecomments = true;
60 $student1 = $this->getDataGenerator()->create_user();
61 $student2 = $this->getDataGenerator()->create_user();
62 $teacher1 = $this->getDataGenerator()->create_user();
63 $course1 = $this->getDataGenerator()->create_course(array('enablecomment' => 1));
64 $studentrole = $DB->get_record('role', array('shortname' => 'student'));
65 $teacherrole = $DB->get_record('role', array('shortname' => 'editingteacher'));
66 $this->getDataGenerator()->enrol_user($student1->id, $course1->id, $studentrole->id);
67 $this->getDataGenerator()->enrol_user($student2->id, $course1->id, $studentrole->id);
68 $this->getDataGenerator()->enrol_user($teacher1->id, $course1->id, $teacherrole->id);
70 // Create a database module instance.
71 $record = new \stdClass();
72 $record->course = $course1->id;
73 $record->name = "Mod data test";
74 $record->intro = "Some intro of some sort";
75 $record->comments = 1;
77 $module1 = $this->getDataGenerator()->create_module('data', $record);
78 $field = data_get_field_new('text', $module1);
80 $fielddetail = new \stdClass();
81 $fielddetail->name = 'Name';
82 $fielddetail->description = 'Some name';
84 $field->define_field($fielddetail);
85 $field->insert_field();
86 $recordid = data_add_record($module1);
88 $datacontent = array();
89 $datacontent['fieldid'] = $field->field->id;
90 $datacontent['recordid'] = $recordid;
91 $datacontent['content'] = 'Asterix';
92 $DB->insert_record('data_content', $datacontent);
94 return [$module1, $recordid, $teacher1, $student1, $student2];
97 /**
98 * Test get_comments
100 public function test_get_comments() {
101 global $CFG;
102 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
104 // Create some comments as student 1.
105 $this->setUser($student1);
106 $inputdata = [
108 'contextlevel' => 'module',
109 'instanceid' => $module1->cmid,
110 'component' => 'mod_data',
111 'content' => 'abc',
112 'itemid' => $recordid,
113 'area' => 'database_entry'
116 'contextlevel' => 'module',
117 'instanceid' => $module1->cmid,
118 'component' => 'mod_data',
119 'content' => 'def',
120 'itemid' => $recordid,
121 'area' => 'database_entry'
124 $result = core_comment_external::add_comments($inputdata);
125 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
126 $ids = array_column($result, 'id');
128 // Verify we can get the comments.
129 $contextlevel = 'module';
130 $instanceid = $module1->cmid;
131 $component = 'mod_data';
132 $itemid = $recordid;
133 $area = 'database_entry';
134 $page = 0;
135 $result = core_comment_external::get_comments($contextlevel, $instanceid, $component, $itemid, $area, $page);
136 $result = external_api::clean_returnvalue(core_comment_external::get_comments_returns(), $result);
138 $this->assertCount(0, $result['warnings']);
139 $this->assertCount(2, $result['comments']);
140 $this->assertEquals(2, $result['count']);
141 $this->assertEquals(15, $result['perpage']);
142 $this->assertTrue($result['canpost']);
144 $this->assertEquals($student1->id, $result['comments'][0]['userid']);
145 $this->assertEquals($student1->id, $result['comments'][1]['userid']);
147 $this->assertEquals($ids[1], $result['comments'][0]['id']); // Default ordering newer first.
148 $this->assertEquals($ids[0], $result['comments'][1]['id']);
150 // Test sort direction and pagination.
151 $CFG->commentsperpage = 1;
152 $result = core_comment_external::get_comments($contextlevel, $instanceid, $component, $itemid, $area, $page, 'ASC');
153 $result = external_api::clean_returnvalue(core_comment_external::get_comments_returns(), $result);
155 $this->assertCount(0, $result['warnings']);
156 $this->assertCount(1, $result['comments']); // Only one per page.
157 $this->assertEquals(2, $result['count']);
158 $this->assertEquals($CFG->commentsperpage, $result['perpage']);
159 $this->assertEquals($ids[0], $result['comments'][0]['id']); // Comments order older first.
161 // Next page.
162 $result = core_comment_external::get_comments($contextlevel, $instanceid, $component, $itemid, $area, $page + 1, 'ASC');
163 $result = external_api::clean_returnvalue(core_comment_external::get_comments_returns(), $result);
165 $this->assertCount(0, $result['warnings']);
166 $this->assertCount(1, $result['comments']);
167 $this->assertEquals(2, $result['count']);
168 $this->assertEquals($CFG->commentsperpage, $result['perpage']);
169 $this->assertEquals($ids[1], $result['comments'][0]['id']);
173 * Test add_comments not enabled site level
175 public function test_add_comments_not_enabled_site_level() {
176 global $CFG;
177 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
179 // Try to add a comment, as student 1, when comments is disabled at site level.
180 $this->setUser($student1);
181 $CFG->usecomments = false;
183 $this->expectException(comment_exception::class);
184 core_comment_external::add_comments([
186 'contextlevel' => 'module',
187 'instanceid' => $module1->cmid,
188 'component' => 'mod_data',
189 'content' => 'abc',
190 'itemid' => $recordid,
191 'area' => 'database_entry'
197 * Test add_comments not enabled module level
199 public function test_add_comments_not_enabled_module_level() {
200 global $DB;
201 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
203 // Disable comments for the module.
204 $DB->set_field('data', 'comments', 0, array('id' => $module1->id));
206 // Verify we can't add a comment.
207 $this->setUser($student1);
208 $this->expectException(comment_exception::class);
209 core_comment_external::add_comments([
211 'contextlevel' => 'module',
212 'instanceid' => $module1->cmid,
213 'component' => 'mod_data',
214 'content' => 'abc',
215 'itemid' => $recordid,
216 'area' => 'database_entry'
222 * Test add_comments
224 public function test_add_comments_single() {
225 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
227 // Add a comment as student 1.
228 $this->setUser($student1);
229 $result = core_comment_external::add_comments([
231 'contextlevel' => 'module',
232 'instanceid' => $module1->cmid,
233 'component' => 'mod_data',
234 'content' => 'abc',
235 'itemid' => $recordid,
236 'area' => 'database_entry'
239 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
241 // Verify the result contains 1 result having the correct structure.
242 $this->assertCount(1, $result);
244 $expectedkeys = [
245 'id',
246 'content',
247 'format',
248 'timecreated',
249 'strftimeformat',
250 'profileurl',
251 'fullname',
252 'time',
253 'avatar',
254 'userid',
255 'delete',
257 foreach ($expectedkeys as $key) {
258 $this->assertArrayHasKey($key, $result[0]);
263 * Test add_comments when one of the comments contains invalid data and cannot be created.
265 * This simply verifies that the entire operation fails.
267 public function test_add_comments_multiple_contains_invalid() {
268 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
270 // Try to create some comments as student 1, but provide a bad area for the second comment.
271 $this->setUser($student1);
272 $this->expectException(comment_exception::class);
273 core_comment_external::add_comments([
275 'contextlevel' => 'module',
276 'instanceid' => $module1->cmid,
277 'component' => 'mod_data',
278 'content' => 'abc',
279 'itemid' => $recordid,
280 'area' => 'database_entry'
283 'contextlevel' => 'module',
284 'instanceid' => $module1->cmid,
285 'component' => 'mod_data',
286 'content' => 'def',
287 'itemid' => $recordid,
288 'area' => 'badarea'
294 * Test add_comments when one of the comments contains invalid data and cannot be created.
296 * This simply verifies that the entire operation fails.
298 public function test_add_comments_multiple_all_valid() {
299 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
301 // Try to create some comments as student 1.
302 $this->setUser($student1);
303 $inputdata = [
305 'contextlevel' => 'module',
306 'instanceid' => $module1->cmid,
307 'component' => 'mod_data',
308 'content' => 'abc',
309 'itemid' => $recordid,
310 'area' => 'database_entry'
313 'contextlevel' => 'module',
314 'instanceid' => $module1->cmid,
315 'component' => 'mod_data',
316 'content' => 'def',
317 'itemid' => $recordid,
318 'area' => 'database_entry'
321 $result = core_comment_external::add_comments($inputdata);
322 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
324 // Two comments should have been created.
325 $this->assertCount(2, $result);
327 // The content for each comment should come back formatted.
328 foreach ($result as $index => $comment) {
329 $formatoptions = array('overflowdiv' => true, 'blanktarget' => true);
330 $expectedcontent = format_text($inputdata[$index]['content'], FORMAT_MOODLE, $formatoptions);
331 $this->assertEquals($expectedcontent, $comment['content']);
336 * Test add_comments invalid area
338 public function test_add_comments_invalid_area() {
339 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
341 // Try to create a comment with an invalid area, verifying failure.
342 $this->setUser($student1);
343 $comments = [
345 'contextlevel' => 'module',
346 'instanceid' => $module1->cmid,
347 'component' => 'mod_data',
348 'content' => 'abc',
349 'itemid' => $recordid,
350 'area' => 'spaghetti'
353 $this->expectException(comment_exception::class);
354 core_comment_external::add_comments($comments);
358 * Test delete_comment invalid comment.
360 public function test_delete_comments_invalid_comment_id() {
361 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
362 $this->setUser($student1);
364 $this->expectException(comment_exception::class);
365 core_comment_external::delete_comments([-1, 0]);
369 * Test delete_comment own user.
371 public function test_delete_comments_own_user() {
372 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
374 // Create a few comments as student 1.
375 $this->setUser($student1);
376 $result = core_comment_external::add_comments([
378 'contextlevel' => 'module',
379 'instanceid' => $module1->cmid,
380 'component' => 'mod_data',
381 'content' => 'abc',
382 'itemid' => $recordid,
383 'area' => 'database_entry'
386 'contextlevel' => 'module',
387 'instanceid' => $module1->cmid,
388 'component' => 'mod_data',
389 'content' => 'def',
390 'itemid' => $recordid,
391 'area' => 'database_entry'
394 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
396 // Delete those comments we just created.
397 $result = core_comment_external::delete_comments([
398 $result[0]['id'],
399 $result[1]['id']
401 $result = external_api::clean_returnvalue(core_comment_external::delete_comments_returns(), $result);
402 $this->assertEquals([], $result);
406 * Test delete_comment other student.
408 public function test_delete_comment_other_student() {
409 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
411 // Create a comment as the student.
412 $this->setUser($student1);
413 $result = core_comment_external::add_comments([
415 'contextlevel' => 'module',
416 'instanceid' => $module1->cmid,
417 'component' => 'mod_data',
418 'content' => 'abc',
419 'itemid' => $recordid,
420 'area' => 'database_entry'
423 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
425 // Now, as student 2, try to delete the comment made by student 1. Verify we can't.
426 $this->setUser($student2);
427 $this->expectException(comment_exception::class);
428 core_comment_external::delete_comments([$result[0]['id']]);
432 * Test delete_comment as teacher.
434 public function test_delete_comments_as_teacher() {
435 [$module1, $recordid, $teacher1, $student1, $student2] = $this->setup_course_and_users_basic();
437 // Create a comment as the student.
438 $this->setUser($student1);
439 $result = core_comment_external::add_comments([
441 'contextlevel' => 'module',
442 'instanceid' => $module1->cmid,
443 'component' => 'mod_data',
444 'content' => 'abc',
445 'itemid' => $recordid,
446 'area' => 'database_entry'
449 $result = external_api::clean_returnvalue(core_comment_external::add_comments_returns(), $result);
451 // Verify teachers can delete the comment.
452 $this->setUser($teacher1);
453 $result = core_comment_external::delete_comments([$result[0]['id']]);
454 $result = external_api::clean_returnvalue(core_comment_external::delete_comments_returns(), $result);
455 $this->assertEquals([], $result);