MDL-68567 badges: add/remove site backpacks
[moodle.git] / badges / tests / badgeslib_test.php
blob596c67e853ba1ca35f858ea989fb96f154976057
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 * Unit tests for badges
20 * @package core
21 * @subpackage badges
22 * @copyright 2013 onwards Totara Learning Solutions Ltd {@link http://www.totaralms.com/}
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
24 * @author Yuliya Bozhko <yuliya.bozhko@totaralms.com>
27 defined('MOODLE_INTERNAL') || die();
29 global $CFG;
30 require_once($CFG->libdir . '/badgeslib.php');
31 require_once($CFG->dirroot . '/badges/lib.php');
33 use core_badges\helper;
35 class core_badges_badgeslib_testcase extends advanced_testcase {
36 protected $badgeid;
37 protected $course;
38 protected $user;
39 protected $module;
40 protected $coursebadge;
41 protected $assertion;
43 /** @var $assertion2 to define json format for Open badge version 2 */
44 protected $assertion2;
46 protected function setUp() {
47 global $DB, $CFG;
48 $this->resetAfterTest(true);
49 $CFG->enablecompletion = true;
50 $user = $this->getDataGenerator()->create_user();
51 $fordb = new stdClass();
52 $fordb->id = null;
53 $fordb->name = "Test badge with 'apostrophe' and other friends (<>&@#)";
54 $fordb->description = "Testing badges";
55 $fordb->timecreated = time();
56 $fordb->timemodified = time();
57 $fordb->usercreated = $user->id;
58 $fordb->usermodified = $user->id;
59 $fordb->issuername = "Test issuer";
60 $fordb->issuerurl = "http://issuer-url.domain.co.nz";
61 $fordb->issuercontact = "issuer@example.com";
62 $fordb->expiredate = null;
63 $fordb->expireperiod = null;
64 $fordb->type = BADGE_TYPE_SITE;
65 $fordb->version = 1;
66 $fordb->language = 'en';
67 $fordb->courseid = null;
68 $fordb->messagesubject = "Test message subject";
69 $fordb->message = "Test message body";
70 $fordb->attachment = 1;
71 $fordb->notification = 0;
72 $fordb->imageauthorname = "Image Author 1";
73 $fordb->imageauthoremail = "author@example.com";
74 $fordb->imageauthorurl = "http://author-url.example.com";
75 $fordb->imagecaption = "Test caption image";
76 $fordb->status = BADGE_STATUS_INACTIVE;
78 $this->badgeid = $DB->insert_record('badge', $fordb, true);
80 // Set the default Issuer (because OBv2 needs them).
81 set_config('badges_defaultissuername', $fordb->issuername);
82 set_config('badges_defaultissuercontact', $fordb->issuercontact);
84 // Create a course with activity and auto completion tracking.
85 $this->course = $this->getDataGenerator()->create_course(array('enablecompletion' => true));
86 $this->user = $this->getDataGenerator()->create_user();
87 $studentrole = $DB->get_record('role', array('shortname' => 'student'));
88 $this->assertNotEmpty($studentrole);
90 // Get manual enrolment plugin and enrol user.
91 require_once($CFG->dirroot.'/enrol/manual/locallib.php');
92 $manplugin = enrol_get_plugin('manual');
93 $maninstance = $DB->get_record('enrol', array('courseid' => $this->course->id, 'enrol' => 'manual'), '*', MUST_EXIST);
94 $manplugin->enrol_user($maninstance, $this->user->id, $studentrole->id);
95 $this->assertEquals(1, $DB->count_records('user_enrolments'));
96 $completionauto = array('completion' => COMPLETION_TRACKING_AUTOMATIC);
97 $this->module = $this->getDataGenerator()->create_module('forum', array('course' => $this->course->id), $completionauto);
99 // Build badge and criteria.
100 $fordb->type = BADGE_TYPE_COURSE;
101 $fordb->courseid = $this->course->id;
102 $fordb->status = BADGE_STATUS_ACTIVE;
103 $this->coursebadge = $DB->insert_record('badge', $fordb, true);
105 // Insert Endorsement.
106 $endorsement = new stdClass();
107 $endorsement->badgeid = $this->coursebadge;
108 $endorsement->issuername = "Issuer 123";
109 $endorsement->issueremail = "issuer123@email.com";
110 $endorsement->issuerurl = "https://example.org/issuer-123";
111 $endorsement->dateissued = 1524567747;
112 $endorsement->claimid = "https://example.org/robotics-badge.json";
113 $endorsement->claimcomment = "Test endorser comment";
114 $DB->insert_record('badge_endorsement', $endorsement, true);
116 // Insert related badges.
117 $badge = new badge($this->coursebadge);
118 $clonedid = $badge->make_clone();
119 $badgeclone = new badge($clonedid);
120 $badgeclone->status = BADGE_STATUS_ACTIVE;
121 $badgeclone->save();
123 $relatebadge = new stdClass();
124 $relatebadge->badgeid = $this->coursebadge;
125 $relatebadge->relatedbadgeid = $clonedid;
126 $relatebadge->relatedid = $DB->insert_record('badge_related', $relatebadge, true);
128 // Insert a aligment.
129 $alignment = new stdClass();
130 $alignment->badgeid = $this->coursebadge;
131 $alignment->targetname = 'CCSS.ELA-Literacy.RST.11-12.3';
132 $alignment->targeturl = 'http://www.corestandards.org/ELA-Literacy/RST/11-12/3';
133 $alignment->targetdescription = 'Test target description';
134 $alignment->targetframework = 'CCSS.RST.11-12.3';
135 $alignment->targetcode = 'CCSS.RST.11-12.3';
136 $DB->insert_record('badge_alignment', $alignment, true);
138 $this->assertion = new stdClass();
139 $this->assertion->badge = '{"uid":"%s","recipient":{"identity":"%s","type":"email","hashed":true,"salt":"%s"},"badge":"%s","verify":{"type":"hosted","url":"%s"},"issuedOn":"%d","evidence":"%s"}';
140 $this->assertion->class = '{"name":"%s","description":"%s","image":"%s","criteria":"%s","issuer":"%s"}';
141 $this->assertion->issuer = '{"name":"%s","url":"%s","email":"%s"}';
142 // Format JSON-LD for Openbadge specification version 2.0.
143 $this->assertion2 = new stdClass();
144 $this->assertion2->badge = '{"recipient":{"identity":"%s","type":"email","hashed":true,"salt":"%s"},' .
145 '"badge":{"name":"%s","description":"%s","image":"%s",' .
146 '"criteria":{"id":"%s","narrative":"%s"},"issuer":{"name":"%s","url":"%s","email":"%s",' .
147 '"@context":"https:\/\/w3id.org\/openbadges\/v2","id":"%s","type":"Issuer"},' .
148 '"@context":"https:\/\/w3id.org\/openbadges\/v2","id":"%s","type":"BadgeClass","version":"%s",' .
149 '"@language":"en","related":[{"id":"%s","version":"%s","@language":"%s"}],"endorsement":"%s",' .
150 '"alignments":[{"targetName":"%s","targetUrl":"%s","targetDescription":"%s","targetFramework":"%s",' .
151 '"targetCode":"%s"}]},"verify":{"type":"hosted","url":"%s"},"issuedOn":"%s","evidence":"%s",' .
152 '"@context":"https:\/\/w3id.org\/openbadges\/v2","type":"Assertion","id":"%s"}';
154 $this->assertion2->class = '{"name":"%s","description":"%s","image":"%s",' .
155 '"criteria":{"id":"%s","narrative":"%s"},"issuer":{"name":"%s","url":"%s","email":"%s",' .
156 '"@context":"https:\/\/w3id.org\/openbadges\/v2","id":"%s","type":"Issuer"},' .
157 '"@context":"https:\/\/w3id.org\/openbadges\/v2","id":"%s","type":"BadgeClass","version":"%s",' .
158 '"@language":"%s","related":[{"id":"%s","version":"%s","@language":"%s"}],"endorsement":"%s",' .
159 '"alignments":[{"targetName":"%s","targetUrl":"%s","targetDescription":"%s","targetFramework":"%s",' .
160 '"targetCode":"%s"}]}';
161 $this->assertion2->issuer = '{"name":"%s","url":"%s","email":"%s",' .
162 '"@context":"https:\/\/w3id.org\/openbadges\/v2","id":"%s","type":"Issuer"}';
165 public function test_create_badge() {
166 $badge = new badge($this->badgeid);
168 $this->assertInstanceOf('badge', $badge);
169 $this->assertEquals($this->badgeid, $badge->id);
172 public function test_clone_badge() {
173 $badge = new badge($this->badgeid);
174 $newid = $badge->make_clone();
175 $clonedbadge = new badge($newid);
177 $this->assertEquals($badge->description, $clonedbadge->description);
178 $this->assertEquals($badge->issuercontact, $clonedbadge->issuercontact);
179 $this->assertEquals($badge->issuername, $clonedbadge->issuername);
180 $this->assertEquals($badge->issuercontact, $clonedbadge->issuercontact);
181 $this->assertEquals($badge->issuerurl, $clonedbadge->issuerurl);
182 $this->assertEquals($badge->expiredate, $clonedbadge->expiredate);
183 $this->assertEquals($badge->expireperiod, $clonedbadge->expireperiod);
184 $this->assertEquals($badge->type, $clonedbadge->type);
185 $this->assertEquals($badge->courseid, $clonedbadge->courseid);
186 $this->assertEquals($badge->message, $clonedbadge->message);
187 $this->assertEquals($badge->messagesubject, $clonedbadge->messagesubject);
188 $this->assertEquals($badge->attachment, $clonedbadge->attachment);
189 $this->assertEquals($badge->notification, $clonedbadge->notification);
190 $this->assertEquals($badge->version, $clonedbadge->version);
191 $this->assertEquals($badge->language, $clonedbadge->language);
192 $this->assertEquals($badge->imagecaption, $clonedbadge->imagecaption);
193 $this->assertEquals($badge->imageauthorname, $clonedbadge->imageauthorname);
194 $this->assertEquals($badge->imageauthoremail, $clonedbadge->imageauthoremail);
195 $this->assertEquals($badge->imageauthorurl, $clonedbadge->imageauthorurl);
198 public function test_badge_status() {
199 $badge = new badge($this->badgeid);
200 $old_status = $badge->status;
201 $badge->set_status(BADGE_STATUS_ACTIVE);
202 $this->assertAttributeNotEquals($old_status, 'status', $badge);
203 $this->assertAttributeEquals(BADGE_STATUS_ACTIVE, 'status', $badge);
206 public function test_delete_badge() {
207 $badge = new badge($this->badgeid);
208 $badge->delete();
209 // We don't actually delete badges. We archive them.
210 $this->assertAttributeEquals(BADGE_STATUS_ARCHIVED, 'status', $badge);
214 * Really delete the badge.
216 public function test_delete_badge_for_real() {
217 global $DB;
219 $badge = new badge($this->badgeid);
221 $newid1 = $badge->make_clone();
222 $newid2 = $badge->make_clone();
223 $newid3 = $badge->make_clone();
225 // Insert related badges to badge 1.
226 $badge->add_related_badges([$newid1, $newid2, $newid3]);
228 // Another badge.
229 $badge2 = new badge($newid2);
230 // Make badge 1 related for badge 2.
231 $badge2->add_related_badges([$this->badgeid]);
233 // Confirm that the records about this badge about its relations have been removed as well.
234 $relatedsql = 'badgeid = :badgeid OR relatedbadgeid = :relatedbadgeid';
235 $relatedparams = array(
236 'badgeid' => $this->badgeid,
237 'relatedbadgeid' => $this->badgeid
239 // Badge 1 has 4 related records. 3 where it's the badgeid, 1 where it's the relatedbadgeid.
240 $this->assertEquals(4, $DB->count_records_select('badge_related', $relatedsql, $relatedparams));
242 // Delete the badge for real.
243 $badge->delete(false);
245 // Confirm that the badge itself has been removed.
246 $this->assertFalse($DB->record_exists('badge', ['id' => $this->badgeid]));
248 // Confirm that the records about this badge about its relations have been removed as well.
249 $this->assertFalse($DB->record_exists_select('badge_related', $relatedsql, $relatedparams));
252 public function test_create_badge_criteria() {
253 $badge = new badge($this->badgeid);
254 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
255 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ALL));
257 $this->assertCount(1, $badge->get_criteria());
259 $criteria_profile = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_PROFILE, 'badgeid' => $badge->id));
260 $params = array('agg' => BADGE_CRITERIA_AGGREGATION_ALL, 'field_address' => 'address');
261 $criteria_profile->save($params);
263 $this->assertCount(2, $badge->get_criteria());
266 public function test_add_badge_criteria_description() {
267 $criteriaoverall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $this->badgeid));
268 $criteriaoverall->save(array(
269 'agg' => BADGE_CRITERIA_AGGREGATION_ALL,
270 'description' => 'Overall description',
271 'descriptionformat' => FORMAT_HTML
274 $criteriaprofile = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_PROFILE, 'badgeid' => $this->badgeid));
275 $params = array(
276 'agg' => BADGE_CRITERIA_AGGREGATION_ALL,
277 'field_address' => 'address',
278 'description' => 'Description',
279 'descriptionformat' => FORMAT_HTML
281 $criteriaprofile->save($params);
283 $badge = new badge($this->badgeid);
284 $this->assertEquals('Overall description', $badge->criteria[BADGE_CRITERIA_TYPE_OVERALL]->description);
285 $this->assertEquals('Description', $badge->criteria[BADGE_CRITERIA_TYPE_PROFILE]->description);
288 public function test_delete_badge_criteria() {
289 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $this->badgeid));
290 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ALL));
291 $badge = new badge($this->badgeid);
293 $this->assertInstanceOf('award_criteria_overall', $badge->criteria[BADGE_CRITERIA_TYPE_OVERALL]);
295 $badge->criteria[BADGE_CRITERIA_TYPE_OVERALL]->delete();
296 $this->assertEmpty($badge->get_criteria());
299 public function test_badge_awards() {
300 global $DB;
301 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
302 $badge = new badge($this->badgeid);
303 $user1 = $this->getDataGenerator()->create_user();
305 $sink = $this->redirectMessages();
307 $DB->set_field_select('message_processors', 'enabled', 0, "name <> 'email'");
308 set_user_preference('message_provider_moodle_badgerecipientnotice_loggedoff', 'email', $user1);
310 $badge->issue($user1->id, false);
311 $this->assertDebuggingCalled(); // Expect debugging while baking a badge via phpunit.
312 $this->assertTrue($badge->is_issued($user1->id));
314 $messages = $sink->get_messages();
315 $sink->close();
316 $this->assertCount(1, $messages);
317 $message = array_pop($messages);
318 // Check we have the expected data.
319 $customdata = json_decode($message->customdata);
320 $this->assertObjectHasAttribute('notificationiconurl', $customdata);
321 $this->assertObjectHasAttribute('hash', $customdata);
323 $user2 = $this->getDataGenerator()->create_user();
324 $badge->issue($user2->id, true);
325 $this->assertTrue($badge->is_issued($user2->id));
327 $this->assertCount(2, $badge->get_awards());
331 * Test the {@link badges_get_user_badges()} function in lib/badgeslib.php
333 public function test_badges_get_user_badges() {
334 global $DB;
336 // Messaging is not compatible with transactions.
337 $this->preventResetByRollback();
339 $badges = array();
340 $user1 = $this->getDataGenerator()->create_user();
341 $user2 = $this->getDataGenerator()->create_user();
343 // Record the current time, we need to be precise about a couple of things.
344 $now = time();
345 // Create 11 badges with which to test.
346 for ($i = 1; $i <= 11; $i++) {
347 // Mock up a badge.
348 $badge = new stdClass();
349 $badge->id = null;
350 $badge->name = "Test badge $i";
351 $badge->description = "Testing badges $i";
352 $badge->timecreated = $now - 12;
353 $badge->timemodified = $now - 12;
354 $badge->usercreated = $user1->id;
355 $badge->usermodified = $user1->id;
356 $badge->issuername = "Test issuer";
357 $badge->issuerurl = "http://issuer-url.domain.co.nz";
358 $badge->issuercontact = "issuer@example.com";
359 $badge->expiredate = null;
360 $badge->expireperiod = null;
361 $badge->type = BADGE_TYPE_SITE;
362 $badge->courseid = null;
363 $badge->messagesubject = "Test message subject for badge $i";
364 $badge->message = "Test message body for badge $i";
365 $badge->attachment = 1;
366 $badge->notification = 0;
367 $badge->status = BADGE_STATUS_INACTIVE;
368 $badge->version = "Version $i";
369 $badge->language = "en";
370 $badge->imagecaption = "Image caption $i";
371 $badge->imageauthorname = "Image author's name $i";
372 $badge->imageauthoremail = "author$i@example.com";
373 $badge->imageauthorname = "Image author's name $i";
375 $badgeid = $DB->insert_record('badge', $badge, true);
376 $badges[$badgeid] = new badge($badgeid);
377 $badges[$badgeid]->issue($user2->id, true);
378 // Check it all actually worked.
379 $this->assertCount(1, $badges[$badgeid]->get_awards());
381 // Hack the database to adjust the time each badge was issued.
382 // The alternative to this is sleep which is a no-no in unit tests.
383 $DB->set_field('badge_issued', 'dateissued', $now - 11 + $i, array('userid' => $user2->id, 'badgeid' => $badgeid));
386 // Make sure the first user has no badges.
387 $result = badges_get_user_badges($user1->id);
388 $this->assertInternalType('array', $result);
389 $this->assertCount(0, $result);
391 // Check that the second user has the expected 11 badges.
392 $result = badges_get_user_badges($user2->id);
393 $this->assertCount(11, $result);
395 // Test pagination.
396 // Ordering is by time issued desc, so things will come out with the last awarded badge first.
397 $result = badges_get_user_badges($user2->id, 0, 0, 4);
398 $this->assertCount(4, $result);
399 $lastbadgeissued = reset($result);
400 $this->assertSame('Test badge 11', $lastbadgeissued->name);
401 // Page 2. Expecting 4 results again.
402 $result = badges_get_user_badges($user2->id, 0, 1, 4);
403 $this->assertCount(4, $result);
404 $lastbadgeissued = reset($result);
405 $this->assertSame('Test badge 7', $lastbadgeissued->name);
406 // Page 3. Expecting just three results here.
407 $result = badges_get_user_badges($user2->id, 0, 2, 4);
408 $this->assertCount(3, $result);
409 $lastbadgeissued = reset($result);
410 $this->assertSame('Test badge 3', $lastbadgeissued->name);
411 // Page 4.... there is no page 4.
412 $result = badges_get_user_badges($user2->id, 0, 3, 4);
413 $this->assertCount(0, $result);
415 // Test search.
416 $result = badges_get_user_badges($user2->id, 0, 0, 0, 'badge 1');
417 $this->assertCount(3, $result);
418 $lastbadgeissued = reset($result);
419 $this->assertSame('Test badge 11', $lastbadgeissued->name);
420 // The term Totara doesn't appear anywhere in the badges.
421 $result = badges_get_user_badges($user2->id, 0, 0, 0, 'Totara');
422 $this->assertCount(0, $result);
424 // Issue a user with a course badge and verify its returned based on if
425 // coursebadges are enabled or disabled.
426 $sitebadgeid = key($badges);
427 $badges[$sitebadgeid]->issue($this->user->id, true);
429 $badge = new stdClass();
430 $badge->id = null;
431 $badge->name = "Test course badge";
432 $badge->description = "Testing course badge";
433 $badge->timecreated = $now;
434 $badge->timemodified = $now;
435 $badge->usercreated = $user1->id;
436 $badge->usermodified = $user1->id;
437 $badge->issuername = "Test issuer";
438 $badge->issuerurl = "http://issuer-url.domain.co.nz";
439 $badge->issuercontact = "issuer@example.com";
440 $badge->expiredate = null;
441 $badge->expireperiod = null;
442 $badge->type = BADGE_TYPE_COURSE;
443 $badge->courseid = $this->course->id;
444 $badge->messagesubject = "Test message subject for course badge";
445 $badge->message = "Test message body for course badge";
446 $badge->attachment = 1;
447 $badge->notification = 0;
448 $badge->status = BADGE_STATUS_ACTIVE;
449 $badge->version = "Version $i";
450 $badge->language = "en";
451 $badge->imagecaption = "Image caption";
452 $badge->imageauthorname = "Image author's name";
453 $badge->imageauthoremail = "author@example.com";
454 $badge->imageauthorname = "Image author's name";
456 $badgeid = $DB->insert_record('badge', $badge, true);
457 $badges[$badgeid] = new badge($badgeid);
458 $badges[$badgeid]->issue($this->user->id, true);
460 // With coursebadges off, we should only get the site badge.
461 set_config('badges_allowcoursebadges', false);
462 $result = badges_get_user_badges($this->user->id);
463 $this->assertCount(1, $result);
465 // With it on, we should get both.
466 set_config('badges_allowcoursebadges', true);
467 $result = badges_get_user_badges($this->user->id);
468 $this->assertCount(2, $result);
472 public function data_for_message_from_template() {
473 return array(
474 array(
475 'This is a message with no variables',
476 array(), // no params
477 'This is a message with no variables'
479 array(
480 'This is a message with %amissing% variables',
481 array(), // no params
482 'This is a message with %amissing% variables'
484 array(
485 'This is a message with %one% variable',
486 array('one' => 'a single'),
487 'This is a message with a single variable'
489 array(
490 'This is a message with %one% %two% %three% variables',
491 array('one' => 'more', 'two' => 'than', 'three' => 'one'),
492 'This is a message with more than one variables'
494 array(
495 'This is a message with %three% %two% %one%',
496 array('one' => 'variables', 'two' => 'ordered', 'three' => 'randomly'),
497 'This is a message with randomly ordered variables'
499 array(
500 'This is a message with %repeated% %one% %repeated% of variables',
501 array('one' => 'and', 'repeated' => 'lots'),
502 'This is a message with lots and lots of variables'
508 * @dataProvider data_for_message_from_template
510 public function test_badge_message_from_template($message, $params, $result) {
511 $this->assertEquals(badge_message_from_template($message, $params), $result);
515 * Test badges observer when course module completion event id fired.
517 public function test_badges_observer_course_module_criteria_review() {
518 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
519 $badge = new badge($this->coursebadge);
520 $this->assertFalse($badge->is_issued($this->user->id));
522 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
523 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY));
524 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_ACTIVITY, 'badgeid' => $badge->id));
525 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY, 'module_'.$this->module->cmid => $this->module->cmid));
527 // Assert the badge will not be issued to the user as is.
528 $badge = new badge($this->coursebadge);
529 $badge->review_all_criteria();
530 $this->assertFalse($badge->is_issued($this->user->id));
532 // Set completion for forum activity.
533 $c = new completion_info($this->course);
534 $activities = $c->get_activities();
535 $this->assertEquals(1, count($activities));
536 $this->assertTrue(isset($activities[$this->module->cmid]));
537 $this->assertEquals($activities[$this->module->cmid]->name, $this->module->name);
539 $current = $c->get_data($activities[$this->module->cmid], false, $this->user->id);
540 $current->completionstate = COMPLETION_COMPLETE;
541 $current->timemodified = time();
542 $sink = $this->redirectEmails();
543 $c->internal_set_data($activities[$this->module->cmid], $current);
544 $this->assertCount(1, $sink->get_messages());
545 $sink->close();
547 // Check if badge is awarded.
548 $this->assertDebuggingCalled('Error baking badge image!');
549 $this->assertTrue($badge->is_issued($this->user->id));
553 * Test badges observer when course_completed event is fired.
555 public function test_badges_observer_course_criteria_review() {
556 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
557 $badge = new badge($this->coursebadge);
558 $this->assertFalse($badge->is_issued($this->user->id));
560 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
561 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY));
562 $criteria_overall1 = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_COURSE, 'badgeid' => $badge->id));
563 $criteria_overall1->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY, 'course_'.$this->course->id => $this->course->id));
565 $ccompletion = new completion_completion(array('course' => $this->course->id, 'userid' => $this->user->id));
567 // Assert the badge will not be issued to the user as is.
568 $badge = new badge($this->coursebadge);
569 $badge->review_all_criteria();
570 $this->assertFalse($badge->is_issued($this->user->id));
572 // Mark course as complete.
573 $sink = $this->redirectEmails();
574 $ccompletion->mark_complete();
575 $this->assertCount(1, $sink->get_messages());
576 $sink->close();
578 // Check if badge is awarded.
579 $this->assertDebuggingCalled('Error baking badge image!');
580 $this->assertTrue($badge->is_issued($this->user->id));
584 * Test badges observer when user_updated event is fired.
586 public function test_badges_observer_profile_criteria_review() {
587 global $CFG, $DB;
588 require_once($CFG->dirroot.'/user/profile/lib.php');
590 // Add a custom field of textarea type.
591 $customprofileid = $DB->insert_record('user_info_field', array(
592 'shortname' => 'newfield', 'name' => 'Description of new field', 'categoryid' => 1,
593 'datatype' => 'textarea'));
595 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
596 $badge = new badge($this->coursebadge);
598 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
599 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY));
600 $criteria_overall1 = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_PROFILE, 'badgeid' => $badge->id));
601 $criteria_overall1->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ALL, 'field_address' => 'address', 'field_aim' => 'aim',
602 'field_' . $customprofileid => $customprofileid));
604 // Assert the badge will not be issued to the user as is.
605 $badge = new badge($this->coursebadge);
606 $badge->review_all_criteria();
607 $this->assertFalse($badge->is_issued($this->user->id));
609 // Set the required fields and make sure the badge got issued.
610 $this->user->address = 'Test address';
611 $this->user->aim = '999999999';
612 $sink = $this->redirectEmails();
613 profile_save_data((object)array('id' => $this->user->id, 'profile_field_newfield' => 'X'));
614 user_update_user($this->user, false);
615 $this->assertCount(1, $sink->get_messages());
616 $sink->close();
617 // Check if badge is awarded.
618 $this->assertDebuggingCalled('Error baking badge image!');
619 $this->assertTrue($badge->is_issued($this->user->id));
623 * Test badges observer when cohort_member_added event is fired.
625 public function test_badges_observer_cohort_criteria_review() {
626 global $CFG;
628 require_once("$CFG->dirroot/cohort/lib.php");
630 $cohort = $this->getDataGenerator()->create_cohort();
632 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
633 $badge = new badge($this->badgeid);
634 $this->assertFalse($badge->is_issued($this->user->id));
636 // Set up the badge criteria.
637 $criteriaoverall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
638 $criteriaoverall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY));
639 $criteriaoverall1 = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_COHORT, 'badgeid' => $badge->id));
640 $criteriaoverall1->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY, 'cohort_cohorts' => array('0' => $cohort->id)));
642 // Make the badge active.
643 $badge->set_status(BADGE_STATUS_ACTIVE);
645 // Add the user to the cohort.
646 cohort_add_member($cohort->id, $this->user->id);
648 // Verify that the badge was awarded.
649 $this->assertDebuggingCalled();
650 $this->assertTrue($badge->is_issued($this->user->id));
655 * Test badges assertion generated when a badge is issued.
657 public function test_badges_assertion() {
658 $this->preventResetByRollback(); // Messaging is not compatible with transactions.
659 $badge = new badge($this->coursebadge);
660 $this->assertFalse($badge->is_issued($this->user->id));
662 $criteria_overall = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_OVERALL, 'badgeid' => $badge->id));
663 $criteria_overall->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ANY));
664 $criteria_overall1 = award_criteria::build(array('criteriatype' => BADGE_CRITERIA_TYPE_PROFILE, 'badgeid' => $badge->id));
665 $criteria_overall1->save(array('agg' => BADGE_CRITERIA_AGGREGATION_ALL, 'field_address' => 'address'));
667 $this->user->address = 'Test address';
668 $sink = $this->redirectEmails();
669 user_update_user($this->user, false);
670 $this->assertCount(1, $sink->get_messages());
671 $sink->close();
672 // Check if badge is awarded.
673 $this->assertDebuggingCalled('Error baking badge image!');
674 $awards = $badge->get_awards();
675 $this->assertCount(1, $awards);
677 // Get assertion.
678 $award = reset($awards);
679 $assertion = new core_badges_assertion($award->uniquehash, OPEN_BADGES_V1);
680 $testassertion = $this->assertion;
682 // Make sure JSON strings have the same structure.
683 $this->assertStringMatchesFormat($testassertion->badge, json_encode($assertion->get_badge_assertion()));
684 $this->assertStringMatchesFormat($testassertion->class, json_encode($assertion->get_badge_class()));
685 $this->assertStringMatchesFormat($testassertion->issuer, json_encode($assertion->get_issuer()));
687 // Test Openbadge specification version 2.
688 // Get assertion version 2.
689 $award = reset($awards);
690 $assertion2 = new core_badges_assertion($award->uniquehash, OPEN_BADGES_V2);
691 $testassertion2 = $this->assertion2;
693 // Make sure JSON strings have the same structure.
694 $this->assertStringMatchesFormat($testassertion2->badge, json_encode($assertion2->get_badge_assertion()));
695 $this->assertStringMatchesFormat($testassertion2->class, json_encode($assertion2->get_badge_class()));
696 $this->assertStringMatchesFormat($testassertion2->issuer, json_encode($assertion2->get_issuer()));
700 * Tests the core_badges_myprofile_navigation() function.
702 public function test_core_badges_myprofile_navigation() {
703 // Set up the test.
704 $tree = new \core_user\output\myprofile\tree();
705 $this->setAdminUser();
706 $badge = new badge($this->badgeid);
707 $badge->issue($this->user->id, true);
708 $iscurrentuser = true;
709 $course = null;
711 // Enable badges.
712 set_config('enablebadges', true);
714 // Check the node tree is correct.
715 core_badges_myprofile_navigation($tree, $this->user, $iscurrentuser, $course);
716 $reflector = new ReflectionObject($tree);
717 $nodes = $reflector->getProperty('nodes');
718 $nodes->setAccessible(true);
719 $this->assertArrayHasKey('localbadges', $nodes->getValue($tree));
723 * Tests the core_badges_myprofile_navigation() function with badges disabled..
725 public function test_core_badges_myprofile_navigation_badges_disabled() {
726 // Set up the test.
727 $tree = new \core_user\output\myprofile\tree();
728 $this->setAdminUser();
729 $badge = new badge($this->badgeid);
730 $badge->issue($this->user->id, true);
731 $iscurrentuser = false;
732 $course = null;
734 // Disable badges.
735 set_config('enablebadges', false);
737 // Check the node tree is correct.
738 core_badges_myprofile_navigation($tree, $this->user, $iscurrentuser, $course);
739 $reflector = new ReflectionObject($tree);
740 $nodes = $reflector->getProperty('nodes');
741 $nodes->setAccessible(true);
742 $this->assertArrayNotHasKey('localbadges', $nodes->getValue($tree));
746 * Tests the core_badges_myprofile_navigation() function with a course badge.
748 public function test_core_badges_myprofile_navigation_with_course_badge() {
749 // Set up the test.
750 $tree = new \core_user\output\myprofile\tree();
751 $this->setAdminUser();
752 $badge = new badge($this->coursebadge);
753 $badge->issue($this->user->id, true);
754 $iscurrentuser = false;
756 // Check the node tree is correct.
757 core_badges_myprofile_navigation($tree, $this->user, $iscurrentuser, $this->course);
758 $reflector = new ReflectionObject($tree);
759 $nodes = $reflector->getProperty('nodes');
760 $nodes->setAccessible(true);
761 $this->assertArrayHasKey('localbadges', $nodes->getValue($tree));
765 * Test insert and update endorsement with a site badge.
767 public function test_badge_endorsement() {
768 $badge = new badge($this->badgeid);
770 // Insert Endorsement.
771 $endorsement = new stdClass();
772 $endorsement->badgeid = $this->badgeid;
773 $endorsement->issuername = "Issuer 123";
774 $endorsement->issueremail = "issuer123@email.com";
775 $endorsement->issuerurl = "https://example.org/issuer-123";
776 $endorsement->dateissued = 1524567747;
777 $endorsement->claimid = "https://example.org/robotics-badge.json";
778 $endorsement->claimcomment = "Test endorser comment";
780 $badge->save_endorsement($endorsement);
781 $endorsement1 = $badge->get_endorsement();
782 $this->assertEquals($endorsement->badgeid, $endorsement1->badgeid);
783 $this->assertEquals($endorsement->issuername, $endorsement1->issuername);
784 $this->assertEquals($endorsement->issueremail, $endorsement1->issueremail);
785 $this->assertEquals($endorsement->issuerurl, $endorsement1->issuerurl);
786 $this->assertEquals($endorsement->dateissued, $endorsement1->dateissued);
787 $this->assertEquals($endorsement->claimid, $endorsement1->claimid);
788 $this->assertEquals($endorsement->claimcomment, $endorsement1->claimcomment);
790 // Update Endorsement.
791 $endorsement1->issuername = "Issuer update";
792 $badge->save_endorsement($endorsement1);
793 $endorsement2 = $badge->get_endorsement();
794 $this->assertEquals($endorsement1->id, $endorsement2->id);
795 $this->assertEquals($endorsement1->issuername, $endorsement2->issuername);
799 * Test insert and delete related badge with a site badge.
801 public function test_badge_related() {
802 $badge = new badge($this->badgeid);
803 $newid1 = $badge->make_clone();
804 $newid2 = $badge->make_clone();
805 $newid3 = $badge->make_clone();
807 // Insert an related badge.
808 $badge->add_related_badges([$newid1, $newid2, $newid3]);
809 $this->assertCount(3, $badge->get_related_badges());
811 // Only get related is active.
812 $clonedbage1 = new badge($newid1);
813 $clonedbage1->status = BADGE_STATUS_ACTIVE;
814 $clonedbage1->save();
815 $this->assertCount(1, $badge->get_related_badges(true));
817 // Delete an related badge.
818 $badge->delete_related_badge($newid2);
819 $this->assertCount(2, $badge->get_related_badges());
823 * Test insert, update, delete alignment with a site badge.
825 public function test_alignments() {
826 $badge = new badge($this->badgeid);
828 // Insert a alignment.
829 $alignment1 = new stdClass();
830 $alignment1->badgeid = $this->badgeid;
831 $alignment1->targetname = 'CCSS.ELA-Literacy.RST.11-12.3';
832 $alignment1->targeturl = 'http://www.corestandards.org/ELA-Literacy/RST/11-12/3';
833 $alignment1->targetdescription = 'Test target description';
834 $alignment1->targetframework = 'CCSS.RST.11-12.3';
835 $alignment1->targetcode = 'CCSS.RST.11-12.3';
836 $alignment2 = clone $alignment1;
837 $newid1 = $badge->save_alignment($alignment1);
838 $newid2 = $badge->save_alignment($alignment2);
839 $alignments1 = $badge->get_alignments();
840 $this->assertCount(2, $alignments1);
842 $this->assertEquals($alignment1->badgeid, $alignments1[$newid1]->badgeid);
843 $this->assertEquals($alignment1->targetname, $alignments1[$newid1]->targetname);
844 $this->assertEquals($alignment1->targeturl, $alignments1[$newid1]->targeturl);
845 $this->assertEquals($alignment1->targetdescription, $alignments1[$newid1]->targetdescription);
846 $this->assertEquals($alignment1->targetframework, $alignments1[$newid1]->targetframework);
847 $this->assertEquals($alignment1->targetcode, $alignments1[$newid1]->targetcode);
849 // Update aligment.
850 $alignments1[$newid1]->targetname = 'CCSS.ELA-Literacy.RST.11-12.3 update';
851 $badge->save_alignment($alignments1[$newid1], $alignments1[$newid1]->id);
852 $alignments2 = $badge->get_alignments();
853 $this->assertEquals($alignments1[$newid1]->id, $alignments2[$newid1]->id);
854 $this->assertEquals($alignments1[$newid1]->targetname, $alignments2[$newid1]->targetname);
856 // Delete alignment.
857 $badge->delete_alignment($alignments1[$newid2]->id);
858 $this->assertCount(1, $badge->get_alignments());
862 * Test badges_delete_site_backpack().
865 public function test_badges_delete_site_backpack(): void {
866 global $DB;
868 $this->setAdminUser();
870 // Create one backpack.
871 $total = $DB->count_records('badge_external_backpack');
872 $this->assertEquals(1, $total);
874 $data = new \stdClass();
875 $data->apiversion = OPEN_BADGES_V2P1;
876 $data->backpackapiurl = 'https://dc.imsglobal.org/obchost/ims/ob/v2p1';
877 $data->backpackweburl = 'https://dc.imsglobal.org';
878 badges_create_site_backpack($data);
879 $backpack = $DB->get_record('badge_external_backpack', ['backpackweburl' => $data->backpackweburl]);
880 $user1 = $this->getDataGenerator()->create_user();
881 $user2 = $this->getDataGenerator()->create_user();
882 // User1 is connected to the backpack to be removed and has 2 collections.
883 $backpackuser1 = helper::create_fake_backpack(['userid' => $user1->id, 'externalbackpackid' => $backpack->id]);
884 helper::create_fake_backpack_collection(['backpackid' => $backpackuser1->id]);
885 helper::create_fake_backpack_collection(['backpackid' => $backpackuser1->id]);
886 // User2 is connected to a different backpack and has 1 collection.
887 $backpackuser2 = helper::create_fake_backpack(['userid' => $user2->id]);
888 helper::create_fake_backpack_collection(['backpackid' => $backpackuser2->id]);
890 $total = $DB->count_records('badge_external_backpack');
891 $this->assertEquals(2, $total);
892 $total = $DB->count_records('badge_backpack');
893 $this->assertEquals(2, $total);
894 $total = $DB->count_records('badge_external');
895 $this->assertEquals(3, $total);
897 // Remove the backpack created previously.
898 $result = badges_delete_site_backpack($backpack->id);
899 $this->assertTrue($result);
901 $total = $DB->count_records('badge_external_backpack');
902 $this->assertEquals(1, $total);
904 $total = $DB->count_records('badge_backpack');
905 $this->assertEquals(1, $total);
907 $total = $DB->count_records('badge_external');
908 $this->assertEquals(1, $total);
910 // Try to remove an non-existent backpack.
911 $result = badges_delete_site_backpack($backpack->id);
912 $this->assertFalse($result);