ILIAS  trunk Revision v12.0_alpha-1227-g7ff6d300864
class.ilLTIConsumerGradeServiceScores.php
Go to the documentation of this file.
1<?php
2
19declare(strict_types=1);
20
31{
33 {
35 $this->id = 'Score.collection';
36 $this->template = '/{context_id}/lineitems/{item_id}/lineitem/scores';
37 $this->variables[] = 'Scores.url';
38 $this->formats[] = 'application/vnd.ims.lis.v1.scorecontainer+json';
39 $this->formats[] = 'application/vnd.ims.lis.v1.score+json';
40 $this->methods[] = 'POST';
41 }
42
47 {
48 $params = $this->parseTemplate();
49 $contextId = $params['context_id'];
50 $itemId = $params['item_id'];
51
52 ilObjLTIConsumer::getLogger()->info("contextId: " . $contextId);
53 ilObjLTIConsumer::getLogger()->info("objId: " . $itemId);
54 ilObjLTIConsumer::getLogger()->info("request data: " . $response->getRequestData());
55
56 // GET is disabled by the moment, but we have the code ready
57 // for a future implementation.
58
59 //$container = empty($contentType) || ($contentType === $this->formats[0]);
60
61 $typeid = 0;
62
64 try {
65 $token = $this->checkTool(array($scope));
66 if (is_null($token)) {
67 throw new Exception('invalid request', 401);
68 }
69
70 // Bug in Moodle as tool provider, should accept only "204 No Content" but schedules grade sync task will notices a failed status if not exactly 200
71 // see: http://www.imsglobal.org/spec/lti-ags/v2p0#score-service-scope-and-allowed-http-methods
72 //$response->setCode(204); // correct
73 $returnCode = 200;
74 $returnCode = $this->checkScore($response->getRequestData(), (int) $itemId);
75 $response->setCode($returnCode); // not really correct
76 } catch (Exception $e) {
77 $response->setCode($e->getCode());
78 $response->setReason($e->getMessage());
79 }
80 }
81
82 protected function checkScore(string $requestData, int $objId): int
83 {
84 global $DIC; /* @var \ILIAS\DI\Container $DIC */
85
86 $logger = $DIC->logger()->root();
87
88 $logger->info('checkScore');
89 $score = json_decode($requestData);
90 //prüfe Userid
91 $userId = ilCmiXapiUser::getUsrIdForObjectAndUsrIdent($objId, $score->userId);
92 if ($userId == null) {
93 ilObjLTIConsumer::getLogger()->info('User not available');
94 throw new Exception('User not available', 404);
95 return 404;
96 }
97
98 ilObjLTIConsumer::getLogger()->dump($score);
99
100 if (empty($score) ||
101 !isset($score->userId) ||
102 !isset($score->gradingProgress) ||
103 !isset($score->activityProgress) ||
104 !isset($score->timestamp) ||
105 isset($score->timestamp) && !self::validate_iso8601_date($score->timestamp) ||
106 (isset($score->scoreGiven) && !is_numeric($score->scoreGiven)) ||
107 (isset($score->scoreGiven) && !isset($score->scoreMaximum)) ||
108 (isset($score->scoreMaximum) && !is_numeric($score->scoreMaximum))
109 ) {
110 ilObjLTIConsumer::getLogger()->info('Incorrect score received');
111 ilObjLTIConsumer::getLogger()->dump($score);
112 throw new Exception('Incorrect score received', 400);
113 return 400;
114 }
115
116 if (!isset($score->scoreMaximum)) {
117 $score->scoreMaximum = 1;
118 }
119 if (!isset($score->scoreGiven)) {
120 $score->scoreGiven = 0;
121 }
122
123 $result = (float) $score->scoreGiven / (float) $score->scoreMaximum;
124
125 $ltiObjRes = new ilLTIConsumerResultService();
126
127 $ltiObjRes->readProperties($objId);
128 // check the object status
129 if (!$ltiObjRes->isAvailable()) {
130 throw new Exception('Tool for Object not available', 404);
131 return 404;
132 }
133
135
136 if ($score->activityProgress === 'InProgress') {
138 } elseif ($score->activityProgress === 'Completed' && $score->gradingProgress === 'FullyGraded') {
139 if ($result >= $ltiObjRes->getMasteryScore()) {
141 } else {
143 }
144 } elseif ($score->activityProgress === 'Completed' && $score->gradingProgress === 'Failed') {
146 }
147
148 $lp_percentage = (int) round(100 * $result);
149
150 ilObjLTIConsumer::getLogger()->info("lp_status: $lp_status, lp_percentage: $lp_percentage, result: $result, mastery_score: " . $ltiObjRes->getMasteryScore());
151
152 $consRes = ilLTIConsumerResult::getByKeys($objId, $userId, false);
153 if (empty($consRes)) {
154 $DIC->database()->insert(
155 'lti_consumer_results',
156 array(
157 'id' => array('integer', $DIC->database()->nextId('lti_consumer_results')),
158 'obj_id' => array('integer', $objId),
159 'usr_id' => array('integer', $userId),
160 'result' => array('float', $result)
161 )
162 );
163 } else {
164 $DIC->database()->replace(
165 'lti_consumer_results',
166 array(
167 'id' => array('integer', $consRes->id)
168 ),
169 array(
170 'obj_id' => array('integer', $objId),
171 'usr_id' => array('integer', $userId),
172 'result' => array('float', $result)
173 )
174 );
175 }
176
177 ilLPStatus::writeStatus($objId, $userId, $lp_status, $lp_percentage, true);
178
179 $ltiTimestamp = DateTimeImmutable::createFromFormat(DateTimeInterface::RFC3339_EXTENDED, $score->timestamp);
180 if (!$ltiTimestamp) { //moodle 4
181 $ltiTimestamp = DateTimeImmutable::createFromFormat(DateTimeInterface::ISO8601, $score->timestamp);
182 }
183 if (!$ltiTimestamp) { //for example nothing
184 $ltiTimestamp = new DateTime('now');
185 }
186 $gradeValues = [
187 'id' => array('integer', $DIC->database()->nextId('lti_consumer_grades')),
188 'obj_id' => array('integer', $objId),
189 'usr_id' => array('integer', $userId),
190 'score_given' => array('float', $score->scoreGiven),
191 'score_maximum' => array('float', $score->scoreMaximum),
192 'activity_progress' => array('text', $score->activityProgress),
193 'grading_progress' => array('text', $score->gradingProgress),
194 'lti_timestamp' => array('timestamp',$ltiTimestamp->format("Y-m-d H:i:s")),
195 'stored' => array('timestamp', date("Y-m-d H:i:s"))
196 ];
197 $DIC->database()->insert('lti_consumer_grades', $gradeValues);
198
199
200
201 return 200;
202 }
203
204 public static function validate_iso8601_date(string $date): bool
205 {
206 if (preg_match('/^([\+-]?\d{4}(?!\d{2}\b))((-?)((0[1-9]|1[0-2])' .
207 '(\3([12]\d|0[1-9]|3[01]))?|W([0-4]\d|5[0-2])(-?[1-7])?|(00[1-9]|0[1-9]\d|[12]\d{2}|3([0-5]\d|6[1-6])))' .
208 '([T\s]((([01]\d|2[0-3])((:?)[0-5]\d)?|24\:?00)([\.,]\d+(?!:))?)?(\17[0-5]\d([\.,]\d+)?)' .
209 '?([zZ]|([\+-])([01]\d|2[0-3]):?([0-5]\d)?)?)?)?$/', $date) > 0) {
210 return true;
211 }
212 return false;
213 }
214}
static getUsrIdForObjectAndUsrIdent(int $objId, string $userIdent)
static writeStatus(int $a_obj_id, int $a_user_id, int $a_status, int $a_percentage=0, bool $a_force_per=false, ?int &$a_old_status=self::LP_STATUS_NOT_ATTEMPTED_NUM)
Write status for user and object.
const LP_STATUS_COMPLETED_NUM
const LP_STATUS_IN_PROGRESS_NUM
const LP_STATUS_NOT_ATTEMPTED_NUM
const LP_STATUS_FAILED_NUM
execute(ilLTIConsumerServiceResponse $response)
Execute the request for this resource.
__construct(ilLTIConsumerServiceBase $service)
Class constructor.
const SCOPE_GRADESERVICE_SCORE
Scope for access to Score service.
array $params
Template variables parsed from the resource template.
checkTool(array $scopes=array())
Check to make sure the request is valid.
ilLTIConsumerServiceBase $service
Service associated with this resource.
parseTemplate()
Parse the template for variables.
static getByKeys(int $a_obj_id, int $a_usr_id, ?bool $a_create=false)
Get a result by object and user key.
$scope
Definition: ltiregstart.php:51
__construct(Container $dic, ilPlugin $plugin)
@inheritDoc
global $DIC
Definition: shib_login.php:26
$token
Definition: xapitoken.php:67
$objId
Definition: xapitoken.php:55
$response
Definition: xapitoken.php:90