ILIAS  trunk Revision v12.0_alpha-377-g3641b37b9db
XapiProxyPolyFill.php
Go to the documentation of this file.
1<?php
2
19declare(strict_types=1);
20
21namespace XapiProxy;
22
24{
25 protected string $client;
26 protected string $token;
27
28 protected bool $plugin = false;
29 protected string $table_prefix;
30 protected ?\ilCmiXapiLrsType $lrsType = null;
31 protected ?\ilCmiXapiAuthToken $authToken = null;
32 protected ?int $objId = null;
33 protected ?array $specificAllowedStatements = null;
34 protected ?array $replacedValues = null;
35 protected bool $blockSubStatements = false;
36 protected array $cmdParts = [];
37 protected string $method;
38
39 protected string $defaultLrsEndpoint = '';
40 protected string $defaultLrsKey = '';
41 protected string $defaultLrsSecret = '';
42
43 protected string $fallbackLrsEndpoint = '';
44 protected string $fallbackLrsKey = '';
45 protected string $fallbackLrsSecret = '';
46
47 public const PARTS_REG = '/^(.*?xapiproxy\.php)(\/([^?]+)?\??(.*))/'; //'/^(.*?xapiproxy\.php)(\/([^\?]+)?\??.*)/';
48
49 protected array $sniffVerbs = array(
50 "http://adlnet.gov/expapi/verbs/completed" => "completed",
51 "http://adlnet.gov/expapi/verbs/passed" => "passed",
52 "http://adlnet.gov/expapi/verbs/failed" => "failed",
53 "http://adlnet.gov/expapi/verbs/satisfied" => "passed"
54 );
55
56 public const TERMINATED_VERB = "http://adlnet.gov/expapi/verbs/terminated";
57
58 public function __construct(string $client, string $token, ?bool $plugin = false)
59 {
60 $this->client = $client;
61 $this->token = $token;
62 $this->plugin = $plugin;
63 $this->table_prefix = $this->plugin ? "xxcf" : "cmix";
64 preg_match(self::PARTS_REG, (string) $GLOBALS['DIC']->http()->request()->getUri(), $this->cmdParts);
65 $this->method = strtolower($GLOBALS['DIC']->http()->request()->getMethod());
66 }
67
68 public function log(): \ilLogger
69 {
70 return \ilLoggerFactory::getLogger('cmix');
71 }
72
73 public function msg(string $msg): string
74 {
75 if ($this->plugin) {
76 return "XapiCmi5Plugin: " . $msg;
77 } else {
78 return $msg;
79 }
80 }
81
82 public function initLrs(): void
83 {
84 $this->log()->debug($this->msg('initLrs'));
85 try {
86 $authToken = \ilCmiXapiAuthToken::getInstanceByToken($this->token);
87 } catch (\ilCmiXapiException $e) {
88 $this->log()->error($this->msg($e->getMessage()));
89 header('HTTP/1.1 401 Unauthorized');
90 header('Access-Control-Allow-Origin: ' . $_SERVER["HTTP_ORIGIN"]);
91 header('Access-Control-Allow-Credentials: true');
92 exit;
93 }
94
95 $this->authToken = $authToken;
96 $this->getLrsType();
97 }
98
102 private function getLrsType(): \ilCmiXapiLrsType
103 { // Core new > 6
104 try {
105 $lrsType = $this->getLrsTypeAndMoreByToken();
106 $this->defaultLrsEndpoint = $lrsType->getLrsEndpoint();
107 $this->defaultLrsKey = $lrsType->getLrsKey();
108 $this->defaultLrsSecret = $lrsType->getLrsSecret();
109 $this->lrsType = $lrsType;
110 // one query IS better :-)
111 // $lrsType = new ilCmiXapiLrsType($authToken->getLrsTypeId());
112 $objId = $this->authToken->getObjId();
113 $this->objId = $objId;
114 if (!$lrsType->isAvailable()) {
115 throw new \ilCmiXapiException(
116 'lrs endpoint (id=' . $this->authToken->getLrsTypeId() . ') unavailable (responded 401-unauthorized)'
117 );
118 }
119 } catch (\ilCmiXapiException $e) {
120 $this->log()->error($this->msg($e->getMessage()));
121 header('Access-Control-Allow-Origin: ' . $_SERVER["HTTP_ORIGIN"]);
122 header('Access-Control-Allow-Credentials: true');
123 header('HTTP/1.1 401 Unauthorized');
124 exit;
125 }
126 \ilCmiXapiUser::saveProxySuccess($this->authToken->getObjId(), $this->authToken->getUsrId(), $this->lrsType->getPrivacyIdent());
127 return $lrsType;
128 }
129
135 {
136 $type_id = null;
137 $lrs = null;
138 $db = $GLOBALS['DIC']->database();
139 $query = "SELECT {$this->table_prefix}_settings.lrs_type_id,
140 {$this->table_prefix}_settings.only_moveon,
141 {$this->table_prefix}_settings.achieved,
142 {$this->table_prefix}_settings.answered,
143 {$this->table_prefix}_settings.completed,
144 {$this->table_prefix}_settings.failed,
145 {$this->table_prefix}_settings.initialized,
146 {$this->table_prefix}_settings.passed,
147 {$this->table_prefix}_settings.progressed,
148 {$this->table_prefix}_settings.satisfied,
149 {$this->table_prefix}_settings.c_terminated,
150 {$this->table_prefix}_settings.hide_data,
151 {$this->table_prefix}_settings.c_timestamp,
152 {$this->table_prefix}_settings.duration,
153 {$this->table_prefix}_settings.no_substatements,
154 {$this->table_prefix}_settings.privacy_ident
155 FROM {$this->table_prefix}_settings, {$this->table_prefix}_token
156 WHERE {$this->table_prefix}_settings.obj_id = {$this->table_prefix}_token.obj_id AND {$this->table_prefix}_token.token = " . $db->quote($this->token, 'text');
157
158 $res = $db->query($query);
159 while ($row = $db->fetchObject($res)) {
160 $type_id = (int) $row->lrs_type_id;
161 if ($type_id) {
162 // $lrs = ($this->plugin) ? new \ilXapiCmi5LrsType($type_id) : new \ilCmiXapiLrsType($type_id);
163 $lrs = new \ilCmiXapiLrsType($type_id);
164 }
165
166 $sarr = [];
167 if ((bool) $row->only_moveon) {
168 if ((bool) $row->achieved) {
169 $sarr[] = "https://w3id.org/xapi/dod-isd/verbs/achieved";
170 }
171 if ((bool) $row->answered) {
172 $sarr[] = "http://adlnet.gov/expapi/verbs/answered";
173 $sarr[] = "https://w3id.org/xapi/dod-isd/verbs/answered";
174 }
175 if ((bool) $row->completed) {
176 $sarr[] = "http://adlnet.gov/expapi/verbs/completed";
177 $sarr[] = "https://w3id.org/xapi/dod-isd/verbs/completed";
178 }
179 if ((bool) $row->failed) {
180 $sarr[] = "http://adlnet.gov/expapi/verbs/failed";
181 }
182 if ((bool) $row->initialized) {
183 $sarr[] = "http://adlnet.gov/expapi/verbs/initialized";
184 $sarr[] = "https://w3id.org/xapi/dod-isd/verbs/initialized";
185 }
186 if ((bool) $row->passed) {
187 $sarr[] = "http://adlnet.gov/expapi/verbs/passed";
188 }
189 if ((bool) $row->progressed) {
190 $sarr[] = "http://adlnet.gov/expapi/verbs/progressed";
191 }
192 if ((bool) $row->satisfied) {
193 $sarr[] = "https://w3id.org/xapi/adl/verbs/satisfied";
194 }
195 if ((bool) $row->c_terminated) {
196 $sarr[] = "http://adlnet.gov/expapi/verbs/terminated";
197 }
198 if (count($sarr) > 0) {
199 $this->specificAllowedStatements = $sarr;
200 $this->log()->debug($this->msg('getSpecificAllowedStatements: ' . var_export($this->specificAllowedStatements, true)));
201 }
202 }
203 if ((bool) $row->hide_data) {
204 $rarr = array();
205 if ((bool) $row->c_timestamp) {
206 $rarr['timestamp'] = '1970-01-01T00:00:00.000Z';
207 }
208 if ((bool) $row->duration) {
209 $rarr['result.duration'] = 'PT00.000S';
210 }
211 if (count($rarr) > 0) {
212 $this->replacedValues = $rarr;
213 $this->log()->debug($this->msg('getReplacedValues: ' . var_export($this->replacedValues, true)));
214 }
215 }
216 if ((bool) $row->no_substatements) {
217 $this->blockSubStatements = true;
218 $this->log()->debug($this->msg('getBlockSubStatements: ' . $this->blockSubStatements));
219 }
220 $lrs->setPrivacyIdent((int) $row->privacy_ident);
221 }
222 return $lrs;
223 }
224}
__construct(string $client, string $token, ?bool $plugin=false)
getLrsTypeAndMoreByToken()
hybrid function, maybe two distinct functions would be better?
static getInstanceByToken(string $token)
static saveProxySuccess(int $objId, int $usrId, int $privacyIdent)
Component logger with individual log levels by component id.
exit
$client
$res
Definition: ltiservices.php:69
static http()
Fetches the global http state from ILIAS.
$_SERVER['HTTP_HOST']
Definition: raiseError.php:26
if(!file_exists('../ilias.ini.php'))
$GLOBALS["DIC"]
Definition: wac.php:54
$token
Definition: xapitoken.php:67
$objId
Definition: xapitoken.php:55