<?php namespace LAM\AJAX; use htmlResponsiveTable; use htmlStatusMessage; use \LAM\TOOLS\IMPORT_EXPORT\Importer; use \LAM\TOOLS\IMPORT_EXPORT\Exporter; use \LAM\TYPES\TypeManager; use \htmlResponsiveRow; use \htmlLink; use \htmlOutputText; use \htmlButton; use \LAM\LOGIN\WEBAUTHN\WebauthnManager; use \LAMCfgMain; /* This code is part of LDAP Account Manager (http://www.ldap-account-manager.org/) Copyright (C) 2011 - 2020 Roland Gruber This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /** * Manages all AJAX requests. * * @author Roland Gruber * @package tools */ /** security functions */ include_once(__DIR__ . "/../../lib/security.inc"); /** LDIF import */ include_once(__DIR__ . "/../../lib/import.inc"); // start session if (isset($_GET['selfservice'])) { // self service uses a different session name session_name('SELFSERVICE'); } // return standard JSON response if session expired if (startSecureSession(false, true) === false) { echo json_encode(array( 'sessionExpired' => "true" )); die(); } setlanguage(); $ajax = new Ajax(); $ajax->handleRequest(); /** * Manages all AJAX requests. */ class Ajax { /** * Manages an AJAX request. */ public function handleRequest() { $this->setHeader(); // check token validateSecurityToken(); $isSelfService = isset($_GET['selfservice']); if (isset($_GET['module']) && isset($_GET['scope']) && in_array($_GET['module'], getAvailableModules($_GET['scope']))) { enforceUserIsLoggedIn(); if (isset($_GET['useContainer']) && ($_GET['useContainer'] == '1')) { $sessionKey = htmlspecialchars($_GET['editKey']); if (!isset($_SESSION[$sessionKey])) { logNewMessage(LOG_ERR, 'Unable to find account container'); die(); } $module = $_SESSION[$sessionKey]->getAccountModule($_GET['module']); $module->handleAjaxRequest(); } else { $module = new $_GET['module']($_GET['scope']); $module->handleAjaxRequest(); } die(); } if (!isset($_GET['function'])) { die(); } $function = $_GET['function']; if (!isset($_POST['jsonInput'])) { die(); } $jsonInput = $_POST['jsonInput']; if ($function == 'passwordStrengthCheck') { $this->checkPasswordStrength($jsonInput); die(); } if ($function === 'webauthn') { enforceUserIsLoggedIn(false); $this->manageWebauthn($isSelfService); die(); } if ($function === 'webauthnDevices') { $this->enforceUserIsLoggedInToMainConfiguration(); $this->manageWebauthnDevices(); die(); } enforceUserIsLoggedIn(); if ($function == 'passwordChange') { $this->managePasswordChange($jsonInput); } elseif ($function === 'import') { include_once('../../lib/import.inc'); $importer = new Importer(); ob_start(); $jsonOut = $importer->doImport(); ob_end_clean(); echo $jsonOut; } elseif ($function === 'export') { include_once('../../lib/export.inc'); $attributes = $_POST['attributes']; $baseDn = $_POST['baseDn']; $ending = $_POST['ending']; $filter = $_POST['filter']; $format = $_POST['format']; $includeSystem = ($_POST['includeSystem'] === 'true'); $saveAsFile = ($_POST['saveAsFile'] === 'true'); $searchScope = $_POST['searchScope']; $exporter = new Exporter($baseDn, $searchScope, $filter, $attributes, $includeSystem, $saveAsFile, $format, $ending); ob_start(); $jsonOut = $exporter->doExport(); ob_end_clean(); echo $jsonOut; } elseif ($function === 'upload') { include_once('../../lib/upload.inc'); $typeManager = new \LAM\TYPES\TypeManager(); $uploader = new \LAM\UPLOAD\Uploader($typeManager->getConfiguredType($_GET['typeId'])); ob_start(); $jsonOut = $uploader->doUpload(); ob_end_clean(); echo $jsonOut; } elseif ($function === 'dnselection') { ob_start(); $jsonOut = $this->dnSelection(); ob_end_clean(); echo $jsonOut; } elseif ($function === 'webauthnOwnDevices') { $this->manageWebauthnOwnDevices(); } } /** * Sets JSON HTTP header. */ private static function setHeader() { if (!headers_sent()) { header('Content-Type: application/json; charset=utf-8'); } } /** * Manages a password change request on the edit account page. * * @param array $input input parameters */ private static function managePasswordChange($input) { $sessionKey = htmlspecialchars($_GET['editKey']); $return = $_SESSION[$sessionKey]->setNewPassword($input); echo json_encode($return); } /** * Checks if a password is accepted by LAM's password policy. * * @param array $input input parameters */ private function checkPasswordStrength($input) { $password = $input['password']; $result = checkPasswordStrength($password, null, null); echo json_encode(array("result" => $result)); } /** * Manages webauthn requests. * * @param bool $isSelfService request is from self service */ private function manageWebauthn($isSelfService) { include_once __DIR__ . '/../../lib/webauthn.inc'; if ($isSelfService) { $userDN = lamDecrypt($_SESSION['selfService_clientDN'], 'SelfService'); } else { $userDN = $_SESSION['ldap']->getUserName(); } $webauthnManager = new WebauthnManager(); $isRegistered = $webauthnManager->isRegistered($userDN); if (!$isRegistered) { $registrationObject = $webauthnManager->getRegistrationObject($userDN, $isSelfService); $_SESSION['webauthn_registration'] = json_encode($registrationObject); echo json_encode( array( 'action' => 'register', 'registration' => $registrationObject ), JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE ); } else { $authenticationObject = $webauthnManager->getAuthenticationObject($userDN, $isSelfService); $_SESSION['webauthn_authentication'] = json_encode($authenticationObject); echo json_encode( array( 'action' => 'authenticate', 'authentication' => $authenticationObject ), JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE ); } die(); } /** * Webauthn device management. */ private function manageWebauthnDevices() { $action = $_POST['action']; if ($action === 'search') { $searchTerm = $_POST['searchTerm']; if (!empty($searchTerm)) { $this->manageWebauthnDevicesSearch($searchTerm); } } elseif ($action === 'delete') { $dn = $_POST['dn']; $credentialId = $_POST['credentialId']; if (!empty($dn) && !empty($credentialId)) { $this->manageWebauthnDevicesDelete($dn, $credentialId); } } } /** * Searches for webauthn devices and prints the results as html. * * @param string $searchTerm search term */ private function manageWebauthnDevicesSearch($searchTerm) { include_once __DIR__ . '/../../lib/webauthn.inc'; $database = new \LAM\LOGIN\WEBAUTHN\PublicKeyCredentialSourceRepositorySQLite(); $results = $database->searchDevices('%' . $searchTerm . '%'); $row = new htmlResponsiveRow(); $row->addVerticalSpacer('0.5rem'); if (empty($results)) { $row->add(new htmlStatusMessage('INFO', _('No devices found.')), 12); } else { $titles = array( _('User'), _('Registration'), _('Last use'), _('Delete') ); $data = array(); $id = 0; foreach ($results as $result) { $delButton = new htmlButton('deleteDevice' . $id, 'delete.png', true); $delButton->addDataAttribute('credential', $result['credentialId']); $delButton->addDataAttribute('dn', $result['dn']); $delButton->addDataAttribute('dialogtitle', _('Remove device')); $delButton->addDataAttribute('oktext', _('Ok')); $delButton->addDataAttribute('canceltext', _('Cancel')); $delButton->setCSSClasses(array('webauthn-delete')); $data[] = array( new htmlOutputText($result['dn']), new htmlOutputText(date('Y-m-d H:i:s', $result['registrationTime'])), new htmlOutputText(date('Y-m-d H:i:s', $result['lastUseTime'])), $delButton ); $id++; } $table = new htmlResponsiveTable($titles, $data); $row->add($table, 12); } $row->addVerticalSpacer('2rem'); $tabindex = 10000; ob_start(); $row->generateHTML('none', array(), array(), false, $tabindex, null); $content = ob_get_contents(); ob_end_clean(); echo json_encode(array('content' => $content)); } /** * Deletes a webauthn device. * * @param string $dn user DN * @param string $credentialId base64 encoded credential id */ private function manageWebauthnDevicesDelete($dn, $credentialId) { include_once __DIR__ . '/../../lib/webauthn.inc'; $database = new \LAM\LOGIN\WEBAUTHN\PublicKeyCredentialSourceRepositorySQLite(); $success = $database->deleteDevice($dn, $credentialId); if ($success) { $message = new htmlStatusMessage('INFO', _('The device was deleted.')); } else { $message = new htmlStatusMessage('ERROR', _('The device was not found.')); } $row = new htmlResponsiveRow(); $row->addVerticalSpacer('0.5rem'); $row->add($message, 12); $row->addVerticalSpacer('2rem'); ob_start(); $tabindex = 50000; $row->generateHTML('none', array(), array(), true, $tabindex, null); $content = ob_get_contents(); ob_end_clean(); echo json_encode(array('content' => $content)); } /** * Manages requests to setup user's own webauthn devices. */ private function manageWebauthnOwnDevices() { $action = $_POST['action']; $dn = $_POST['dn']; $sessionDn = $_SESSION['ldap']->getUserName(); if ($sessionDn !== $dn) { logNewMessage(LOG_ERR, 'WebAuthn delete canceled, DN does not match.'); die(); } if ($action === 'delete') { $credentialId = $_POST['credentialId']; $this->manageWebauthnDevicesDelete($sessionDn, $credentialId); } } /** * Handles DN selection fields. * * @return string JSON output */ private function dnSelection() { $dn = trim($_POST['dn']); if (empty($dn) || !get_preg($dn, 'dn')) { $dnList = $this->getDefaultDns(); $dn = null; } else { $dnList = $this->getSubDns($dn); } $html = $this->buildDnSelectionHtml($dnList, $dn); return json_encode(array('dialogData' => $html)); } /** * Returns a list of default DNs from account types + tree suffix. * * @return string[] default DNs */ private function getDefaultDns() { $typeManager = new TypeManager(); $baseDnList = array(); foreach ($typeManager->getConfiguredTypes() as $type) { $suffix = $type->getSuffix(); if (!empty($suffix)) { $baseDnList[] = $suffix; } } $treeSuffix = $_SESSION['config']->get_Suffix('tree'); if (!empty($treeSuffix)) { $baseDnList[] = $suffix; } $baseDnList = array_unique($baseDnList); usort($baseDnList, 'compareDN'); return $baseDnList; } /** * Returns the HTML to build the DN selection list. * * @param string[] $dnList DN list * @param string $currentDn current DN */ private function buildDnSelectionHtml($dnList, $currentDn) { $fieldId = trim($_POST['fieldId']); $mainRow = new htmlResponsiveRow(); $onclickUp = 'window.lam.html.updateDnSelection(this, \'' . htmlspecialchars($fieldId) . '\', \'' . getSecurityTokenName() . '\', \'' . getSecurityTokenValue() . '\')'; if (!empty($currentDn)) { $row = new htmlResponsiveRow(); $row->addDataAttribute('dn', $currentDn); $text = new htmlOutputText($currentDn); $text->setIsBold(true); $row->add($text, 12, 9); $row->setCSSClasses(array('text-right')); $buttonId = base64_encode($currentDn); $buttonId = str_replace('=', '', $buttonId); $button = new htmlButton($buttonId, _('Ok')); $button->setIconClass('okButton'); $button->setOnClick('window.lam.html.selectDn(this, \'' . htmlspecialchars($fieldId) . '\')'); $row->add($button, 12, 3); $mainRow->add($row, 12); // back up $row = new htmlResponsiveRow(); $row->addDataAttribute('dn', extractDNSuffix($currentDn)); $text = new htmlLink('..', '#'); $text->setCSSClasses(array('bold')); $text->setOnClick($onclickUp); $row->add($text, 12, 9); $row->setCSSClasses(array('text-right')); $buttonId = base64_encode('..'); $buttonId = str_replace('=', '', $buttonId); $button = new htmlButton($buttonId, _('Up')); $button->setIconClass('upButton'); $button->setOnClick($onclickUp); $row->add($button, 12, 3); $mainRow->add($row, 12); } foreach ($dnList as $dn) { $row = new htmlResponsiveRow(); $row->addDataAttribute('dn', $dn); $link = new htmlLink($dn, '#'); $link->setOnClick($onclickUp); $row->add($link, 12, 9); $row->setCSSClasses(array('text-right')); $buttonId = base64_encode($dn); $buttonId = str_replace('=', '', $buttonId); $button = new htmlButton($buttonId, _('Ok')); $button->setIconClass('okButton'); $button->setOnClick('window.lam.html.selectDn(this, \'' . htmlspecialchars($fieldId) . '\')'); $row->add($button, 12, 3); $mainRow->add($row, 12); } $tabindex = 1000; ob_start(); parseHtml(null, $mainRow, array(), false, $tabindex, 'user'); $out = ob_get_contents(); ob_end_clean(); return $out; } /** * Returns the sub DNs of given DN. * * @param string $dn DN * @return string[] sub DNs */ private function getSubDns($dn) { $dnEntries = ldapListDN($dn); $dnList = array(); foreach ($dnEntries as $entry) { $dnList[] = $entry['dn']; } usort($dnList, 'compareDN'); return $dnList; } /** * Checks if the user entered the configuration master password. * Dies if password is not set. */ private function enforceUserIsLoggedInToMainConfiguration() { if (!isset($_SESSION['cfgMain'])) { $cfg = new LAMCfgMain(); } else { $cfg = $_SESSION['cfgMain']; } if (isset($_SESSION["mainconf_password"]) && ($cfg->checkPassword($_SESSION["mainconf_password"]))) { return; } die(); } } ?>