Proyectos de Subversion Moodle

Rev

Rev 11 | | Comparar con el anterior | Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1 efrain 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.
13
//
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/>.
16
 
17
namespace mod_forum;
18
 
19
use core_external\external_api;
20
use externallib_advanced_testcase;
21
use mod_forum_external;
22
 
23
defined('MOODLE_INTERNAL') || die();
24
 
25
global $CFG;
26
 
27
require_once($CFG->dirroot . '/webservice/tests/helpers.php');
28
require_once($CFG->dirroot . '/mod/forum/lib.php');
29
 
30
/**
31
 * The module forums external functions unit tests
32
 *
33
 * @package    mod_forum
34
 * @category   external
35
 * @copyright  2012 Mark Nelson <markn@moodle.com>
36
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
37
 */
38
final class externallib_test extends externallib_advanced_testcase {
39
 
40
    /**
41
     * Tests set up
42
     */
43
    protected function setUp(): void {
44
        global $CFG;
1441 ariadna 45
        parent::setUp();
1 efrain 46
 
47
        // We must clear the subscription caches. This has to be done both before each test, and after in case of other
48
        // tests using these functions.
49
        \mod_forum\subscriptions::reset_forum_cache();
50
 
51
        require_once($CFG->dirroot . '/mod/forum/externallib.php');
52
    }
53
 
54
    public function tearDown(): void {
55
        // We must clear the subscription caches. This has to be done both before each test, and after in case of other
56
        // tests using these functions.
57
        \mod_forum\subscriptions::reset_forum_cache();
1441 ariadna 58
        parent::tearDown();
1 efrain 59
    }
60
 
61
    /**
62
     * Get the expected attachment.
63
     *
64
     * @param \stored_file $file
65
     * @param array $values
66
     * @param \moodle_url|null $url
67
     * @return array
68
     */
69
    protected function get_expected_attachment(\stored_file $file, array $values  = [], ?\moodle_url $url = null): array {
70
        if (!$url) {
71
            $url = \moodle_url::make_pluginfile_url(
72
                $file->get_contextid(),
73
                $file->get_component(),
74
                $file->get_filearea(),
75
                $file->get_itemid(),
76
                $file->get_filepath(),
77
                $file->get_filename()
78
            );
79
            $url->param('forcedownload', 1);
80
        }
81
 
82
        return array_merge(
83
            [
84
                'contextid' => $file->get_contextid(),
85
                'component' => $file->get_component(),
86
                'filearea' => $file->get_filearea(),
87
                'itemid' => $file->get_itemid(),
88
                'filepath' => $file->get_filepath(),
89
                'filename' => $file->get_filename(),
90
                'isdir' => $file->is_directory(),
91
                'isimage' => $file->is_valid_image(),
92
                'timemodified' => $file->get_timemodified(),
93
                'timecreated' => $file->get_timecreated(),
94
                'filesize' => $file->get_filesize(),
95
                'author' => $file->get_author(),
96
                'license' => $file->get_license(),
97
                'filenameshort' => $file->get_filename(),
98
                'filesizeformatted' => display_size((int) $file->get_filesize()),
99
                'icon' => $file->is_directory() ? file_folder_icon() : file_file_icon($file),
100
                'timecreatedformatted' => userdate($file->get_timecreated()),
101
                'timemodifiedformatted' => userdate($file->get_timemodified()),
102
                'url' => $url->out(),
103
            ], $values);
104
    }
105
 
106
    /**
107
     * Test get forums
108
     */
11 efrain 109
    public function test_mod_forum_get_forums_by_courses(): void {
1 efrain 110
        global $USER, $CFG, $DB;
111
 
112
        $this->resetAfterTest(true);
113
 
114
        // Create a user.
115
        $user = self::getDataGenerator()->create_user(array('trackforums' => 1));
116
 
117
        // Set to the user.
118
        self::setUser($user);
119
 
120
        // Create courses to add the modules.
121
        $course1 = self::getDataGenerator()->create_course();
122
        $course2 = self::getDataGenerator()->create_course();
123
 
124
        // First forum.
125
        $record = new \stdClass();
126
        $record->introformat = FORMAT_HTML;
127
        $record->course = $course1->id;
128
        $record->trackingtype = FORUM_TRACKING_FORCED;
129
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
130
 
131
        // Second forum.
132
        $record = new \stdClass();
133
        $record->introformat = FORMAT_HTML;
134
        $record->course = $course2->id;
135
        $record->trackingtype = FORUM_TRACKING_OFF;
136
        $forum2 = self::getDataGenerator()->create_module('forum', $record);
137
        $forum2->introfiles = [];
138
 
139
        // Add discussions to the forums.
140
        $record = new \stdClass();
141
        $record->course = $course1->id;
142
        $record->userid = $user->id;
143
        $record->forum = $forum1->id;
144
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
145
        // Expect one discussion.
146
        $forum1->numdiscussions = 1;
147
        $forum1->cancreatediscussions = true;
148
        $forum1->istracked = true;
149
        $forum1->unreadpostscount = 0;
150
        $forum1->introfiles = [];
151
        $forum1->lang = '';
152
 
153
        $record = new \stdClass();
154
        $record->course = $course2->id;
155
        $record->userid = $user->id;
156
        $record->forum = $forum2->id;
157
        $discussion2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
158
        $discussion3 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
159
        // Expect two discussions.
160
        $forum2->numdiscussions = 2;
161
        // Default limited role, no create discussion capability enabled.
162
        $forum2->cancreatediscussions = false;
163
        $forum2->istracked = false;
164
        $forum2->lang = '';
165
 
166
        // Check the forum was correctly created.
167
        $this->assertEquals(2, $DB->count_records_select('forum', 'id = :forum1 OR id = :forum2',
168
                array('forum1' => $forum1->id, 'forum2' => $forum2->id)));
169
 
170
        // Enrol the user in two courses.
171
        // DataGenerator->enrol_user automatically sets a role for the user with the permission mod/form:viewdiscussion.
172
        $this->getDataGenerator()->enrol_user($user->id, $course1->id, null, 'manual');
173
        // Execute real Moodle enrolment as we'll call unenrol() method on the instance later.
174
        $enrol = enrol_get_plugin('manual');
175
        $enrolinstances = enrol_get_instances($course2->id, true);
176
        foreach ($enrolinstances as $courseenrolinstance) {
177
            if ($courseenrolinstance->enrol == "manual") {
178
                $instance2 = $courseenrolinstance;
179
                break;
180
            }
181
        }
182
        $enrol->enrol_user($instance2, $user->id);
183
 
184
        // Assign capabilities to view forums for forum 2.
185
        $cm2 = get_coursemodule_from_id('forum', $forum2->cmid, 0, false, MUST_EXIST);
186
        $context2 = \context_module::instance($cm2->id);
187
        $newrole = create_role('Role 2', 'role2', 'Role 2 description');
188
        $roleid2 = $this->assignUserCapability('mod/forum:viewdiscussion', $context2->id, $newrole);
189
 
190
        // Create what we expect to be returned when querying the two courses.
191
        unset($forum1->displaywordcount);
192
        unset($forum2->displaywordcount);
193
 
194
        $expectedforums = array();
195
        $expectedforums[$forum1->id] = (array) $forum1;
196
        $expectedforums[$forum2->id] = (array) $forum2;
197
 
198
        // Call the external function passing course ids.
199
        $forums = mod_forum_external::get_forums_by_courses(array($course1->id, $course2->id));
200
        $forums = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $forums);
201
        $this->assertCount(2, $forums);
202
        foreach ($forums as $forum) {
203
            $this->assertEquals($expectedforums[$forum['id']], $forum);
204
        }
205
 
206
        // Call the external function without passing course id.
207
        $forums = mod_forum_external::get_forums_by_courses();
208
        $forums = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $forums);
209
        $this->assertCount(2, $forums);
210
        foreach ($forums as $forum) {
211
            $this->assertEquals($expectedforums[$forum['id']], $forum);
212
        }
213
 
214
        // Unenrol user from second course and alter expected forums.
215
        $enrol->unenrol_user($instance2, $user->id);
216
        unset($expectedforums[$forum2->id]);
217
 
218
        // Call the external function without passing course id.
219
        $forums = mod_forum_external::get_forums_by_courses();
220
        $forums = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $forums);
221
        $this->assertCount(1, $forums);
222
        $this->assertEquals($expectedforums[$forum1->id], $forums[0]);
223
        $this->assertTrue($forums[0]['cancreatediscussions']);
224
 
225
        // Change the type of the forum, the user shouldn't be able to add discussions.
226
        $DB->set_field('forum', 'type', 'news', array('id' => $forum1->id));
227
        $forums = mod_forum_external::get_forums_by_courses();
228
        $forums = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $forums);
229
        $this->assertFalse($forums[0]['cancreatediscussions']);
230
 
231
        // Call for the second course we unenrolled the user from.
232
        $forums = mod_forum_external::get_forums_by_courses(array($course2->id));
233
        $forums = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $forums);
234
        $this->assertCount(0, $forums);
235
    }
236
 
237
    /**
238
     * Test the toggle favourite state
239
     */
11 efrain 240
    public function test_mod_forum_toggle_favourite_state(): void {
1 efrain 241
        global $USER, $CFG, $DB;
242
 
243
        $this->resetAfterTest(true);
244
 
245
        // Create a user.
246
        $user = self::getDataGenerator()->create_user(array('trackforums' => 1));
247
 
248
        // Set to the user.
249
        self::setUser($user);
250
 
251
        // Create courses to add the modules.
252
        $course1 = self::getDataGenerator()->create_course();
253
        $this->getDataGenerator()->enrol_user($user->id, $course1->id);
254
 
255
        $record = new \stdClass();
256
        $record->introformat = FORMAT_HTML;
257
        $record->course = $course1->id;
258
        $record->trackingtype = FORUM_TRACKING_OFF;
259
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
260
        $forum1->introfiles = [];
261
 
262
        // Add discussions to the forums.
263
        $record = new \stdClass();
264
        $record->course = $course1->id;
265
        $record->userid = $user->id;
266
        $record->forum = $forum1->id;
267
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
268
 
269
        $response = mod_forum_external::toggle_favourite_state($discussion1->id, 1);
270
        $response = external_api::clean_returnvalue(mod_forum_external::toggle_favourite_state_returns(), $response);
271
        $this->assertTrue($response['userstate']['favourited']);
272
 
273
        $response = mod_forum_external::toggle_favourite_state($discussion1->id, 0);
274
        $response = external_api::clean_returnvalue(mod_forum_external::toggle_favourite_state_returns(), $response);
275
        $this->assertFalse($response['userstate']['favourited']);
276
 
277
        $this->setUser(0);
278
        try {
279
            $response = mod_forum_external::toggle_favourite_state($discussion1->id, 0);
280
        } catch (\moodle_exception $e) {
281
            $this->assertEquals('requireloginerror', $e->errorcode);
282
        }
283
    }
284
 
285
    /**
286
     * Test the toggle pin state
287
     */
11 efrain 288
    public function test_mod_forum_set_pin_state(): void {
1 efrain 289
        $this->resetAfterTest(true);
290
 
291
        // Create a user.
292
        $user = self::getDataGenerator()->create_user(array('trackforums' => 1));
293
 
294
        // Set to the user.
295
        self::setUser($user);
296
 
297
        // Create courses to add the modules.
298
        $course1 = self::getDataGenerator()->create_course();
299
        $this->getDataGenerator()->enrol_user($user->id, $course1->id);
300
 
301
        $record = new \stdClass();
302
        $record->introformat = FORMAT_HTML;
303
        $record->course = $course1->id;
304
        $record->trackingtype = FORUM_TRACKING_OFF;
305
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
306
        $forum1->introfiles = [];
307
 
308
        // Add discussions to the forums.
309
        $record = new \stdClass();
310
        $record->course = $course1->id;
311
        $record->userid = $user->id;
312
        $record->forum = $forum1->id;
313
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
314
 
315
        try {
316
            $response = mod_forum_external::set_pin_state($discussion1->id, 1);
317
        } catch (\Exception $e) {
318
            $this->assertEquals('cannotpindiscussions', $e->errorcode);
319
        }
320
 
321
        self::setAdminUser();
322
        $response = mod_forum_external::set_pin_state($discussion1->id, 1);
323
        $response = external_api::clean_returnvalue(mod_forum_external::set_pin_state_returns(), $response);
324
        $this->assertTrue($response['pinned']);
325
 
326
        $response = mod_forum_external::set_pin_state($discussion1->id, 0);
327
        $response = external_api::clean_returnvalue(mod_forum_external::set_pin_state_returns(), $response);
328
        $this->assertFalse($response['pinned']);
329
    }
330
 
331
    /**
332
     * Test get forum posts
333
     *
334
     * Tests is similar to the get_forum_discussion_posts only utilizing the new return structure and entities
335
     */
11 efrain 336
    public function test_mod_forum_get_discussion_posts(): void {
1 efrain 337
        global $CFG;
338
 
339
        $this->resetAfterTest(true);
340
 
341
        // Set the CFG variable to allow track forums.
342
        $CFG->forum_trackreadposts = true;
343
 
344
        $urlfactory = \mod_forum\local\container::get_url_factory();
345
        $legacyfactory = \mod_forum\local\container::get_legacy_data_mapper_factory();
346
        $entityfactory = \mod_forum\local\container::get_entity_factory();
347
 
348
        // Create course to add the module.
349
        $course1 = self::getDataGenerator()->create_course();
350
 
351
        // Create a user who can track forums.
352
        $record = new \stdClass();
353
        $record->trackforums = true;
354
        $user1 = self::getDataGenerator()->create_user($record);
355
        // Create a bunch of other users to post.
356
        $user2 = self::getDataGenerator()->create_user();
357
        $user2entity = $entityfactory->get_author_from_stdClass($user2);
358
        $exporteduser2 = [
359
            'id' => (int) $user2->id,
360
            'fullname' => fullname($user2),
361
            'isdeleted' => false,
362
            'groups' => [],
363
            'urls' => [
364
                'profile' => $urlfactory->get_author_profile_url($user2entity, $course1->id)->out(false),
365
                'profileimage' => $urlfactory->get_author_profile_image_url($user2entity),
366
            ]
367
        ];
368
        $user2->fullname = $exporteduser2['fullname'];
369
 
370
        $user3 = self::getDataGenerator()->create_user(['fullname' => "Mr Pants 1"]);
371
        $user3entity = $entityfactory->get_author_from_stdClass($user3);
372
        $exporteduser3 = [
373
            'id' => (int) $user3->id,
374
            'fullname' => fullname($user3),
375
            'groups' => [],
376
            'isdeleted' => false,
377
            'urls' => [
378
                'profile' => $urlfactory->get_author_profile_url($user3entity, $course1->id)->out(false),
379
                'profileimage' => $urlfactory->get_author_profile_image_url($user3entity),
380
            ]
381
        ];
382
        $user3->fullname = $exporteduser3['fullname'];
383
        $forumgenerator = self::getDataGenerator()->get_plugin_generator('mod_forum');
384
 
385
        // Set the first created user to the test user.
386
        self::setUser($user1);
387
 
388
        // Forum with tracking off.
389
        $record = new \stdClass();
390
        $record->course = $course1->id;
391
        $record->trackingtype = FORUM_TRACKING_OFF;
392
        // Display word count. Otherwise, word and char counts will be set to null by the forum post exporter.
393
        $record->displaywordcount = true;
394
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
395
        $forum1context = \context_module::instance($forum1->cmid);
396
 
397
        // Forum with tracking enabled.
398
        $record = new \stdClass();
399
        $record->course = $course1->id;
400
        $forum2 = self::getDataGenerator()->create_module('forum', $record);
401
        $forum2cm = get_coursemodule_from_id('forum', $forum2->cmid);
402
        $forum2context = \context_module::instance($forum2->cmid);
403
 
404
        // Add discussions to the forums.
405
        $record = new \stdClass();
406
        $record->course = $course1->id;
407
        $record->userid = $user1->id;
408
        $record->forum = $forum1->id;
409
        $discussion1 = $forumgenerator->create_discussion($record);
410
 
411
        $record = new \stdClass();
412
        $record->course = $course1->id;
413
        $record->userid = $user2->id;
414
        $record->forum = $forum1->id;
415
        $discussion2 = $forumgenerator->create_discussion($record);
416
 
417
        $record = new \stdClass();
418
        $record->course = $course1->id;
419
        $record->userid = $user2->id;
420
        $record->forum = $forum2->id;
421
        $discussion3 = $forumgenerator->create_discussion($record);
422
 
423
        // Add 2 replies to the discussion 1 from different users.
424
        $record = new \stdClass();
425
        $record->discussion = $discussion1->id;
426
        $record->parent = $discussion1->firstpost;
427
        $record->userid = $user2->id;
428
        $discussion1reply1 = $forumgenerator->create_post($record);
429
        $filename = 'shouldbeanimage.jpg';
430
        // Add a fake inline image to the post.
431
        $filerecordinline = array(
432
            'contextid' => $forum1context->id,
433
            'component' => 'mod_forum',
434
            'filearea'  => 'post',
435
            'itemid'    => $discussion1reply1->id,
436
            'filepath'  => '/',
437
            'filename'  => $filename,
438
        );
439
        $fs = get_file_storage();
440
        $timepost = time();
441
        $file = $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
442
 
443
        $record->parent = $discussion1reply1->id;
444
        $record->userid = $user3->id;
445
        $discussion1reply2 = $forumgenerator->create_post($record);
446
 
447
        // Enrol the user in the  course.
448
        $enrol = enrol_get_plugin('manual');
449
        // Following line enrol and assign default role id to the user.
450
        // So the user automatically gets mod/forum:viewdiscussion on all forums of the course.
451
        $this->getDataGenerator()->enrol_user($user1->id, $course1->id);
452
        $this->getDataGenerator()->enrol_user($user2->id, $course1->id);
453
 
454
        // Delete one user, to test that we still receive posts by this user.
455
        delete_user($user3);
456
        $exporteduser3 = [
457
            'id' => (int) $user3->id,
458
            'fullname' => get_string('deleteduser', 'mod_forum'),
459
            'groups' => [],
460
            'isdeleted' => true,
461
            'urls' => [
462
                'profile' => $urlfactory->get_author_profile_url($user3entity, $course1->id)->out(false),
463
                'profileimage' => $urlfactory->get_author_profile_image_url($user3entity),
464
            ]
465
        ];
466
 
467
        // Create what we expect to be returned when querying the discussion.
468
        $expectedposts = array(
469
            'posts' => array(),
470
            'courseid' => $course1->id,
471
            'forumid' => $forum1->id,
472
            'ratinginfo' => array(
473
                'contextid' => $forum1context->id,
474
                'component' => 'mod_forum',
475
                'ratingarea' => 'post',
476
                'canviewall' => null,
477
                'canviewany' => null,
478
                'scales' => array(),
479
                'ratings' => array(),
480
            ),
481
            'warnings' => array(),
482
        );
483
 
484
        // User pictures are initially empty, we should get the links once the external function is called.
485
        $isolatedurl = $urlfactory->get_discussion_view_url_from_discussion_id($discussion1reply2->discussion);
486
        $isolatedurl->params(['parent' => $discussion1reply2->id]);
487
        $message = file_rewrite_pluginfile_urls($discussion1reply2->message, 'pluginfile.php',
488
            $forum1context->id, 'mod_forum', 'post', $discussion1reply2->id);
489
        $expectedposts['posts'][] = array(
490
            'id' => $discussion1reply2->id,
491
            'discussionid' => $discussion1reply2->discussion,
492
            'parentid' => $discussion1reply2->parent,
493
            'hasparent' => true,
494
            'timecreated' => $discussion1reply2->created,
495
            'timemodified' => $discussion1reply2->modified,
496
            'subject' => $discussion1reply2->subject,
1441 ariadna 497
            'replysubject' => $discussion1reply2->subject,
1 efrain 498
            'message' => $message,
499
            'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
500
            'unread' => null,
501
            'isdeleted' => false,
502
            'isprivatereply' => false,
503
            'haswordcount' => true,
504
            'wordcount' => count_words($message),
505
            'charcount' => count_letters($message),
506
            'author'=> $exporteduser3,
507
            'attachments' => [],
508
            'messageinlinefiles' => [],
509
            'tags' => [],
510
            'html' => [
511
                'rating' => null,
512
                'taglist' => null,
513
                'authorsubheading' => $forumgenerator->get_author_subheading_html((object)$exporteduser3, $discussion1reply2->created)
514
            ],
515
            'capabilities' => [
516
                'view' => 1,
517
                'edit' => 0,
518
                'delete' => 0,
519
                'split' => 0,
520
                'reply' => 1,
521
                'export' => 0,
522
                'controlreadstatus' => 0,
523
                'canreplyprivately' => 0,
524
                'selfenrol' => 0
525
            ],
526
            'urls' => [
527
                'view' => $urlfactory->get_view_post_url_from_post_id($discussion1reply2->discussion, $discussion1reply2->id),
528
                'viewisolated' => $isolatedurl->out(false),
529
                'viewparent' => $urlfactory->get_view_post_url_from_post_id($discussion1reply2->discussion, $discussion1reply2->parent),
530
                'edit' => null,
531
                'delete' =>null,
532
                'split' => null,
533
                'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
534
                    'reply' => $discussion1reply2->id
535
                ]))->out(false),
536
                'export' => null,
537
                'markasread' => null,
538
                'markasunread' => null,
539
                'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id($discussion1reply2->discussion),
540
            ],
541
        );
542
 
543
 
544
        $isolatedurl = $urlfactory->get_discussion_view_url_from_discussion_id($discussion1reply1->discussion);
545
        $isolatedurl->params(['parent' => $discussion1reply1->id]);
546
        $message = file_rewrite_pluginfile_urls($discussion1reply1->message, 'pluginfile.php',
547
            $forum1context->id, 'mod_forum', 'post', $discussion1reply1->id);
548
        $expectedposts['posts'][] = array(
549
            'id' => $discussion1reply1->id,
550
            'discussionid' => $discussion1reply1->discussion,
551
            'parentid' => $discussion1reply1->parent,
552
            'hasparent' => true,
553
            'timecreated' => $discussion1reply1->created,
554
            'timemodified' => $discussion1reply1->modified,
555
            'subject' => $discussion1reply1->subject,
1441 ariadna 556
            'replysubject' => $discussion1reply1->subject,
1 efrain 557
            'message' => $message,
558
            'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
559
            'unread' => null,
560
            'isdeleted' => false,
561
            'isprivatereply' => false,
562
            'haswordcount' => true,
563
            'wordcount' => count_words($message),
564
            'charcount' => count_letters($message),
565
            'author'=> $exporteduser2,
566
            'attachments' => [],
567
            'messageinlinefiles' => [
568
 
569
            ],
570
            'tags' => [],
571
            'html' => [
572
                'rating' => null,
573
                'taglist' => null,
574
                'authorsubheading' => $forumgenerator->get_author_subheading_html((object)$exporteduser2, $discussion1reply1->created)
575
            ],
576
            'capabilities' => [
577
                'view' => 1,
578
                'edit' => 0,
579
                'delete' => 0,
580
                'split' => 0,
581
                'reply' => 1,
582
                'export' => 0,
583
                'controlreadstatus' => 0,
584
                'canreplyprivately' => 0,
585
                'selfenrol' => 0
586
            ],
587
            'urls' => [
588
                'view' => $urlfactory->get_view_post_url_from_post_id($discussion1reply1->discussion, $discussion1reply1->id),
589
                'viewisolated' => $isolatedurl->out(false),
590
                'viewparent' => $urlfactory->get_view_post_url_from_post_id($discussion1reply1->discussion, $discussion1reply1->parent),
591
                'edit' => null,
592
                'delete' =>null,
593
                'split' => null,
594
                'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
595
                    'reply' => $discussion1reply1->id
596
                ]))->out(false),
597
                'export' => null,
598
                'markasread' => null,
599
                'markasunread' => null,
600
                'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id($discussion1reply1->discussion),
601
            ],
602
        );
603
 
604
        // Test a discussion with two additional posts (total 3 posts).
605
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'DESC', true);
606
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
607
        $this->assertEquals(3, count($posts['posts']));
608
 
609
        // Unset the initial discussion post.
610
        array_pop($posts['posts']);
611
        $this->assertEquals($expectedposts, $posts);
612
 
613
        // Check we receive the unread count correctly on tracked forum.
614
        forum_tp_count_forum_unread_posts($forum2cm, $course1, true);    // Reset static cache.
615
        $result = mod_forum_external::get_forums_by_courses(array($course1->id));
616
        $result = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $result);
617
        foreach ($result as $f) {
618
            if ($f['id'] == $forum2->id) {
619
                $this->assertEquals(1, $f['unreadpostscount']);
620
            }
621
        }
622
 
623
        // Test discussion without additional posts. There should be only one post (the one created by the discussion).
624
        $posts = mod_forum_external::get_discussion_posts($discussion2->id, 'modified', 'DESC');
625
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
626
        $this->assertEquals(1, count($posts['posts']));
627
 
628
        // Test discussion tracking on not tracked forum.
629
        $result = mod_forum_external::view_forum_discussion($discussion1->id);
630
        $result = external_api::clean_returnvalue(mod_forum_external::view_forum_discussion_returns(), $result);
631
        $this->assertTrue($result['status']);
632
        $this->assertEmpty($result['warnings']);
633
 
634
        // Test posts have not been marked as read.
635
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'DESC');
636
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
637
        foreach ($posts['posts'] as $post) {
638
            $this->assertNull($post['unread']);
639
        }
640
 
641
        // Test discussion tracking on tracked forum.
642
        $result = mod_forum_external::view_forum_discussion($discussion3->id);
643
        $result = external_api::clean_returnvalue(mod_forum_external::view_forum_discussion_returns(), $result);
644
        $this->assertTrue($result['status']);
645
        $this->assertEmpty($result['warnings']);
646
 
647
        // Test posts have been marked as read.
648
        $posts = mod_forum_external::get_discussion_posts($discussion3->id, 'modified', 'DESC');
649
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
650
        foreach ($posts['posts'] as $post) {
651
            $this->assertFalse($post['unread']);
652
        }
653
 
654
        // Check we receive 0 unread posts.
655
        forum_tp_count_forum_unread_posts($forum2cm, $course1, true);    // Reset static cache.
656
        $result = mod_forum_external::get_forums_by_courses(array($course1->id));
657
        $result = external_api::clean_returnvalue(mod_forum_external::get_forums_by_courses_returns(), $result);
658
        foreach ($result as $f) {
659
            if ($f['id'] == $forum2->id) {
660
                $this->assertEquals(0, $f['unreadpostscount']);
661
            }
662
        }
663
    }
664
 
665
    /**
666
     * Test get forum posts
667
     */
11 efrain 668
    public function test_mod_forum_get_discussion_posts_deleted(): void {
1 efrain 669
        global $CFG, $PAGE;
670
 
671
        $this->resetAfterTest(true);
672
        $generator = self::getDataGenerator()->get_plugin_generator('mod_forum');
673
 
674
        // Create a course and enrol some users in it.
675
        $course1 = self::getDataGenerator()->create_course();
676
 
677
        // Create users.
678
        $user1 = self::getDataGenerator()->create_user();
679
        $this->getDataGenerator()->enrol_user($user1->id, $course1->id);
680
        $user2 = self::getDataGenerator()->create_user();
681
        $this->getDataGenerator()->enrol_user($user2->id, $course1->id);
682
 
683
        // Set the first created user to the test user.
684
        self::setUser($user1);
685
 
686
        // Create test data.
687
        $forum1 = self::getDataGenerator()->create_module('forum', (object) [
688
            'course' => $course1->id,
689
        ]);
690
        $forum1context = \context_module::instance($forum1->cmid);
691
 
692
        // Add discussions to the forum.
693
        $discussion = $generator->create_discussion((object) [
694
            'course' => $course1->id,
695
            'userid' => $user1->id,
696
            'forum' => $forum1->id,
697
        ]);
698
 
699
        $discussion2 = $generator->create_discussion((object) [
700
            'course' => $course1->id,
701
            'userid' => $user2->id,
702
            'forum' => $forum1->id,
703
        ]);
704
 
705
        // Add replies to the discussion.
706
        $discussionreply1 = $generator->create_post((object) [
707
            'discussion' => $discussion->id,
708
            'parent' => $discussion->firstpost,
709
            'userid' => $user2->id,
710
        ]);
711
        $discussionreply2 = $generator->create_post((object) [
712
            'discussion' => $discussion->id,
713
            'parent' => $discussionreply1->id,
714
            'userid' => $user2->id,
715
            'subject' => '',
716
            'message' => '',
717
            'messageformat' => FORMAT_PLAIN,
718
            'deleted' => 1,
719
        ]);
720
        $discussionreply3 = $generator->create_post((object) [
721
            'discussion' => $discussion->id,
722
            'parent' => $discussion->firstpost,
723
            'userid' => $user2->id,
724
        ]);
725
 
726
        // Test where some posts have been marked as deleted.
727
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'modified', 'DESC');
728
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
729
        $deletedsubject = get_string('forumsubjectdeleted', 'mod_forum');
730
        $deletedmessage = get_string('forumbodydeleted', 'mod_forum');
731
 
732
        foreach ($posts['posts'] as $post) {
733
            if ($post['id'] == $discussionreply2->id) {
734
                $this->assertTrue($post['isdeleted']);
735
                $this->assertEquals($deletedsubject, $post['subject']);
736
                $this->assertEquals($deletedmessage, $post['message']);
737
            } else {
738
                $this->assertFalse($post['isdeleted']);
739
                $this->assertNotEquals($deletedsubject, $post['subject']);
740
                $this->assertNotEquals($deletedmessage, $post['message']);
741
            }
742
        }
743
    }
744
 
745
    /**
746
     * Test get forum posts returns inline attachments.
747
     */
11 efrain 748
    public function test_mod_forum_get_discussion_posts_inline_attachments(): void {
1 efrain 749
        global $CFG;
750
 
751
        $this->resetAfterTest(true);
752
 
753
        // Create a course and enrol some users in it.
754
        $course = self::getDataGenerator()->create_course();
755
 
756
        // Create users.
757
        $user = self::getDataGenerator()->create_user();
758
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
759
 
760
 
761
        // Set the first created user to the test user.
762
        self::setUser($user);
763
 
764
        // Create test data.
765
        $forum = self::getDataGenerator()->create_module('forum', (object) [
766
            'course' => $course->id,
767
        ]);
768
 
769
        // Create a file in a draft area for inline attachments.
770
        $draftidinlineattach = file_get_unused_draft_itemid();
771
        $draftidattach = file_get_unused_draft_itemid();
772
        self::setUser($user);
773
        $usercontext = \context_user::instance($user->id);
774
        $filepath = '/';
775
        $filearea = 'draft';
776
        $component = 'user';
777
        $filenameimg = 'fakeimage.png';
778
        $filerecordinline = [
779
            'contextid' => $usercontext->id,
780
            'component' => $component,
781
            'filearea'  => $filearea,
782
            'itemid'    => $draftidinlineattach,
783
            'filepath'  => $filepath,
784
            'filename'  => $filenameimg,
785
        ];
786
        $fs = get_file_storage();
787
        $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
788
 
789
        // Create discussion.
790
        $dummytext = 'Here is an inline image: <img src="' . $CFG->wwwroot .
791
            "/draftfile.php/{$usercontext->id}/user/draft/{$draftidinlineattach}/{$filenameimg}" .
792
            '" alt="inlineimage">.';
793
        $options = [
794
            [
795
                'name' => 'inlineattachmentsid',
796
                'value' => $draftidinlineattach
797
            ],
798
            [
799
                'name' => 'attachmentsid',
800
                'value' => $draftidattach
801
            ]
802
        ];
803
        $discussion = mod_forum_external::add_discussion($forum->id, 'the inline attachment subject', $dummytext,
804
            -1, $options);
805
 
806
        $posts = mod_forum_external::get_discussion_posts($discussion['discussionid'], 'modified', 'DESC');
807
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
808
        $post = $posts['posts'][0];
809
        $this->assertCount(0, $post['messageinlinefiles']);
810
        $this->assertEmpty($post['messageinlinefiles']);
811
 
812
        $posts = mod_forum_external::get_discussion_posts($discussion['discussionid'], 'modified', 'DESC',
813
            true);
814
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
815
        $post = $posts['posts'][0];
816
        $this->assertCount(1, $post['messageinlinefiles']);
817
        $this->assertEquals('fakeimage.png', $post['messageinlinefiles'][0]['filename']);
818
    }
819
 
820
    /**
821
     * Test get forum posts (qanda forum)
822
     */
11 efrain 823
    public function test_mod_forum_get_discussion_posts_qanda(): void {
1 efrain 824
        global $CFG, $DB;
825
 
826
        $this->resetAfterTest(true);
827
 
828
        $record = new \stdClass();
829
        $user1 = self::getDataGenerator()->create_user($record);
830
        $user2 = self::getDataGenerator()->create_user();
831
 
832
        // Set the first created user to the test user.
833
        self::setUser($user1);
834
 
835
        // Create course to add the module.
836
        $course1 = self::getDataGenerator()->create_course();
837
        $this->getDataGenerator()->enrol_user($user1->id, $course1->id);
838
        $this->getDataGenerator()->enrol_user($user2->id, $course1->id);
839
 
840
        // Forum with tracking off.
841
        $record = new \stdClass();
842
        $record->course = $course1->id;
843
        $record->type = 'qanda';
844
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
845
        $forum1context = \context_module::instance($forum1->cmid);
846
 
847
        // Add discussions to the forums.
848
        $record = new \stdClass();
849
        $record->course = $course1->id;
850
        $record->userid = $user2->id;
851
        $record->forum = $forum1->id;
852
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
853
 
854
        // Add 1 reply (not the actual user).
855
        $record = new \stdClass();
856
        $record->discussion = $discussion1->id;
857
        $record->parent = $discussion1->firstpost;
858
        $record->userid = $user2->id;
859
        $discussion1reply1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
860
 
861
        // We still see only the original post.
862
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'DESC');
863
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
864
        $this->assertEquals(1, count($posts['posts']));
865
 
866
        // Add a new reply, the user is going to be able to see only the original post and their new post.
867
        $record = new \stdClass();
868
        $record->discussion = $discussion1->id;
869
        $record->parent = $discussion1->firstpost;
870
        $record->userid = $user1->id;
871
        $discussion1reply2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
872
 
873
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'DESC');
874
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
875
        $this->assertEquals(2, count($posts['posts']));
876
 
877
        // Now, we can fake the time of the user post, so he can se the rest of the discussion posts.
878
        $discussion1reply2->created -= $CFG->maxeditingtime * 2;
879
        $DB->update_record('forum_posts', $discussion1reply2);
880
 
881
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'DESC');
882
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
883
        $this->assertEquals(3, count($posts['posts']));
884
    }
885
 
886
    /**
887
     * Test get forum discussions
888
     */
11 efrain 889
    public function test_mod_forum_get_forum_discussions(): void {
1 efrain 890
        global $CFG, $DB, $PAGE;
891
 
892
        $this->resetAfterTest(true);
893
 
894
        // Set the CFG variable to allow track forums.
895
        $CFG->forum_trackreadposts = true;
896
 
897
        // Create a user who can track forums.
898
        $record = new \stdClass();
899
        $record->trackforums = true;
900
        $user1 = self::getDataGenerator()->create_user($record);
901
        // Create a bunch of other users to post.
902
        $user2 = self::getDataGenerator()->create_user();
903
        $user3 = self::getDataGenerator()->create_user();
904
        $user4 = self::getDataGenerator()->create_user();
905
 
906
        // Set the first created user to the test user.
907
        self::setUser($user1);
908
 
909
        // Create courses to add the modules.
910
        $course1 = self::getDataGenerator()->create_course();
911
 
912
        // First forum with tracking off.
913
        $record = new \stdClass();
914
        $record->course = $course1->id;
915
        $record->trackingtype = FORUM_TRACKING_OFF;
916
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
917
 
918
        // Add discussions to the forums.
919
        $record = new \stdClass();
920
        $record->course = $course1->id;
921
        $record->userid = $user1->id;
922
        $record->forum = $forum1->id;
923
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
924
 
925
        // Add three replies to the discussion 1 from different users.
926
        $record = new \stdClass();
927
        $record->discussion = $discussion1->id;
928
        $record->parent = $discussion1->firstpost;
929
        $record->userid = $user2->id;
930
        $discussion1reply1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
931
 
932
        $record->parent = $discussion1reply1->id;
933
        $record->userid = $user3->id;
934
        $discussion1reply2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
935
 
936
        $record->userid = $user4->id;
937
        $discussion1reply3 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
938
 
939
        // Enrol the user in the first course.
940
        $enrol = enrol_get_plugin('manual');
941
 
942
        // We don't use the dataGenerator as we need to get the $instance2 to unenrol later.
943
        $enrolinstances = enrol_get_instances($course1->id, true);
944
        foreach ($enrolinstances as $courseenrolinstance) {
945
            if ($courseenrolinstance->enrol == "manual") {
946
                $instance1 = $courseenrolinstance;
947
                break;
948
            }
949
        }
950
        $enrol->enrol_user($instance1, $user1->id);
951
 
952
        // Delete one user.
953
        delete_user($user4);
954
 
955
        // Assign capabilities to view discussions for forum 1.
956
        $cm = get_coursemodule_from_id('forum', $forum1->cmid, 0, false, MUST_EXIST);
957
        $context = \context_module::instance($cm->id);
958
        $newrole = create_role('Role 2', 'role2', 'Role 2 description');
959
        $this->assignUserCapability('mod/forum:viewdiscussion', $context->id, $newrole);
960
 
961
        // Create what we expect to be returned when querying the forums.
962
 
963
        $post1 = $DB->get_record('forum_posts', array('id' => $discussion1->firstpost), '*', MUST_EXIST);
964
 
965
        // User pictures are initially empty, we should get the links once the external function is called.
966
        $expecteddiscussions = array(
967
            'id' => $discussion1->firstpost,
968
            'name' => $discussion1->name,
969
            'groupid' => (int) $discussion1->groupid,
970
            'timemodified' => (int) $discussion1reply3->created,
971
            'usermodified' => (int) $discussion1reply3->userid,
972
            'timestart' => (int) $discussion1->timestart,
973
            'timeend' => (int) $discussion1->timeend,
974
            'discussion' => (int) $discussion1->id,
975
            'parent' => 0,
976
            'userid' => (int) $discussion1->userid,
977
            'created' => (int) $post1->created,
978
            'modified' => (int) $post1->modified,
979
            'mailed' => (int) $post1->mailed,
980
            'subject' => $post1->subject,
981
            'message' => $post1->message,
982
            'messageformat' => (int) $post1->messageformat,
983
            'messagetrust' => (int) $post1->messagetrust,
984
            'attachment' => $post1->attachment,
985
            'totalscore' => (int) $post1->totalscore,
986
            'mailnow' => (int) $post1->mailnow,
987
            'userfullname' => fullname($user1),
988
            'usermodifiedfullname' => fullname($user4),
989
            'userpictureurl' => '',
990
            'usermodifiedpictureurl' => '',
991
            'numreplies' => 3,
992
            'numunread' => 0,
993
            'pinned' => (bool) FORUM_DISCUSSION_UNPINNED,
994
            'locked' => false,
995
            'canreply' => false,
996
            'canlock' => false,
997
            'starred' => false,
998
            'canfavourite' => true
999
        );
1000
 
1001
        // Call the external function passing forum id.
1002
        $discussions = mod_forum_external::get_forum_discussions($forum1->id);
1003
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1004
        $expectedreturn = array(
1005
            'discussions' => array($expecteddiscussions),
1006
            'warnings' => array()
1007
        );
1008
 
1009
        // Wait the theme to be loaded (the external_api call does that) to generate the user profiles.
1010
        $userpicture = new \user_picture($user1);
1011
        $userpicture->size = 2; // Size f2.
1012
        $expectedreturn['discussions'][0]['userpictureurl'] = $userpicture->get_url($PAGE)->out(false);
1013
 
1014
        $userpicture = new \user_picture($user4);
1015
        $userpicture->size = 2; // Size f2.
1016
        $expectedreturn['discussions'][0]['usermodifiedpictureurl'] = $userpicture->get_url($PAGE)->out(false);
1017
 
1018
        $this->assertEquals($expectedreturn, $discussions);
1019
 
1020
        // Test the starring functionality return.
1021
        $t = mod_forum_external::toggle_favourite_state($discussion1->id, 1);
1022
        $expectedreturn['discussions'][0]['starred'] = true;
1023
        $discussions = mod_forum_external::get_forum_discussions($forum1->id);
1024
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1025
        $this->assertEquals($expectedreturn, $discussions);
1026
 
1027
        // Call without required view discussion capability.
1028
        $this->unassignUserCapability('mod/forum:viewdiscussion', $context->id, $newrole);
1029
        try {
1030
            mod_forum_external::get_forum_discussions($forum1->id);
1031
            $this->fail('Exception expected due to missing capability.');
1032
        } catch (\moodle_exception $e) {
1033
            $this->assertEquals('noviewdiscussionspermission', $e->errorcode);
1034
        }
1035
 
1036
        // Unenrol user from second course.
1037
        $enrol->unenrol_user($instance1, $user1->id);
1038
 
1039
        // Call for the second course we unenrolled the user from, make sure exception thrown.
1040
        try {
1041
            mod_forum_external::get_forum_discussions($forum1->id);
1042
            $this->fail('Exception expected due to being unenrolled from the course.');
1043
        } catch (\moodle_exception $e) {
1044
            $this->assertEquals('requireloginerror', $e->errorcode);
1045
        }
1046
 
1047
        $this->setAdminUser();
1048
        $discussions = mod_forum_external::get_forum_discussions($forum1->id);
1049
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1050
        $this->assertTrue($discussions['discussions'][0]['canlock']);
1051
    }
1052
 
1053
    /**
1054
     * Test the sorting in get forum discussions
1055
     */
11 efrain 1056
    public function test_mod_forum_get_forum_discussions_sorting(): void {
1 efrain 1057
        global $CFG, $DB, $PAGE;
1058
 
1059
        $this->resetAfterTest(true);
1060
 
1061
        $clock = $this->mock_clock_with_frozen();
1062
 
1063
        // Set the CFG variable to allow track forums.
1064
        $CFG->forum_trackreadposts = true;
1065
 
1066
        // Create a user who can track forums.
1067
        $record = new \stdClass();
1068
        $record->trackforums = true;
1069
        $user1 = self::getDataGenerator()->create_user($record);
1070
        // Create a bunch of other users to post.
1071
        $user2 = self::getDataGenerator()->create_user();
1072
        $user3 = self::getDataGenerator()->create_user();
1073
        $user4 = self::getDataGenerator()->create_user();
1074
 
1075
        // Set the first created user to the test user.
1076
        self::setUser($user1);
1077
 
1078
        // Create courses to add the modules.
1079
        $course1 = self::getDataGenerator()->create_course();
1080
 
1081
        // Enrol the user in the first course.
1082
        $enrol = enrol_get_plugin('manual');
1083
 
1084
        // We don't use the dataGenerator as we need to get the $instance2 to unenrol later.
1085
        $enrolinstances = enrol_get_instances($course1->id, true);
1086
        foreach ($enrolinstances as $courseenrolinstance) {
1087
            if ($courseenrolinstance->enrol == "manual") {
1088
                $instance1 = $courseenrolinstance;
1089
                break;
1090
            }
1091
        }
1092
        $enrol->enrol_user($instance1, $user1->id);
1093
 
1094
        // First forum with tracking off.
1095
        $record = new \stdClass();
1096
        $record->course = $course1->id;
1097
        $record->trackingtype = FORUM_TRACKING_OFF;
1098
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
1099
 
1100
        // Assign capabilities to view discussions for forum 1.
1101
        $cm = get_coursemodule_from_id('forum', $forum1->cmid, 0, false, MUST_EXIST);
1102
        $context = \context_module::instance($cm->id);
1103
        $newrole = create_role('Role 2', 'role2', 'Role 2 description');
1104
        $this->assignUserCapability('mod/forum:viewdiscussion', $context->id, $newrole);
1105
 
1106
        // Add discussions to the forums.
1107
        $record = new \stdClass();
1108
        $record->course = $course1->id;
1109
        $record->userid = $user1->id;
1110
        $record->forum = $forum1->id;
1111
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1112
        $clock->bump();
1113
 
1114
        // Add three replies to the discussion 1 from different users.
1115
        $record = new \stdClass();
1116
        $record->discussion = $discussion1->id;
1117
        $record->parent = $discussion1->firstpost;
1118
        $record->userid = $user2->id;
1119
        $discussion1reply1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
1120
        $clock->bump();
1121
 
1122
        $record->parent = $discussion1reply1->id;
1123
        $record->userid = $user3->id;
1124
        $discussion1reply2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
1125
        $clock->bump();
1126
 
1127
        $record->userid = $user4->id;
1128
        $discussion1reply3 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
1129
        $clock->bump();
1130
 
1131
        // Create discussion2.
1132
        $record2 = new \stdClass();
1133
        $record2->course = $course1->id;
1134
        $record2->userid = $user1->id;
1135
        $record2->forum = $forum1->id;
1136
        $discussion2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record2);
1137
        $clock->bump();
1138
 
1139
        // Add one reply to the discussion 2.
1140
        $record2 = new \stdClass();
1141
        $record2->discussion = $discussion2->id;
1142
        $record2->parent = $discussion2->firstpost;
1143
        $record2->userid = $user2->id;
1144
        $discussion2reply1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record2);
1145
        $clock->bump();
1146
 
1147
        // Create discussion 3.
1148
        $record3 = new \stdClass();
1149
        $record3->course = $course1->id;
1150
        $record3->userid = $user1->id;
1151
        $record3->forum = $forum1->id;
1152
        $discussion3 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record3);
1153
        $clock->bump();
1154
 
1155
        // Add two replies to the discussion 3.
1156
        $record3 = new \stdClass();
1157
        $record3->discussion = $discussion3->id;
1158
        $record3->parent = $discussion3->firstpost;
1159
        $record3->userid = $user2->id;
1160
        $discussion3reply1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record3);
1161
        $clock->bump();
1162
 
1163
        $record3->parent = $discussion3reply1->id;
1164
        $record3->userid = $user3->id;
1165
        $discussion3reply2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record3);
1166
 
1167
        // Call the external function passing forum id.
1168
        $discussions = mod_forum_external::get_forum_discussions($forum1->id);
1169
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1170
        // Discussions should be ordered by last post date in descending order by default.
1171
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion3->id);
1172
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion2->id);
1173
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion1->id);
1174
 
1175
        $vaultfactory = \mod_forum\local\container::get_vault_factory();
1176
        $discussionlistvault = $vaultfactory->get_discussions_in_forum_vault();
1177
 
1178
        // Call the external function passing forum id and sort order parameter.
1179
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_LASTPOST_ASC);
1180
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1181
        // Discussions should be ordered by last post date in ascending order.
1182
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion1->id);
1183
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion2->id);
1184
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion3->id);
1185
 
1186
        // Call the external function passing forum id and sort order parameter.
1187
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_CREATED_DESC);
1188
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1189
        // Discussions should be ordered by discussion creation date in descending order.
1190
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion3->id);
1191
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion2->id);
1192
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion1->id);
1193
 
1194
        // Call the external function passing forum id and sort order parameter.
1195
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_CREATED_ASC);
1196
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1197
        // Discussions should be ordered by discussion creation date in ascending order.
1198
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion1->id);
1199
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion2->id);
1200
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion3->id);
1201
 
1202
        // Call the external function passing forum id and sort order parameter.
1203
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_REPLIES_DESC);
1204
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1205
        // Discussions should be ordered by the number of replies in descending order.
1206
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion1->id);
1207
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion3->id);
1208
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion2->id);
1209
 
1210
        // Call the external function passing forum id and sort order parameter.
1211
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_REPLIES_ASC);
1212
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1213
        // Discussions should be ordered by the number of replies in ascending order.
1214
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion2->id);
1215
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion3->id);
1216
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion1->id);
1217
 
1218
        // Pin discussion2.
1219
        $DB->update_record('forum_discussions',
1220
            (object) array('id' => $discussion2->id, 'pinned' => FORUM_DISCUSSION_PINNED));
1221
 
1222
        // Call the external function passing forum id.
1223
        $discussions = mod_forum_external::get_forum_discussions($forum1->id);
1224
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1225
        // Discussions should be ordered by last post date in descending order by default.
1226
        // Pinned discussions should be at the top of the list.
1227
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion2->id);
1228
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion3->id);
1229
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion1->id);
1230
 
1231
        // Call the external function passing forum id and sort order parameter.
1232
        $discussions = mod_forum_external::get_forum_discussions($forum1->id, $discussionlistvault::SORTORDER_LASTPOST_ASC);
1233
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1234
        // Discussions should be ordered by last post date in ascending order.
1235
        // Pinned discussions should be at the top of the list.
1236
        $this->assertEquals($discussions['discussions'][0]['discussion'], $discussion2->id);
1237
        $this->assertEquals($discussions['discussions'][1]['discussion'], $discussion1->id);
1238
        $this->assertEquals($discussions['discussions'][2]['discussion'], $discussion3->id);
1239
    }
1240
 
1241
    /**
1242
     * Test add_discussion_post
1243
     */
11 efrain 1244
    public function test_add_discussion_post(): void {
1 efrain 1245
        global $CFG;
1246
 
1247
        $this->resetAfterTest(true);
1248
 
1249
        $user = self::getDataGenerator()->create_user();
1250
        $otheruser = self::getDataGenerator()->create_user();
1251
 
1252
        self::setAdminUser();
1253
 
1254
        // Create course to add the module.
1255
        $course = self::getDataGenerator()->create_course(array('groupmode' => VISIBLEGROUPS, 'groupmodeforce' => 0));
1256
 
1257
        // Forum with tracking off.
1258
        $record = new \stdClass();
1259
        $record->course = $course->id;
1260
        $forum = self::getDataGenerator()->create_module('forum', $record);
1261
        $cm = get_coursemodule_from_id('forum', $forum->cmid, 0, false, MUST_EXIST);
1262
        $forumcontext = \context_module::instance($forum->cmid);
1263
 
1264
        // Add discussions to the forums.
1265
        $record = new \stdClass();
1266
        $record->course = $course->id;
1267
        $record->userid = $user->id;
1268
        $record->forum = $forum->id;
1269
        $discussion = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1270
 
1271
        // Try to post (user not enrolled).
1272
        self::setUser($user);
1273
        try {
1274
            mod_forum_external::add_discussion_post($discussion->firstpost, 'some subject', 'some text here...');
1275
            $this->fail('Exception expected due to being unenrolled from the course.');
1276
        } catch (\moodle_exception $e) {
1277
            $this->assertEquals('requireloginerror', $e->errorcode);
1278
        }
1279
 
1280
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
1281
        $this->getDataGenerator()->enrol_user($otheruser->id, $course->id);
1282
 
1283
        $createdpost = mod_forum_external::add_discussion_post($discussion->firstpost, 'some subject', 'some text here...');
1284
        $createdpost = external_api::clean_returnvalue(mod_forum_external::add_discussion_post_returns(), $createdpost);
1285
 
1286
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'modified', 'ASC');
1287
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1288
        // We receive the discussion and the post.
1289
        $this->assertEquals(2, count($posts['posts']));
1290
 
1291
        $tested = false;
1292
        foreach ($posts['posts'] as $thispost) {
1293
            if ($createdpost['postid'] == $thispost['id']) {
1294
                $this->assertEquals('some subject', $thispost['subject']);
1295
                $this->assertEquals('some text here...', $thispost['message']);
1296
                $this->assertEquals(FORMAT_HTML, $thispost['messageformat']); // This is the default if format was not specified.
1297
                $tested = true;
1298
            }
1299
        }
1300
        $this->assertTrue($tested);
1301
 
1302
        // Let's simulate a call with any other format, it should be stored that way.
1303
        global $DB; // Yes, we are going to use DB facilities too, because cannot rely on other functions for checking
1304
                    // the format. They eat it completely (going back to FORMAT_HTML. So we only can trust DB for further
1305
                    // processing.
1306
        $formats = [FORMAT_PLAIN, FORMAT_MOODLE, FORMAT_MARKDOWN, FORMAT_HTML];
1307
        $options = [];
1308
        foreach ($formats as $format) {
1309
            $createdpost = mod_forum_external::add_discussion_post($discussion->firstpost,
1310
                'with some format', 'some formatted here...', $options, $format);
1311
            $createdpost = external_api::clean_returnvalue(mod_forum_external::add_discussion_post_returns(), $createdpost);
1312
            $dbformat = $DB->get_field('forum_posts', 'messageformat', ['id' => $createdpost['postid']]);
1313
            $this->assertEquals($format, $dbformat);
1314
        }
1315
 
1316
        // Now let's try the 'topreferredformat' option. That should end with the content
1317
        // transformed and the format being FORMAT_HTML (when, like in this case,  user preferred
1318
        // format is HTML, inferred from editor in preferences).
1319
        $options = [['name' => 'topreferredformat', 'value' => true]];
1320
        $createdpost = mod_forum_external::add_discussion_post($discussion->firstpost,
1321
            'interesting subject', 'with some https://example.com link', $options, FORMAT_MOODLE);
1322
        $createdpost = external_api::clean_returnvalue(mod_forum_external::add_discussion_post_returns(), $createdpost);
1323
        $dbpost = $DB->get_record('forum_posts', ['id' => $createdpost['postid']]);
1324
        // Format HTML and content converted, we should get.
1325
        $this->assertEquals(FORMAT_HTML, $dbpost->messageformat);
1326
        $this->assertEquals('<div class="text_to_html">with some https://example.com link</div>', $dbpost->message);
1327
 
1328
        // Test inline and regular attachment in post
1329
        // Create a file in a draft area for inline attachments.
1330
        $draftidinlineattach = file_get_unused_draft_itemid();
1331
        $draftidattach = file_get_unused_draft_itemid();
1332
        self::setUser($user);
1333
        $usercontext = \context_user::instance($user->id);
1334
        $filepath = '/';
1335
        $filearea = 'draft';
1336
        $component = 'user';
1337
        $filenameimg = 'shouldbeanimage.txt';
1338
        $filerecordinline = array(
1339
            'contextid' => $usercontext->id,
1340
            'component' => $component,
1341
            'filearea'  => $filearea,
1342
            'itemid'    => $draftidinlineattach,
1343
            'filepath'  => $filepath,
1344
            'filename'  => $filenameimg,
1345
        );
1346
        $fs = get_file_storage();
1347
 
1348
        // Create a file in a draft area for regular attachments.
1349
        $filerecordattach = $filerecordinline;
1350
        $attachfilename = 'attachment.txt';
1351
        $filerecordattach['filename'] = $attachfilename;
1352
        $filerecordattach['itemid'] = $draftidattach;
1353
        $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
1354
        $fs->create_file_from_string($filerecordattach, 'simple text attachment');
1355
 
1356
        $options = array(array('name' => 'inlineattachmentsid', 'value' => $draftidinlineattach),
1357
                         array('name' => 'attachmentsid', 'value' => $draftidattach));
1358
        $dummytext = 'Here is an inline image: <img src="' . $CFG->wwwroot
1359
                     . "/draftfile.php/{$usercontext->id}/user/draft/{$draftidinlineattach}/{$filenameimg}"
1360
                     . '" alt="inlineimage">.';
1361
        $createdpost = mod_forum_external::add_discussion_post($discussion->firstpost, 'new post inline attachment',
1362
                                                               $dummytext, $options);
1363
        $createdpost = external_api::clean_returnvalue(mod_forum_external::add_discussion_post_returns(), $createdpost);
1364
 
1365
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'modified', 'ASC');
1366
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1367
        // We receive the discussion and the post.
1368
        // Can't guarantee order of posts during tests.
1369
        $postfound = false;
1370
        foreach ($posts['posts'] as $thispost) {
1371
            if ($createdpost['postid'] == $thispost['id']) {
1372
                $this->assertEquals($createdpost['postid'], $thispost['id']);
1373
                $this->assertCount(1, $thispost['attachments']);
1374
                $this->assertEquals('attachment.txt', $thispost['attachments'][0]['filename']);
1375
                $this->assertEquals($thispost['attachments'][0]['filename'], $attachfilename, "There should be 1 attachment");
1376
                $this->assertStringContainsString('pluginfile.php', $thispost['message']);
1377
                $postfound = true;
1378
                break;
1379
            }
1380
        }
1381
 
1382
        $this->assertTrue($postfound);
1383
 
1384
        // Check not posting in groups the user is not member of.
1385
        $group = $this->getDataGenerator()->create_group(array('courseid' => $course->id));
1386
        groups_add_member($group->id, $otheruser->id);
1387
 
1388
        $forum = self::getDataGenerator()->create_module('forum', $record, array('groupmode' => SEPARATEGROUPS));
1389
        $record->forum = $forum->id;
1390
        $record->userid = $otheruser->id;
1391
        $record->groupid = $group->id;
1392
        $discussion = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1393
 
1394
        try {
1395
            mod_forum_external::add_discussion_post($discussion->firstpost, 'some subject', 'some text here...');
1396
            $this->fail('Exception expected due to invalid permissions for posting.');
1397
        } catch (\moodle_exception $e) {
1398
            $this->assertEquals('nopostforum', $e->errorcode);
1399
        }
1400
    }
1401
 
1402
    /**
1403
     * Test add_discussion_post and auto subscription to a discussion.
1404
     */
11 efrain 1405
    public function test_add_discussion_post_subscribe_discussion(): void {
1 efrain 1406
        global $USER;
1407
 
1408
        $this->resetAfterTest(true);
1409
 
1410
        self::setAdminUser();
1411
 
1412
        $user = self::getDataGenerator()->create_user();
1413
        $admin = get_admin();
1414
        // Create course to add the module.
1415
        $course = self::getDataGenerator()->create_course(array('groupmode' => VISIBLEGROUPS, 'groupmodeforce' => 0));
1416
 
1417
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
1418
 
1419
        // Forum with tracking off.
1420
        $record = new \stdClass();
1421
        $record->course = $course->id;
1422
        $forum = self::getDataGenerator()->create_module('forum', $record);
1423
        $cm = get_coursemodule_from_id('forum', $forum->cmid, 0, false, MUST_EXIST);
1424
 
1425
        // Add discussions to the forums.
1426
        $record = new \stdClass();
1427
        $record->course = $course->id;
1428
        $record->userid = $admin->id;
1429
        $record->forum = $forum->id;
1430
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1431
        $discussion2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1432
 
1433
        // Try to post as user.
1434
        self::setUser($user);
1435
        // Enable auto subscribe discussion.
1436
        $USER->autosubscribe = true;
1437
        // Add a discussion post in a forum discussion where the user is not subscribed (auto-subscribe preference enabled).
1438
        mod_forum_external::add_discussion_post($discussion1->firstpost, 'some subject', 'some text here...');
1439
 
1440
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'ASC');
1441
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1442
        // We receive the discussion and the post.
1443
        $this->assertEquals(2, count($posts['posts']));
1444
        // The user should be subscribed to the discussion after adding a discussion post.
1445
        $this->assertTrue(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion1->id, $cm));
1446
 
1447
        // Disable auto subscribe discussion.
1448
        $USER->autosubscribe = false;
1449
        $this->assertTrue(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion1->id, $cm));
1450
        // Add a discussion post in a forum discussion where the user is subscribed (auto-subscribe preference disabled).
1451
        mod_forum_external::add_discussion_post($discussion1->firstpost, 'some subject 1', 'some text here 1...');
1452
 
1453
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'ASC');
1454
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1455
        // We receive the discussion and the post.
1456
        $this->assertEquals(3, count($posts['posts']));
1457
        // The user should still be subscribed to the discussion after adding a discussion post.
1458
        $this->assertTrue(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion1->id, $cm));
1459
 
1460
        $this->assertFalse(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion2->id, $cm));
1461
        // Add a discussion post in a forum discussion where the user is not subscribed (auto-subscribe preference disabled).
1462
        mod_forum_external::add_discussion_post($discussion2->firstpost, 'some subject 2', 'some text here 2...');
1463
 
1464
        $posts = mod_forum_external::get_discussion_posts($discussion2->id, 'modified', 'ASC');
1465
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1466
        // We receive the discussion and the post.
1467
        $this->assertEquals(2, count($posts['posts']));
1468
        // The user should still not be subscribed to the discussion after adding a discussion post.
1469
        $this->assertFalse(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion2->id, $cm));
1470
 
1471
        // Passing a value for the discussionsubscribe option parameter.
1472
        $this->assertFalse(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion2->id, $cm));
1473
        // Add a discussion post in a forum discussion where the user is not subscribed (auto-subscribe preference disabled),
1474
        // and the option parameter 'discussionsubscribe' => true in the webservice.
1475
        $option = array('name' => 'discussionsubscribe', 'value' => true);
1476
        $options[] = $option;
1477
        mod_forum_external::add_discussion_post($discussion2->firstpost, 'some subject 2', 'some text here 2...',
1478
            $options);
1479
 
1480
        $posts = mod_forum_external::get_discussion_posts($discussion2->id, 'modified', 'ASC');
1481
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1482
        // We receive the discussion and the post.
1483
        $this->assertEquals(3, count($posts['posts']));
1484
        // The user should now be subscribed to the discussion after adding a discussion post.
1485
        $this->assertTrue(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion2->id, $cm));
1486
    }
1487
 
1488
    /*
1489
     * Test add_discussion. A basic test since all the API functions are already covered by unit tests.
1490
     */
11 efrain 1491
    public function test_add_discussion(): void {
1 efrain 1492
        global $CFG, $USER;
1493
        $this->resetAfterTest(true);
1494
 
1495
        // Create courses to add the modules.
1496
        $course = self::getDataGenerator()->create_course();
1497
 
1498
        $user1 = self::getDataGenerator()->create_user();
1499
        $user2 = self::getDataGenerator()->create_user();
1500
 
1501
        // First forum with tracking off.
1502
        $record = new \stdClass();
1503
        $record->course = $course->id;
1504
        $record->type = 'news';
1505
        $forum = self::getDataGenerator()->create_module('forum', $record);
1506
 
1507
        self::setUser($user1);
1508
        $this->getDataGenerator()->enrol_user($user1->id, $course->id);
1509
 
1510
        try {
1511
            mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...');
1512
            $this->fail('Exception expected due to invalid permissions.');
1513
        } catch (\moodle_exception $e) {
1514
            $this->assertEquals('cannotcreatediscussion', $e->errorcode);
1515
        }
1516
 
1517
        self::setAdminUser();
1518
        $createddiscussion = mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...');
1519
        $createddiscussion = external_api::clean_returnvalue(mod_forum_external::add_discussion_returns(), $createddiscussion);
1520
 
1521
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1522
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1523
 
1524
        $this->assertCount(1, $discussions['discussions']);
1525
        $this->assertCount(0, $discussions['warnings']);
1526
 
1527
        $this->assertEquals($createddiscussion['discussionid'], $discussions['discussions'][0]['discussion']);
1528
        $this->assertEquals(-1, $discussions['discussions'][0]['groupid']);
1529
        $this->assertEquals('the subject', $discussions['discussions'][0]['subject']);
1530
        $this->assertEquals('some text here...', $discussions['discussions'][0]['message']);
1531
 
1532
        $discussion2pinned = mod_forum_external::add_discussion($forum->id, 'the pinned subject', 'some 2 text here...', -1,
1533
                                                                array('options' => array('name' => 'discussionpinned',
1534
                                                                                         'value' => true)));
1535
        $discussion3 = mod_forum_external::add_discussion($forum->id, 'the non pinnedsubject', 'some 3 text here...');
1536
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1537
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1538
        $this->assertCount(3, $discussions['discussions']);
1539
        $this->assertEquals($discussion2pinned['discussionid'], $discussions['discussions'][0]['discussion']);
1540
 
1541
        // Test inline and regular attachment in new discussion
1542
        // Create a file in a draft area for inline attachments.
1543
 
1544
        $fs = get_file_storage();
1545
 
1546
        $draftidinlineattach = file_get_unused_draft_itemid();
1547
        $draftidattach = file_get_unused_draft_itemid();
1548
 
1549
        $usercontext = \context_user::instance($USER->id);
1550
        $filepath = '/';
1551
        $filearea = 'draft';
1552
        $component = 'user';
1553
        $filenameimg = 'shouldbeanimage.txt';
1554
        $filerecord = array(
1555
            'contextid' => $usercontext->id,
1556
            'component' => $component,
1557
            'filearea'  => $filearea,
1558
            'itemid'    => $draftidinlineattach,
1559
            'filepath'  => $filepath,
1560
            'filename'  => $filenameimg,
1561
        );
1562
 
1563
        // Create a file in a draft area for regular attachments.
1564
        $filerecordattach = $filerecord;
1565
        $attachfilename = 'attachment.txt';
1566
        $filerecordattach['filename'] = $attachfilename;
1567
        $filerecordattach['itemid'] = $draftidattach;
1568
        $fs->create_file_from_string($filerecord, 'image contents (not really)');
1569
        $fs->create_file_from_string($filerecordattach, 'simple text attachment');
1570
 
1571
        $dummytext = 'Here is an inline image: <img src="' . $CFG->wwwroot .
1572
                    "/draftfile.php/{$usercontext->id}/user/draft/{$draftidinlineattach}/{$filenameimg}" .
1573
                    '" alt="inlineimage">.';
1574
 
1575
        $options = array(array('name' => 'inlineattachmentsid', 'value' => $draftidinlineattach),
1576
                         array('name' => 'attachmentsid', 'value' => $draftidattach));
1577
        $createddiscussion = mod_forum_external::add_discussion($forum->id, 'the inline attachment subject',
1578
                                                                $dummytext, -1, $options);
1579
        $createddiscussion = external_api::clean_returnvalue(mod_forum_external::add_discussion_returns(), $createddiscussion);
1580
 
1581
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1582
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1583
 
1584
        $this->assertCount(4, $discussions['discussions']);
1585
        $this->assertCount(0, $createddiscussion['warnings']);
1586
        // Can't guarantee order of posts during tests.
1587
        $postfound = false;
1588
        foreach ($discussions['discussions'] as $thisdiscussion) {
1589
            if ($createddiscussion['discussionid'] == $thisdiscussion['discussion']) {
1590
                $this->assertEquals($thisdiscussion['attachment'], 1, "There should be a non-inline attachment");
1591
                $this->assertCount(1, $thisdiscussion['attachments'], "There should be 1 attachment");
1592
                $this->assertEquals($thisdiscussion['attachments'][0]['filename'], $attachfilename, "There should be 1 attachment");
1593
                $this->assertStringNotContainsString('draftfile.php', $thisdiscussion['message']);
1594
                $this->assertStringContainsString('pluginfile.php', $thisdiscussion['message']);
1595
                $postfound = true;
1596
                break;
1597
            }
1598
        }
1599
 
1600
        $this->assertTrue($postfound);
1601
    }
1602
 
1603
    /**
1604
     * Test adding discussions in a course with gorups
1605
     */
11 efrain 1606
    public function test_add_discussion_in_course_with_groups(): void {
1 efrain 1607
        global $CFG;
1608
 
1609
        $this->resetAfterTest(true);
1610
 
1611
        // Create course to add the module.
1612
        $course = self::getDataGenerator()->create_course(array('groupmode' => VISIBLEGROUPS, 'groupmodeforce' => 0));
1613
        $user = self::getDataGenerator()->create_user();
1614
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
1615
 
1616
        // Forum forcing separate gropus.
1617
        $record = new \stdClass();
1618
        $record->course = $course->id;
1619
        $forum = self::getDataGenerator()->create_module('forum', $record, array('groupmode' => SEPARATEGROUPS));
1620
 
1621
        // Try to post (user not enrolled).
1622
        self::setUser($user);
1623
 
1624
        // The user is not enroled in any group, try to post in a forum with separate groups.
1625
        try {
1626
            mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...');
1627
            $this->fail('Exception expected due to invalid group permissions.');
1628
        } catch (\moodle_exception $e) {
1629
            $this->assertEquals('cannotcreatediscussion', $e->errorcode);
1630
        }
1631
 
1632
        try {
1633
            mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...', 0);
1634
            $this->fail('Exception expected due to invalid group permissions.');
1635
        } catch (\moodle_exception $e) {
1636
            $this->assertEquals('cannotcreatediscussion', $e->errorcode);
1637
        }
1638
 
1639
        // Create a group.
1640
        $group = $this->getDataGenerator()->create_group(array('courseid' => $course->id));
1641
 
1642
        // Try to post in a group the user is not enrolled.
1643
        try {
1644
            mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...', $group->id);
1645
            $this->fail('Exception expected due to invalid group permissions.');
1646
        } catch (\moodle_exception $e) {
1647
            $this->assertEquals('cannotcreatediscussion', $e->errorcode);
1648
        }
1649
 
1650
        // Add the user to a group.
1651
        groups_add_member($group->id, $user->id);
1652
 
1653
        // Try to post in a group the user is not enrolled.
1654
        try {
1655
            mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...', $group->id + 1);
1656
            $this->fail('Exception expected due to invalid group.');
1657
        } catch (\moodle_exception $e) {
1658
            $this->assertEquals('cannotcreatediscussion', $e->errorcode);
1659
        }
1660
 
1661
        // Nost add the discussion using a valid group.
1662
        $discussion = mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...', $group->id);
1663
        $discussion = external_api::clean_returnvalue(mod_forum_external::add_discussion_returns(), $discussion);
1664
 
1665
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1666
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1667
 
1668
        $this->assertCount(1, $discussions['discussions']);
1669
        $this->assertCount(0, $discussions['warnings']);
1670
        $this->assertEquals($discussion['discussionid'], $discussions['discussions'][0]['discussion']);
1671
        $this->assertEquals($group->id, $discussions['discussions'][0]['groupid']);
1672
 
1673
        // Now add a discussions without indicating a group. The function should guess the correct group.
1674
        $discussion = mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...');
1675
        $discussion = external_api::clean_returnvalue(mod_forum_external::add_discussion_returns(), $discussion);
1676
 
1677
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1678
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1679
 
1680
        $this->assertCount(2, $discussions['discussions']);
1681
        $this->assertCount(0, $discussions['warnings']);
1682
        $this->assertEquals($group->id, $discussions['discussions'][0]['groupid']);
1683
        $this->assertEquals($group->id, $discussions['discussions'][1]['groupid']);
1684
 
1685
        // Enrol the same user in other group.
1686
        $group2 = $this->getDataGenerator()->create_group(array('courseid' => $course->id));
1687
        groups_add_member($group2->id, $user->id);
1688
 
1689
        // Now add a discussions without indicating a group. The function should guess the correct group (the first one).
1690
        $discussion = mod_forum_external::add_discussion($forum->id, 'the subject', 'some text here...');
1691
        $discussion = external_api::clean_returnvalue(mod_forum_external::add_discussion_returns(), $discussion);
1692
 
1693
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
1694
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
1695
 
1696
        $this->assertCount(3, $discussions['discussions']);
1697
        $this->assertCount(0, $discussions['warnings']);
1698
        $this->assertEquals($group->id, $discussions['discussions'][0]['groupid']);
1699
        $this->assertEquals($group->id, $discussions['discussions'][1]['groupid']);
1700
        $this->assertEquals($group->id, $discussions['discussions'][2]['groupid']);
1701
 
1702
    }
1703
 
1704
    /*
1705
     * Test set_lock_state.
1706
     *
1707
     * @covers \mod_forum\event\discussion_lock_updated
1708
     */
11 efrain 1709
    public function test_set_lock_state(): void {
1 efrain 1710
        global $DB;
1711
        $this->resetAfterTest(true);
1712
 
1713
        // Create courses to add the modules.
1714
        $course = self::getDataGenerator()->create_course();
1715
        $user = self::getDataGenerator()->create_user();
1716
        $studentrole = $DB->get_record('role', array('shortname' => 'student'));
1717
 
1718
        // First forum with tracking off.
1719
        $record = new \stdClass();
1720
        $record->course = $course->id;
1721
        $record->type = 'news';
1722
        $forum = self::getDataGenerator()->create_module('forum', $record);
1723
        $context = \context_module::instance($forum->cmid);
1724
 
1725
        $record = new \stdClass();
1726
        $record->course = $course->id;
1727
        $record->userid = $user->id;
1728
        $record->forum = $forum->id;
1729
        $discussion = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1730
 
1731
        // User who is a student.
1732
        self::setUser($user);
1733
        $this->getDataGenerator()->enrol_user($user->id, $course->id, $studentrole->id, 'manual');
1734
 
1735
        // Only a teacher should be able to lock a discussion.
1736
        try {
1737
            $result = mod_forum_external::set_lock_state($forum->id, $discussion->id, 0);
1738
            $this->fail('Exception expected due to missing capability.');
1739
        } catch (\moodle_exception $e) {
1740
            $this->assertEquals('errorcannotlock', $e->errorcode);
1741
        }
1742
 
1743
        // Set the lock.
1744
        self::setAdminUser();
1745
        $sink = $this->redirectEvents(); // Capturing the event.
1746
        $result = mod_forum_external::set_lock_state($forum->id, $discussion->id, 0);
1747
        $result = external_api::clean_returnvalue(mod_forum_external::set_lock_state_returns(), $result);
1748
        $this->assertTrue($result['locked']);
1749
        $this->assertNotEquals(0, $result['times']['locked']);
1750
 
1751
        // Check that the event contains the expected values.
1752
        $events = $sink->get_events();
1753
        $this->assertCount(1, $events);
1754
        $event = reset($events);
1755
        $this->assertInstanceOf('\mod_forum\event\discussion_lock_updated', $event);
1756
        $this->assertEquals($context, $event->get_context());
1757
        $this->assertEventContextNotUsed($event);
1758
        $this->assertNotEmpty($event->get_name());
1759
        $this->assertStringContainsString(' locked the discussion:', $event->get_description());
1760
 
1761
        // Unset the lock.
1762
        $sink = $this->redirectEvents(); // Capturing the event.
1763
        $result = mod_forum_external::set_lock_state($forum->id, $discussion->id, time());
1764
        $result = external_api::clean_returnvalue(mod_forum_external::set_lock_state_returns(), $result);
1765
        $this->assertFalse($result['locked']);
1766
        $this->assertEquals('0', $result['times']['locked']);
1767
 
1768
        // Check that the event contains the expected values.
1769
        $events = $sink->get_events();
1770
        $this->assertCount(1, $events);
1771
        $event = reset($events);
1772
        $this->assertInstanceOf('\mod_forum\event\discussion_lock_updated', $event);
1773
        $this->assertEquals($context, $event->get_context());
1774
        $this->assertEventContextNotUsed($event);
1775
        $this->assertNotEmpty($event->get_name());
1776
        $this->assertStringContainsString(' unlocked the discussion:', $event->get_description());
1777
    }
1778
 
1779
    /*
1780
     * Test can_add_discussion. A basic test since all the API functions are already covered by unit tests.
1781
     */
11 efrain 1782
    public function test_can_add_discussion(): void {
1 efrain 1783
        global $DB;
1784
        $this->resetAfterTest(true);
1785
 
1786
        // Create courses to add the modules.
1787
        $course = self::getDataGenerator()->create_course();
1788
 
1789
        $user = self::getDataGenerator()->create_user();
1790
 
1791
        // First forum with tracking off.
1792
        $record = new \stdClass();
1793
        $record->course = $course->id;
1794
        $record->type = 'news';
1795
        $forum = self::getDataGenerator()->create_module('forum', $record);
1796
 
1797
        // User with no permissions to add in a news forum.
1798
        self::setUser($user);
1799
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
1800
 
1801
        $result = mod_forum_external::can_add_discussion($forum->id);
1802
        $result = external_api::clean_returnvalue(mod_forum_external::can_add_discussion_returns(), $result);
1803
        $this->assertFalse($result['status']);
1804
        $this->assertFalse($result['canpindiscussions']);
1805
        $this->assertTrue($result['cancreateattachment']);
1806
 
1807
        // Disable attachments.
1808
        $DB->set_field('forum', 'maxattachments', 0, array('id' => $forum->id));
1809
        $result = mod_forum_external::can_add_discussion($forum->id);
1810
        $result = external_api::clean_returnvalue(mod_forum_external::can_add_discussion_returns(), $result);
1811
        $this->assertFalse($result['status']);
1812
        $this->assertFalse($result['canpindiscussions']);
1813
        $this->assertFalse($result['cancreateattachment']);
1814
        $DB->set_field('forum', 'maxattachments', 1, array('id' => $forum->id));    // Enable attachments again.
1815
 
1816
        self::setAdminUser();
1817
        $result = mod_forum_external::can_add_discussion($forum->id);
1818
        $result = external_api::clean_returnvalue(mod_forum_external::can_add_discussion_returns(), $result);
1819
        $this->assertTrue($result['status']);
1820
        $this->assertTrue($result['canpindiscussions']);
1821
        $this->assertTrue($result['cancreateattachment']);
1822
    }
1823
 
1824
    /*
1825
     * A basic test to make sure users cannot post to forum after the cutoff date.
1826
     */
11 efrain 1827
    public function test_can_add_discussion_after_cutoff(): void {
1 efrain 1828
        $this->resetAfterTest(true);
1829
 
1830
        // Create courses to add the modules.
1831
        $course = self::getDataGenerator()->create_course();
1832
 
1833
        $user = self::getDataGenerator()->create_user();
1834
 
1835
        // Create a forum with cutoff date set to a past date.
1836
        $forum = self::getDataGenerator()->create_module('forum', ['course' => $course->id, 'cutoffdate' => time() - 1]);
1837
 
1838
        // User with no mod/forum:canoverridecutoff capability.
1839
        self::setUser($user);
1840
        $this->getDataGenerator()->enrol_user($user->id, $course->id);
1841
 
1842
        $result = mod_forum_external::can_add_discussion($forum->id);
1843
        $result = external_api::clean_returnvalue(mod_forum_external::can_add_discussion_returns(), $result);
1844
        $this->assertFalse($result['status']);
1845
 
1846
        self::setAdminUser();
1847
        $result = mod_forum_external::can_add_discussion($forum->id);
1848
        $result = external_api::clean_returnvalue(mod_forum_external::can_add_discussion_returns(), $result);
1849
        $this->assertTrue($result['status']);
1850
    }
1851
 
1852
    /**
1853
     * Test get posts discussions including rating information.
1854
     */
11 efrain 1855
    public function test_mod_forum_get_discussion_rating_information(): void {
1 efrain 1856
        global $DB, $CFG, $PAGE;
1857
        require_once($CFG->dirroot . '/rating/lib.php');
1858
        $PAGE->set_url('/my/index.php');    // Need this because some internal API calls require the $PAGE url to be set.
1859
        $this->resetAfterTest(true);
1860
 
1861
        $user1 = self::getDataGenerator()->create_user();
1862
        $user2 = self::getDataGenerator()->create_user();
1863
        $user3 = self::getDataGenerator()->create_user();
1864
        $teacher = self::getDataGenerator()->create_user();
1865
 
1866
        // Create course to add the module.
1867
        $course = self::getDataGenerator()->create_course();
1868
 
1869
        $studentrole = $DB->get_record('role', array('shortname' => 'student'));
1870
        $teacherrole = $DB->get_record('role', array('shortname' => 'editingteacher'));
1871
        $this->getDataGenerator()->enrol_user($user1->id, $course->id, $studentrole->id, 'manual');
1872
        $this->getDataGenerator()->enrol_user($user2->id, $course->id, $studentrole->id, 'manual');
1873
        $this->getDataGenerator()->enrol_user($user3->id, $course->id, $studentrole->id, 'manual');
1874
        $this->getDataGenerator()->enrol_user($teacher->id, $course->id, $teacherrole->id, 'manual');
1875
 
1876
        // Create the forum.
1877
        $record = new \stdClass();
1878
        $record->course = $course->id;
1879
        // Set Aggregate type = Average of ratings.
1880
        $record->assessed = RATING_AGGREGATE_AVERAGE;
1881
        $record->scale = 100;
1882
        $forum = self::getDataGenerator()->create_module('forum', $record);
1883
        $context = \context_module::instance($forum->cmid);
1884
 
1885
        // Add discussion to the forum.
1886
        $record = new \stdClass();
1887
        $record->course = $course->id;
1888
        $record->userid = $user1->id;
1889
        $record->forum = $forum->id;
1890
        $discussion = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
1891
 
1892
        // Retrieve the first post.
1893
        $post = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
1894
 
1895
        // Rate the discussion as user2.
1896
        $rating1 = new \stdClass();
1897
        $rating1->contextid = $context->id;
1898
        $rating1->component = 'mod_forum';
1899
        $rating1->ratingarea = 'post';
1900
        $rating1->itemid = $post->id;
1901
        $rating1->rating = 50;
1902
        $rating1->scaleid = 100;
1903
        $rating1->userid = $user2->id;
1904
        $rating1->timecreated = time();
1905
        $rating1->timemodified = time();
1906
        $rating1->id = $DB->insert_record('rating', $rating1);
1907
 
1908
        // Rate the discussion as user3.
1909
        $rating2 = new \stdClass();
1910
        $rating2->contextid = $context->id;
1911
        $rating2->component = 'mod_forum';
1912
        $rating2->ratingarea = 'post';
1913
        $rating2->itemid = $post->id;
1914
        $rating2->rating = 100;
1915
        $rating2->scaleid = 100;
1916
        $rating2->userid = $user3->id;
1917
        $rating2->timecreated = time() + 1;
1918
        $rating2->timemodified = time() + 1;
1919
        $rating2->id = $DB->insert_record('rating', $rating2);
1920
 
1921
        // Retrieve the rating for the post as student.
1922
        $this->setUser($user1);
1923
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'DESC');
1924
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1925
        $this->assertCount(1, $posts['ratinginfo']['ratings']);
1926
        $this->assertTrue($posts['ratinginfo']['ratings'][0]['canviewaggregate']);
1927
        $this->assertFalse($posts['ratinginfo']['canviewall']);
1928
        $this->assertFalse($posts['ratinginfo']['ratings'][0]['canrate']);
1929
        $this->assertEquals(2, $posts['ratinginfo']['ratings'][0]['count']);
1930
        $this->assertEquals(($rating1->rating + $rating2->rating) / 2, $posts['ratinginfo']['ratings'][0]['aggregate']);
1931
 
1932
        // Retrieve the rating for the post as teacher.
1933
        $this->setUser($teacher);
1934
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'DESC');
1935
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
1936
        $this->assertCount(1, $posts['ratinginfo']['ratings']);
1937
        $this->assertTrue($posts['ratinginfo']['ratings'][0]['canviewaggregate']);
1938
        $this->assertTrue($posts['ratinginfo']['canviewall']);
1939
        $this->assertTrue($posts['ratinginfo']['ratings'][0]['canrate']);
1940
        $this->assertEquals(2, $posts['ratinginfo']['ratings'][0]['count']);
1941
        $this->assertEquals(($rating1->rating + $rating2->rating) / 2, $posts['ratinginfo']['ratings'][0]['aggregate']);
1942
    }
1943
 
1944
    /**
1945
     * Test mod_forum_get_forum_access_information.
1946
     */
11 efrain 1947
    public function test_mod_forum_get_forum_access_information(): void {
1 efrain 1948
        global $DB;
1949
 
1950
        $this->resetAfterTest(true);
1951
 
1952
        $student = self::getDataGenerator()->create_user();
1953
        $course = self::getDataGenerator()->create_course();
1954
        // Create the forum.
1955
        $record = new \stdClass();
1956
        $record->course = $course->id;
1957
        $forum = self::getDataGenerator()->create_module('forum', $record);
1958
 
1959
        $studentrole = $DB->get_record('role', array('shortname' => 'student'));
1960
        $this->getDataGenerator()->enrol_user($student->id, $course->id, $studentrole->id, 'manual');
1961
 
1962
        self::setUser($student);
1963
        $result = mod_forum_external::get_forum_access_information($forum->id);
1964
        $result = external_api::clean_returnvalue(mod_forum_external::get_forum_access_information_returns(), $result);
1965
 
1966
        // Check default values for capabilities.
1967
        $enabledcaps = array('canviewdiscussion', 'canstartdiscussion', 'canreplypost', 'canviewrating', 'cancreateattachment',
1968
            'canexportownpost', 'cancantogglefavourite', 'cancanmailnow', 'candeleteownpost', 'canallowforcesubscribe');
1969
 
1970
        unset($result['warnings']);
1971
        foreach ($result as $capname => $capvalue) {
1972
            if (in_array($capname, $enabledcaps)) {
1973
                $this->assertTrue($capvalue);
1974
            } else {
1975
                $this->assertFalse($capvalue);
1976
            }
1977
        }
1978
        // Now, unassign some capabilities.
1979
        unassign_capability('mod/forum:deleteownpost', $studentrole->id);
1980
        unassign_capability('mod/forum:allowforcesubscribe', $studentrole->id);
1981
        array_pop($enabledcaps);
1982
        array_pop($enabledcaps);
1983
        accesslib_clear_all_caches_for_unit_testing();
1984
 
1985
        $result = mod_forum_external::get_forum_access_information($forum->id);
1986
        $result = external_api::clean_returnvalue(mod_forum_external::get_forum_access_information_returns(), $result);
1987
        unset($result['warnings']);
1988
        foreach ($result as $capname => $capvalue) {
1989
            if (in_array($capname, $enabledcaps)) {
1990
                $this->assertTrue($capvalue);
1991
            } else {
1992
                $this->assertFalse($capvalue);
1993
            }
1994
        }
1995
    }
1996
 
1997
    /**
1998
     * Test add_discussion_post
1999
     */
11 efrain 2000
    public function test_add_discussion_post_private(): void {
1 efrain 2001
        global $DB;
2002
 
2003
        $this->resetAfterTest(true);
2004
 
2005
        self::setAdminUser();
2006
 
2007
        // Create course to add the module.
2008
        $course = self::getDataGenerator()->create_course();
2009
 
2010
        // Standard forum.
2011
        $record = new \stdClass();
2012
        $record->course = $course->id;
2013
        $forum = self::getDataGenerator()->create_module('forum', $record);
2014
        $cm = get_coursemodule_from_id('forum', $forum->cmid, 0, false, MUST_EXIST);
2015
        $forumcontext = \context_module::instance($forum->cmid);
2016
        $generator = self::getDataGenerator()->get_plugin_generator('mod_forum');
2017
 
2018
        // Create an enrol users.
2019
        $student1 = self::getDataGenerator()->create_user();
2020
        $this->getDataGenerator()->enrol_user($student1->id, $course->id, 'student');
2021
        $student2 = self::getDataGenerator()->create_user();
2022
        $this->getDataGenerator()->enrol_user($student2->id, $course->id, 'student');
2023
        $teacher1 = self::getDataGenerator()->create_user();
2024
        $this->getDataGenerator()->enrol_user($teacher1->id, $course->id, 'editingteacher');
2025
        $teacher2 = self::getDataGenerator()->create_user();
2026
        $this->getDataGenerator()->enrol_user($teacher2->id, $course->id, 'editingteacher');
2027
 
2028
        // Add a new discussion to the forum.
2029
        self::setUser($student1);
2030
        $record = new \stdClass();
2031
        $record->course = $course->id;
2032
        $record->userid = $student1->id;
2033
        $record->forum = $forum->id;
2034
        $discussion = $generator->create_discussion($record);
2035
 
2036
        // Have the teacher reply privately.
2037
        self::setUser($teacher1);
2038
        $post = mod_forum_external::add_discussion_post($discussion->firstpost, 'some subject', 'some text here...', [
2039
                [
2040
                    'name' => 'private',
2041
                    'value' => true,
2042
                ],
2043
            ]);
2044
        $post = external_api::clean_returnvalue(mod_forum_external::add_discussion_post_returns(), $post);
2045
        $privatereply = $DB->get_record('forum_posts', array('id' => $post['postid']));
2046
        $this->assertEquals($student1->id, $privatereply->privatereplyto);
2047
        // Bump the time of the private reply to ensure order.
2048
        $privatereply->created++;
2049
        $privatereply->modified = $privatereply->created;
2050
        $DB->update_record('forum_posts', $privatereply);
2051
 
2052
        // The teacher will receive their private reply.
2053
        self::setUser($teacher1);
2054
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'DESC');
2055
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2056
        $this->assertEquals(2, count($posts['posts']));
2057
        $this->assertTrue($posts['posts'][0]['isprivatereply']);
2058
 
2059
        // Another teacher on the course will also receive the private reply.
2060
        self::setUser($teacher2);
2061
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'DESC');
2062
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2063
        $this->assertEquals(2, count($posts['posts']));
2064
        $this->assertTrue($posts['posts'][0]['isprivatereply']);
2065
 
2066
        // The student will receive the private reply.
2067
        self::setUser($student1);
2068
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'DESC');
2069
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2070
        $this->assertEquals(2, count($posts['posts']));
2071
        $this->assertTrue($posts['posts'][0]['isprivatereply']);
2072
 
2073
        // Another student will not receive the private reply.
2074
        self::setUser($student2);
2075
        $posts = mod_forum_external::get_discussion_posts($discussion->id, 'id', 'ASC');
2076
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2077
        $this->assertEquals(1, count($posts['posts']));
2078
        $this->assertFalse($posts['posts'][0]['isprivatereply']);
2079
 
2080
        // A user cannot reply to a private reply.
2081
        self::setUser($teacher2);
2082
        $this->expectException('coding_exception');
2083
        $post = mod_forum_external::add_discussion_post($privatereply->id, 'some subject', 'some text here...', [
2084
                'options' => [
2085
                    'name' => 'private',
2086
                    'value' => false,
2087
                ],
2088
            ]);
2089
    }
2090
 
2091
    /**
2092
     * Test trusted text enabled.
2093
     */
11 efrain 2094
    public function test_trusted_text_enabled(): void {
1 efrain 2095
        global $USER, $CFG;
2096
 
2097
        $this->resetAfterTest(true);
2098
        $CFG->enabletrusttext = 1;
2099
 
2100
        $dangeroustext = '<button>Untrusted text</button>';
2101
        $cleantext = 'Untrusted text';
2102
 
2103
        // Create courses to add the modules.
2104
        $course = self::getDataGenerator()->create_course();
2105
        $user1 = self::getDataGenerator()->create_user();
2106
 
2107
        // First forum with tracking off.
2108
        $record = new \stdClass();
2109
        $record->course = $course->id;
2110
        $record->type = 'qanda';
2111
        $forum = self::getDataGenerator()->create_module('forum', $record);
2112
        $context = \context_module::instance($forum->cmid);
2113
 
2114
        // Add discussions to the forums.
2115
        $discussionrecord = new \stdClass();
2116
        $discussionrecord->course = $course->id;
2117
        $discussionrecord->userid = $user1->id;
2118
        $discussionrecord->forum = $forum->id;
2119
        $discussionrecord->message = $dangeroustext;
2120
        $discussionrecord->messagetrust  = trusttext_trusted($context);
2121
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($discussionrecord);
2122
 
2123
        self::setAdminUser();
2124
        $discussionrecord->userid = $USER->id;
2125
        $discussionrecord->messagetrust  = trusttext_trusted($context);
2126
        $discussion2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($discussionrecord);
2127
 
2128
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
2129
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
2130
 
2131
        $this->assertCount(2, $discussions['discussions']);
2132
        $this->assertCount(0, $discussions['warnings']);
2133
        // Admin message is fully trusted.
2134
        $this->assertEquals(1, $discussions['discussions'][0]['messagetrust']);
2135
        $this->assertEquals($dangeroustext, $discussions['discussions'][0]['message']);
2136
        // Student message is not trusted.
2137
        $this->assertEquals(0, $discussions['discussions'][1]['messagetrust']);
2138
        $this->assertEquals($cleantext, $discussions['discussions'][1]['message']);
2139
 
2140
        // Get posts now.
2141
        $posts = mod_forum_external::get_discussion_posts($discussion2->id, 'modified', 'DESC');
2142
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2143
        // Admin message is fully trusted.
2144
        $this->assertEquals($dangeroustext, $posts['posts'][0]['message']);
2145
 
2146
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'ASC');
2147
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2148
        // Student message is not trusted.
2149
        $this->assertEquals($cleantext, $posts['posts'][0]['message']);
2150
    }
2151
 
2152
    /**
2153
     * Test trusted text disabled.
2154
     */
11 efrain 2155
    public function test_trusted_text_disabled(): void {
1 efrain 2156
        global $USER, $CFG;
2157
 
2158
        $this->resetAfterTest(true);
2159
        $CFG->enabletrusttext = 0;
2160
 
2161
        $dangeroustext = '<button>Untrusted text</button>';
2162
        $cleantext = 'Untrusted text';
2163
 
2164
        // Create courses to add the modules.
2165
        $course = self::getDataGenerator()->create_course();
2166
        $user1 = self::getDataGenerator()->create_user();
2167
 
2168
        // First forum with tracking off.
2169
        $record = new \stdClass();
2170
        $record->course = $course->id;
2171
        $record->type = 'qanda';
2172
        $forum = self::getDataGenerator()->create_module('forum', $record);
2173
        $context = \context_module::instance($forum->cmid);
2174
 
2175
        // Add discussions to the forums.
2176
        $discussionrecord = new \stdClass();
2177
        $discussionrecord->course = $course->id;
2178
        $discussionrecord->userid = $user1->id;
2179
        $discussionrecord->forum = $forum->id;
2180
        $discussionrecord->message = $dangeroustext;
2181
        $discussionrecord->messagetrust = trusttext_trusted($context);
2182
        $discussion1 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($discussionrecord);
2183
 
2184
        self::setAdminUser();
2185
        $discussionrecord->userid = $USER->id;
2186
        $discussionrecord->messagetrust = trusttext_trusted($context);
2187
        $discussion2 = self::getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($discussionrecord);
2188
 
2189
        $discussions = mod_forum_external::get_forum_discussions($forum->id);
2190
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_forum_discussions_returns(), $discussions);
2191
 
2192
        $this->assertCount(2, $discussions['discussions']);
2193
        $this->assertCount(0, $discussions['warnings']);
2194
        // Admin message is not trusted because enabletrusttext is disabled.
2195
        $this->assertEquals(0, $discussions['discussions'][0]['messagetrust']);
2196
        $this->assertEquals($cleantext, $discussions['discussions'][0]['message']);
2197
        // Student message is not trusted.
2198
        $this->assertEquals(0, $discussions['discussions'][1]['messagetrust']);
2199
        $this->assertEquals($cleantext, $discussions['discussions'][1]['message']);
2200
 
2201
        // Get posts now.
2202
        $posts = mod_forum_external::get_discussion_posts($discussion2->id, 'modified', 'ASC');
2203
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2204
        // Admin message is not trusted because enabletrusttext is disabled.
2205
        $this->assertEquals($cleantext, $posts['posts'][0]['message']);
2206
 
2207
        $posts = mod_forum_external::get_discussion_posts($discussion1->id, 'modified', 'ASC');
2208
        $posts = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $posts);
2209
        // Student message is not trusted.
2210
        $this->assertEquals($cleantext, $posts['posts'][0]['message']);
2211
    }
2212
 
2213
    /**
2214
     * Test delete a discussion.
2215
     */
11 efrain 2216
    public function test_delete_post_discussion(): void {
1 efrain 2217
        global $DB;
2218
        $this->resetAfterTest(true);
2219
 
2220
        // Setup test data.
2221
        $course = $this->getDataGenerator()->create_course();
2222
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2223
        $user = $this->getDataGenerator()->create_user();
2224
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2225
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2226
 
2227
        // Add a discussion.
2228
        $record = new \stdClass();
2229
        $record->course = $course->id;
2230
        $record->userid = $user->id;
2231
        $record->forum = $forum->id;
2232
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2233
 
2234
        $this->setUser($user);
2235
        $result = mod_forum_external::delete_post($discussion->firstpost);
2236
        $result = external_api::clean_returnvalue(mod_forum_external::delete_post_returns(), $result);
2237
        $this->assertTrue($result['status']);
2238
        $this->assertEquals(0, $DB->count_records('forum_posts', array('id' => $discussion->firstpost)));
2239
        $this->assertEquals(0, $DB->count_records('forum_discussions', array('id' => $discussion->id)));
2240
    }
2241
 
2242
    /**
2243
     * Test delete a post.
2244
     */
11 efrain 2245
    public function test_delete_post_post(): void {
1 efrain 2246
        global $DB;
2247
        $this->resetAfterTest(true);
2248
 
2249
        // Setup test data.
2250
        $course = $this->getDataGenerator()->create_course();
2251
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2252
        $user = $this->getDataGenerator()->create_user();
2253
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2254
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2255
 
2256
        // Add a discussion.
2257
        $record = new \stdClass();
2258
        $record->course = $course->id;
2259
        $record->userid = $user->id;
2260
        $record->forum = $forum->id;
2261
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2262
        $parentpost = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
2263
 
2264
        // Add a post.
2265
        $record = new \stdClass();
2266
        $record->course = $course->id;
2267
        $record->userid = $user->id;
2268
        $record->forum = $forum->id;
2269
        $record->discussion = $discussion->id;
2270
        $record->parent = $parentpost->id;
2271
        $post = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
2272
 
2273
        $this->setUser($user);
2274
        $result = mod_forum_external::delete_post($post->id);
2275
        $result = external_api::clean_returnvalue(mod_forum_external::delete_post_returns(), $result);
2276
        $this->assertTrue($result['status']);
2277
        $this->assertEquals(1, $DB->count_records('forum_posts', array('discussion' => $discussion->id)));
2278
        $this->assertEquals(1, $DB->count_records('forum_discussions', array('id' => $discussion->id)));
2279
    }
2280
 
2281
    /**
2282
     * Test delete a different user post.
2283
     */
11 efrain 2284
    public function test_delete_post_other_user_post(): void {
1 efrain 2285
        global $DB;
2286
        $this->resetAfterTest(true);
2287
 
2288
        // Setup test data.
2289
        $course = $this->getDataGenerator()->create_course();
2290
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2291
        $user = $this->getDataGenerator()->create_user();
2292
        $otheruser = $this->getDataGenerator()->create_user();
2293
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2294
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2295
        self::getDataGenerator()->enrol_user($otheruser->id, $course->id, $role->id);
2296
 
2297
        // Add a discussion.
2298
        $record = array();
2299
        $record['course'] = $course->id;
2300
        $record['forum'] = $forum->id;
2301
        $record['userid'] = $user->id;
2302
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2303
        $parentpost = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
2304
 
2305
        // Add a post.
2306
        $record = new \stdClass();
2307
        $record->course = $course->id;
2308
        $record->userid = $user->id;
2309
        $record->forum = $forum->id;
2310
        $record->discussion = $discussion->id;
2311
        $record->parent = $parentpost->id;
2312
        $post = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
2313
 
2314
        $this->setUser($otheruser);
2315
        $this->expectExceptionMessage(get_string('cannotdeletepost', 'forum'));
2316
        mod_forum_external::delete_post($post->id);
2317
    }
2318
 
2319
    /*
2320
     * Test get forum posts by user id.
2321
     */
11 efrain 2322
    public function test_mod_forum_get_discussion_posts_by_userid(): void {
1 efrain 2323
        global $DB;
2324
        $this->resetAfterTest(true);
2325
 
2326
        $urlfactory = \mod_forum\local\container::get_url_factory();
2327
        $entityfactory = \mod_forum\local\container::get_entity_factory();
2328
        $vaultfactory = \mod_forum\local\container::get_vault_factory();
2329
        $postvault = $vaultfactory->get_post_vault();
2330
        $legacydatamapper = \mod_forum\local\container::get_legacy_data_mapper_factory();
2331
        $legacypostmapper = $legacydatamapper->get_post_data_mapper();
2332
 
2333
        // Create course to add the module.
2334
        $course1 = self::getDataGenerator()->create_course();
2335
 
2336
        $user1 = self::getDataGenerator()->create_user();
2337
        $user1entity = $entityfactory->get_author_from_stdClass($user1);
2338
        $exporteduser1 = [
2339
            'id' => (int) $user1->id,
2340
            'fullname' => fullname($user1),
2341
            'groups' => [],
2342
            'urls' => [
2343
                'profile' => $urlfactory->get_author_profile_url($user1entity, $course1->id)->out(false),
2344
                'profileimage' => $urlfactory->get_author_profile_image_url($user1entity),
2345
            ],
2346
            'isdeleted' => false,
2347
        ];
2348
        // Create a bunch of other users to post.
2349
        $user2 = self::getDataGenerator()->create_user();
2350
        $user2entity = $entityfactory->get_author_from_stdClass($user2);
2351
        $exporteduser2 = [
2352
            'id' => (int) $user2->id,
2353
            'fullname' => fullname($user2),
2354
            'groups' => [],
2355
            'urls' => [
2356
                'profile' => $urlfactory->get_author_profile_url($user2entity, $course1->id)->out(false),
2357
                'profileimage' => $urlfactory->get_author_profile_image_url($user2entity),
2358
            ],
2359
            'isdeleted' => false,
2360
        ];
2361
        $user2->fullname = $exporteduser2['fullname'];
2362
 
2363
        $forumgenerator = self::getDataGenerator()->get_plugin_generator('mod_forum');
2364
 
2365
        // Set the first created user to the test user.
2366
        self::setUser($user1);
2367
 
2368
        // Forum with tracking off.
2369
        $record = new \stdClass();
2370
        $record->course = $course1->id;
2371
        $forum1 = self::getDataGenerator()->create_module('forum', $record);
2372
        $forum1context = \context_module::instance($forum1->cmid);
2373
 
2374
        // Add discussions to the forums.
2375
        $time = time();
2376
        $record = new \stdClass();
2377
        $record->course = $course1->id;
2378
        $record->userid = $user1->id;
2379
        $record->forum = $forum1->id;
2380
        $record->timemodified = $time + 100;
2381
        $discussion1 = $forumgenerator->create_discussion($record);
2382
        $discussion1firstpost = $postvault->get_first_post_for_discussion_ids([$discussion1->id]);
2383
        $discussion1firstpost = $discussion1firstpost[$discussion1->firstpost];
2384
        $discussion1firstpostobject = $legacypostmapper->to_legacy_object($discussion1firstpost);
2385
 
2386
        $record = new \stdClass();
2387
        $record->course = $course1->id;
2388
        $record->userid = $user1->id;
2389
        $record->forum = $forum1->id;
2390
        $record->timemodified = $time + 200;
2391
        $discussion2 = $forumgenerator->create_discussion($record);
2392
        $discussion2firstpost = $postvault->get_first_post_for_discussion_ids([$discussion2->id]);
2393
        $discussion2firstpost = $discussion2firstpost[$discussion2->firstpost];
2394
        $discussion2firstpostobject = $legacypostmapper->to_legacy_object($discussion2firstpost);
2395
 
2396
        // Add 1 reply to the discussion 1 from a different user.
2397
        $record = new \stdClass();
2398
        $record->discussion = $discussion1->id;
2399
        $record->parent = $discussion1->firstpost;
2400
        $record->userid = $user2->id;
2401
        $discussion1reply1 = $forumgenerator->create_post($record);
2402
        $filename = 'shouldbeanimage.jpg';
2403
        // Add a fake inline image to the post.
2404
        $filerecordinline = array(
2405
                'contextid' => $forum1context->id,
2406
                'component' => 'mod_forum',
2407
                'filearea'  => 'post',
2408
                'itemid'    => $discussion1reply1->id,
2409
                'filepath'  => '/',
2410
                'filename'  => $filename,
2411
        );
2412
        $fs = get_file_storage();
2413
        $file1 = $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
2414
 
2415
        // Add 1 reply to the discussion 2 from a different user.
2416
        $record = new \stdClass();
2417
        $record->discussion = $discussion2->id;
2418
        $record->parent = $discussion2->firstpost;
2419
        $record->userid = $user2->id;
2420
        $discussion2reply1 = $forumgenerator->create_post($record);
2421
        $filename = 'shouldbeanimage.jpg';
2422
        // Add a fake inline image to the post.
2423
        $filerecordinline = array(
2424
                'contextid' => $forum1context->id,
2425
                'component' => 'mod_forum',
2426
                'filearea'  => 'post',
2427
                'itemid'    => $discussion2reply1->id,
2428
                'filepath'  => '/',
2429
                'filename'  => $filename,
2430
        );
2431
        $fs = get_file_storage();
2432
        $file2 = $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
2433
 
2434
        // Following line enrol and assign default role id to the user.
2435
        // So the user automatically gets mod/forum:viewdiscussion on all forums of the course.
2436
        $this->getDataGenerator()->enrol_user($user1->id, $course1->id, 'teacher');
2437
        $this->getDataGenerator()->enrol_user($user2->id, $course1->id);
2438
        // Changed display period for the discussions in past.
2439
        $discussion = new \stdClass();
2440
        $discussion->id = $discussion1->id;
2441
        $discussion->timestart = $time - 200;
2442
        $discussion->timeend = $time - 100;
2443
        $DB->update_record('forum_discussions', $discussion);
2444
        $discussion = new \stdClass();
2445
        $discussion->id = $discussion2->id;
2446
        $discussion->timestart = $time - 200;
2447
        $discussion->timeend = $time - 100;
2448
        $DB->update_record('forum_discussions', $discussion);
2449
        // Create what we expect to be returned when querying the discussion.
2450
        $expectedposts = array(
2451
            'discussions' => array(),
2452
            'warnings' => array(),
2453
        );
2454
 
2455
        $isolatedurluser = $urlfactory->get_discussion_view_url_from_discussion_id($discussion1reply1->discussion);
2456
        $isolatedurluser->params(['parent' => $discussion1reply1->id]);
2457
        $isolatedurlparent = $urlfactory->get_discussion_view_url_from_discussion_id($discussion1firstpostobject->discussion);
2458
        $isolatedurlparent->params(['parent' => $discussion1firstpostobject->id]);
2459
 
2460
        $expectedposts['discussions'][0] = [
2461
            'name' => $discussion1->name,
2462
            'id' => $discussion1->id,
2463
            'timecreated' => $discussion1firstpost->get_time_created(),
2464
            'authorfullname' => $user1entity->get_full_name(),
2465
            'posts' => [
2466
                'userposts' => [
2467
                    [
2468
                        'id' => $discussion1reply1->id,
2469
                        'discussionid' => $discussion1reply1->discussion,
2470
                        'parentid' => $discussion1reply1->parent,
2471
                        'hasparent' => true,
2472
                        'timecreated' => $discussion1reply1->created,
2473
                        'timemodified' => $discussion1reply1->modified,
2474
                        'subject' => $discussion1reply1->subject,
1441 ariadna 2475
                        'replysubject' => $discussion1reply1->subject,
1 efrain 2476
                        'message' => file_rewrite_pluginfile_urls($discussion1reply1->message, 'pluginfile.php',
2477
                        $forum1context->id, 'mod_forum', 'post', $discussion1reply1->id),
2478
                        'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
2479
                        'unread' => null,
2480
                        'isdeleted' => false,
2481
                        'isprivatereply' => false,
2482
                        'haswordcount' => false,
2483
                        'wordcount' => null,
2484
                        'author' => $exporteduser2,
2485
                        'attachments' => [],
2486
                        'messageinlinefiles' => [],
2487
                        'tags' => [],
2488
                        'html' => [
2489
                            'rating' => null,
2490
                            'taglist' => null,
2491
                            'authorsubheading' => $forumgenerator->get_author_subheading_html(
2492
                                (object)$exporteduser2, $discussion1reply1->created)
2493
                        ],
2494
                        'charcount' => null,
2495
                        'capabilities' => [
2496
                            'view' => true,
2497
                            'edit' => true,
2498
                            'delete' => true,
2499
                            'split' => true,
2500
                            'reply' => true,
2501
                            'export' => false,
2502
                            'controlreadstatus' => false,
2503
                            'canreplyprivately' => true,
2504
                            'selfenrol' => false
2505
                        ],
2506
                        'urls' => [
2507
                            'view' => $urlfactory->get_view_post_url_from_post_id(
2508
                                $discussion1reply1->discussion, $discussion1reply1->id)->out(false),
2509
                            'viewisolated' => $isolatedurluser->out(false),
2510
                            'viewparent' => $urlfactory->get_view_post_url_from_post_id(
2511
                                $discussion1reply1->discussion, $discussion1reply1->parent)->out(false),
2512
                            'edit' => (new \moodle_url('/mod/forum/post.php', [
2513
                                'edit' => $discussion1reply1->id
2514
                            ]))->out(false),
2515
                            'delete' => (new \moodle_url('/mod/forum/post.php', [
2516
                                'delete' => $discussion1reply1->id
2517
                            ]))->out(false),
2518
                            'split' => (new \moodle_url('/mod/forum/post.php', [
2519
                                'prune' => $discussion1reply1->id
2520
                            ]))->out(false),
2521
                            'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
2522
                                'reply' => $discussion1reply1->id
2523
                            ]))->out(false),
2524
                            'export' => null,
2525
                            'markasread' => null,
2526
                            'markasunread' => null,
2527
                            'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id(
2528
                                $discussion1reply1->discussion)->out(false),
2529
                        ],
2530
                    ]
2531
                ],
2532
                'parentposts' => [
2533
                    [
2534
                        'id' => $discussion1firstpostobject->id,
2535
                        'discussionid' => $discussion1firstpostobject->discussion,
2536
                        'parentid' => null,
2537
                        'hasparent' => false,
2538
                        'timecreated' => $discussion1firstpostobject->created,
2539
                        'timemodified' => $discussion1firstpostobject->modified,
2540
                        'subject' => $discussion1firstpostobject->subject,
1441 ariadna 2541
                        'replysubject' => $discussion1firstpostobject->subject,
1 efrain 2542
                        'message' => file_rewrite_pluginfile_urls($discussion1firstpostobject->message, 'pluginfile.php',
2543
                            $forum1context->id, 'mod_forum', 'post', $discussion1firstpostobject->id),
2544
                        'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
2545
                        'unread' => null,
2546
                        'isdeleted' => false,
2547
                        'isprivatereply' => false,
2548
                        'haswordcount' => false,
2549
                        'wordcount' => null,
2550
                        'author' => $exporteduser1,
2551
                        'attachments' => [],
2552
                        'messageinlinefiles' => [],
2553
                        'tags' => [],
2554
                        'html' => [
2555
                            'rating' => null,
2556
                            'taglist' => null,
2557
                            'authorsubheading' => $forumgenerator->get_author_subheading_html(
2558
                                (object)$exporteduser1, $discussion1firstpostobject->created)
2559
                        ],
2560
                        'charcount' => null,
2561
                        'capabilities' => [
2562
                            'view' => true,
2563
                            'edit' => true,
2564
                            'delete' => true,
2565
                            'split' => false,
2566
                            'reply' => true,
2567
                            'export' => false,
2568
                            'controlreadstatus' => false,
2569
                            'canreplyprivately' => true,
2570
                            'selfenrol' => false
2571
                        ],
2572
                        'urls' => [
2573
                            'view' => $urlfactory->get_view_post_url_from_post_id(
2574
                                $discussion1firstpostobject->discussion, $discussion1firstpostobject->id)->out(false),
2575
                            'viewisolated' => $isolatedurlparent->out(false),
2576
                            'viewparent' => null,
2577
                            'edit' => (new \moodle_url('/mod/forum/post.php', [
2578
                                'edit' => $discussion1firstpostobject->id
2579
                            ]))->out(false),
2580
                            'delete' => (new \moodle_url('/mod/forum/post.php', [
2581
                                'delete' => $discussion1firstpostobject->id
2582
                            ]))->out(false),
2583
                            'split' => null,
2584
                            'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
2585
                                'reply' => $discussion1firstpostobject->id
2586
                            ]))->out(false),
2587
                            'export' => null,
2588
                            'markasread' => null,
2589
                            'markasunread' => null,
2590
                            'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id(
2591
                                $discussion1firstpostobject->discussion)->out(false),
2592
                        ],
2593
                    ]
2594
                ],
2595
            ],
2596
        ];
2597
 
2598
        $isolatedurluser = $urlfactory->get_discussion_view_url_from_discussion_id($discussion2reply1->discussion);
2599
        $isolatedurluser->params(['parent' => $discussion2reply1->id]);
2600
        $isolatedurlparent = $urlfactory->get_discussion_view_url_from_discussion_id($discussion2firstpostobject->discussion);
2601
        $isolatedurlparent->params(['parent' => $discussion2firstpostobject->id]);
2602
 
2603
        $expectedposts['discussions'][1] = [
2604
            'name' => $discussion2->name,
2605
            'id' => $discussion2->id,
2606
            'timecreated' => $discussion2firstpost->get_time_created(),
2607
            'authorfullname' => $user1entity->get_full_name(),
2608
            'posts' => [
2609
                'userposts' => [
2610
                    [
2611
                        'id' => $discussion2reply1->id,
2612
                        'discussionid' => $discussion2reply1->discussion,
2613
                        'parentid' => $discussion2reply1->parent,
2614
                        'hasparent' => true,
2615
                        'timecreated' => $discussion2reply1->created,
2616
                        'timemodified' => $discussion2reply1->modified,
2617
                        'subject' => $discussion2reply1->subject,
1441 ariadna 2618
                        'replysubject' => $discussion2reply1->subject,
1 efrain 2619
                        'message' => file_rewrite_pluginfile_urls($discussion2reply1->message, 'pluginfile.php',
2620
                            $forum1context->id, 'mod_forum', 'post', $discussion2reply1->id),
2621
                        'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
2622
                        'unread' => null,
2623
                        'isdeleted' => false,
2624
                        'isprivatereply' => false,
2625
                        'haswordcount' => false,
2626
                        'wordcount' => null,
2627
                        'author' => $exporteduser2,
2628
                        'attachments' => [],
2629
                        'messageinlinefiles' => [],
2630
                        'tags' => [],
2631
                        'html' => [
2632
                            'rating' => null,
2633
                            'taglist' => null,
2634
                            'authorsubheading' => $forumgenerator->get_author_subheading_html(
2635
                                (object)$exporteduser2, $discussion2reply1->created)
2636
                        ],
2637
                        'charcount' => null,
2638
                        'capabilities' => [
2639
                            'view' => true,
2640
                            'edit' => true,
2641
                            'delete' => true,
2642
                            'split' => true,
2643
                            'reply' => true,
2644
                            'export' => false,
2645
                            'controlreadstatus' => false,
2646
                            'canreplyprivately' => true,
2647
                            'selfenrol' => false
2648
                        ],
2649
                        'urls' => [
2650
                            'view' => $urlfactory->get_view_post_url_from_post_id(
2651
                                $discussion2reply1->discussion, $discussion2reply1->id)->out(false),
2652
                            'viewisolated' => $isolatedurluser->out(false),
2653
                            'viewparent' => $urlfactory->get_view_post_url_from_post_id(
2654
                                $discussion2reply1->discussion, $discussion2reply1->parent)->out(false),
2655
                            'edit' => (new \moodle_url('/mod/forum/post.php', [
2656
                                'edit' => $discussion2reply1->id
2657
                            ]))->out(false),
2658
                            'delete' => (new \moodle_url('/mod/forum/post.php', [
2659
                                'delete' => $discussion2reply1->id
2660
                            ]))->out(false),
2661
                            'split' => (new \moodle_url('/mod/forum/post.php', [
2662
                                'prune' => $discussion2reply1->id
2663
                            ]))->out(false),
2664
                            'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
2665
                                'reply' => $discussion2reply1->id
2666
                            ]))->out(false),
2667
                            'export' => null,
2668
                            'markasread' => null,
2669
                            'markasunread' => null,
2670
                            'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id(
2671
                                $discussion2reply1->discussion)->out(false),
2672
                        ],
2673
                    ]
2674
                ],
2675
                'parentposts' => [
2676
                    [
2677
                        'id' => $discussion2firstpostobject->id,
2678
                        'discussionid' => $discussion2firstpostobject->discussion,
2679
                        'parentid' => null,
2680
                        'hasparent' => false,
2681
                        'timecreated' => $discussion2firstpostobject->created,
2682
                        'timemodified' => $discussion2firstpostobject->modified,
2683
                        'subject' => $discussion2firstpostobject->subject,
1441 ariadna 2684
                        'replysubject' => $discussion2firstpostobject->subject,
1 efrain 2685
                        'message' => file_rewrite_pluginfile_urls($discussion2firstpostobject->message, 'pluginfile.php',
2686
                            $forum1context->id, 'mod_forum', 'post', $discussion2firstpostobject->id),
2687
                        'messageformat' => 1,   // This value is usually changed by \core_external\util::format_text() function.
2688
                        'unread' => null,
2689
                        'isdeleted' => false,
2690
                        'isprivatereply' => false,
2691
                        'haswordcount' => false,
2692
                        'wordcount' => null,
2693
                        'author' => $exporteduser1,
2694
                        'attachments' => [],
2695
                        'messageinlinefiles' => [],
2696
                        'tags' => [],
2697
                        'html' => [
2698
                            'rating' => null,
2699
                            'taglist' => null,
2700
                            'authorsubheading' => $forumgenerator->get_author_subheading_html(
2701
                                (object)$exporteduser1, $discussion2firstpostobject->created)
2702
                        ],
2703
                        'charcount' => null,
2704
                        'capabilities' => [
2705
                            'view' => true,
2706
                            'edit' => true,
2707
                            'delete' => true,
2708
                            'split' => false,
2709
                            'reply' => true,
2710
                            'export' => false,
2711
                            'controlreadstatus' => false,
2712
                            'canreplyprivately' => true,
2713
                            'selfenrol' => false
2714
                        ],
2715
                        'urls' => [
2716
                            'view' => $urlfactory->get_view_post_url_from_post_id(
2717
                                $discussion2firstpostobject->discussion, $discussion2firstpostobject->id)->out(false),
2718
                            'viewisolated' => $isolatedurlparent->out(false),
2719
                            'viewparent' => null,
2720
                            'edit' => (new \moodle_url('/mod/forum/post.php', [
2721
                                'edit' => $discussion2firstpostobject->id
2722
                            ]))->out(false),
2723
                            'delete' => (new \moodle_url('/mod/forum/post.php', [
2724
                                'delete' => $discussion2firstpostobject->id
2725
                            ]))->out(false),
2726
                            'split' => null,
2727
                            'reply' => (new \moodle_url('/mod/forum/post.php#mformforum', [
2728
                                'reply' => $discussion2firstpostobject->id
2729
                            ]))->out(false),
2730
                            'export' => null,
2731
                            'markasread' => null,
2732
                            'markasunread' => null,
2733
                            'discuss' => $urlfactory->get_discussion_view_url_from_discussion_id(
2734
                                $discussion2firstpostobject->discussion)->out(false),
2735
 
2736
                        ]
2737
                    ],
2738
                ]
2739
            ],
2740
        ];
2741
 
2742
        // Test discussions with one additional post each (total 2 posts).
2743
        // Also testing that we get the parent posts too.
2744
        $discussions = mod_forum_external::get_discussion_posts_by_userid($user2->id, $forum1->cmid, 'modified', 'DESC');
2745
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_by_userid_returns(), $discussions);
2746
 
2747
        $this->assertEquals(2, count($discussions['discussions']));
2748
 
2749
        $this->assertEquals($expectedposts, $discussions);
2750
 
2751
        // When groupmode is SEPARATEGROUPS, even there is no groupid specified, the post not for the user shouldn't be seen.
2752
        $group1 = self::getDataGenerator()->create_group(['courseid' => $course1->id]);
2753
        $group2 = self::getDataGenerator()->create_group(['courseid' => $course1->id]);
2754
        // Update discussion with group.
2755
        $discussion = new \stdClass();
2756
        $discussion->id = $discussion1->id;
2757
        $discussion->groupid = $group1->id;
2758
        $DB->update_record('forum_discussions', $discussion);
2759
        $discussion = new \stdClass();
2760
        $discussion->id = $discussion2->id;
2761
        $discussion->groupid = $group2->id;
2762
        $DB->update_record('forum_discussions', $discussion);
2763
        $cm = get_coursemodule_from_id('forum', $forum1->cmid);
2764
        $cm->groupmode = SEPARATEGROUPS;
2765
        $DB->update_record('course_modules', $cm);
2766
        $teacher = self::getDataGenerator()->create_user();
2767
        $role = $DB->get_record('role', array('shortname' => 'teacher'), '*', MUST_EXIST);
2768
        self::getDataGenerator()->enrol_user($teacher->id, $course1->id, $role->id);
2769
        groups_add_member($group2->id, $teacher->id);
2770
        self::setUser($teacher);
2771
        $discussions = mod_forum_external::get_discussion_posts_by_userid($user2->id, $forum1->cmid, 'modified', 'DESC');
2772
        $discussions = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_by_userid_returns(), $discussions);
2773
        // Discussion is only 1 record (group 2).
2774
        $this->assertEquals(1, count($discussions['discussions']));
2775
        $this->assertEquals($expectedposts['discussions'][1], $discussions['discussions'][0]);
2776
    }
2777
 
2778
    /**
2779
     * Test get_discussion_post a discussion.
2780
     */
11 efrain 2781
    public function test_get_discussion_post_discussion(): void {
1 efrain 2782
        global $DB;
2783
        $this->resetAfterTest(true);
2784
        // Setup test data.
2785
        $course = $this->getDataGenerator()->create_course();
2786
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2787
        $user = $this->getDataGenerator()->create_user();
2788
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2789
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2790
        // Add a discussion.
2791
        $record = new \stdClass();
2792
        $record->course = $course->id;
2793
        $record->userid = $user->id;
2794
        $record->forum = $forum->id;
2795
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2796
        $this->setUser($user);
2797
        $result = mod_forum_external::get_discussion_post($discussion->firstpost);
2798
        $result = external_api::clean_returnvalue(mod_forum_external::get_discussion_post_returns(), $result);
2799
        $this->assertEquals($discussion->firstpost, $result['post']['id']);
2800
        $this->assertFalse($result['post']['hasparent']);
2801
        $this->assertEquals($discussion->message, $result['post']['message']);
2802
    }
2803
 
2804
    /**
2805
     * Test get_discussion_post a post.
2806
     */
11 efrain 2807
    public function test_get_discussion_post_post(): void {
1 efrain 2808
        global $DB;
2809
        $this->resetAfterTest(true);
2810
        // Setup test data.
2811
        $course = $this->getDataGenerator()->create_course();
2812
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2813
        $user = $this->getDataGenerator()->create_user();
2814
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2815
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2816
        // Add a discussion.
2817
        $record = new \stdClass();
2818
        $record->course = $course->id;
2819
        $record->userid = $user->id;
2820
        $record->forum = $forum->id;
2821
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2822
        $parentpost = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
2823
        // Add a post.
2824
        $record = new \stdClass();
2825
        $record->course = $course->id;
2826
        $record->userid = $user->id;
2827
        $record->forum = $forum->id;
2828
        $record->discussion = $discussion->id;
2829
        $record->parent = $parentpost->id;
2830
        $post = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
2831
        $this->setUser($user);
2832
        $result = mod_forum_external::get_discussion_post($post->id);
2833
        $result = external_api::clean_returnvalue(mod_forum_external::get_discussion_post_returns(), $result);
2834
        $this->assertEquals($post->id, $result['post']['id']);
2835
        $this->assertTrue($result['post']['hasparent']);
2836
        $this->assertEquals($post->message, $result['post']['message']);
2837
    }
2838
 
2839
    /**
2840
     * Test get_discussion_post a different user post.
2841
     */
11 efrain 2842
    public function test_get_discussion_post_other_user_post(): void {
1 efrain 2843
        global $DB;
2844
        $this->resetAfterTest(true);
2845
        // Setup test data.
2846
        $course = $this->getDataGenerator()->create_course();
2847
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2848
        $user = $this->getDataGenerator()->create_user();
2849
        $otheruser = $this->getDataGenerator()->create_user();
2850
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2851
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2852
        self::getDataGenerator()->enrol_user($otheruser->id, $course->id, $role->id);
2853
        // Add a discussion.
2854
        $record = array();
2855
        $record['course'] = $course->id;
2856
        $record['forum'] = $forum->id;
2857
        $record['userid'] = $user->id;
2858
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2859
        $parentpost = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
2860
        // Add a post.
2861
        $record = new \stdClass();
2862
        $record->course = $course->id;
2863
        $record->userid = $user->id;
2864
        $record->forum = $forum->id;
2865
        $record->discussion = $discussion->id;
2866
        $record->parent = $parentpost->id;
2867
        $post = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
2868
        // Check other user post.
2869
        $this->setUser($otheruser);
2870
        $result = mod_forum_external::get_discussion_post($post->id);
2871
        $result = external_api::clean_returnvalue(mod_forum_external::get_discussion_post_returns(), $result);
2872
        $this->assertEquals($post->id, $result['post']['id']);
2873
        $this->assertTrue($result['post']['hasparent']);
2874
        $this->assertEquals($post->message, $result['post']['message']);
2875
    }
2876
 
2877
    /**
2878
     * Test prepare_draft_area_for_post a different user post.
2879
     */
11 efrain 2880
    public function test_prepare_draft_area_for_post(): void {
1 efrain 2881
        global $DB;
2882
        $this->resetAfterTest(true);
2883
        // Setup test data.
2884
        $course = $this->getDataGenerator()->create_course();
2885
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2886
        $user = $this->getDataGenerator()->create_user();
2887
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
2888
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
2889
        // Add a discussion.
2890
        $record = array();
2891
        $record['course'] = $course->id;
2892
        $record['forum'] = $forum->id;
2893
        $record['userid'] = $user->id;
2894
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2895
        $parentpost = $DB->get_record('forum_posts', array('discussion' => $discussion->id));
2896
        // Add a post.
2897
        $record = new \stdClass();
2898
        $record->course = $course->id;
2899
        $record->userid = $user->id;
2900
        $record->forum = $forum->id;
2901
        $record->discussion = $discussion->id;
2902
        $record->parent = $parentpost->id;
2903
        $post = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
2904
 
2905
        // Add some files only in the attachment area.
2906
        $filename = 'faketxt.txt';
2907
        $filerecordinline = array(
2908
            'contextid' => \context_module::instance($forum->cmid)->id,
2909
            'component' => 'mod_forum',
2910
            'filearea'  => 'attachment',
2911
            'itemid'    => $post->id,
2912
            'filepath'  => '/',
2913
            'filename'  => $filename,
2914
        );
2915
        $fs = get_file_storage();
2916
        $fs->create_file_from_string($filerecordinline, 'fake txt contents 1.');
2917
        $filerecordinline['filename'] = 'otherfaketxt.txt';
2918
        $fs->create_file_from_string($filerecordinline, 'fake txt contents 2.');
2919
 
2920
        $this->setUser($user);
2921
 
2922
        // Check attachment area.
2923
        $result = mod_forum_external::prepare_draft_area_for_post($post->id, 'attachment');
2924
        $result = external_api::clean_returnvalue(mod_forum_external::prepare_draft_area_for_post_returns(), $result);
2925
        $this->assertCount(2, $result['files']);
2926
        $this->assertEquals($filename, $result['files'][0]['filename']);
2927
        $this->assertCount(5, $result['areaoptions']);
2928
        $this->assertEmpty($result['messagetext']);
2929
 
2930
        // Check again using existing draft item id.
2931
        $result = mod_forum_external::prepare_draft_area_for_post($post->id, 'attachment', $result['draftitemid']);
2932
        $result = external_api::clean_returnvalue(mod_forum_external::prepare_draft_area_for_post_returns(), $result);
2933
        $this->assertCount(2, $result['files']);
2934
 
2935
        // Keep only certain files in the area.
2936
        $filestokeep = array(array('filename' => $filename, 'filepath' => '/'));
2937
        $result = mod_forum_external::prepare_draft_area_for_post($post->id, 'attachment', $result['draftitemid'], $filestokeep);
2938
        $result = external_api::clean_returnvalue(mod_forum_external::prepare_draft_area_for_post_returns(), $result);
2939
        $this->assertCount(1, $result['files']);
2940
        $this->assertEquals($filename, $result['files'][0]['filename']);
2941
 
2942
        // Check editor (post) area.
2943
        $filerecordinline['filearea'] = 'post';
2944
        $filerecordinline['filename'] = 'fakeimage.png';
2945
        $fs->create_file_from_string($filerecordinline, 'fake image.');
2946
        $result = mod_forum_external::prepare_draft_area_for_post($post->id, 'post');
2947
        $result = external_api::clean_returnvalue(mod_forum_external::prepare_draft_area_for_post_returns(), $result);
2948
        $this->assertCount(1, $result['files']);
2949
        $this->assertEquals($filerecordinline['filename'], $result['files'][0]['filename']);
2950
        $this->assertCount(5, $result['areaoptions']);
2951
        $this->assertEquals($post->message, $result['messagetext']);
2952
    }
2953
 
2954
    /**
2955
     * Test update_discussion_post with a discussion.
2956
     */
11 efrain 2957
    public function test_update_discussion_post_discussion(): void {
1 efrain 2958
        global $DB, $USER;
2959
        $this->resetAfterTest(true);
2960
        // Setup test data.
2961
        $course = $this->getDataGenerator()->create_course();
2962
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
2963
 
2964
        $this->setAdminUser();
2965
 
2966
        // Add a discussion.
2967
        $record = new \stdClass();
2968
        $record->course = $course->id;
2969
        $record->userid = $USER->id;
2970
        $record->forum = $forum->id;
2971
        $record->pinned = FORUM_DISCUSSION_UNPINNED;
2972
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
2973
 
2974
        $subject = 'Hey subject updated';
2975
        $message = 'Hey message updated';
2976
        $messageformat = FORMAT_HTML;
2977
        $options = [
2978
            ['name' => 'pinned', 'value' => true],
2979
        ];
2980
 
2981
        $result = mod_forum_external::update_discussion_post($discussion->firstpost, $subject, $message, $messageformat,
2982
            $options);
2983
        $result = external_api::clean_returnvalue(mod_forum_external::update_discussion_post_returns(), $result);
2984
        $this->assertTrue($result['status']);
2985
 
2986
        // Get the post from WS.
2987
        $result = mod_forum_external::get_discussion_post($discussion->firstpost);
2988
        $result = external_api::clean_returnvalue(mod_forum_external::get_discussion_post_returns(), $result);
2989
        $this->assertEquals($subject, $result['post']['subject']);
2990
        $this->assertEquals($message, $result['post']['message']);
2991
        $this->assertEquals($messageformat, $result['post']['messageformat']);
2992
 
2993
        // Get discussion object from DB.
2994
        $discussion = $DB->get_record('forum_discussions', ['id' => $discussion->id]);
2995
        $this->assertEquals($subject, $discussion->name);   // Check discussion subject.
2996
        $this->assertEquals(FORUM_DISCUSSION_PINNED, $discussion->pinned);  // Check discussion pinned.
2997
    }
2998
 
2999
    /**
3000
     * Test update_discussion_post with a post.
3001
     */
11 efrain 3002
    public function test_update_discussion_post_post(): void {
1 efrain 3003
        global $DB, $USER;
3004
        $this->resetAfterTest(true);
3005
        // Setup test data.
3006
        $course = $this->getDataGenerator()->create_course();
3007
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
3008
        $cm = get_coursemodule_from_id('forum', $forum->cmid, 0, false, MUST_EXIST);
3009
        $user = $this->getDataGenerator()->create_user();
3010
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
3011
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
3012
 
3013
        $this->setUser($user);
3014
        // Enable auto subscribe discussion.
3015
        $USER->autosubscribe = true;
3016
 
3017
        // Add a discussion.
3018
        $record = new \stdClass();
3019
        $record->course = $course->id;
3020
        $record->userid = $user->id;
3021
        $record->forum = $forum->id;
3022
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
3023
 
3024
        // Add a post via WS (so discussion subscription works).
3025
        $result = mod_forum_external::add_discussion_post($discussion->firstpost, 'some subject', 'some text here...');
3026
        $newpost = $result['post'];
3027
        $this->assertTrue(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion->id, $cm));
3028
 
3029
        // Test inline and regular attachment in post
3030
        // Create a file in a draft area for inline attachments.
3031
        $draftidinlineattach = file_get_unused_draft_itemid();
3032
        $draftidattach = file_get_unused_draft_itemid();
3033
        self::setUser($user);
3034
        $usercontext = \context_user::instance($user->id);
3035
        $filepath = '/';
3036
        $filearea = 'draft';
3037
        $component = 'user';
3038
        $filenameimg = 'fakeimage.png';
3039
        $filerecordinline = array(
3040
            'contextid' => $usercontext->id,
3041
            'component' => $component,
3042
            'filearea'  => $filearea,
3043
            'itemid'    => $draftidinlineattach,
3044
            'filepath'  => $filepath,
3045
            'filename'  => $filenameimg,
3046
        );
3047
        $fs = get_file_storage();
3048
 
3049
        // Create a file in a draft area for regular attachments.
3050
        $filerecordattach = $filerecordinline;
3051
        $attachfilename = 'faketxt.txt';
3052
        $filerecordattach['filename'] = $attachfilename;
3053
        $filerecordattach['itemid'] = $draftidattach;
3054
        $fs->create_file_from_string($filerecordinline, 'image contents (not really)');
3055
        $fs->create_file_from_string($filerecordattach, 'simple text attachment');
3056
 
3057
        // Do not update subject.
3058
        $message = 'Hey message updated';
3059
        $messageformat = FORMAT_HTML;
3060
        $options = [
3061
            ['name' => 'discussionsubscribe', 'value' => false],
3062
            ['name' => 'inlineattachmentsid', 'value' => $draftidinlineattach],
3063
            ['name' => 'attachmentsid', 'value' => $draftidattach],
3064
        ];
3065
 
3066
        $result = mod_forum_external::update_discussion_post($newpost->id, '', $message, $messageformat, $options);
3067
        $result = external_api::clean_returnvalue(mod_forum_external::update_discussion_post_returns(), $result);
3068
        $this->assertTrue($result['status']);
3069
        // Check subscription status.
3070
        $this->assertFalse(\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussion->id, $cm));
3071
 
3072
        // Get the post from WS.
3073
        $result = mod_forum_external::get_discussion_posts($discussion->id, 'modified', 'DESC', true);
3074
        $result = external_api::clean_returnvalue(mod_forum_external::get_discussion_posts_returns(), $result);
3075
        $found = false;
3076
        foreach ($result['posts'] as $post) {
3077
            if ($post['id'] == $newpost->id) {
3078
                $this->assertEquals($newpost->subject, $post['subject']);
3079
                $this->assertEquals($message, $post['message']);
3080
                $this->assertEquals($messageformat, $post['messageformat']);
3081
                $this->assertCount(1, $post['messageinlinefiles']);
3082
                $this->assertEquals('fakeimage.png', $post['messageinlinefiles'][0]['filename']);
3083
                $this->assertCount(1, $post['attachments']);
3084
                $this->assertEquals('faketxt.txt', $post['attachments'][0]['filename']);
3085
                $found = true;
3086
            }
3087
        }
3088
        $this->assertTrue($found);
3089
    }
3090
 
3091
    /**
3092
     * Test update_discussion_post with other user post (no permissions).
3093
     */
11 efrain 3094
    public function test_update_discussion_post_other_user_post(): void {
1 efrain 3095
        global $DB, $USER;
3096
        $this->resetAfterTest(true);
3097
        // Setup test data.
3098
        $course = $this->getDataGenerator()->create_course();
3099
        $forum = $this->getDataGenerator()->create_module('forum', array('course' => $course->id));
3100
        $user = $this->getDataGenerator()->create_user();
3101
        $role = $DB->get_record('role', array('shortname' => 'student'), '*', MUST_EXIST);
3102
        self::getDataGenerator()->enrol_user($user->id, $course->id, $role->id);
3103
 
3104
        $this->setAdminUser();
3105
        // Add a discussion.
3106
        $record = new \stdClass();
3107
        $record->course = $course->id;
3108
        $record->userid = $USER->id;
3109
        $record->forum = $forum->id;
3110
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion($record);
3111
 
3112
        // Add a post.
3113
        $record = new \stdClass();
3114
        $record->course = $course->id;
3115
        $record->userid = $USER->id;
3116
        $record->forum = $forum->id;
3117
        $record->discussion = $discussion->id;
3118
        $record->parent = $discussion->firstpost;
3119
        $newpost = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_post($record);
3120
 
3121
        $this->setUser($user);
3122
        $subject = 'Hey subject updated';
3123
        $message = 'Hey message updated';
3124
        $messageformat = FORMAT_HTML;
3125
 
3126
        $this->expectExceptionMessage(get_string('cannotupdatepost', 'forum'));
3127
        mod_forum_external::update_discussion_post($newpost->id, $subject, $message, $messageformat);
3128
    }
3129
 
3130
    /**
3131
     * Test that we can update the subject of a post to the string '0'
3132
     */
3133
    public function test_update_discussion_post_set_subject_to_zero(): void {
3134
        global $DB, $USER;
3135
 
3136
        $this->resetAfterTest(true);
3137
        $this->setAdminUser();
3138
 
3139
        // Setup test data.
3140
        $course = $this->getDataGenerator()->create_course();
3141
        $forum = $this->getDataGenerator()->create_module('forum', ['course' => $course->id]);
3142
 
3143
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion((object) [
3144
            'userid' => $USER->id,
3145
            'course' => $course->id,
3146
            'forum' => $forum->id,
3147
            'name' => 'Test discussion subject',
3148
        ]);
3149
 
3150
        // Update discussion post subject.
3151
        $result = external_api::clean_returnvalue(
3152
            mod_forum_external::update_discussion_post_returns(),
3153
            mod_forum_external::update_discussion_post($discussion->firstpost, '0')
3154
        );
3155
        $this->assertTrue($result['status']);
3156
 
3157
        // Get updated discussion post subject from DB.
3158
        $postsubject = $DB->get_field('forum_posts', 'subject', ['id' => $discussion->firstpost]);
3159
        $this->assertEquals('0', $postsubject);
3160
    }
3161
 
3162
    /**
3163
     * Test that we can update the message of a post to the string '0'
3164
     */
3165
    public function test_update_discussion_post_set_message_to_zero(): void {
3166
        global $DB, $USER;
3167
 
3168
        $this->resetAfterTest(true);
3169
        $this->setAdminUser();
3170
 
3171
        // Setup test data.
3172
        $course = $this->getDataGenerator()->create_course();
3173
        $forum = $this->getDataGenerator()->create_module('forum', ['course' => $course->id]);
3174
 
3175
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion((object) [
3176
            'userid' => $USER->id,
3177
            'course' => $course->id,
3178
            'forum' => $forum->id,
3179
            'message' => 'Test discussion message',
3180
            'messageformat' => FORMAT_HTML,
3181
        ]);
3182
 
3183
        // Update discussion post message.
3184
        $result = external_api::clean_returnvalue(
3185
            mod_forum_external::update_discussion_post_returns(),
3186
            mod_forum_external::update_discussion_post($discussion->firstpost, '', '0', FORMAT_HTML)
3187
        );
3188
        $this->assertTrue($result['status']);
3189
 
3190
        // Get updated discussion post subject from DB.
3191
        $postmessage = $DB->get_field('forum_posts', 'message', ['id' => $discussion->firstpost]);
3192
        $this->assertEquals('0', $postmessage);
3193
    }
3194
 
3195
    /**
3196
     * Test that we can update the message format of a post to {@see FORMAT_MOODLE}
3197
     */
3198
    public function test_update_discussion_post_set_message_format_moodle(): void {
3199
        global $DB, $USER;
3200
 
3201
        $this->resetAfterTest(true);
3202
        $this->setAdminUser();
3203
 
3204
        // Setup test data.
3205
        $course = $this->getDataGenerator()->create_course();
3206
        $forum = $this->getDataGenerator()->create_module('forum', ['course' => $course->id]);
3207
 
3208
        $discussion = $this->getDataGenerator()->get_plugin_generator('mod_forum')->create_discussion((object) [
3209
            'userid' => $USER->id,
3210
            'course' => $course->id,
3211
            'forum' => $forum->id,
3212
            'message' => 'Test discussion message',
3213
            'messageformat' => FORMAT_HTML,
3214
        ]);
3215
 
3216
        // Update discussion post message & messageformat.
3217
        $result = external_api::clean_returnvalue(
3218
            mod_forum_external::update_discussion_post_returns(),
3219
            mod_forum_external::update_discussion_post($discussion->firstpost, '', 'Update discussion message', FORMAT_MOODLE)
3220
        );
3221
        $this->assertTrue($result['status']);
3222
 
3223
        // Get updated discussion post from DB.
3224
        $updatedpost = $DB->get_record('forum_posts', ['id' => $discussion->firstpost], 'message,messageformat');
3225
        $this->assertEquals((object) [
3226
            'message' => 'Update discussion message',
3227
            'messageformat' => FORMAT_MOODLE,
3228
        ], $updatedpost);
3229
    }
3230
}