ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilLTIViewGUI.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
33 {
37  public const CHECK_HTTP_REFERER = true;
38 
43  private ?int $user = null;
44  private ?ilLogger $log = null;
45  private string $link_dir = "";
46 
47  private ?int $effectiveRefId = null;
48  private \ILIAS\HTTP\Wrapper\WrapperFactory $wrapper;
51 
55  public ?ilLanguage $lng = null;
56 
60  public function __construct()
61  {
62  global $DIC;
63  $this->dic = $DIC;
64  $this->log = ilLoggerFactory::getLogger('ltis');
65  $this->lng = $this->dic->language();
66  $this->lng->loadLanguageModule('lti');
67  $this->wrapper = $DIC->http()->wrapper();
68  $this->kindlyTo = $DIC->refinery()->kindlyTo();
69  $this->locator = $DIC['ilLocator'];
70  }
71 
75  public function init(): void
76  {
77  $this->link_dir = (defined("ILIAS_MODULE")) ? "../" : "";
78  if ($this->isLTIUser()) {
79  $context = $this->dic->globalScreen()->tool()->context();
80  $context->claim()->lti();
81  $this->initGUI();
82  }
83  }
84 
89  public static function getInstance()
90  {
91  global $DIC;
92  return $DIC["lti"];
93  }
94 
99  private function isLTIUser(): bool
100  {
101  if (!$this->dic->user() instanceof ilObjUser) {
102  return false;
103  }
104  if ($this->dic->user()->getAuthMode() == null) {
105  return false;
106  }
107  return (strpos($this->dic->user()->getAuthMode(), 'lti_') === 0);
108  }
109 
113  public function executeCommand(): void
114  {
115  global $ilCtrl;
116  $cmd = $ilCtrl->getCmd();
117  switch ($cmd) {
118  case 'exit':
119  $this->exitLti();
120  break;
121  }
122  }
123 
127  public function isActive(): bool
128  {
129  return $this->isLTIUser();
130  }
131 
135  public function initGUI(): void
136  {
137  $this->log->debug("initGUI");
138  $baseclass = '';
139  if ($this->wrapper->query()->has('baseClass')) {
140  $baseclass = strtolower($this->wrapper->query()->retrieve('baseClass', $this->kindlyTo->string()));
141  }
142  if ($this->wrapper->query()->has('cmdClass')) {
143  $cmdclass = strtolower($this->wrapper->query()->retrieve('cmdClass', $this->kindlyTo->string()));
144  }
145  if ($baseclass == 'illtiroutergui') {
146  return;
147  }
148  }
149 
153  protected function getContextId(): ?int
154  {
155  global $DIC;
156 
157  // forced lti_context_id for example request command in exitLTI
158  if ($this->wrapper->query()->has('lti_context_id') &&
159  $this->wrapper->query()->retrieve('lti_context_id', $this->kindlyTo->string()) !== '') {
160  $contextId = (int) $this->wrapper->query()->retrieve('lti_context_id', $this->kindlyTo->int());
161  $this->log->debug("find context_id by GET param: " . (string) $contextId);
162  return $contextId;
163  }
164 
165  $this->findEffectiveRefId();
167  // ???
168  if (empty($ref_id)) {
169  return 0;
170  }
171 
172  $this->log->debug("Effective ref_id: " . $ref_id);
173  // context_id = ref_id in request
174  if (ilSession::has('lti_' . $ref_id . '_post_data')) {
175  $this->log->debug("lti context session exists for " . $ref_id);
176  // return $ref_id;
177  }
178  // sub item request
179  $this->log->debug("ref_id not exists as context_id, walking tree backwards to find a valid context_id");
180  $locator_items = $this->locator->getItems();
181  if (is_array($locator_items) && count($locator_items) > 0) {
182  for ($i = count($locator_items) - 1;$i >= 0;$i--) {
183  if (ilSession::has('lti_' . $locator_items[$i]['ref_id'] . '_post_data')) {
184  $this->log->debug("found valid ref_id in locator: " . $locator_items[$i]['ref_id']);
185  return $locator_items[$i]['ref_id'];
186  }
187  }
188  }
189  $this->log->warning("no valid context_id found for ref_id request: " . $ref_id);
190 
193  $obj_type = ilObject::_lookupType($ref_id, true);
194  $context_id = 0;
195  $referer = 0;
196 
197  // first try to get real http referer
198  if (isset($_SERVER['HTTP_REFERER'])) {
199  $this->findEffectiveRefId($_SERVER['HTTP_REFERER']);
200  } else { // only fallback and not reliable on multiple browser LTi contexts
201  if (ilSession::has('referer_ref_id')) {
202  $this->effectiveRefId = ilSession::get('referer_ref_id');
203  }
204  }
205 
206  $referrer = (int) $this->effectiveRefId;
207 
208  if ($referer > 0) {
209  if (ilSession::has('lti_' . $referer . '_post_data')) {
210  $ref_id = $referer;
211  $context_id = $referer;
212  $obj_type = ilObject::_lookupType($ref_id, true);
213  $this->log->debug("referer obj_type: " . $obj_type);
214  } else {
215  $this->log->debug("search tree of referer...");
216  if ($this->dic->repositoryTree()->isInTree($referer)) {
217  $path = $this->dic->repositoryTree()->getPathId($referer);
218  for ($i = count($path) - 1;$i >= 0;$i--) {
219  if (ilSession::has('lti_' . $path[$i] . '_post_data')) {
220  // redirect to referer, because it is valid
221  $ref_id = $referer;
222  $context_id = $path[$i];
223  $obj_type = ilObject::_lookupType($ref_id, true);
224  break;
225  }
226  }
227  }
228  }
229  }
230  if ($ref_id > 0 && $obj_type != '') {
231  if (
232  (
233  $this->wrapper->query()->has('baseClass') &&
234  $this->wrapper->query()->retrieve('baseClass', $this->kindlyTo->string()) === 'ilDashboardGUI'
235  )
236  &&
237  (
238  $this->wrapper->query()->has('cmdClass') &&
239  $this->wrapper->query()->retrieve('cmdClass', $this->kindlyTo->string()) === 'ilpersonalprofilegui'
240  )
241  ) {
242  return $context_id;
243  }
244  // $this->dic->ui()->mainTemplate()->setOnScreenMessage('failure', $this->lng->txt('permission_denied'), true);
245  $redirect = $this->link_dir . "goto.php?target=" . $obj_type . "_" . $ref_id . "&lti_context_id=" . $context_id;
246  $this->log->debug("redirect: " . $redirect);
247  ilUtil::redirect($redirect);
248  }
249  }
250  $lti_context_ids = ilSession::get('lti_context_ids');
251  if (is_array($lti_context_ids) && count($lti_context_ids) > 0) {
252  if (count($lti_context_ids) == 1) {
253  $this->log->debug("using context_id from only LTI session");
254  return $lti_context_ids[0];
255  } else {
256  $this->log->warning("Multiple LTI sessions exists. The context_id can not be clearly detected");
257  }
258  }
259  return null;
260  }
261 
265  public function getPostData(): ?array
266  {
267  $context_id = $this->getContextId();
268  if ($context_id == 0) {
269  $this->log->warning("could not find any valid context_id!");
270  return null;
271  }
272  $post_data = ilSession::get('lti_' . $context_id . '_post_data');
273  if (!is_array($post_data)) {
274  $this->log->warning("no session post_data: " . "lti_" . $context_id . "_post_data");
275  return null;
276  }
277  return $post_data;
278  }
279 
283  public function getExternalCss(): string
284  {
285  $post_data = $this->getPostData();
286  if ($post_data !== null) {
287  return (isset($post_data['launch_presentation_css_url'])) ? $post_data['launch_presentation_css_url'] : '';
288  }
289  return '';
290  }
291 
295  public function getTitle(): string
296  {
297  $post_data = $this->getPostData();
298  if ($post_data !== null) {
299  return (isset($post_data['resource_link_title'])) ? "LTI - " . $post_data['resource_link_title'] : "LTI";
300  }
301  return "LTI";
302  }
303 
307  public function getTitleForExitPage(): string
308  {
309  return $this->lng->txt('lti_exited');
310  }
311 
315  public function getShortTitle(): string
316  {
317  return $this->lng->txt('lti_mode');
318  }
319 
324  public function exitLti(): void
325  {
326  $this->log->info("exitLTI");
327  $force_ilias_logout = false;
328  $context_id = $this->getContextId();
329  if ($context_id == 0) {
330  $this->log->warning("could not find any valid context_id!");
331  $force_ilias_logout = true;
332  }
333  $post_data = $this->getPostData();
334 
335  $return_url = '';
336  if (isset($post_data['launch_presentation_return_url'])) {
337  $return_url = $post_data['launch_presentation_return_url'];
338  }
339  $this->removeContextFromSession((string) $context_id);
340 
341  if (ilSession::has('lti_' . $context_id . '_post_data')) {
342  ilSession::clear('lti_' . $context_id . '_post_data');
343  $this->log->debug('unset SESSION["' . 'lti_' . $context_id . '_post_data"]');
344  }
345 
346  if (!isset($return_url) || $return_url === '') {
347  $cc = $this->dic->globalScreen()->tool()->context()->current();
348  $cc->addAdditionalData(LtiViewLayoutProvider::GS_EXIT_LTI, true);
349  $ui_factory = $this->dic->ui()->factory();
350  $renderer = $this->dic->ui()->renderer();
351  $content = [
352  $ui_factory->messageBox()->info($this->lng->txt('lti_exited_info'))
353  ];
354  $tpl = $this->dic["tpl"];
355  $tpl->setContent($renderer->render($content));
356  $this->logout($force_ilias_logout);
357  $tpl->printToStdout();
358  } else {
359  $this->logout($force_ilias_logout);
360  header('Location: ' . $return_url);
361  }
362  }
363 
367  public function logout(bool $force_ilias_logout = false): void
368  {
369  if ($force_ilias_logout) {
370  $this->log->warning("forcing logout ilias session, maybe a broken LTI context");
371  } else {
372  if (is_array(ilSession::get('lti_context_ids')) && count(ilSession::get('lti_context_ids')) > 0) {
373  $this->log->debug("there is another valid consumer session: ilias session logout refused.");
374  return;
375  }
376  }
377  $this->log->info("logout");
378  $this->dic->user()->setAuthMode((string) ilAuthUtils::AUTH_LOCAL);
379  //ilSession::setClosingContext(ilSession::SESSION_CLOSE_USER); // needed?
380  $auth = $this->dic['ilAuthSession'];
381  //$auth->logout(); // needed?
382  // $auth->setExpired($auth::SESSION_AUTH_EXPIRED, ilAuthStatus::STATUS_UNDEFINED);
383  $auth->setExpired(true);
384  session_destroy();
385  ilUtil::setCookie("ilClientId", "");
386  ilUtil::setCookie("PHPSESSID", "");
387  }
388 
394  public function getCmdLink(String $cmd): String
395  {
396  global $ilCtrl;
397  $lti_context_id = $this->getContextId();
398  $lti_context_id_param = ($lti_context_id != '') ? "&lti_context_id=" . $lti_context_id : '';
399  $targetScript = "";
400  return $this->link_dir . $targetScript . $this->dic->ctrl()->getLinkTargetByClass(array('illtiroutergui',strtolower(get_class($this))), $cmd) . "&baseClass=illtiroutergui" . $lti_context_id_param;
401  }
402 
403  private function getSessionValue(string $sess_key): string
404  {
405  if (ilSession::has($sess_key) && ilSession::get($sess_key) != '') {
406  return ilSession::get($sess_key);
407  } else {
408  return '';
409  }
410  }
411 
412  private function getCookieValue(string $cookie_key): string
413  {
414  if ($this->dic->wrapper->cookie()->has($cookie_key) && $this->dic->wrapper->cookie()->retrieve($cookie_key, $this->dic->refinery()->kindlyTo()->string() != '')) {
415  return $this->dic->wrapper->cookie()->retrieve($cookie_key, $this->dic->refinery()->kindlyTo()->string());
416  } else {
417  return '';
418  }
419  }
420 
421  private function removeContextFromSession(string $context_id): void
422  {
423  $lti_context_ids = ilSession::get('lti_context_ids');
424  if (is_array($lti_context_ids) && in_array($context_id, $lti_context_ids)) {
425  array_splice($lti_context_ids, array_search($context_id, $lti_context_ids), 1);
426  ilSession::set('lti_context_ids', $lti_context_ids);
427  }
428  }
429 
434  private function findEffectiveRefId(?string $url = null): void
435  {
436  $query = [];
437  if ($url === null) {
438  if ($this->wrapper->query()->has('ref_id')) {
439  $query['ref_id'] = $this->wrapper->query()->retrieve('ref_id', $this->kindlyTo->string());
440  }
441  if ($this->wrapper->query()->has('target')) {
442  $query['target'] = $this->wrapper->query()->retrieve('target', $this->kindlyTo->string());
443  }
444  } else {
445  parse_str((string) parse_url($url, PHP_URL_QUERY), $query);
446  }
447  if (isset($query['ref_id']) && (int) $query['ref_id']) {
448  $this->effectiveRefId = (int) $query['ref_id'];
449  return;
450  }
451  if (ilSession::has('lti_init_target') && ilSession::get('lti_init_target') != "") {
452  $target_arr = explode('_', ilSession::get('lti_init_target'));
453  ilSession::set('lti_init_target', "");
454  } else {
455  if (isset($query['target'])) {
456  $target_arr = explode('_', (string) $query['target']);
457  }
458  }
459  if (isset($target_arr[1]) and (int) $target_arr[1]) {
460  $this->effectiveRefId = (int) $target_arr[1];
461  }
462  }
463 }
static get(string $a_var)
$renderer
$context
Definition: webdav.php:31
logout(bool $force_ilias_logout=false)
logout ILIAS and destroys Session and ilClientId cookie if no consumer is still open in the LTI User ...
static getLogger(string $a_component_id)
Get component logger.
$url
Definition: shib_logout.php:66
removeContextFromSession(string $context_id)
$path
Definition: ltiservices.php:29
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
ILIAS HTTP Wrapper WrapperFactory $wrapper
getSessionValue(string $sess_key)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
isLTIUser()
get LTI Mode from Users->getAuthMode
$ref_id
Definition: ltiauth.php:65
init()
Init LTI mode for lti authenticated users.
static setCookie(string $a_cookie_name, string $a_cookie_value='', bool $a_also_set_super_global=true, bool $a_set_cookie_invalid=false)
$_SERVER['HTTP_HOST']
Definition: raiseError.php:26
exitLti()
exit LTI session and if defined redirecting to returnUrl ToDo: Standard Template with delos ...
global $DIC
Definition: shib_login.php:22
ILIAS DI Container $dic
private variables
static has($a_var)
getCookieValue(string $cookie_key)
static redirect(string $a_script)
const CHECK_HTTP_REFERER
contstants
ilLanguage $lng
public variables
ILIAS Refinery KindlyTo Group $kindlyTo
getCmdLink(String $cmd)
static getInstance()
for compatiblity with ilLTIRouterGUI
static _lookupType(int $id, bool $reference=false)
class for ILIAS ViewLTI
header()
expected output: > ILIAS shows the rendered Component.
Definition: header.php:29
static clear(string $a_var)
ilLocatorGUI $locator
static set(string $a_var, $a_val)
Set a value.
findEffectiveRefId(?string $url=null)
Find effective ref_id for request.