ILIAS  release_8 Revision v8.19-1-g4e8f2f9140c
All Data Structures Namespaces Files Functions Variables Modules Pages
ilLTIConsumerGradeServiceScores Class Reference
+ Inheritance diagram for ilLTIConsumerGradeServiceScores:
+ Collaboration diagram for ilLTIConsumerGradeServiceScores:

Public Member Functions

 __construct (ilLTIConsumerServiceBase $service)
 
 execute (ilLTIConsumerServiceResponse $response)
 Execute the request for this resource. More...
 
- Public Member Functions inherited from ilLTIConsumerResourceBase
 __construct (ilLTIConsumerServiceBase $service)
 Class constructor. More...
 
 getTemplate ()
 Get the resource template. More...
 
 getFormats ()
 Get the resource media types. More...
 
 getMethods ()
 Get the resource methods. More...
 
 getService ()
 Get the resource's service. More...
 
 checkTool (array $scopes=array())
 Check to make sure the request is valid. More...
 

Static Public Member Functions

static validate_iso8601_date (string $date)
 

Protected Member Functions

 checkScore (string $requestData, int $objId)
 
- Protected Member Functions inherited from ilLTIConsumerResourceBase
 parseTemplate ()
 Parse the template for variables. More...
 

Static Protected Member Functions

static getUsrIdForObjectAndUsrIdent (int $objId, string $userIdent)
 

Additional Inherited Members

- Data Fields inherited from ilLTIConsumerResourceBase
const HTTP_POST = 'POST'
 HTTP Post method. More...
 
const HTTP_GET = 'GET'
 HTTP Get method. More...
 
const HTTP_PUT = 'PUT'
 HTTP Put method. More...
 
const HTTP_DELETE = 'DELETE'
 HTTP Delete method. More...
 
- Protected Attributes inherited from ilLTIConsumerResourceBase
string $type
 Type for this resource. More...
 
string $id
 ID for this resource. More...
 
string $template
 Template for this resource. More...
 
array $variables
 Custom parameter substitution variables associated with this resource. More...
 
array $formats
 Media types supported by this resource. More...
 
array $methods
 HTTP actions supported by this resource. More...
 
array $params
 Template variables parsed from the resource template. More...
 

Detailed Description

Definition at line 30 of file class.ilLTIConsumerGradeServiceScores.php.

Constructor & Destructor Documentation

◆ __construct()

ilLTIConsumerGradeServiceScores::__construct ( ilLTIConsumerServiceBase  $service)

Definition at line 32 of file class.ilLTIConsumerGradeServiceScores.php.

References ILIAS\GlobalScreen\Provider\__construct().

33  {
34  parent::__construct($service);
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  }
__construct(Container $dic, ilPlugin $plugin)
+ Here is the call graph for this function:

Member Function Documentation

◆ checkScore()

ilLTIConsumerGradeServiceScores::checkScore ( string  $requestData,
int  $objId 
)
protected

Definition at line 82 of file class.ilLTIConsumerGradeServiceScores.php.

References $DIC, ilLTIConsumerResult\getByKeys(), ilObjLTIConsumer\getLogger(), ILIAS\Repository\int(), ilLPStatus\LP_STATUS_COMPLETED_NUM, ilLPStatus\LP_STATUS_IN_PROGRESS_NUM, and ilLPStatus\writeStatus().

Referenced by execute().

82  : int
83  {
84  global $DIC; /* @var \ILIAS\DI\Container $DIC */
85  $score = json_decode($requestData);
86  //prüfe Userid
87  $userId = self::getUsrIdForObjectAndUsrIdent($objId, $score->userId);
88  if ($userId == null) {
89  ilObjLTIConsumer::getLogger()->debug('User not available');
90  throw new Exception('User not available', 404);
91  return 404;
92  }
93 
94  if (empty($score) ||
95  !isset($score->userId) ||
96  !isset($score->gradingProgress) ||
97  !isset($score->activityProgress) ||
98  !isset($score->timestamp) ||
99  isset($score->timestamp) && !self::validate_iso8601_date($score->timestamp) ||
100  (isset($score->scoreGiven) && !is_numeric($score->scoreGiven)) ||
101  (isset($score->scoreGiven) && !isset($score->scoreMaximum)) ||
102  (isset($score->scoreMaximum) && !is_numeric($score->scoreMaximum))
103  ) {
104  ilObjLTIConsumer::getLogger()->debug('Incorrect score received');
105  ilObjLTIConsumer::getLogger()->dump($score);
106  throw new Exception('Incorrect score received', 400);
107  return 400;
108  }
109  //Achtung Ggfs. Timestamp prüfen falls schon was ankam
110  if (!isset($score->scoreMaximum)) {
111  $score->scoreMaximum = 1;
112  }
113  if (isset($score->scoreGiven)) {
114  if ($score->gradingProgress != 'FullyGraded') {
115  $score->scoreGiven = null;
116  }
117  }
118  $result = (float)$score->scoreGiven / (float)$score->scoreMaximum;
119  ilObjLTIConsumer::getLogger()->debug("result: " . $result);
120 
121  $ltiObjRes = new ilLTIConsumerResultService();
122 
123  $ltiObjRes->readProperties($objId);
124  // check the object status
125  if (!$ltiObjRes->isAvailable()) {
126  throw new Exception('Tool for Object not available', 404);
127  return 404;
128  }
129 
130  if ($result >= $ltiObjRes->getMasteryScore()) {
132  } else {
134  }
135  $lp_percentage = (int) round(100 * $result);
136 
137  $consRes = ilLTIConsumerResult::getByKeys($objId, $userId, false);
138  if (empty($consRes)) {
139  ilObjLTIConsumer::getLogger()->debug("lti_consumer_results_id not found!");
140  // throw new Exception('lti_consumer_results_id not found!', 404);
141  // return 404;
142  }
143  if (!isset($consRes->id)) {
144  $consRes->id = $DIC->database()->nextId('lti_consumer_results');
145  }
146  $DIC->database()->replace(
147  'lti_consumer_results',
148  array(
149  'id' => array('integer', $consRes->id)
150  ),
151  array(
152  'obj_id' => array('integer', $objId),
153  'usr_id' => array('integer', $userId),
154  'result' => array('float', $result)
155  )
156  );
157 
158  ilLPStatus::writeStatus($objId, $userId, $lp_status, $lp_percentage, true);
159 
160  $ltiTimestamp = DateTimeImmutable::createFromFormat(DateTimeInterface::RFC3339_EXTENDED, $score->timestamp);
161  if (!$ltiTimestamp) { //moodle 4
162  $ltiTimestamp = DateTimeImmutable::createFromFormat(DateTimeInterface::ISO8601, $score->timestamp);
163  }
164  if (!$ltiTimestamp) { //for example nothing
165  $ltiTimestamp = new DateTime('now');
166  }
167  $gradeValues = [
168  'id' => array('integer', $DIC->database()->nextId('lti_consumer_grades')),
169  'obj_id' => array('integer', $objId),
170  'usr_id' => array('integer', $userId),
171  'score_given' => array('float', $score->scoreGiven),
172  'score_maximum' => array('float', $score->scoreMaximum),
173  'activity_progress' => array('text', $score->activityProgress),
174  'grading_progress' => array('text', $score->gradingProgress),
175  'lti_timestamp' => array('timestamp',$ltiTimestamp->format("Y-m-d H:i:s")),
176  'stored' => array('timestamp', date("Y-m-d H:i:s"))
177  ];
178  $DIC->database()->insert('lti_consumer_grades', $gradeValues);
179 
180 
181 
182  return 200;
183  }
const LP_STATUS_COMPLETED_NUM
const LP_STATUS_IN_PROGRESS_NUM
$objId
Definition: xapitoken.php:57
global $DIC
Definition: feed.php:28
static getByKeys(int $a_obj_id, int $a_usr_id, ?bool $a_create=false)
Get a result by object and user key.
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.
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ execute()

ilLTIConsumerGradeServiceScores::execute ( ilLTIConsumerServiceResponse  $response)

Execute the request for this resource.

Definition at line 46 of file class.ilLTIConsumerGradeServiceScores.php.

References Vendor\Package\$e, ilLTIConsumerResourceBase\$params, $scope, $token, checkScore(), ilLTIConsumerResourceBase\checkTool(), ilObjLTIConsumer\getLogger(), ilLTIConsumerServiceResponse\getRequestData(), ILIAS\Repository\int(), ilLTIConsumerResourceBase\parseTemplate(), ilLTIConsumerGradeService\SCOPE_GRADESERVICE_SCORE, ilLTIConsumerServiceResponse\setCode(), and ilLTIConsumerServiceResponse\setReason().

46  : void
47  {
48  $params = $this->parseTemplate();
49  $contextId = $params['context_id'];
50  $itemId = $params['item_id'];
51 
52  ilObjLTIConsumer::getLogger()->debug("contextId: " . $contextId);
53  ilObjLTIConsumer::getLogger()->debug("objId: " . $itemId);
54  ilObjLTIConsumer::getLogger()->debug("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  }
array $params
Template variables parsed from the resource template.
$scope
Definition: ltiregstart.php:53
parseTemplate()
Parse the template for variables.
const SCOPE_GRADESERVICE_SCORE
Scope for access to Score service.
checkTool(array $scopes=array())
Check to make sure the request is valid.
$token
Definition: xapitoken.php:70
setCode(int $code)
Set the response code.
setReason(string $reason)
Set the response reason.
+ Here is the call graph for this function:

◆ getUsrIdForObjectAndUsrIdent()

static ilLTIConsumerGradeServiceScores::getUsrIdForObjectAndUsrIdent ( int  $objId,
string  $userIdent 
)
staticprotected

Definition at line 196 of file class.ilLTIConsumerGradeServiceScores.php.

References $DIC, $query, $res, and ILIAS\Repository\int().

196  : ?int
197  {
198  global $DIC; /* @var \ILIAS\DI\Container $DIC */
199  $atExist = strpos($userIdent, '@');
200 
201  $query = "SELECT usr_id FROM cmix_users WHERE obj_id = " . $DIC->database()->quote($objId, 'integer');
202 
203  if ($atExist > 1) {
204  $query .= " AND usr_ident = " . $DIC->database()->quote($userIdent, 'text');
205  } else { //LTI 1.1
206  $query .= " AND" . $DIC->database()->like('usr_ident', 'text', $userIdent . '@%');
207  }
208  $res = $DIC->database()->query($query);
209 
210  $usrId = null;
211  while ($row = $DIC->database()->fetchAssoc($res)) {
212  $usrId = (int) $row['usr_id'];
213  }
214 
215  return $usrId;
216  }
$res
Definition: ltiservices.php:69
$objId
Definition: xapitoken.php:57
global $DIC
Definition: feed.php:28
$query
+ Here is the call graph for this function:

◆ validate_iso8601_date()

static ilLTIConsumerGradeServiceScores::validate_iso8601_date ( string  $date)
static

Definition at line 185 of file class.ilLTIConsumerGradeServiceScores.php.

185  : bool
186  {
187  if (preg_match('/^([\+-]?\d{4}(?!\d{2}\b))((-?)((0[1-9]|1[0-2])' .
188  '(\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])))' .
189  '([T\s]((([01]\d|2[0-3])((:?)[0-5]\d)?|24\:?00)([\.,]\d+(?!:))?)?(\17[0-5]\d([\.,]\d+)?)' .
190  '?([zZ]|([\+-])([01]\d|2[0-3]):?([0-5]\d)?)?)?)?$/', $date) > 0) {
191  return true;
192  }
193  return false;
194  }

The documentation for this class was generated from the following file: