summaryrefslogblamecommitdiffstats
path: root/modules-available/minilinux/page.inc.php
blob: faea759982ba21580e1bb9051affff897239f76c (plain) (tree)
1
2
3
4
5
6
7
8
9

     

                                 
 


                                         
 




                                                                





                                                                       

                                                           



                                                        
                                               

                                                                                                                 
         
 

                                     




                                                                                                                      






                                                                                                                                                 
                                                                                                     


                                                                                                                          
                         
















                                                                                                                                                       
                         


                                                                        
                 
         


                                   


                                                               
                                                    

                                                 
                 

         

                                                     

                                                                                 

                                                                                        
                                                                                                                            
                                                                                                                                        








                                                                                      
                 
                                                                                             

         
                                             






                                                                       
                                                                                                                                            












                                                                                                                   
                                      




                                                                              
                                 


                                                                                     
                                                       
                                 











                                                                                                         
                                 
                         

                                                                         

                                                               





                                                                                        
                 


                                                                          






                                                      
                                    







                                                                                 

                                                       
                              







                                                                                                    
                         












                                                                                 

                                                                                

















                                                                       
                                                    














                                                                                                                       
                               
                 
                                                                           
















                                                                                                      
                                                                                                               

                                               




                                                               
                 
         
 

                                     
                                                 










                                                                                                                       
                                                                    

         
 
<?php

class Page_MiniLinux extends Page
{

	protected function doPreprocess()
	{
		User::load();

		if (!User::isLoggedIn()) {
			Message::addError('main.no-permission');
			Util::redirect('?do=Main');
		}

		if (Request::isPost()) {
			$show = Request::post('show', false, 'string');
			if ($show === 'delete') {
				$this->deleteVersion();
			} elseif ($show === 'updatesources') {
				$this->updateSources();
			} elseif ($show === 'setdefault') {
				$this->setDefault();
			}
			Util::redirect('?do=minilinux');
		}

		User::assertPermission('view');
		Dashboard::addSubmenu('?do=minilinux', Dictionary::translate('menu-versions', true));
		Dashboard::addSubmenu('?do=minilinux&show=sources', Dictionary::translate('menu-sources', true));
	}

	protected function doRender()
	{
		Render::addTemplate('page-minilinux', ['default' => Property::get(MiniLinux::PROPERTY_DEFAULT_BOOT)]);
		// Warning
		if (!MiniLinux::updateCurrentBootSetting()) {
			Message::addError('default-not-installed', Property::get(MiniLinux::PROPERTY_DEFAULT_BOOT));
		}
		$show = Request::get('show', 'list', 'string');
		if ($show === 'list') {
			// List branches and versions
			$branches = Database::queryAll('SELECT sourceid, branchid, title, description FROM minilinux_branch ORDER BY title ASC');
			$versions = MiniLinux::queryAllVersionsByBranch();
			// Group by branch for detailed listing
			foreach ($branches as &$branch) {
				$branch['bid'] = 'div-' . str_replace('/', '-', $branch['branchid']);
				if (isset($versions[$branch['branchid']])) {
					$branch['versionlist'] = $this->renderVersionList($versions[$branch['branchid']]);
				}
			}
			unset($branch);
			Render::addTemplate('branches', ['branches' => $branches]);
		} elseif ($show === 'sources') {
			// List sources
			$res = Database::simpleQuery('SELECT sourceid, title, url, lastupdate, pubkey FROM minilinux_source ORDER BY title, sourceid');
			$data = ['list' => [], 'show_refresh' => true];
			$tooOld = strtotime('-7 days');
			$showRefresh = strtotime('-10 minutes');
			while ($row = $res->fetch(PDO::FETCH_ASSOC)) {
				$row['lastupdate_s'] = Util::prettyTime($row['lastupdate']);
				if ($row['lastupdate'] != 0 && $row['lastupdate'] < $tooOld) {
					$row['update_class'] = 'text-danger';
				}
				if ($row['lastupdate'] > $showRefresh) {
					$data['show_refresh'] = false;
				}
				$data['list'][] = $row;
			}
			Render::addTemplate('sources', $data);
		} else {
			Message::addError('main.invalid-action', $show);
		}
	}
	
	protected function doAjax()
	{
		User::load();
		$show = Request::post('show', false, 'string');
		if ($show === 'version') {
			$this->ajaxVersionDetails();
		} elseif ($show === 'download') {
			$this->ajaxDownload();
		}
	}

	private function renderVersionList($versions)
	{
		$def = Property::get(MiniLinux::PROPERTY_DEFAULT_BOOT);
		$eff = Property::get(MiniLinux::PROPERTY_DEFAULT_BOOT_EFFECTIVE);
		foreach ($versions as &$version) {
			$version['dateline_s'] = Util::prettyTime($version['dateline']);
			$version['orphan'] = ($version['orphan'] > 0 && !$version['installed']) || ($version['orphan'] > 1);
			$version['downloading'] = $version['taskid'] && Taskmanager::isRunning(Taskmanager::status($version['taskid']));
			if ($version['installed'] && $version['versionid'] !== $def) {
				$version['showsetdefault'] = true;
			}
			if ($version['versionid'] === $def) {
				$version['isdefault'] = true;
				if (!$version['installed']) {
					$version['default_class'] = 'bg-danger';
				}
			}
		}
		return Render::parse('versionlist', ['versions' => array_values($versions)]);
	}

	private function ajaxVersionDetails()
	{
		User::assertPermission('view');
		$verify = Request::post('verify', false, 'bool');
		$versionid = Request::post('version', false, 'string');
		if ($versionid === false) {
			die('What!');
		}
		$ver = Database::queryFirst('SELECT versionid, taskid, data, installed FROM minilinux_version WHERE versionid = :versionid',
			['versionid' => $versionid]);
		if ($ver === false) {
			die('No such version');
		}
		$versionid = $ver['versionid']; // Just to be sure -- should be safe for building a path either way
		$data = json_decode($ver['data'], true);
		if (!is_array($data)) {
			die('Corrupted data');
		}
		$data['versionid'] = $versionid;
		$data['dltask'] = MiniLinux::validateDownloadTask($versionid, $ver['taskid']);
		$data['verify_button'] = !$verify && $data['dltask'] === false;
		if (is_array($data['files'])) {
			$valid = true;
			$sort = [];
			foreach ($data['files'] as &$file) {
				if (empty($file['name'])) {
					$sort[] = 'zzz' . implode(',', $file);
					continue;
				}
				$sort[] = $file['name'];
				$s = $this->getFileState($versionid, $file, $verify);
				if ($s !== self::FILE_OK) {
					$valid = false;
				}
				if ($s !== self::FILE_MISSING) {
					$data['delete_button'] = true;
				}
				$file['state'] = $this->fileStateToString($s);
				if (isset($file['size'])) {
					$file['size_s'] = Util::readableFileSize($file['size']);
				}
				if (isset($file['mtime'])) {
					$file['mtime_s'] = Util::prettyTime($file['mtime']);
				}
				if ($data['dltask']) {
					$file['fileid'] = MiniLinux::fileToId($versionid, $file['name']);
				}
			}
			unset($file);
			array_multisort($sort, SORT_ASC, $data['files']);
			if (!$valid) {
				$data['verify_button'] = false;
				if ($ver['installed']) {
					MiniLinux::setInstalledState($versionid, false);
				}
			} elseif (!$ver['installed'] && $verify) {
				MiniLinux::setInstalledState($versionid, true);
			}
		}
		if ($data['dltask'] !== false || $ver['installed']) {
			MiniLinux::checkStage4($data, $data['s4_errors']);
		}
		echo Render::parse('filelist', $data);
	}

	const FILE_OK = 0;
	const FILE_MISSING = 1;
	const FILE_SIZE_MISMATCH = 2;
	const FILE_CHECKSUM_BAD = 3;
	const FILE_NOT_READABLE = 4;

	private function getFileState($versionid, $file, $verify)
	{
		$path = CONFIG_HTTP_DIR . '/' . $versionid . '/' . $file['name'];
		if (!is_file($path))
			return self::FILE_MISSING;
		if (isset($file['size']) && filesize($path) != $file['size'])
			return self::FILE_SIZE_MISMATCH;
		if (!is_readable($path))
			return self::FILE_NOT_READABLE;
		if ($verify) {
			foreach (['sha512', 'sha384', 'sha256', 'sha224', 'sha1', 'md5'] as $algo) {
				if (isset($file[$algo])) {
					$calced = hash_file($algo, $path);
					if ($calced === false)
						continue; // Algo not supported?
					if ($calced !== $file['md5'])
						return self::FILE_CHECKSUM_BAD;
				}
			}
		}
		return self::FILE_OK;
	}

	private function fileStateToString($state)
	{
		switch ($state) {
		case self::FILE_CHECKSUM_BAD:
			return Dictionary::translate('file-checksum-bad', true);
		case self::FILE_SIZE_MISMATCH:
			return Dictionary::translate('file-size-mismatch', true);
		case self::FILE_MISSING:
			return Dictionary::translate('file-missing', true);
		case self::FILE_NOT_READABLE:
			return Dictionary::translate('file-not-readable', true);
		case self::FILE_OK:
			return Dictionary::translate('file-ok', true);
		}
		return '???';
	}

	private function ajaxDownload()
	{
		User::assertPermission('update');
		$version = Request::post('version', false, 'string');
		if ($version === false) {
			die('No version');
		}
		$task = MiniLinux::downloadVersion($version);
		if ($task === false) {
			Message::addError('no-such-version', $version);
			Message::renderList();
		} else {
			$this->ajaxVersionDetails();
		}
	}

	private function deleteVersion()
	{
		User::assertPermission('delete');
		$versionid = Request::post('version', false, 'string');
		if ($versionid === false) {
			Message::addError('main.parameter-missing', 'versionid');
			return;
		}
		$version = Database::queryFirst('SELECT versionid FROM minilinux_version WHERE versionid = :versionid',
			['versionid' => $versionid]);
		if ($version === false) {
			Message::addError('no-such-version');
			return;
		}
		MiniLinux::setInstalledState($version['versionid'], false);
		$path = CONFIG_HTTP_DIR . '/' . $version['versionid'];
		$task = Taskmanager::submit('DeleteDirectory', [
			'path' => $path,
			'recursive' => true,
		]);
		if ($task !== false) {
			$task = Taskmanager::waitComplete($task, 2500);
			if (Taskmanager::isFailed($task)) {
				Message::addError('delete-error', $versionid, $task['data']['error']);
			} else {
				Message::addSuccess('version-deleted', $versionid);
			}
		}
	}

	private function updateSources()
	{
		User::assertPermission('view'); // As it doesn't really change anything, accept view permission
		$ret = MiniLinux::updateList();
		if ($ret > 0) {
			for ($i = 0; $i < 6; ++$i) {
				sleep(1);
				if (!Trigger::checkCallbacks())
					break;
			}
		}
	}

	private function setDefault()
	{
		User::assertPermission('update');
		$versionid = Request::post('version', false, 'string');
		if ($versionid === false) {
			Message::addError('main.parameter-missing', 'versionid');
			return;
		}
		$version = Database::queryFirst('SELECT versionid FROM minilinux_version WHERE versionid = :versionid',
			['versionid' => $versionid]);
		if ($version === false) {
			Message::addError('no-such-version');
			return;
		}
		MiniLinux::setDefaultVersion($version['versionid']);
	}

}