compileTask !== null) return $this->compileTask; $this->compileTask = Property::get(IPxeBuilder::PROP_IPXE_COMPILE_TASKID); if ($this->compileTask !== false) { $this->compileTask = Taskmanager::status($this->compileTask); if (!Taskmanager::isTask($this->compileTask) || Taskmanager::isFinished($this->compileTask)) { $this->compileTask = false; } } return $this->compileTask; } protected function doPreprocess() { User::load(); if (!User::isLoggedIn()) { Message::addError('main.no-permission'); Util::redirect('?do=Main'); } if (Request::any('action') === 'getimage') { User::assertPermission("download"); $this->handleGetImage(); } if (User::hasPermission('edit.address')) { Taskmanager::submit('IpxeVersion', ['id' => IPxeBuilder::VERSION_LIST_TASK, 'action' => 'LIST'], true); } $action = Request::post('action'); if ($action === false) { $this->currentAddress = Property::getServerIp(); $this->getLocalAddresses(); } if ($action === 'compile') { User::assertPermission("edit.address"); if ($this->getCompileTask() === false) { $taskId = IPxeBuilder::setIpxeVersion(Request::post('version', false, 'string')); Trigger::ipxe($taskId); } Util::redirect('?do=serversetup&show=address&sv=1'); } if ($action === 'fetch' || $action === 'reset') { User::assertPermission("edit.address"); if ($this->getCompileTask() === false) { $task = Taskmanager::submit('IpxeVersion', ['action' => strtoupper($action)]); if (Taskmanager::isTask($task)) { Property::set(IPxeBuilder::PROP_VERSION_SELECT_TASKID, $task['id'], 2); } } Util::redirect('?do=serversetup&show=address&sv=1'); } if ($action === 'ip') { User::assertPermission("edit.address"); // New address is to be set $this->getLocalAddresses(); $this->updateLocalAddress(); } if ($action === 'savebootentry') { User::assertPermission('ipxe.bootentry.edit'); $this->saveBootEntry(); } if ($action === 'deleteBootentry') { User::assertPermission('ipxe.bootentry.edit'); $this->deleteBootEntry(); } if ($action === 'savemenu') { User::assertPermission('ipxe.menu.edit'); $this->saveMenu(); } if ($action === 'savelocation') { // Permcheck in function $this->saveLocationMenu(); Util::redirect('?do=locations'); } if ($action === 'savelocalboot') { User::assertPermission('ipxe.localboot.edit'); $this->saveLocalboot(); } if ($action === 'import') { User::assertPermission('ipxe.bootentry.edit'); $this->execImportPxeMenu(); } if ($action === 'deleteMenu') { // Permcheck in function $this->deleteMenu(); } if ($action === 'setDefaultMenu') { User::assertPermission('ipxe.menu.edit', 0); $this->setDefaultMenu(); } if (Request::isPost()) { Util::redirect('?do=serversetup'); } User::assertPermission('access-page'); $addr = false; if (User::hasPermission('ipxe.menu.view')) { Dashboard::addSubmenu('?do=serversetup&show=menu', Dictionary::translate('submenu_menu')); } if (User::hasPermission('ipxe.bootentry.view')) { Dashboard::addSubmenu('?do=serversetup&show=bootentry', Dictionary::translate('submenu_bootentry')); } if (User::hasPermission('edit.address')) { Dashboard::addSubmenu('?do=serversetup&show=address', Dictionary::translate('submenu_address')); $addr = true; } if (User::hasPermission('download')) { Dashboard::addSubmenu('?do=serversetup&show=download', Dictionary::translate('submenu_download')); } if (User::hasPermission('ipxe.localboot.*')) { Dashboard::addSubmenu('?do=serversetup&show=localboot', Dictionary::translate('submenu_localboot')); } if (User::hasPermission('ipxe.bootentry.edit')) { Dashboard::addSubmenu('?do=serversetup&show=import', Dictionary::translate('submenu_import')); } if (Request::get('show') === false) { $subs = Dashboard::getSubmenus(); $sv = Request::get('sv') ? '&sv=' . Request::get('sv') : ''; if (empty($subs)) { User::assertPermission('download'); } elseif ($addr && !preg_match('/^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/', Property::getServerIp())) { Util::redirect('?do=serversetup&show=address' . $sv); } else { Util::redirect($subs[0]['url'] . $sv); } } } protected function doRender() { Render::addTemplate("heading"); $show = Request::get('show'); if (in_array($show, ['menu', 'address', 'download'])) { $selectTask = Taskmanager::status(Property::get(IPxeBuilder::PROP_VERSION_SELECT_TASKID)); $buildTask = $this->getCompileTask(); if ($buildTask !== false || Taskmanager::isRunning($selectTask) || Request::get('sv')) { Render::addTemplate('git_task', [ 'selectTask' => Property::get(IPxeBuilder::PROP_VERSION_SELECT_TASKID), 'reload' => Taskmanager::isRunning($selectTask), ]); } if ($buildTask !== false) { $files = []; if ($buildTask['data'] && $buildTask['data']['files']) { foreach ($buildTask['data']['files'] as $k => $v) { $files[] = ['name' => $k, 'namehyphen' => str_replace(['/', '.'], '-', $k)]; } } Render::addTemplate('ipxe_update', ['taskid' => $buildTask['id'], 'files' => $files]); } } switch ($show) { case 'editbootentry': User::assertPermission('ipxe.bootentry.*'); $this->showEditBootEntry(); break; case 'editmenu': User::assertPermission('ipxe.menu.view'); $this->showEditMenu(); break; case 'download': User::assertPermission('download'); $this->showDownload(); break; case 'menu': User::assertPermission('ipxe.menu.view'); $this->showMenuList(); break; case 'bootentry': User::assertPermission('ipxe.bootentry.view'); $this->showBootentryList(); break; case 'address': User::assertPermission('edit.address'); $this->showEditAddress(); break; case 'assignlocation': // Permcheck in function $this->showEditLocation(); break; case 'localboot': User::assertPermission('ipxe.localboot.*'); $this->showLocalbootConfig(); break; case 'import': User::assertPermission('ipxe.bootentry.edit'); $this->showImportMenu(); break; default: Util::redirect('?do=serversetup'); } } private function showDownload() { $list = glob('/srv/openslx/www/boot/download/*', GLOB_NOSORT); usort($list, function ($a, $b) { return strcmp(substr($a, -4), substr($b, -4)) * 100 + strcmp($a, $b); }); $files = []; $strings = [ 'efi' => [Dictionary::translate('dl-efi') => 50], 'pcbios' => [Dictionary::translate('dl-pcbios') => 51], 'usb' => [Dictionary::translate('dl-usb') => 80], 'hd' => [Dictionary::translate('dl-hd') => 81], 'lkrn' => [Dictionary::translate('dl-lkrn') => 82], 'i386' => [Dictionary::translate('dl-i386') => 10], 'x86_64' => [Dictionary::translate('dl-x86_64') => 11], 'ecm' => [Dictionary::translate('dl-usbnic') => 60], 'ncm' => [Dictionary::translate('dl-usbnic') => 61], 'ipxe' => [Dictionary::translate('dl-pcinic') => 62], 'snp' => [Dictionary::translate('dl-snp') => 63], ]; foreach ($list as $file) { if ($file[0] === '.') continue; if (is_file($file)) { $base = basename($file); $features = []; foreach (preg_split('/[\-.\/]+/', $base, -1, PREG_SPLIT_NO_EMPTY) as $p) { if (array_key_exists($p, $strings)) { $features += $strings[$p]; } } asort($features); $files[] = [ 'name' => $base, 'size' => Util::readableFileSize(filesize($file)), 'modified' => Util::prettyTime(filemtime($file)), 'class' => substr($base, -4) === '.usb' ? 'slx-bold' : '', 'features' => implode(', ', array_keys($features)), ]; } } Render::addTemplate('download', ['files' => $files]); } /** * @return array{EFI: array, PCBIOS: array} */ private function makeSelectArray(array $list, array $defaults): array { $ret = ['EFI' => [], 'PCBIOS' => []]; foreach (['PCBIOS', 'EFI'] as $m) { foreach (array_keys($list[$m]) as $k) { $ret[$m][] = [ 'key' => $k, 'selected' => ($k === $defaults[$m] ? 'selected' : ''), ]; } } return $ret; } private function showLocalbootConfig() { // Default setting $default = Localboot::getDefault(); $optionList = $this->makeSelectArray(Localboot::BOOT_METHODS, $default); // Exceptions $models = []; $res = Database::simpleQuery('SELECT m.systemmodel, cnt, sl.pcbios AS PCBIOS, sl.efi AS EFI FROM ( SELECT m2.systemmodel, Count(*) AS cnt FROM machine m2 GROUP BY systemmodel ) m LEFT JOIN serversetup_localboot sl USING (systemmodel) ORDER BY systemmodel'); foreach ($res as $row) { $row['modelesc'] = urlencode($row['systemmodel']); $row['options'] = $this->makeSelectArray(Localboot::BOOT_METHODS, $row); $models[] = $row; } // Output $data = [ 'default' => $default, 'options' => $optionList, 'exceptions' => $models, ]; Render::addTemplate('localboot', $data); } private function showImportMenu() { Render::addTemplate('page-import'); } private function showBootentryList() { $allowEdit = User::hasPermission('ipxe.bootentry.edit'); $bootentryTable = Database::queryAll("SELECT be.entryid, be.hotkey, be.title, be.builtin, be.module, Count(sme.menuid) AS refs, GROUP_CONCAT(sm.title SEPARATOR ', ') as menuname FROM serversetup_bootentry be LEFT JOIN serversetup_menuentry sme USING (entryid) LEFT JOIN serversetup_menu sm USING (menuid) WHERE be.module <> '.special' GROUP BY be.entryid, be.title"); if (empty($bootentryTable)) { if (Property::getServerIp() === 'invalid') { Message::addError('no-ip-set'); Util::redirect('?do=serversetup&show=address'); } IPxe::importLegacyMenu(true); $num = IPxe::importSubnetPxeMenus('/srv/openslx/tftp/pxelinux.cfg'); if ($num > 0) { EventLog::info('Imported old PXELinux menu, with ' . $num . ' additional IP-range based menus.'); } else { EventLog::info('Imported old PXELinux menu.'); } Util::redirect('?do=serversetup&show=bootentry'); } Module::isAvailable('js_stupidtable'); Render::addTemplate('bootentry-list', array( 'bootentryTable' => $bootentryTable, 'allowEdit' => $allowEdit, )); } private function showMenuList() { $allowedEdit = User::getAllowedLocations('ipxe.menu.edit'); // TODO Permission::addGlobalTags($perms, null, ['edit.menu', 'edit.address', 'download']); $res = Database::simpleQuery("SELECT m.menuid, m.title, m.isdefault, GROUP_CONCAT(l.locationid) AS locations, GROUP_CONCAT(ll.locationname SEPARATOR ', ') AS locnames FROM serversetup_menu m LEFT JOIN serversetup_menu_location l USING (menuid) LEFT JOIN location ll USING (locationid) GROUP BY menuid, title ORDER BY title"); $menuTable = []; foreach ($res as $row) { if (empty($row['locations'])) { $locations = []; $row['allowEdit'] = in_array(0, $allowedEdit); } else { $locations = explode(',', $row['locations']); $row['allowEdit'] = in_array(0, $allowedEdit) || empty(array_diff($locations, $allowedEdit)); } $row['locationCount'] = empty($locations) ? '' : count($locations); $menuTable[] = $row; } Render::addTemplate('menu-list', array( 'menuTable' => $menuTable, 'showSetDefault' => User::hasPermission('ipxe.menu.edit', 0) )); } private function hasMenuPermission(int $menuid, string $permission): bool { $allowedEditLocations = User::getAllowedLocations($permission); $allowEdit = in_array(0, $allowedEditLocations); if (!$allowEdit) { // Get locations $locations = Database::queryColumnArray('SELECT locationid FROM serversetup_menu_location WHERE menuid = :menuid', compact('menuid')); if (!empty($locations)) { $allowEdit = count(array_diff($locations, $allowedEditLocations)) === 0; } } return $allowEdit; } private function showEditMenu() { $id = Request::get('id', false, 'int'); // if = edit, else = add new if ($id !== 0) { $menu = Database::queryFirst("SELECT menuid, timeoutms, title, defaultentryid, isdefault FROM serversetup_menu m WHERE menuid = :id", compact('id')); } else { $menu = []; $menu['menuid'] = 0; $menu['timeoutms'] = 0; $menu['title'] = ''; $menu['defaultentryid'] = null; $menu['isdefault'] = false; } if ($menu === false) { Message::addError('invalid-menu-id', $id); Util::redirect('?do=serversetup&show=menu'); } $highlight = Request::get('highlight', false, 'string'); if ($id !== 0 && !$this->hasMenuPermission($id, 'ipxe.menu.edit')) { $menu['readonly'] = 'readonly'; $menu['disabled'] = 'disabled'; $menu['plainpass'] = ''; } if (!User::hasPermission('ipxe.menu.edit', 0)) { $menu['globalMenuWarning'] = true; } $menu['timeout'] = round($menu['timeoutms'] / 1000); $menu['entries'] = []; $res = Database::simpleQuery("SELECT me.menuentryid, me.entryid, me.refmenuid, me.hotkey, me.title, me.hidden, me.sortval, me.plainpass, be.title AS parenttitle FROM serversetup_menuentry me LEFT JOIN serversetup_bootentry be USING (entryid) WHERE menuid = :id ORDER BY sortval ASC", compact('id')); foreach ($res as $row) { if ($row['entryid'] === null && $row['refmenuid'] !== null) { $row['entryid'] = 'menu:' . $row['refmenuid']; } if ($row['entryid'] == $highlight) { $row['highlight'] = 'active'; } $menu['entries'][] = $row; } $menu['keys'] = array_map(function ($item) { return ['key' => $item]; }, MenuEntry::getKeyList()); $menu['entrylist'] = array_merge( Database::queryAll("SELECT entryid, title, hotkey, module, data FROM serversetup_bootentry ORDER BY title ASC"), // Add all menus, so we can link Database::queryAll("SELECT Concat('menu:', menuid) AS entryid, title, 1 AS no_edit FROM serversetup_menu ORDER BY title ASC") ); foreach ($menu['entrylist'] as &$bootentry) { if (!isset($bootentry['data']) || !isset($bootentry['module'])) continue; if ($bootentry['module'][0] !== '.') { // Hook from other module $bootentry['moduleName'] = Dictionary::translateFileModule($bootentry['module'], 'module', 'module_name'); if (!$bootentry['moduleName']) { $bootentry['moduleName'] = $bootentry['module']; } $bootentry['ishook'] = true; $data = json_decode($bootentry['data'], true); unset($bootentry['data']); $bootentry['id'] = $data['id']; $bootentry['otherFields'] = []; foreach ($data as $k => $v) { if ($k === 'id') continue; $bootentry['otherFields'][] = [ 'key' => Dictionary::translateFileModule($bootentry['module'], 'module', 'ipxe-' . $k), 'value' => is_bool($v) ? Util::boolToString($v) : $v, ]; } continue; } $entry = BootEntry::fromJson($bootentry['module'], $bootentry['data']); if ($entry === null) { error_log('WARNING: Ignoring NULL menu entry: ' . $bootentry['data']); continue; } $bootentry['data'] = $entry->toArray(); // Transform stuff suitable for mustache if (!array_key_exists('arch', $bootentry['data'])) continue; // Naming and agnostic if ($bootentry['data']['arch'] === BootEntry::BIOS) { $bootentry['data']['arch'] = Dictionary::translateFile('template-tags','lang_biosOnly'); unset($bootentry['data']['EFI']); } elseif ($bootentry['data']['arch'] === BootEntry::EFI) { $bootentry['data']['arch'] = Dictionary::translateFile('template-tags','lang_efiOnly'); unset($bootentry['data']['PCBIOS']); } elseif ($bootentry['data']['arch'] === BootEntry::AGNOSTIC) { $bootentry['data']['arch'] = Dictionary::translateFile('template-tags','lang_archAgnostic'); unset($bootentry['data']['EFI']); } else { $bootentry['data']['arch'] = Dictionary::translateFile('template-tags','lang_archBoth'); } foreach ($bootentry['data'] as &$e) { if (isset($e['initRd'])) { $e['initRd'] = implode(',', $e['initRd']); } } } foreach ($menu['entries'] as &$entry) { $entry['isdefault'] = ($entry['menuentryid'] == $menu['defaultentryid']); // TODO: plainpass only when permissions } Permission::addGlobalTags($menu['perms'], 0, ['ipxe.menu.edit']); Render::addTemplate('menu-edit', $menu); } private function showEditBootEntry() { $params = ['hooks' => []]; foreach (Hook::load('ipxe-bootentry') as $hook) { $var = $hook->run(); if ($var instanceof BootEntryHook) { $var->moduleId = $hook->moduleId; $params['hooks'][] = $var; } } $id = Request::get('id', false, 'string'); if ($id === false) { $params['exec_checked'] = 'checked'; $params['entryid'] = 'u-' . dechex(mt_rand(0x1000, 0xffff)) . '-' . dechex(time()); } else { // Query existing entry $row = Database::queryFirst('SELECT entryid, title, builtin, module, data FROM serversetup_bootentry WHERE entryid = :id LIMIT 1', ['id' => $id]); if ($row === false) { Message::addError('invalid-boot-entry', $id); Util::redirect('?do=serversetup'); } if ($row['module'] === '.special') { Message::addError('cannot-edit-special', $id); Util::redirect('?do=serversetup'); } if ($row['module'][0] === '.') { // either script or exec entry $entry = BootEntry::fromJson($row['module'], $row['data']); if ($entry === null) { Message::addError('unknown-bootentry-type', $id); Util::redirect('?do=serversetup&show=bootentry'); } $entry->addFormFields($params); } else { // Hook from another module if (Module::get($row['module']) === false) { Message::addError('unknown-hook-module', $row['module']); } else { foreach ($params['hooks'] as $he) { /** @var BootEntryHook $he */ if ($he->moduleId === $row['module']) { $he->setSelected($row['data']); $he->checked = 'checked'; if (!$he->isValidId($he->getSelected())) { Message::addError('invalid-custom-entry-id', $row['module'], $row['data']); } break; } } } } $params['title'] = $row['title']; if (!Request::get('copy')) { $params['oldentryid'] = $params['entryid'] = $row['entryid']; $params['builtin'] = $row['builtin']; } $params['menus'] = Database::queryAll('SELECT m.menuid, m.title FROM serversetup_menu m INNER JOIN serversetup_menuentry me ON (me.menuid = m.menuid) WHERE me.entryid = :entryid', ['entryid' => $row['entryid']]); } if (!isset($params['entries']) || !is_array($params['entries'])) { $params['entries'] = []; } $f = []; foreach ($params['entries'] as $e) { if (isset($e['mode'])) { $f[] = $e['mode']; } } if (!in_array('PCBIOS', $f)) { $params['entries'][] = (new ExecData)->toFormFields('PCBIOS'); } if (!in_array('EFI', $f)) { $params['entries'][] = (new ExecData)->toFormFields('EFI'); } $params['disabled'] = User::hasPermission('ipxe.bootentry.edit') ? '' : 'disabled'; Render::addTemplate('ipxe-new-boot-entry', $params); } private function showEditAddress() { $status = IPxeBuilder::getVersionTaskResult(); $versions = false; if ($status === null) { $error = 'Taskmanager down'; } elseif (!empty($status['versions'])) { $versions = $status['versions']; foreach ($versions as &$version) { if ($version['hash'] === Property::get(IPxeBuilder::PROP_IPXE_HASH)) { $version['hash_selected'] = 'selected'; } $version['date_s'] = date('Y-m-d H:i', $version['date']); $version['hash_s'] = substr($version['hash'], 0, 7); } $error = false; } else { $error = $status['error'] ?? 'Unknown error'; } Render::addTemplate('ipaddress', array( 'ips' => $this->addrListTask['data']['addresses'] ?? [], 'chooseHintClass' => $this->hasIpSet ? '' : 'alert alert-danger', 'disabled' => ($this->getCompileTask() === false) ? '' : 'disabled', 'versions' => $versions, 'error' => $error, 'lastBuild' => Property::get(IPxeBuilder::PROP_IPXE_BUILDSTRING), )); } // ----------------------------------------------------------------------------------------------- private function getLocalAddresses(): void { $this->addrListTask = Taskmanager::submit('LocalAddressesList', array()); if ($this->addrListTask === false) { $this->addrListTask['data']['addresses'] = false; return; } if (!Taskmanager::isFinished($this->addrListTask)) { // TODO: Async if just displaying $this->addrListTask = Taskmanager::waitComplete($this->addrListTask['id'], 4000); } if (Taskmanager::isFailed($this->addrListTask) || !isset($this->addrListTask['data']['addresses'])) { $this->addrListTask['data']['addresses'] = false; return; } $sortIp = array(); foreach (array_keys($this->addrListTask['data']['addresses']) as $key) { $item =& $this->addrListTask['data']['addresses'][$key]; if (!isset($item['ip']) || !preg_match('/^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/', $item['ip']) || substr($item['ip'], 0, 4) === '127.') { unset($this->addrListTask['data']['addresses'][$key]); continue; } if ($this->currentAddress === $item['ip']) { $item['default'] = true; $this->hasIpSet = true; } $sortIp[] = $item['ip']; } unset($item); array_multisort($sortIp, SORT_STRING, $this->addrListTask['data']['addresses']); } private function deleteBootEntry() { $id = Request::post('deleteid', false, 'string'); if ($id === false) { Message::addError('main.parameter-missing', 'deleteid'); return; } $res = Database::exec("DELETE FROM serversetup_bootentry WHERE entryid = :entryid AND builtin = 0", array("entryid" => $id)); if ($res > 0) { Message::addSuccess('bootentry-deleted'); } Util::redirect('?do=serversetup&show=bootentry'); } private function setDefaultMenu() { $id = Request::post('menuid', false, 'int'); if ($id === false) { Message::addError('main.parameter-missing', 'menuid'); return; } Database::exec('UPDATE serversetup_menu SET isdefault = (menuid = :menuid)', ['menuid' => $id]); Message::addSuccess('menu-set-default'); } private function deleteMenu() { $id = Request::post('deleteid', false, 'int'); if ($id === false) { Message::addError('main.parameter-missing', 'deleteid'); return; } if (!$this->hasMenuPermission($id, 'ipxe.menu.edit')) { Message::addError('locations.no-permission-location', $id); return; } Database::exec("DELETE FROM serversetup_menu WHERE menuid = :menuid", array("menuid" => $id)); Message::addSuccess('menu-deleted'); } private function saveMenu() { $id = Request::post('menuid', false, 'int'); if ($id === false) { Message::addError('main.parameter-missing', 'menuid'); return; } $insertParams = [ 'title' => IPxe::sanitizeIpxeString(Request::post('title', '', 'string')), 'timeoutms' => abs(Request::post('timeout', 0, 'int') * 1000), ]; if ($id === 0) { $num = Database::queryFirst("SELECT Count(*) AS cnt FROM serversetup_menu"); $insertParams['def'] = ($num['cnt'] == 0) ? 1 : 0; Database::exec("INSERT INTO serversetup_menu (title, timeoutms, isdefault) VALUES (:title, :timeoutms, :def)", $insertParams); $menu['menuid'] = $id = Database::lastInsertId(); } else { $menu = Database::queryFirst("SELECT m.menuid FROM serversetup_menu m WHERE menuid = :id", compact('id')); if ($menu === false) { Message::addError('invalid-menu-id', $id); return; } $insertParams['menuid'] = $id; Database::exec('UPDATE serversetup_menu SET title = :title, timeoutms = :timeoutms WHERE menuid = :menuid', $insertParams); } $keepIds = []; $entries = Request::post('entry', false, 'array'); $wantedDefaultEntryId = Request::post('defaultentry', null, 'string'); $defaultEntryId = null; if ($entries) { foreach ($entries as $key => $entry) { if (!isset($entry['sortval'])) { error_log("Incomplete entry $key with: " . print_r($entry, true)); continue; } // Fallback defaults $entry += [ 'entryid' => null, 'title' => '', 'hidden' => 0, 'plainpass' => '', ]; $params = [ 'entryid' => null, 'refmenuid' => null, 'title' => IPxe::sanitizeIpxeString($entry['title']), 'sortval' => (int)$entry['sortval'], 'menuid' => $menu['menuid'], ]; $spacer = empty($entry['entryid']); if ($spacer) { // Spacer $params += [ 'hotkey' => '', 'hidden' => 0, // Doesn't make any sense 'plainpass' => '', // Doesn't make any sense ]; } else { $params += [ 'hotkey' => MenuEntry::filterKeyName($entry['hotkey']), 'hidden' => (int)$entry['hidden'], // TODO (needs hotkey to make sense) 'plainpass' => $entry['plainpass'], ]; if (preg_match('/^menu:(\d+)$/', $entry['entryid'], $out)) { $params['refmenuid'] = $out[1]; } else { $params['entryid'] = $entry['entryid']; } } if (is_numeric($key)) { // Update, use known key if (!$spacer && ((string)$key === $wantedDefaultEntryId || $defaultEntryId === null)) { // if still null, use whatever as fallback, in case user didn't select any $defaultEntryId = $key; } $keepIds[] = $key; $params['menuentryid'] = $key; $params['md5pass'] = IPxe::makeMd5Pass($entry['plainpass'], $key); $ret = Database::exec('UPDATE serversetup_menuentry SET entryid = :entryid, refmenuid = :refmenuid, hotkey = :hotkey, title = :title, hidden = :hidden, sortval = :sortval, plainpass = :plainpass, md5pass = :md5pass WHERE menuid = :menuid AND menuentryid = :menuentryid', $params, true); } else { $ret = Database::exec("INSERT INTO serversetup_menuentry (menuid, entryid, refmenuid, hotkey, title, hidden, sortval, plainpass, md5pass) VALUES (:menuid, :entryid, :refmenuid, :hotkey, :title, :hidden, :sortval, :plainpass, '')", $params, true); if ($ret) { $newKey = Database::lastInsertId(); // Check now that we have generated our key if (!$spacer && ((string)$key === $wantedDefaultEntryId || $defaultEntryId === null)) { // if still null, use whatever as fallback, in case user didn't select any $defaultEntryId = $newKey; } $keepIds[] = $newKey; if (!empty($entry['plainpass'])) { Database::exec('UPDATE serversetup_menuentry SET md5pass = :md5pass WHERE menuentryid = :id', [ 'md5pass' => IPxe::makeMd5Pass($entry['plainpass'], $newKey), 'id' => $newKey, ]); } } } if ($ret === false) { Message::addWarning('error-saving-entry', $entry['title'], Database::lastError()); } } Database::exec('DELETE FROM serversetup_menuentry WHERE menuid = :menuid AND menuentryid NOT IN (:keep)', ['menuid' => $menu['menuid'], 'keep' => $keepIds]); // Set default entry Database::exec('UPDATE serversetup_menu SET defaultentryid = :default WHERE menuid = :menuid', ['menuid' => $menu['menuid'], 'default' => $defaultEntryId]); } else { Database::exec('DELETE FROM serversetup_menuentry WHERE menuid = :menuid', ['menuid' => $menu['menuid']]); Database::exec('UPDATE serversetup_menu SET defaultentryid = NULL WHERE menuid = :menuid', ['menuid' => $menu['menuid']]); } Message::addSuccess('menu-saved'); if (Request::post('next') === 'reload') { Util::redirect('?do=serversetup&show=editmenu&id=' . $menu['menuid']); } } private function updateLocalAddress() { $newAddress = Request::post('ip', 'none', 'string'); $valid = false; foreach ($this->addrListTask['data']['addresses'] ?? [] as $item) { if ($item['ip'] !== $newAddress) continue; $valid = true; break; } if ($valid) { Property::setServerIp($newAddress); Util::redirect('?do=ServerSetup'); } else { Message::addError('invalid-ip', $newAddress); } Util::redirect(); } private function handleGetImage() { $file = "/opt/openslx/ipxe/openslx-bootstick.raw"; if (!is_readable($file)) { Message::addError('image-not-found'); return; } Header('Content-Type: application/octet-stream'); Header('Content-Disposition: attachment; filename="openslx-bootstick-' . Property::getServerIp() . '-raw.img"'); readfile($file); exit; } private function saveBootEntry() { $oldEntryId = Request::post('entryid', false, 'string'); $newId = Request::post('newid', false, 'string'); if (!preg_match('/^[a-z0-9\-_]{1,16}$/', $newId)) { Message::addError('main.parameter-empty', 'newid'); return; } $data = Request::post('entry', false); if (!is_array($data)) { Message::addError('missing-bootentry-data'); return; } $type = Request::post('type', false, 'string'); if ($type[0] === '.') { // Exec or script if ($type === '.exec') { $entry = BootEntry::newStandardBootEntry($data); } elseif ($type === '.script') { $entry = BootEntry::newCustomBootEntry($data); } else { $entry = null; } if ($entry === null) { Message::addError('main.empty-field'); Util::redirect('?do=serversetup&show=bootentry'); } $entryData = json_encode($entry->toArray()); } else { // Module hook $hook = Hook::loadSingle($type, 'ipxe-bootentry'); if ($hook === null) { Message::addError('unknown-bootentry-type', $type); return; } /** @var BootEntryHook $module */ $module = $hook->run(); $id = Request::post('selection-' . $type, false, 'string'); if (!$module->isValidId($id)) { Message::addError('invalid-custom-entry-id', $type, $id); return; } $entryData = ['id' => $id]; foreach ($module->extraFields() as $field) { if ($field->name === 'id') continue; $entryData[$field->name] = $field->fromPost($type); } $entryData = json_encode($entryData); } $params = [ 'entryid' => $newId, 'title' => Request::post('title', '', 'string'), 'module' => $type, 'data' => $entryData, ]; // New or update? if (empty($oldEntryId)) { // New entry Database::exec("INSERT INTO serversetup_bootentry (entryid, title, hotkey, builtin, module, data) VALUES (:entryid, :title, '', 0, :module, :data)", $params); Message::addSuccess('boot-entry-created', $newId); } else { // Edit existing entry $params['oldid'] = $oldEntryId; // Ignore .special, must never update $aff = Database::exec("UPDATE serversetup_bootentry SET entryid = If(builtin = 0, :entryid, entryid), title = :title, module = :module, data = :data WHERE entryid = :oldid AND module <> '.special'", $params); if ($aff > 0) { Message::addSuccess('boot-entry-updated', $newId); } else { Message::addWarning('nothing-changed-or-protected', $newId); } } if (Request::post('next') === 'reload') { Util::redirect('?do=serversetup&show=editbootentry&id=' . $newId); } Util::redirect('?do=serversetup&show=bootentry'); } private function showEditLocation() { $locationId = Request::get('locationid', false, 'int'); $loc = Location::get($locationId); if ($loc === false) { Message::addError('locations.invalid-location-id', $locationId); return; } User::assertPermission('ipxe.menu.assign', $locationId); // List of menu entries $res = Database::simpleQuery('SELECT me.menuentryid, If(Length(me.title) = 0, be.title, me.title) FROM serversetup_menuentry me INNER JOIN serversetup_bootentry be USING (entryid)'); $menuEntries = $res->fetchAll(PDO::FETCH_KEY_PAIR); // List of menus $data = [ 'locationid' => $locationId, 'locationName' => $loc['locationname'], ]; $menu = IPxeMenu::forLocation($loc['parentlocationid']); $data['defaultMenu'] = $menu; $res = Database::simpleQuery('SELECT m.defaultentryid AS menu_default, m.menuid, m.title, ml.locationid, ml.defaultentryid, GROUP_CONCAT(me.menuentryid) AS entries FROM serversetup_menu m LEFT JOIN serversetup_menu_location ml ON (m.menuid = ml.menuid AND ml.locationid = :locationid) INNER JOIN serversetup_menuentry me ON (m.menuid = me.menuid AND me.entryid IS NOT NULL) GROUP BY m.menuid, m.title ORDER BY m.title ASC', ['locationid' => $locationId]); $menus = []; $hasDefault = false; foreach ($res as $row) { $eids = explode(',', $row['entries']); $row['default_entry_title'] = $menuEntries[$row['menu_default']] ?? ''; $row['entries'] = []; foreach ($eids as $eid) { $row['entries'][] = [ 'id' => $eid, 'title' => $menuEntries[$eid], 'selected' => ($eid == $row['defaultentryid'] ? 'selected' : ''), ]; } if ($row['locationid'] !== null) { $hasDefault = true; $row['menu_selected'] = 'checked'; } $menus[] = $row; } if (!$hasDefault) { $data['default_selected'] = 'checked'; } $data['list'] = $menus; Render::addTemplate('menu-assign-location', $data); } private function saveLocationMenu() { $locationId = Request::post('locationid', false, 'int'); $loc = Location::get($locationId); if ($loc === false) { Message::addError('locations.invalid-location-id', $locationId); return; } User::assertPermission('ipxe.menu.assign', $locationId); $menuId = Request::post('menuid', false, 'int'); if ($menuId === 0) { Database::exec('DELETE FROM serversetup_menu_location WHERE locationid = :locationid', ['locationid' => $locationId]); Message::addSuccess('location-use-default', $loc['locationname']); return; } $defaultEntryId = Request::post('defaultentryid-' . $menuId, 0, 'int'); if ($defaultEntryId === 0) { $defaultEntryId = null; } Database::exec('INSERT INTO serversetup_menu_location (menuid, locationid, defaultentryid) VALUES (:menuid, :locationid, :defaultentryid) ON DUPLICATE KEY UPDATE menuid = :menuid, defaultentryid = :defaultentryid', [ 'menuid' => $menuId, 'locationid' => $locationId, 'defaultentryid' => $defaultEntryId ]); Message::addSuccess('location-menu-assigned', $loc['locationname']); } private function saveLocalboot() { $pcbios = Request::post('default-pcbios', 'SANBOOT', 'string'); $efi = Request::post('default-efi', 'EXIT', 'string'); if (!array_key_exists($pcbios, Localboot::BOOT_METHODS['PCBIOS'])) { Message::addError('localboot-invalid-method', $pcbios); return; } if (!array_key_exists($efi, Localboot::BOOT_METHODS['EFI'])) { Message::addError('localboot-invalid-method', $efi); return; } Localboot::setDefault($pcbios, $efi); $overrides = Request::post('override', [], 'array'); Database::exec('TRUNCATE TABLE serversetup_localboot'); foreach ($overrides as $model => $modes) { if (empty($modes)) // No override continue; $params = ['model' => urldecode($model), 'EFI' => null, 'PCBIOS' => null]; foreach (['EFI', 'PCBIOS'] as $m) { if (empty($modes[$m])) continue; if (!array_key_exists($modes[$m], Localboot::BOOT_METHODS[$m])) { Message::addWarning('localboot-invalid-method', $modes[$m]); continue; } $params[$m] = $modes[$m]; } if ($params['EFI'] === null && $params['PCBIOS'] === null) continue; Database::exec('INSERT INTO serversetup_localboot (systemmodel, pcbios, efi) VALUES (:model, :PCBIOS, :EFI)', $params); } Message::addSuccess('localboot-saved'); Util::redirect('?do=serversetup&show=localboot'); } private function execImportPxeMenu() { $content = Request::post('pxemenu', false, 'string'); if (empty($content)) { Message::addError('main.parameter-empty', 'pxemenu'); Util::redirect('?do=serversetup&show=import'); } $menu = PxeLinux::parsePxeLinux($content, false); if ($menu === null) { Message::addWarning('import-no-entries'); Util::redirect('?do=serversetup&show=import'); } if (Request::post('entries-only', 0, 'int') !== 0) { $foo = []; IPxe::importPxeMenuEntries($menu, $foo); Util::redirect('?do=serversetup&show=bootentry'); } else { $id = IPxe::insertMenu($menu, 'Imported Menu', null, 0, [], []); if ($id === null) { Message::addError('import-error'); Util::redirect('?do=serversetup&show=import'); } else { Util::redirect('?do=serversetup&show=editmenu&id=' . $id); } } } }