19 declare(strict_types=1);
77 $this->ref_id = $a_ref_id;
79 $this->
lng = $DIC->language();
80 $this->
lng->loadLanguageModule(
'auth');
83 $this->
tabs = $DIC->tabs();
84 $this->
ctrl = $DIC->ctrl();
85 $this->
logger = $DIC->logger()->auth();
86 $this->
access = $DIC->access();
87 $this->review = $DIC->rbac()->review();
88 $this->error = $DIC[
'ilErr'];
89 $this->
upload = $DIC->upload();
90 $this->body = $DIC->http()->request()->getParsedBody();
92 $http_wrapper = $DIC->http()->wrapper();
93 $this->
toolbar = $DIC->toolbar();
94 $refinery = $DIC->refinery();
95 $this->tpl = $DIC->ui()->mainTemplate();
96 $this->
http = $DIC->http();
97 $this->
ui = $DIC->ui()->factory();
98 $this->
renderer = $DIC->ui()->renderer();
100 $this->
factory = $DIC->ui()->factory();
101 $this->request = $DIC->http()->request();
104 if ($http_wrapper->query()->has(self::POST_VALUE) && $http_wrapper->query()->retrieve(
106 $refinery->kindlyTo()->int()
108 $this->mapping_template = $http_wrapper->query()->retrieve(self::POST_VALUE, $refinery->kindlyTo()->int());
115 $this->error->raiseError($this->
lng->txt(
'msg_no_perm_read'), $this->error->WARNING);
121 return $this->
access->checkAccess($a_permission,
'', $this->ref_id);
128 switch ($this->
ctrl->getNextClass()) {
130 $cmd = $this->
ctrl->getCmd(self::DEFAULT_CMD);
152 $form->setTitle($this->
lng->txt(
'auth_oidc_settings_title'));
153 $form->setFormAction($this->
ctrl->getFormAction($this));
156 $this->
lng->txt(
'auth_oidc_settings_activation'),
159 $activation->setChecked($this->
settings->getActive());
160 $form->addItem($activation);
163 $this->
lng->txt(
'auth_oidc_settings_provider'),
171 $this->
lng->txt(
'auth_oidc_settings_client_id'),
179 $this->
lng->txt(
'auth_oidc_settings_secret'),
183 $secret->setRetype(
false);
184 $secret->setRequired(
false);
185 if ($this->
settings->getSecret() !==
'') {
186 $secret->setValue(
'******');
188 $form->addItem($secret);
191 $this->
lng->txt(
'auth_oidc_settings_le'),
195 $login_element->setValue((
string) $this->
settings->getLoginElementType());
196 $form->addItem($login_element);
199 $this->
lng->txt(
'auth_oidc_settings_txt'),
202 $login_element->addOption($text_option);
208 $text->setValue($this->
settings->getLoginElemenText());
209 $text->setMaxLength(120);
210 $text->setInfo($this->
lng->txt(
'auth_oidc_settings_txt_val_info'));
211 $text_option->addSubItem($text);
214 $this->
lng->txt(
'auth_oidc_settings_img'),
217 $login_element->addOption($img_option);
223 $image->setAllowDeletion(
false);
225 if ($this->
settings->hasImageFile()) {
226 $image->setImage($this->
settings->getImageFilePath());
228 $image->setInfo($this->
lng->txt(
'auth_oidc_settings_img_file_info'));
229 $img_option->addSubItem($image);
232 $this->
lng->txt(
'auth_oidc_settings_login_options'),
238 $this->
lng->txt(
'auth_oidc_settings_login_option_enforce'),
241 $enforce->setInfo($this->
lng->txt(
'auth_oidc_settings_login_option_enforce_info'));
242 $login_options->addOption($enforce);
245 $this->
lng->txt(
'auth_oidc_settings_login_option_default'),
248 $default->setInfo($this->
lng->txt(
'auth_oidc_settings_login_option_default_info'));
249 $login_options->addOption($default);
251 $form->addItem($login_options);
254 $this->
lng->txt(
'auth_oidc_settings_logout_scope'),
260 $this->
lng->txt(
'auth_oidc_settings_logout_scope_global'),
263 $global_scope->setInfo($this->
lng->txt(
'auth_oidc_settings_logout_scope_global_info'));
264 $logout_scope->addOption($global_scope);
267 $this->
lng->txt(
'auth_oidc_settings_logout_scope_local'),
270 $ilias_scope->setInfo($this->
lng->txt(
'auth_oidc_settings_logout_scope_local_info'));
271 $logout_scope->addOption($ilias_scope);
273 $form->addItem($logout_scope);
276 $this->
lng->txt(
'auth_oidc_settings_custom_session_duration_type'),
279 $use_custom_session->setOptionTitle(
280 $this->
lng->txt(
'auth_oidc_settings_custom_session_duration_option')
282 $use_custom_session->setChecked($this->
settings->isCustomSession());
283 $form->addItem($use_custom_session);
286 $this->
lng->txt(
'auth_oidc_settings_session_duration'),
290 $session->setSuffix($this->
lng->txt(
'minutes'));
291 $session->setMinValue(5);
292 $session->setMaxValue(1440);
293 $session->setRequired(
true);
294 $use_custom_session->addSubItem($session);
297 $form->addCommandButton(
'saveSettings', $this->
lng->txt(
'save'));
301 $user_sync->setTitle($this->
lng->txt(
'auth_oidc_settings_section_user_sync'));
302 $form->addItem($user_sync);
305 $this->
lng->txt(
'auth_oidc_settings_user_sync'),
308 $sync->setChecked($this->
settings->isSyncAllowed());
309 $sync->setInfo($this->
lng->txt(
'auth_oidc_settings_user_sync_info'));
310 $sync->setValue(
'1');
311 $form->addItem($sync);
314 $this->
lng->txt(
'auth_oidc_settings_default_role'),
318 $roles->setInfo($this->
lng->txt(
'auth_oidc_settings_default_role_info'));
320 $roles->setRequired(
true);
321 $sync->addSubItem($roles);
324 $this->
lng->txt(
'auth_oidc_settings_user_attr'),
327 $user_attr->setValue($this->
settings->getUidField());
328 $user_attr->setRequired(
true);
329 $form->addItem($user_attr);
339 if (!$form->checkInput()) {
340 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->
lng->txt(
'err_check_input'));
341 $form->setValuesByPost();
346 $this->
settings->setActive((
bool) $form->getInput(
'activation'));
347 $this->
settings->setProvider((
string) $form->getInput(
'provider'));
348 $this->
settings->setClientId((
string) $form->getInput(
'client_id'));
349 if ((
string) $form->getInput(
'secret') !==
'' && strcmp($form->getInput(
'secret'),
'******') !== 0) {
350 $this->
settings->setSecret((
string) $form->getInput(
'secret'));
353 $this->
settings->setLoginElementType((
int) $form->getInput(
'le'));
354 $this->
settings->setLoginElementText((
string) $form->getInput(
'le_text'));
355 $this->
settings->setLoginPromptType((
int) $form->getInput(
'login_prompt'));
356 $this->
settings->setLogoutScope((
int) $form->getInput(
'logout_scope'));
357 $this->
settings->useCustomSession((
bool) $form->getInput(
'custom_session'));
358 $this->
settings->setSessionDuration((
int) $form->getInput(
'session_duration'));
359 $this->
settings->allowSync((
bool) $form->getInput(
'sync'));
360 $this->
settings->setRole((
int) $form->getInput(
'role'));
361 $this->
settings->setUidField((
string) $form->getInput(
'username'));
363 $fileData = (array) $form->getInput(
'le_img');
365 if ((
string) ($fileData[
'tmp_name'] ??
'') !==
'') {
371 $this->
mainTemplate->setOnScreenMessage(
'success', $this->
lng->txt(
'settings_saved'),
true);
372 $this->
ctrl->redirect($this,
'settings');
378 if (!$this->
upload->hasBeenProcessed()) {
382 foreach ($this->
upload->getResults() as $single_file_upload) {
383 if ($single_file_upload->isOK()) {
385 $this->
upload->moveFilesTo(
389 $this->
settings->setLoginElementImage($single_file_upload->getName());
393 $this->
logger->warning(
'Upload failed with message: ' . $e->getMessage());
403 $this->review->getGlobalRoles(),
410 if ($a_with_select_option) {
411 $select[0] = $this->
lng->txt(
'links_select_one');
413 foreach ($global_roles as $role_id) {
442 $this->
toolbar->setFormAction($this->
ctrl->getFormAction($this));
443 $this->
toolbar->addFormButton($this->
lng->txt(
'auth_oidc_discover_scopes'),
'discoverScopesFromServer');
446 $form = $this->initScopesForm();
447 $this->tpl->setContent($this->
renderer->render($form));
450 private function initScopesForm():
Form 458 $form = $this->
ui->input()->container()->form()->standard(
459 $this->
ctrl->getFormAction($this,
'saveScopes'),
469 $type = $this->
settings->getValidateScopes();
471 $url = $this->
settings->getProvider() . self::URL_VALIDATION_PROVIDER_STRING;
477 $found_scopes = $this->
settings->getSupportedScopesFromUrl(
$url);
478 if ($found_scopes ===
true) {
479 $this->
mainTemplate->setOnScreenMessage(
'success', $this->
lng->txt(
'auth_oidc_discover_scopes_info'));
492 $disabled_input = $this->
ui 495 ->text($this->
lng->txt(
'auth_oidc_settings_default_scopes'),
'')
498 ->withDedicatedName(
'default_scope')
499 ->withDisabled(
true);
501 $scopeValues = $this->
settings->getAdditionalScopes();
503 $tag_input = $this->
ui 507 $this->
lng->txt(
'auth_oidc_settings_additional_scopes'),
510 ->withDedicatedName(
'custom_scope')
511 ->withByline($this->
lng->txt(
'auth_oidc_settings_additional_scopes_info'));
512 $group1 = $this->
ui->input()->field()->group(
514 $this->
lng->txt(
'auth_oidc_settings_validate_scope_default')
516 $group2 = $this->
ui->input()->field()->group(
518 $this->
lng->txt(
'auth_oidc_settings_discovery_url') => $this->
ui 522 $this->
lng->txt(
'auth_oidc_settings_discovery_url')
526 $this->
settings->getCustomDiscoveryUrl() ??
'' 529 $this->
lng->txt(
'auth_oidc_settings_validate_scope_custom')
531 $group3 = $this->
ui->input()->field()->group(
533 $this->
lng->txt(
'auth_oidc_settings_validate_scope_none')
535 $url_validation = $this->
ui->input()->field()->switchableGroup(
541 $this->
lng->txt(
'auth_oidc_settings_validate_scopes')
542 )->withDedicatedName(
'validate_scopes')->withValue($this->
settings->getValidateScopes());
543 $group = $this->
ui->input()->field()->group(
544 [$disabled_input, $tag_input, $url_validation]
546 $ui_container[] = $group;
548 return $ui_container;
560 $form = $this->initScopesForm();
561 if ($this->request->getMethod() ===
'POST') {
562 $request_form = $form->withRequest($this->request);
563 $result = $request_form->getData();
564 if ($result ===
null) {
565 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->
lng->txt(
'err_check_input'));
570 foreach ($form->getInputs() as $group => $groups) {
571 foreach ($groups->getInputs() as $key => $input) {
572 $dedicated_name = $input->getDedicatedName();
573 $result_data = $result[$group][$key];
574 if ($dedicated_name ===
'validate_scopes') {
575 $type = (
int) $result_data[0];
576 $url = array_pop($result_data[1]);
577 } elseif ($dedicated_name ===
'custom_scope') {
578 $custom_scopes = $result_data;
590 $this->
settings->setAdditionalScopes((array) $custom_scopes);
591 $this->
settings->setValidateScopes((
int) $type);
596 $this->
mainTemplate->setOnScreenMessage(
'success', $this->
lng->txt(
'settings_saved'),
true);
597 $this->
ctrl->redirect($this,
'scopes');
600 if ($this->failed_validation_messages !==
'') {
601 $this->failed_validation_messages = $this->
lng->txt(
605 $this->failed_validation_messages = $this->
lng->txt(
'err_check_input');
608 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->failed_validation_messages,
true);
609 $this->
ctrl->redirect($this,
'scopes');
620 $discoveryURL = $url . self::URL_VALIDATION_PROVIDER_STRING;
623 $discoveryURL =
$url;
626 $discoveryURL =
null;
630 $validation_result = $discoveryURL !==
null ? $this->
settings->validateScopes(
634 if (!empty($validation_result)) {
636 $this->failed_validation_messages =
638 $this->
lng->txt(
'auth_oidc_settings_invalid_scopes'),
639 implode(
',', $validation_result[1])
642 $this->failed_validation_messages = sprintf(
643 $this->
lng->txt(
'auth_oidc_settings_discovery_error'),
644 $validation_result[1]
657 $this->failed_validation_messages = $e->getMessage();
670 $form = $this->initUserMappingForm();
671 if ($this->request->getMethod() ===
'POST' &&
672 $this->request->getQueryParams()[
'opic'] ===
'opic_user_data_mapping') {
673 $request_form = $form->withRequest($this->request);
674 $result = $request_form->getData();
675 if ($result ===
null) {
676 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->
lng->txt(
'err_check_input'));
681 foreach ($this->
settings->getProfileMappingFields() as $field => $lng_key) {
685 foreach ($this->udf->getDefinitions() as $definition) {
686 $field = self::UDF_STRING . $definition[
'field_id'];
693 $this->
mainTemplate->setOnScreenMessage(
'success', $this->
lng->txt(
'settings_saved'),
true);
694 $this->
ctrl->redirect($this, self::STAB_PROFILE);
699 $form = $this->initUserMappingForm();
700 $request_form = $form->withRequest($this->request);
701 $result = $request_form->getData();
702 foreach ($form->getInputs() as $group => $groups) {
703 foreach ($groups->getInputs() as $key => $input) {
704 $dedicated_name = $input->getDedicatedName();
705 $result_data = $result[$group][$key];
707 if ($dedicated_name === $field . self::VALUE_STRING) {
708 $this->
settings->setProfileMappingFieldValue(
712 } elseif ($dedicated_name === $field . self::UPDATE_STRING) {
713 $this->
settings->setProfileMappingFieldUpdate(
740 $form->setTitle($this->
lng->txt(
'auth_oidc_role_mapping_table'));
741 $form->setFormAction($this->
ctrl->getFormAction($this, self::STAB_ROLES));
746 'role_map_' . $role_id
748 $role_map->setInfo($this->
lng->txt(
'auth_oidc_role_info'));
749 $role_map->setValue($this->
settings->getRoleMappingValueForId((
int) $role_id));
750 $form->addItem($role_map);
754 'role_map_update_' . $role_id
756 $update->setOptionTitle($this->
lng->txt(
'auth_oidc_update_role_info'));
757 $update->setValue(
'1');
758 $update->setChecked(!$this->
settings->getRoleMappingUpdateForId((
int) $role_id));
759 $form->addItem($update);
763 $form->addCommandButton(
'saveRoles', $this->
lng->txt(
'save'));
774 if ($form->checkInput()) {
780 $role_settings[(
int) $role_id][
'update'] = !$form->getInput(
'role_map_update_' . $role_id);
781 $role_settings[(
int) $role_id][
'value'] =
'';
783 $input_role = trim($form->getInput(
'role_map_' . $role_id));
784 if ($input_role ===
'') {
788 $role_params = explode(
'::', $input_role);
791 if (count($role_params) !== 2) {
792 if ($form->getItemByPostVar(
'role_map_' . $role_id)) {
793 $form->getItemByPostVar(
'role_map_' . $role_id)->setAlert($this->
lng->txt(
'msg_wrong_format'));
798 $role_settings[(
int) $role_id][
'value'] = $input_role;
802 $form->setValuesByPost();
803 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->
lng->txt(
'err_check_input'));
808 $this->
settings->setRoleMappings($role_settings);
810 $this->
mainTemplate->setOnScreenMessage(
'success', $this->
lng->txt(
'settings_saved'),
true);
811 $this->
ctrl->redirect($this,
'roles');
814 $form->setValuesByPost();
816 $this->
mainTemplate->setOnScreenMessage(
'failure', $this->
lng->txt(
'err_check_input'));
823 $this->
tabs->addSubTab(
825 $this->
lng->txt(
'auth_oidc_' . self::STAB_SETTINGS),
826 $this->
ctrl->getLinkTarget($this, self::STAB_SETTINGS)
831 $this->
tabs->addSubTab(
833 $this->
lng->txt(
'auth_oidc_' . self::STAB_SCOPES),
834 $this->
ctrl->getLinkTarget($this, self::STAB_SCOPES)
837 $this->
tabs->addSubTab(
839 $this->
lng->txt(
'auth_oidc_' . self::STAB_PROFILE),
840 $this->
ctrl->getLinkTarget($this, self::STAB_PROFILE)
842 $this->
tabs->addSubTab(
844 $this->
lng->txt(
'auth_oidc_' . self::STAB_ROLES),
845 $this->
ctrl->getLinkTarget($this, self::STAB_ROLES)
849 $this->
tabs->activateSubTab($active_tab);
858 if ((
int) $this->mapping_template === self::VIEW_TAB_EFFECTIVE_MAPPING) {
868 if ($this->mapping_template === self::VIEW_TAB_EFFECTIVE_MAPPING) {
870 $this->
factory->link()->standard(
871 $this->
lng->txt(
'auth_oidc_here'),
872 'https://openid.net/specs/openid-connect-core-1_0.html#StandardClaims' 878 $this->
factory->link()->standard(
879 $this->
lng->txt(
'auth_oidc_here'),
880 $this->
ctrl->getLinkTarget($this, self::STAB_SCOPES)
883 $tab_name = $this->
lng->txt(
'auth_oidc_configured_scopes');
884 $message = sprintf($this->
lng->txt(
'auth_odic_scope_info'),
$url, $tab_name);
892 $mapping = $this->attribute_mapping_template->getMappingRulesByAdditionalScopes(
893 $this->
settings->getAdditionalScopes()
896 if (count($mapping) > 0) {
897 $this->
settings->clearProfileMaps();
900 foreach ($mapping as $field => $item) {
901 $this->
settings->setProfileMappingFieldValue(
910 private function initUserMappingForm():
Form 915 foreach ($this->
settings->getProfileMappingFields() as $mapping =>
$lang) {
919 foreach ($this->udf->getDefinitions() as $definition) {
923 $this->
ctrl->setParameter(
926 'opic_user_data_mapping' 935 $this->
ctrl->getFormAction($this,
'saveProfileMapping'),
949 $value = $this->
settings->getProfileMappingFieldValue(self::UDF_STRING . $definition[
'field_id']);
950 $update = $this->
settings->getProfileMappingFieldUpdate(self::UDF_STRING . $definition[
'field_id']);
952 $text_input = $this->
ui 955 ->text($definition[
'field_name'],
'')
958 ->withDedicatedName(self::UDF_STRING . $definition[
'field_id'] . self::VALUE_STRING);
959 $checkbox_input = $this->
ui 961 ->field()->checkbox(
'', $this->
lng->txt(
'auth_oidc_update_field_info'))
964 self::UDF_STRING . $definition[
'field_id'] . self::UPDATE_STRING
966 $group = $this->
ui->input()->field()->group(
967 [$text_input, $checkbox_input]
969 $ui_container[] = $group;
971 return $ui_container;
980 $value = $this->
settings->getProfileMappingFieldValue($mapping);
981 $update = $this->
settings->getProfileMappingFieldUpdate($mapping);
983 $text_input = $this->
ui 989 ->withDedicatedName($mapping . self::VALUE_STRING);
990 $checkbox_input = $this->
ui 993 ->checkbox(
'', $this->
lng->txt(
'auth_oidc_update_field_info'))
995 ->withDedicatedName($mapping . self::UPDATE_STRING);
996 $group = $this->
ui->input()->field()->group(
1002 $ui_container[] = $group;
1004 return $ui_container;
1009 if ($this->udf ===
null) {
1016 $form = $this->initUserMappingForm();
1019 $active = self::EFFECTIVE_ATTRIBUTE_MAPPING_TAB;
1021 $target = $this->
http->request()->getRequestTarget();
1030 $this->
lng->txt(
'auth_oidc_saved_values') =>
"$target&" . self::POST_VALUE .
'=' . self::SAVED_VALUES,
1033 ) =>
"$target&" . self::POST_VALUE .
'=' . self::DEFAULT_VALUES,
1036 $aria_label =
'change_the_currently_displayed_mode';
1037 $active_label = $this->
lng->txt(
'auth_oidc_saved_values');
1038 if ($active !== self::EFFECTIVE_ATTRIBUTE_MAPPING_TAB) {
1041 $view_control = $this->
factory->viewControl()->mode($actions, $aria_label)->withActive($active_label);
1043 $this->tpl->setContent($this->
renderer->render([$view_control, $form]));
1050 $this->tpl->setOnScreenMessage(
1051 $this->tpl::MESSAGE_TYPE_FAILURE,
1052 $this->
lng->txt(
'permission_denied'),
1055 $this->
ctrl->redirect($this,
'settings');
1061 return $this->
refinery->custom()->transformation(
static function (array $values): array {
1068 return $this->
refinery->custom()->transformation(
static function ($value) {
1069 if (is_string($value)) {
1070 $value = trim($value);
setSubTabs(string $active_tab)
const URL_VALIDATION_NONE
checkAccessBool(string $a_permission)
This class represents an option in a radio group.
__construct(int $a_ref_id)
const LOGOUT_SCOPE_GLOBAL
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
readonly FileUpload $upload
static stripSlashesRecursive($a_data, bool $a_strip_html=true, string $a_allow="")
buildUserMappingInputFormUDF($definition, array $ui_container)
saveImageFromHttpRequest()
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
prepareRoleSelection(bool $a_with_select_option=true)
checkAccess(string $a_permission)
Additional user data fields definition.
Interface Observer Contains several chained tasks and infos about them.
const URL_VALIDATION_PROVIDER
readonly ilErrorHandling $error
readonly ilAccessHandler $access
const LOGIN_ELEMENT_TYPE_TXT
ilOpenIdAttributeMappingTemplate $attribute_mapping_template
buildUserMappingInputForUserData(string $lang, string $mapping, array $ui_container)
readonly ilGlobalTemplateInterface $mainTemplate
final const OPEN_ID_CONFIGURED_SCOPES
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static http()
Fetches the global http state from ILIAS.
const EFFECTIVE_ATTRIBUTE_MAPPING_TAB
saniziteArrayElementsTrafo()
static _lookupTitle(int $obj_id)
This is how the factory for UI elements looks.
const VALIDATION_ISSUE_INVALID_SCOPE
string $failed_validation_messages
roles(?ilPropertyFormGUI $form=null)
ILIAS Refinery Factory $refinery
const URL_VALIDATION_PROVIDER_STRING
buildScopeSelection(array $ui_container)
static _sortIds(array $a_ids, string $a_table, string $a_field, string $a_id_name)
Function that sorts ids by a given table field using WHERE IN E.g: __sort(array(6,7),'usr_data','lastname','usr_id') => sorts by lastname.
redirectToSettingsScreenIfNoURLIsConfigured()
ilGlobalTemplateInterface $tpl
readonly ilLogger $logger
discoverScopesFromServer()
readonly ilRbacReview $review
const URL_VALIDATION_CUSTOM
const VIEW_TAB_EFFECTIVE_MAPPING
updateProfileMappingFieldValue(string $field)
const LOGIN_ELEMENT_TYPE_IMG
settings(?ilPropertyFormGUI $form=null)
validateDiscoveryUrl(int $type, ?string $url, array $scopes)
ServerRequestInterface $request
readonly ilOpenIdConnectSettings $settings