Merge branch 'MDL-36754-master' of git://github.com/andrewnicols/moodle
[moodle.git] / calendar / tests / privacy_test.php
blob4a47348c6237e388dbca79e7dc747a38210db74e
1 <?php
2 // This file is part of Moodle - http://moodle.org/
3 //
4 // Moodle is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
8 //
9 // Moodle is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
17 /**
18 * Privacy tests for core_calendar.
20 * @package core_calendar
21 * @category test
22 * @copyright 2018 Zig Tan <zig@moodle.com>
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
26 defined('MOODLE_INTERNAL') || die();
27 global $CFG;
29 require_once($CFG->dirroot . '/calendar/lib.php');
30 require_once($CFG->dirroot . '/calendar/tests/externallib_test.php');
32 use \core_calendar\privacy\provider;
33 use \core_privacy\local\metadata\collection;
34 use \core_privacy\local\request\approved_contextlist;
35 use \core_privacy\local\request\writer;
36 use \core_privacy\tests\provider_testcase;
38 /**
39 * Unit tests for calendar/classes/privacy/provider
41 * @copyright 2018 Zig Tan <zig@moodle.com>
42 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
44 class core_calendar_privacy_testcase extends provider_testcase {
46 /**
47 * Overriding setUp() function to always reset after tests.
49 public function setUp() {
50 $this->resetAfterTest(true);
53 /**
54 * Test for provider::get_contexts_for_userid().
56 * @throws coding_exception
58 public function test_get_contexts_for_userid() {
59 // Create test user to create Calendar Events and Subscriptions.
60 $user = $this->getDataGenerator()->create_user();
61 $this->setUser($user);
63 // Create a Category and Courses to assign Calendar Events and Subscriptions.
64 $category = $this->getDataGenerator()->create_category();
65 $course1 = $this->getDataGenerator()->create_course();
66 $course2 = $this->getDataGenerator()->create_course();
67 $course3 = $this->getDataGenerator()->create_course();
68 $grouprecord = (object)[
69 'courseid' => $course3->id,
70 'name' => 'test_group'
72 $course3group = $this->getDataGenerator()->create_group($grouprecord);
74 // Get contexts.
75 $usercontext = context_user::instance($user->id);
76 $categorycontext = context_coursecat::instance($category->id);
77 $course1context = context_course::instance($course1->id);
78 $course2context = context_course::instance($course2->id);
79 $course3context = context_course::instance($course3->id);
81 // Add Category Calendar Events for Category.
82 $this->create_test_standard_calendar_event('category', $user->id, time(), '', $category->id);
83 $this->create_test_standard_calendar_event('category', $user->id, time(), '', $category->id);
85 // Add User Calendar Events for User.
86 $this->create_test_standard_calendar_event('user', $user->id, time(), '');
87 $this->create_test_standard_calendar_event('user', $user->id, time(), '', 0, $course1->id);
88 $this->create_test_standard_calendar_event('user', $user->id, time(), '', 0, $course2->id);
90 // Add a Course Calendar Event for Course 1.
91 $this->create_test_standard_calendar_event('course', $user->id, time(), '', 0, $course1->id);
93 // Add a Course Assignment Action Calendar Event for Course 2.
94 $generator = $this->getDataGenerator()->get_plugin_generator('mod_assign');
95 $params['course'] = $course2->id;
96 $params['assignsubmission_onlinetext_enabled'] = 1;
97 $instance = $generator->create_instance($params);
98 $cm = get_coursemodule_from_instance('assign', $instance->id);
99 $modulecontext = context_module::instance($cm->id);
100 $assign = new assign($modulecontext, $cm, $course2);
101 $this->create_test_action_calendar_event('duedate', $course2->id, $instance->id, 'assign', $user->id, time());
102 $this->create_test_action_calendar_event('gradingduedate', $course2->id, $instance->id, 'assign', $user->id, time());
104 // Add a Calendar Subscription and Group Calendar Event to Course 3.
105 $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user->id, 0, $course3->id);
106 $this->create_test_standard_calendar_event('group', $user->id, time(), '', 0, $course1->id, $course3group->id);
108 // The user will be in these contexts.
109 $usercontextids = [
110 $usercontext->id,
111 $categorycontext->id,
112 $course1context->id,
113 $modulecontext->id,
114 $course3context->id
116 // Retrieve the user's context ids.
117 $contextids = provider::get_contexts_for_userid($user->id);
119 // Check the user context list and retrieved user context lists contains the same number of records.
120 $this->assertEquals(count($usercontextids), count($contextids->get_contextids()));
121 // There should be no difference between the contexts.
122 $this->assertEmpty(array_diff($usercontextids, $contextids->get_contextids()));
126 * Test for provider::export_user_data().
128 * @throws coding_exception
130 public function test_export_user_data() {
131 global $DB;
133 // Create test user to create Calendar Events and Subscriptions with.
134 $user = $this->getDataGenerator()->create_user();
135 $this->setUser($user);
137 // Create a Category to test creating a Category Calendar Event.
138 $category = $this->getDataGenerator()->create_category();
139 $course1 = $this->getDataGenerator()->create_course();
140 $course2 = $this->getDataGenerator()->create_course();
141 $course3 = $this->getDataGenerator()->create_course();
142 $grouprecord = (object)[
143 'courseid' => $course3->id,
144 'name' => 'test_group'
146 $course3group = $this->getDataGenerator()->create_group($grouprecord);
148 // Add User Calendar Events for User.
149 $event1 = $this->create_test_standard_calendar_event('user', $user->id, time(), '');
151 // Add Category Calendar Events for Category.
152 $event2 = $this->create_test_standard_calendar_event('category', $user->id, time(), '', $category->id);
154 // Add two Course Calendar Event for Course 1 and set the same time (1 day a head).
155 $time = strtotime('+1 day', time());
156 $event3 = $this->create_test_standard_calendar_event('course', $user->id, $time, 'ABC', 0, $course1->id);
157 $event4 = $this->create_test_standard_calendar_event('course', $user->id, $time, 'DEF', 0, $course1->id);
159 // Add a Course Assignment Action Calendar Event for Course 2.
160 $generator = $this->getDataGenerator()->get_plugin_generator('mod_assign');
161 $params['course'] = $course2->id;
162 $params['assignsubmission_onlinetext_enabled'] = 1;
163 $instance = $generator->create_instance($params);
164 $cm = get_coursemodule_from_instance('assign', $instance->id);
165 $modulecontext = context_module::instance($cm->id);
166 $assign = new assign($modulecontext, $cm, $course2);
167 $event5 = $this->create_test_action_calendar_event('duedate', $course2->id, $instance->id, 'assign', $user->id, time());
169 // Add a Calendar Subscription and Group Calendar Event to Course 3.
170 $subscription1 = $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user->id, 0, $course3->id);
171 $event6 = $this->create_test_standard_calendar_event('group', $user->id, time(), '', 0, $course3->id, $course3group->id);
173 // Retrieve the user's context ids.
174 $contextlist = provider::get_contexts_for_userid($user->id);
175 $approvedcontextlist = new approved_contextlist($user, 'core_calendar', $contextlist->get_contextids());
177 // Retrieve Calendar Event and Subscriptions data only for this user.
178 provider::export_user_data($approvedcontextlist);
180 foreach ($contextlist as $context) {
181 $writer = writer::with_context($context);
182 $this->assertTrue($writer->has_any_data());
184 // Test event 1 that was created for the test User.
185 if ($context->instanceid == $user->id && $context->contextlevel == CONTEXT_USER) {
186 // Test the content contains Calendar Event user data.
187 $subcontexts = [
188 get_string('calendar', 'calendar'),
189 get_string('events', 'calendar'),
190 date('c', $event1->timestart)
192 $name = "user-event";
193 $data = $writer->get_related_data($subcontexts, $name);
194 $this->assertEquals('Standard Calendar Event user', $data->name);
197 // Test event 2 that was created for the test Category.
198 if ($context->instanceid == $category->id && $context->contextlevel == CONTEXT_COURSECAT) {
199 // Test the content contains Calendar Event category data.
200 $subcontexts = [
201 get_string('calendar', 'calendar'),
202 get_string('events', 'calendar'),
203 date('c', $event2->timestart)
205 $name = "category-event";
206 $data = $writer->get_related_data($subcontexts, $name);
207 $this->assertEquals('Standard Calendar Event category', $data->name);
210 // Test events 3, 4, and 5 that were created for the test Course 1.
211 if ($context->instanceid == $course1->id && $context->contextlevel == CONTEXT_COURSE) {
212 // Test the content contains Calendar Event course data set with the same time, and the exported files are uniquely identified.
213 $subcontext1 = [
214 get_string('calendar', 'calendar'),
215 get_string('events', 'calendar'),
216 date('c', $event3->timestart)
218 $name1 = "course-event-1";
219 $data1 = $writer->get_related_data($subcontext1, $name1);
220 $this->assertEquals('Standard Calendar Event course -- ABC', $data1->name);
222 $subcontext2 = [
223 get_string('calendar', 'calendar'),
224 get_string('events', 'calendar'),
225 date('c', $event4->timestart)
227 $name2 = "course-event-2";
228 $data2 = $writer->get_related_data($subcontext2, $name2);
229 $this->assertEquals('Standard Calendar Event course -- DEF', $data2->name);
232 // Test action event that were created for the test Course 2.
233 if ($context->instanceid == $cm->id && $context->contextlevel == CONTEXT_MODULE) {
234 // Test the content contains Calendar Action Event course data.
235 $subcontexts = [
236 get_string('calendar', 'calendar'),
237 get_string('events', 'calendar'),
238 date('c', $event5->timestart)
240 $name = "duedate-event";
241 $data = $writer->get_related_data($subcontexts, $name);
242 $this->assertEquals('Action Calendar Event duedate -- assign', $data->name);
245 // Test Calendar Subscription and Event that were created for the test Course 3.
246 if ($context->instanceid == $course3->id && $context->contextlevel == CONTEXT_COURSE) {
247 // Test the content contains Calendar Subscription data also created for the test Course 3.
248 $subcontexts = [
249 get_string('calendar', 'calendar'),
250 get_string('subscriptions', 'calendar')
252 $name = "course-subscription";
253 $data = $writer->get_related_data($subcontexts, $name);
254 $this->assertEquals('Calendar Subscription course', $data->name);
256 // Test the content contains Calendar Event group data also created for the test Course 3.
257 $subcontexts = [
258 get_string('calendar', 'calendar'),
259 get_string('events', 'calendar'),
260 date('c', $event6->timestart)
262 $name = "group-event";
263 $data = $writer->get_related_data($subcontexts, $name);
264 $this->assertEquals('Standard Calendar Event group', $data->name);
271 * Test for provider::test_export_user_preferences().
273 public function test_export_user_preferences() {
274 global $DB;
276 // Test setup.
277 $user = $this->getDataGenerator()->create_user();
278 $this->setUser($user);
280 // Add a user home page preference for the User.
281 set_user_preference('calendar_savedflt', 'true', $user);
283 // Test the user preference exists.
284 $params = [
285 'userid' => $user->id,
286 'name' => 'calendar_savedflt'
289 // Test the user preferences export contains 1 user preference record for the User.
290 provider::export_user_preferences($user->id);
291 $contextuser = context_user::instance($user->id);
292 $writer = writer::with_context($contextuser);
293 $this->assertTrue($writer->has_any_data());
295 $exportedpreferences = $writer->get_user_preferences('core_calendar');
296 $this->assertCount(1, (array) $exportedpreferences);
297 $this->assertEquals('true', $exportedpreferences->calendarsavedflt->value);
301 * Test for provider::delete_data_for_all_users_in_context().
303 * @throws dml_exception
305 public function test_delete_data_for_all_users_in_context() {
306 global $DB;
308 // Create test user to create Calendar Events and Subscriptions with.
309 $user1 = $this->getDataGenerator()->create_user();
310 $user2 = $this->getDataGenerator()->create_user();
312 // Create a Course to test creating a Category Calendar Event.
313 $course1 = $this->getDataGenerator()->create_course();
314 $course2 = $this->getDataGenerator()->create_course();
316 // Get contexts.
317 $course1context = context_course::instance($course1->id);
318 $course2context = context_course::instance($course2->id);
320 // Add a Course Calendar Event by User 1 for Course 1 and Course 2.
321 $this->setUser($user1);
322 $this->create_test_standard_calendar_event('course', $user1->id, time(), '', 0, $course1->id);
323 $this->create_test_standard_calendar_event('course', $user1->id, time(), '', 0, $course2->id);
325 // Add a Calendar Subscription by User 1 for Course 1.
326 $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user1->id, 0, $course1->id);
328 // Add a Course Calendar Event by User 2 for Course 1 and Course 2.
329 $this->setUser($user2);
330 $this->create_test_standard_calendar_event('course', $user2->id, time(), '', 0, $course1->id);
331 $this->create_test_standard_calendar_event('course', $user2->id, time(), '', 0, $course2->id);
333 // Add a Calendar Subscription by User 2 for Course 2.
334 $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user2->id, 0, $course2->id);
336 // Add a Course Assignment Action Calendar Event by User 2 for Course 2.
337 $generator = $this->getDataGenerator()->get_plugin_generator('mod_assign');
338 $params['course'] = $course2->id;
339 $params['assignsubmission_onlinetext_enabled'] = 1;
340 $instance = $generator->create_instance($params);
341 $cm = get_coursemodule_from_instance('assign', $instance->id);
342 $modulecontext = context_module::instance($cm->id);
343 $assign = new assign($modulecontext, $cm, $course2);
344 $this->create_test_action_calendar_event('duedate', $course2->id, $instance->id, 'assign', $user2->id, time());
345 $this->create_test_action_calendar_event('gradingduedate', $course2->id, $instance->id, 'assign', $user2->id, time());
347 // Delete all Calendar Events for all Users by Context for Course 1.
348 provider::delete_data_for_all_users_in_context($course1context);
350 // Verify all Calendar Events for Course 1 were deleted.
351 $events = $DB->get_records('event', array('courseid' => $course1->id));
352 $this->assertCount(0, $events);
353 // Verify all Calendar Subscriptions for Course 1 were deleted.
354 $subscriptions = $DB->get_records('event_subscriptions', array('courseid' => $course1->id));
355 $this->assertCount(0, $subscriptions);
357 // Verify all Calendar Events for Course 2 exists still.
358 $events = $DB->get_records('event', array('courseid' => $course2->id));
359 $this->assertCount(4, $events);
360 // Verify all Calendar Subscriptions for Course 2 exists still.
361 $subscriptions = $DB->get_records('event_subscriptions', array('courseid' => $course2->id));
362 $this->assertCount(1, $subscriptions);
364 // Delete all Calendar Events for all Users by Context for Course 2.
365 provider::delete_data_for_all_users_in_context($course2context);
367 // Verify all Calendar Events for Course 2 were deleted.
368 $events = $DB->get_records('event', array('courseid' => $course2->id));
369 $this->assertCount(0, $events);
370 // Verify all Calendar Subscriptions for Course 2 were deleted.
371 $subscriptions = $DB->get_records('event_subscriptions', array('courseid' => $course2->id));
372 $this->assertCount(0, $subscriptions);
376 * Test for provider::delete_data_for_user().
378 * @throws dml_exception
380 public function test_delete_data_for_user() {
381 global $DB;
383 // Create test user to create Calendar Events and Subscriptions with.
384 $user1 = $this->getDataGenerator()->create_user();
385 $user2 = $this->getDataGenerator()->create_user();
387 // Create a Category and Courses to test creating a Category Calendar Event.
388 $category = $this->getDataGenerator()->create_category();
389 $course1 = $this->getDataGenerator()->create_course();
390 $course2 = $this->getDataGenerator()->create_course();
392 // Add 5 Calendar Events for User 1 for various contexts.
393 $this->setUser($user1);
394 $this->create_test_standard_calendar_event('user', $user1->id, time(), '');
395 $this->create_test_standard_calendar_event('site', $user1->id, time(), '', 0, 1);
396 $this->create_test_standard_calendar_event('category', $user1->id, time(), '', $category->id);
397 $this->create_test_standard_calendar_event('course', $user1->id, time(), '', 0, $course1->id);
398 $this->create_test_standard_calendar_event('course', $user1->id, time(), '', 0, $course2->id);
400 // Add 1 Calendar Subscription for User 1 at course context.
401 $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user1->id, 0, $course2->id);
403 // Add 3 Calendar Events for User 2 for various contexts.
404 $this->setUser($user2);
405 $this->create_test_standard_calendar_event('user', $user2->id, time(), '');
406 $this->create_test_standard_calendar_event('category', $user2->id, time(), '', $category->id);
407 $this->create_test_standard_calendar_event('course', $user2->id, time(), '', 0, $course1->id);
409 // Add 1 Calendar Subscription for User 2 at course context.
410 $this->create_test_calendar_subscription('course', 'https://calendar.google.com/', $user2->id, 0, $course2->id);
412 // Retrieve the user's context ids.
413 $contextlist = provider::get_contexts_for_userid($user1->id);
414 $approvedcontextlist = new approved_contextlist($user1, 'core_calendar', $contextlist->get_contextids());
416 // Delete all Calendar data for User 1.
417 provider::delete_data_for_user($approvedcontextlist);
419 // Test all Calendar Events and Subscriptions for User 1 equals zero.
420 $events = $DB->get_records('event', ['userid' => $user1->id]);
421 $this->assertCount(0, $events);
422 $eventsubscriptions = $DB->get_records('event_subscriptions', ['userid' => $user1->id]);
423 $this->assertCount(0, $eventsubscriptions);
425 // Test all Calendar Events and Subscriptions for User 2 still exists and matches the same number created.
426 $events = $DB->get_records('event', ['userid' => $user2->id]);
427 $this->assertCount(3, $events);
428 $eventsubscriptions = $DB->get_records('event_subscriptions', ['userid' => $user2->id]);
429 $this->assertCount(1, $eventsubscriptions);
432 // Start of helper functions.
435 * Helper function to create a Standard Calendar Event.
437 * @param string $eventtype Calendar event type
438 * @param int $userid User Id
439 * @param int $time Timestamp value
440 * @param string $customname Custom name
441 * @param int $categoryid Course Category Id
442 * @param int $courseid Course Id
443 * @param int $groupid Group Id
444 * @return bool|calendar_event Standard Calendar Event created.
445 * @throws coding_exception
447 protected function create_test_standard_calendar_event($eventtype, $userid, $time, $customname = '', $categoryid = 0, $courseid = 0, $groupid = 0) {
448 // Create a standard calendar event.
449 $name = "Standard Calendar Event $eventtype";
450 if ($customname != '') {
451 $name .= " -- $customname";
454 $event = (object)[
455 'name' => $name,
456 'categoryid' => $categoryid,
457 'courseid' => $courseid,
458 'groupid' => $groupid,
459 'userid' => $userid,
460 'modulename' => 0,
461 'instance' => 0,
462 'eventtype' => $eventtype,
463 'type' => CALENDAR_EVENT_TYPE_STANDARD,
464 'timestart' => $time,
465 'visible' => 1
467 return calendar_event::create($event, false);
471 * Helper function to create an Action Calendar Event.
473 * @param string $eventtype Calendar event type
474 * @param int $courseid Course Id
475 * @param int $instanceid Activity Module instance id
476 * @param string $modulename Activity Module name
477 * @param int $userid User Id
478 * @param int $time Timestamp value
479 * @return bool|calendar_event Action Calendar Event created.
480 * @throws coding_exception
482 protected function create_test_action_calendar_event($eventtype, $courseid, $instanceid, $modulename, $userid, $time) {
483 // Create an action calendar event.
484 $event = (object)[
485 'name' => "Action Calendar Event $eventtype -- $modulename",
486 'categoryid' => 0,
487 'courseid' => $courseid,
488 'groupid' => 0,
489 'userid' => $userid,
490 'modulename' => $modulename,
491 'instance' => $instanceid,
492 'eventtype' => $eventtype,
493 'type' => CALENDAR_EVENT_TYPE_ACTION,
494 'timestart' => $time,
495 'visible' => 1
497 return calendar_event::create($event, false);
501 * Helper function to create a Calendar Subscription.
503 * @param string $eventtype Calendar Subscription event type
504 * @param string $url Calendar Subscription URL
505 * @param int $userid User Id
506 * @param int $categoryid Category Id
507 * @param int $courseid Course Id
508 * @param int $groupid Group Id
509 * @return int Calendar Subscription Id
511 protected function create_test_calendar_subscription($eventtype, $url, $userid, $categoryid = 0, $courseid = 0, $groupid = 0) {
512 // Create a subscription calendar event.
513 $subscription = (object)[
514 'name' => "Calendar Subscription " . $eventtype,
515 'url' => $url,
516 'categoryid' => $categoryid,
517 'courseid' => $courseid,
518 'groupid' => $groupid,
519 'userid' => $userid,
520 'eventtype' => $eventtype
523 return calendar_add_subscription($subscription);