36 public bool $ok =
true;
88 public ?
string $kid = null;
95 public ?
string $jku = null;
255 if (array_key_exists($name, $this->
settings)) {
272 if ($value !== $old_value) {
273 if (!empty($value)) {
278 $this->settingsChanged =
true;
308 if ($this->settingsChanged) {
324 return !empty($this->jwt) && $this->jwt->hasJwt();
344 if (is_null($this->rawParameters)) {
345 $this->rawParameters = LTIOAuth\OAuthUtil::parse_parameters(file_get_contents(LTIOAuth\OAuthRequest::$POST_INPUT));
359 $messageClaims = null;
360 if (!is_null($this->messageParameters)) {
363 if (!empty($messageParameters[
'lti_message_type'])) {
367 $messageType = $messageParameters[
'lti_message_type'];
369 if (!empty($messageParameters[
'accept_media_types'])) {
370 $mediaTypes = array_filter(explode(
',', str_replace(
' ',
'', $messageParameters[
'accept_media_types'])),
'strlen');
372 if (!empty($messageParameters[
'accept_types'])) {
373 $types = array_filter(explode(
',', str_replace(
' ',
'', $messageParameters[
'accept_types'])),
'strlen');
374 foreach ($mediaTypes as $mediaType) {
375 if (strpos($mediaType,
'application/vnd.ims.lti.') === 0) {
376 unset($mediaTypes[array_search($mediaType, $mediaTypes)]);
379 $messageParameters[
'accept_media_types'] = implode(
',', $mediaTypes);
381 foreach ($mediaTypes as $mediaType) {
384 $messageParameters[
'accept_media_types'] = implode(
',', $mediaTypes);
388 $messageParameters[
'accept_media_types'] = implode(
',', $mediaTypes);
390 } elseif (substr($mediaType, 0, 6) ===
'image/') {
394 } elseif ($mediaType ===
'text/html') {
398 } elseif ($mediaType ===
'*/*') {
407 $types = array_unique($types);
408 $messageParameters[
'accept_types'] = implode(
',', $types);
411 if (!empty($messageParameters[
'accept_presentation_document_targets'])) {
412 $documentTargets = array_filter(explode(
414 str_replace(
' ',
'', $messageParameters[
'accept_presentation_document_targets'])
417 foreach ($documentTargets as $documentTarget) {
418 switch ($documentTarget) {
425 $targets[] = $documentTarget;
429 $targets = array_unique($targets);
430 $messageParameters[
'accept_presentation_document_targets'] = implode(
',', $targets);
432 $messageClaims = array();
433 if (!empty($messageParameters[
'oauth_consumer_key'])) {
434 $messageClaims[
'aud'] = array($messageParameters[
'oauth_consumer_key']);
436 foreach ($messageParameters as $key => $value) {
444 if (isset($mapping[
'isObject']) && $mapping[
'isObject']) {
445 $value = json_decode($value);
446 } elseif (isset($mapping[
'isArray']) && $mapping[
'isArray']) {
447 $value = array_filter(explode(
',', str_replace(
' ',
'', $value)),
'strlen');
449 } elseif (isset($mapping[
'isBoolean']) && $mapping[
'isBoolean']) {
450 $value = (is_bool($value)) ? $value : $value ===
'true';
451 } elseif (isset($mapping[
'isInteger']) && $mapping[
'isInteger']) {
452 $value = intval($value);
453 } elseif (is_bool($value)) {
454 $value = ($value) ?
'true' :
'false';
456 $value = strval($value);
460 if (!empty($mapping[
'suffix'])) {
461 $claim .=
"-{$mapping['suffix']}";
464 if (is_null($mapping[
'group'])) {
465 $claim = $mapping[
'claim'];
466 } elseif (empty($mapping[
'group'])) {
467 $claim .= $mapping[
'claim'];
469 $group = $claim . $mapping[
'group'];
470 $claim = $mapping[
'claim'];
472 } elseif (substr($key, 0, 7) ===
'custom_') {
474 $claim = substr($key, 7);
475 } elseif (substr($key, 0, 4) ===
'ext_') {
477 $claim = substr($key, 4);
478 } elseif (substr($key, 0, 7) ===
'lti1p1_') {
480 $claim = substr($key, 7);
484 $json = json_decode($value);
485 if (!is_null($json)) {
493 if ($fullyQualified) {
499 } elseif (empty($group)) {
500 $messageClaims[$claim] = $value;
502 $messageClaims[$group][$claim] = $value;
506 if (!empty($messageParameters[
'unmapped_claims'])) {
507 $claims = json_decode($messageParameters[
'unmapped_claims']);
508 foreach (
$claims as $claim => $value) {
509 if ($fullyQualified) {
511 } elseif (!is_object($value)) {
512 $messageClaims[$claim] = $value;
513 } elseif (!isset($messageClaims[$claim])) {
514 $messageClaims[$claim] = $value;
516 $objVars = get_object_vars($value);
517 foreach ($objVars as $attrName => $attrValue) {
518 if (is_object($messageClaims[$claim])) {
519 $messageClaims[$claim]->{$attrName} = $attrValue;
521 $messageClaims[$claim][$attrName] = $attrValue;
529 return $messageClaims;
540 if (!is_array($roles)) {
541 $roles = array_filter(explode(
',', str_replace(
' ',
'', $roles)),
'strlen');
543 $parsedRoles = array();
544 foreach ($roles as $role) {
548 if ((substr($role, 0, 4) !==
'urn:') &&
549 (substr($role, 0, 7) !==
'http://') && (substr($role, 0, 8) !==
'https://')) {
550 $role =
'urn:lti:role:ims/lis/' . $role;
552 } elseif ((substr($role, 0, 7) !==
'http://') && (substr($role, 0, 8) !==
'https://')) {
553 $role =
'http://purl.imsglobal.org/vocab/lis/v2/membership#' . $role;
555 $parsedRoles[] = $role;
575 $params[
'lti_message_type'] =
$type;
577 $params = $this->
addSignature($url, $params,
'POST',
'application/x-www-form-urlencoded');
598 if (!isset($loginHint) || (strlen($loginHint) <= 0)) {
599 if (isset($params[
'user_id']) && (strlen($params[
'user_id']) > 0)) {
600 $loginHint = $params[
'user_id'];
602 $loginHint =
'Anonymous';
607 $params[
'lti_message_type'] =
$type;
611 'iss' => $this->platformId,
612 'target_link_uri' => $url,
613 'login_hint' => $loginHint
618 if (!empty($this->clientId)) {
621 if (!empty($this->deploymentId)) {
622 $params[
'lti_deployment_id'] = $this->deploymentId;
644 public function sendMessage(
string $url,
string $type, array $messageParams,
string $target =
'', ?
string $userId = null,
string $hint =
''): string
646 $sendParams = $this->
signMessage($url, $type, $this->ltiVersion, $messageParams, $userId, $hint);
683 $http =
new HttpMessage($service->endpoint, $method,
$data, $header);
685 if (
$http->send() && !empty(
$http->response)) {
686 $http->responseJson = json_decode(
$http->response);
700 return empty($this->signatureMethod) || (substr($this->signatureMethod, 0, 2) !==
'RS');
730 $this->ok =
$_SERVER[
'REQUEST_METHOD'] ===
'POST';
732 $this->reason =
'LTI messages must use HTTP POST';
733 } elseif (!empty($this->jwt) && !empty($this->jwt->hasJwt())) {
735 if (is_null($this->messageParameters[
'oauth_consumer_key']) || (strlen($this->messageParameters[
'oauth_consumer_key']) <= 0)) {
736 $this->reason =
'Missing iss claim';
737 } elseif (empty($this->jwt->getClaim(
'iat',
''))) {
738 $this->reason =
'Missing iat claim';
739 } elseif (empty($this->jwt->getClaim(
'exp',
''))) {
740 $this->reason =
'Missing exp claim';
741 } elseif (intval($this->jwt->getClaim(
'iat')) > intval($this->jwt->getClaim(
'exp'))) {
742 $this->reason =
'iat claim must not have a value greater than exp claim';
743 } elseif (empty($this->jwt->getClaim(
'nonce',
''))) {
744 $this->reason =
'Missing nonce claim';
750 if (isset($this->messageParameters[
'oauth_signature_method'])) {
751 $this->signatureMethod = $this->messageParameters[
'oauth_signature_method'];
752 if (($this instanceof
Tool) && !empty($this->platform)) {
758 $this->ok = isset($this->messageParameters[
'lti_message_type']);
760 $this->reason =
'Missing lti_message_type parameter.';
764 $this->ok = isset($this->messageParameters[
'lti_version']) && in_array(
765 $this->messageParameters[
'lti_version'],
769 $this->reason =
'Invalid or missing lti_version parameter.';
788 } elseif (($this instanceof
Tool) && !empty($this->platform)) {
789 $key = $this->platform->getKey();
790 $secret = $this->platform->secret;
795 if ($this instanceof Tool) {
796 $platform = $this->platform;
797 $publicKey = $this->platform->rsaKey;
798 $jku = $this->platform->jku;
810 if (empty($this->jwt) || empty($this->jwt->hasJwt())) {
826 $server->add_signature_method($method);
828 $request = LTIOAuth\OAuthRequest::from_request();
829 $server->verify_request($request);
832 if (empty($this->reason)) {
834 $oauthConsumer =
new LTIOAuth\OAuthConsumer(
$key,
$secret);
836 $signature = $request->build_signature($method, $oauthConsumer, null);
837 if ($this->debugMode) {
838 $this->reason = $e->getMessage();
840 if (empty($this->reason)) {
841 $this->reason =
'OAuth signature check failed - perhaps an incorrect secret or timestamp.';
843 $this->details[] =
"Shared secret: '{$secret}'";
844 $this->details[] =
'Current timestamp: ' . time();
845 $this->details[] =
"Expected signature: {$signature}";
846 $this->details[] =
"Base string: {$request->base_string}";
850 $nonce =
new PlatformNonce($platform, $this->jwt->getClaim(
'nonce'));
852 $ok = !$nonce->
load();
854 $ok = $nonce->
save();
857 $this->reason =
'Invalid nonce.';
859 if (empty($publicKey)) {
865 $ok = $this->jwt->verify($publicKey,
$jku);
867 $this->reason =
'JWT signature check failed - perhaps an invalid public key or timestamp';
871 $this->reason =
'Unable to verify JWT signature as neither a public key nor a JSON Web Key URL is specified';
887 if (is_null($this->messageParameters)) {
889 if (isset($this->rawParameters[
'id_token']) || isset($this->rawParameters[
'JWT'])) {
892 if (isset($this->rawParameters[
'id_token'])) {
893 $this->ok = $this->jwt->load($this->rawParameters[
'id_token'], $this->rsaKey);
895 $this->ok = $this->jwt->load($this->rawParameters[
'JWT'], $this->rsaKey);
898 $this->reason =
'Message does not contain a valid JWT';
900 $this->ok = $this->jwt->hasClaim(
'iss') && $this->jwt->hasClaim(
'aud') &&
903 $iss = $this->jwt->getClaim(
'iss');
904 $aud = $this->jwt->getClaim(
'aud');
906 $this->ok = !empty($iss) && !empty($aud) && !empty($deploymentId);
908 $this->reason =
'iss, aud and/or deployment_id claim is empty';
909 } elseif (is_array($aud)) {
910 if ($this->jwt->hasClaim(
'azp')) {
911 $this->ok = !empty($this->jwt->getClaim(
'azp'));
913 $this->reason =
'azp claim is empty';
915 $this->ok = in_array($this->jwt->getClaim(
'azp'), $aud);
917 $aud = $this->jwt->getClaim(
'azp');
919 $this->reason =
'azp claim value is not included in aud claim';
924 $this->ok = !empty($aud);
926 $this->reason =
'First element of aud claim is empty';
929 } elseif ($this->jwt->hasClaim(
'azp')) {
930 $this->ok = $this->jwt->getClaim(
'azp') === $aud;
932 $this->reason =
'aud claim does not match the azp claim';
936 if ($this instanceof
Tool) {
938 $this->platform->platformId = $iss;
939 if (isset($this->rawParameters[
'id_token'])) {
940 $this->ok = !empty($this->rawParameters[
'state']);
942 $nonce =
new PlatformNonce($this->platform, $this->rawParameters[
'state']);
943 $this->ok = $nonce->load();
946 $nonce =
new PlatformNonce($platform, $this->rawParameters[
'state']);
947 $this->ok = $nonce->load();
951 $nonce =
new PlatformNonce($platform, $this->rawParameters[
'state']);
952 $this->ok = $nonce->load();
955 $this->ok = $nonce->delete();
961 $this->messageParameters = array();
962 $this->messageParameters[
'oauth_consumer_key'] = $aud;
963 $this->messageParameters[
'oauth_signature_method'] = $this->jwt->getHeader(
'alg');
966 $this->reason =
'state parameter is invalid or missing';
970 $this->reason =
'iss, aud and/or deployment_id claim not found';
975 $this->reason =
'Message does not contain a valid JWT';
977 } elseif (isset($this->rawParameters[
'error'])) {
979 $this->reason = $this->rawParameters[
'error'];
980 if (!empty($this->rawParameters[
'error_description'])) {
981 $this->reason .=
": {$this->rawParameters['error_description']}";
984 if (isset($this->rawParameters[
'oauth_consumer_key']) && ($this instanceof
Tool)) {
1003 if (!empty($mapping[
'suffix'])) {
1004 $claim .=
"-{$mapping['suffix']}";
1006 $claim .=
'/claim/';
1007 if (is_null($mapping[
'group'])) {
1008 $claim = $mapping[
'claim'];
1009 } elseif (empty($mapping[
'group'])) {
1010 $claim .= $mapping[
'claim'];
1012 $claim .= $mapping[
'group'];
1014 if ($this->jwt->hasClaim($claim)) {
1016 if (empty($mapping[
'group'])) {
1018 $value = $this->jwt->getClaim($claim);
1020 $group = $this->jwt->getClaim($claim);
1021 if (is_array($group) && array_key_exists($mapping[
'claim'], $group)) {
1022 unset(
$payload->{$claim}[$mapping[
'claim']]);
1023 $value = $group[$mapping[
'claim']];
1024 } elseif (is_object($group) && isset($group->{$mapping[
'claim']})) {
1025 unset(
$payload->{$claim}->{$mapping[
'claim']});
1026 $value = $group->{$mapping[
'claim']};
1029 if (!is_null($value)) {
1030 if (isset($mapping[
'isArray']) && $mapping[
'isArray']) {
1031 if (!is_array($value)) {
1032 $errors[] =
"'{$claim}' claim must be an array";
1034 $value = implode(
',', $value);
1036 } elseif (isset($mapping[
'isObject']) && $mapping[
'isObject']) {
1037 $value = json_encode($value);
1038 } elseif (isset($mapping[
'isBoolean']) && $mapping[
'isBoolean']) {
1039 $value = $value ?
'true' :
'false';
1040 } elseif (isset($mapping[
'isInteger']) && $mapping[
'isInteger']) {
1041 $value = strval($value);
1044 if (!is_null($value) && is_string($value)) {
1045 $this->messageParameters[
$key] = $value;
1049 if (!empty($this->messageParameters[
'lti_message_type']) &&
1051 $this->messageParameters[
'lti_message_type'] = array_search(
1052 $this->messageParameters[
'lti_message_type'],
1056 if (!empty($this->messageParameters[
'accept_types'])) {
1057 $types = array_filter(explode(
',', str_replace(
' ',
'', $this->messageParameters[
'accept_types'])),
'strlen');
1058 $mediaTypes = array();
1059 if (!empty($this->messageParameters[
'accept_media_types'])) {
1060 $mediaTypes = array_filter(
1061 explode(
',', str_replace(
' ',
'', $this->messageParameters[
'accept_media_types'])),
1071 if (in_array(
'html', $types) && !in_array(
'*/*', $mediaTypes)) {
1072 $mediaTypes[] =
'text/html';
1074 if (in_array(
'image', $types) && !in_array(
'*/*', $mediaTypes)) {
1075 $mediaTypes[] =
'image/*';
1077 $mediaTypes = array_unique($mediaTypes);
1078 $this->messageParameters[
'accept_media_types'] = implode(
',', $mediaTypes);
1081 if ($this->jwt->hasClaim($claim)) {
1083 $custom = $this->jwt->getClaim($claim);
1084 if (!is_array($custom) && !is_object($custom)) {
1085 $errors[] =
"'{$claim}' claim must be an object";
1087 foreach ($custom as
$key => $value) {
1088 $this->messageParameters[
"custom_{$key}"] = $value;
1093 if ($this->jwt->hasClaim($claim)) {
1095 $ext = $this->jwt->getClaim($claim);
1096 if (!is_array($ext) && !is_object($ext)) {
1097 $errors[] =
"'{$claim}' claim must be an object";
1099 foreach ($ext as
$key => $value) {
1100 $this->messageParameters[
"ext_{$key}"] = $value;
1105 if ($this->jwt->hasClaim($claim)) {
1107 $lti1p1 = $this->jwt->getClaim($claim);
1108 if (!is_array($lti1p1) && !is_object($lti1p1)) {
1109 $errors[] =
"'{$claim}' claim must be an object";
1111 foreach ($lti1p1 as
$key => $value) {
1112 if (is_null($value)) {
1114 } elseif (is_object($value)) {
1115 $value = json_encode($value);
1117 $this->messageParameters[
"lti1p1_{$key}"] = $value;
1122 $objVars = get_object_vars(
$payload);
1123 foreach ($objVars as $attrName => $attrValue) {
1124 if (empty((array) $attrValue)) {
1128 $this->messageParameters[
'unmapped_claims'] = json_encode(
$payload);
1132 $this->reason =
'Invalid JWT: ' . implode(
', ',
$errors);
1143 if (array_key_exists($this->messageParameters[
'lti_message_type'],
Util::$METHOD_NAMES)) {
1146 $callback =
"on{$this->messageParameters['lti_message_type']}";
1148 if (method_exists($this, $callback)) {
1150 } elseif ($this->ok) {
1152 $this->reason =
"Message type not supported: {$this->messageParameters['lti_message_type']}";
1169 if (is_array(
$data)) {
1171 $params[
'oauth_callback'] =
'about:blank';
1174 $queryString = parse_url($endpoint, PHP_URL_QUERY);
1176 $queryParams = LTIOAuth\OAuthUtil::parse_parameters($queryString);
1179 if (!is_array(
$data)) {
1181 switch ($this->signatureMethod) {
1183 $hash = base64_encode(
hash(
'sha224',
$data,
true));
1186 $hash = base64_encode(
hash(
'sha256',
$data,
true));
1189 $hash = base64_encode(
hash(
'sha384',
$data,
true));
1192 $hash = base64_encode(
hash(
'sha512',
$data,
true));
1195 $hash = base64_encode(sha1(
$data,
true));
1199 $params[
'oauth_body_hash'] = $hash;
1201 if (!empty($timestamp)) {
1206 switch ($this->signatureMethod) {
1228 if (($this instanceof
Tool) && !empty($this->platform)) {
1229 $key = $this->platform->getKey();
1230 $secret = $this->platform->secret;
1237 $oauthConsumer =
new LTIOAuth\OAuthConsumer(
$key,
$secret, null);
1239 $oauthReq = LTIOAuth\OAuthRequest::from_consumer_and_token($oauthConsumer, null, $method, $endpoint,
$params);
1240 $oauthReq->sign_request($hmacMethod, $oauthConsumer, null);
1241 if (!is_array(
$data)) {
1242 $header = $oauthReq->to_header();
1244 if (!empty($type)) {
1245 $header .=
"\nAccept: {$type}";
1247 } elseif (isset($type)) {
1248 $header .=
"\nContent-Type: {$type}";
1249 $header .=
"\nContent-Length: " . strlen(
$data);
1254 $params = $oauthReq->get_parameters();
1255 foreach ($queryParams as
$key => $value) {
1256 if (!is_array($value)) {
1258 if (
$params[$key] === $value) {
1265 foreach ($value as $element) {
1272 if (is_array($value)) {
1273 if (count($value) <= 0) {
1275 } elseif (count($value) === 1) {
1297 if (is_array(
$data)) {
1299 if (empty($nonce)) {
1303 if (!array_key_exists(
'grant_type',
$data)) {
1304 $this->messageParameters =
$data;
1313 $payload[
'iss'] = $this->platformId;
1314 $payload[
'aud'] = array($this->clientId);
1318 $paramName =
'id_token';
1320 if (!empty($this->platform)) {
1321 $publicKey = $this->platform->rsaKey;
1322 $payload[
'iss'] = $this->platform->clientId;
1323 $payload[
'aud'] = array($this->platform->platformId);
1324 $payload[
'azp'] = $this->platform->platformId;
1331 $authorizationId =
'';
1332 if ($this instanceof
Tool) {
1334 if (!empty($this->platform)) {
1335 $sub = $this->platform->clientId;
1336 $authorizationId = $this->platform->authorizationServerId;
1337 $publicKey = $this->platform->rsaKey;
1353 if (empty($authorizationId)) {
1354 $authorizationId = $endpoint;
1356 $payload[
'aud'] = array($authorizationId);
1359 $paramName =
'client_assertion';
1363 if (empty($timestamp)) {
1364 $timestamp = time();
1370 $params[$paramName] = $jwt::sign(
1372 $this->signatureMethod,
1376 $this->encryptionMethod,
1386 if ($this instanceof
Tool) {
1387 $platform = $this->platform;
1391 $accessToken = $platform->getAccessToken();
1392 if (empty($accessToken)) {
1394 $platform->setAccessToken($accessToken);
1396 if (!$accessToken->hasScope()) {
1397 $accessToken->get();
1399 if (!empty($accessToken->token)) {
1400 $header =
"Authorization: Bearer {$accessToken->token}";
1402 if (empty(
$data) && ($method !==
'DELETE')) {
1403 if (!empty($type)) {
1404 $header .=
"\nAccept: {$type}";
1406 } elseif (isset($type)) {
1407 $header .=
"\nContent-Type: {$type}";
1409 $header .=
"\nContent-Length: " . strlen(
$data);
1427 if (is_object($value)) {
1428 foreach ($value as
$c => $v) {
getSetting(string $a_keyword, ?string $a_default_value=null)
read one value from settingstable
if(! $DIC->user() ->getId()||!ilLTIConsumerAccess::hasCustomProviderCreationAccess()) $params
if(count($parts) !=3) $payload
const LTI_ASSIGNMENT_MEDIA_TYPE
Media type for LTI assignment links.
OAuth PECL extension includes an OAuth Exception class, so we need to wrap the definition of this cla...
setSetting(string $a_key, string $a_val)
const TYPE_LTI_LINK
Type for LTI link content-item.
foreach($mandatory_scripts as $file) $timestamp
const TYPE_LTI_ASSIGNMENT
Type for LTI assignment content-item.
const LTI_LINK_MEDIA_TYPE
Media type for LTI launch links.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...