2013-06-03 17:02:08 +02:00
|
|
|
<?php
|
2013-06-07 13:29:11 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
2013-06-03 17:02:08 +02:00
|
|
|
|
|
|
|
namespace Icinga\Web;
|
|
|
|
|
|
|
|
use Icinga\Authentication\Auth;
|
|
|
|
use Icinga\Application\Benchmark;
|
|
|
|
use Icinga\Exception;
|
|
|
|
use Icinga\Application\Config;
|
2013-06-07 13:35:03 +02:00
|
|
|
use Icinga\Pdf\File;
|
2013-06-03 17:02:08 +02:00
|
|
|
use Icinga\Web\Notification;
|
|
|
|
use Zend_Layout as ZfLayout;
|
|
|
|
use Zend_Controller_Action as ZfController;
|
|
|
|
use Zend_Controller_Request_Abstract as ZfRequest;
|
|
|
|
use Zend_Controller_Response_Abstract as ZfResponse;
|
|
|
|
use Zend_Controller_Action_HelperBroker as ZfActionHelper;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base class for all core action controllers
|
|
|
|
*
|
|
|
|
* All Icinga Web core controllers should extend this class
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2013 Icinga-Web Team <info@icinga.org>
|
|
|
|
* @author Icinga-Web Team <info@icinga.org>
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU General Public License
|
2013-06-07 13:29:11 +02:00
|
|
|
* @package Icinga\Web
|
2013-06-03 17:02:08 +02:00
|
|
|
*/
|
|
|
|
class ActionController extends ZfController
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The Icinga Config object is available in all controllers. This is the
|
|
|
|
* modules config for module action controllers.
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $config;
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected $replaceLayout = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The current module name. TODO: Find out whether this shall be null for
|
|
|
|
* non-module actions
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $module_name;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The current controller name
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $controller_name;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The current action name
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $action_name;
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected $handlesAuthentication = false;
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected $modifiesSession = false;
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected $allowAccess = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The constructor starts benchmarking, loads the configuration and sets
|
|
|
|
* other useful controller properties
|
|
|
|
*
|
|
|
|
* @param ZfRequest $request
|
|
|
|
* @param ZfResponse $response
|
|
|
|
* @param array $invokeArgs Any additional invocation arguments
|
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
ZfRequest $request,
|
|
|
|
ZfResponse $response,
|
|
|
|
array $invokeArgs = array()
|
|
|
|
) {
|
|
|
|
Benchmark::measure('Action::__construct()');
|
|
|
|
if (Auth::getInstance()->isAuthenticated()
|
2013-06-07 13:29:11 +02:00
|
|
|
&& !$this->modifiesSession
|
|
|
|
&& !Notification::getInstance()->hasMessages()
|
2013-06-03 17:02:08 +02:00
|
|
|
) {
|
|
|
|
session_write_close();
|
|
|
|
}
|
2013-06-07 13:29:11 +02:00
|
|
|
$this->module_name = $request->getModuleName();
|
2013-06-03 17:02:08 +02:00
|
|
|
$this->controller_name = $request->getControllerName();
|
2013-06-07 13:29:11 +02:00
|
|
|
$this->action_name = $request->getActionName();
|
2013-06-03 17:02:08 +02:00
|
|
|
|
|
|
|
$this->loadConfig();
|
|
|
|
$this->setRequest($request)
|
2013-06-07 13:29:11 +02:00
|
|
|
->setResponse($response)
|
|
|
|
->_setInvokeArgs($invokeArgs);
|
2013-06-03 17:02:08 +02:00
|
|
|
$this->_helper = new ZfActionHelper($this);
|
|
|
|
|
|
|
|
if ($this->handlesAuthentication()
|
2013-06-07 13:29:11 +02:00
|
|
|
|| Auth::getInstance()->isAuthenticated()
|
|
|
|
) {
|
2013-06-03 17:02:08 +02:00
|
|
|
$this->allowAccess = true;
|
|
|
|
$this->init();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This is where the configuration is going to be loaded
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
protected function loadConfig()
|
|
|
|
{
|
|
|
|
$this->config = Config::getInstance();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Translates the given string with the global translation catalog
|
|
|
|
*
|
|
|
|
* @param string $string The string that should be translated
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function translate($string)
|
|
|
|
{
|
|
|
|
return t($string);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function creating a new widget
|
|
|
|
*
|
|
|
|
* @param string $name The widget name
|
2013-06-07 13:29:11 +02:00
|
|
|
* @param array|string $properties Optional widget properties
|
2013-06-03 17:02:08 +02:00
|
|
|
*
|
|
|
|
* @return Widget\AbstractWidget
|
|
|
|
*/
|
|
|
|
public function widget($name, $properties = array())
|
|
|
|
{
|
|
|
|
return Widget::create($name, $properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the current user has the given permission
|
|
|
|
*
|
|
|
|
* TODO: This has not been implemented yet
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @param $uri
|
2013-06-03 17:02:08 +02:00
|
|
|
* @param string $permission Permission name
|
2013-06-07 13:29:11 +02:00
|
|
|
* @internal param string $object No idea what this should have been :-)
|
2013-06-03 17:02:08 +02:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
final protected function hasPermission($uri, $permission = 'read')
|
|
|
|
{
|
|
|
|
return Auth::getInstance()->hasPermission($uri, $permission);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assert the current user has the given permission
|
|
|
|
*
|
|
|
|
* TODO: This has not been implemented yet
|
|
|
|
*
|
|
|
|
* @param string $permission Permission name
|
|
|
|
* @param string $object No idea what this should have been :-)
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @throws \Exception
|
2013-06-03 17:02:08 +02:00
|
|
|
* @return self
|
|
|
|
*/
|
|
|
|
final protected function assertPermission($permission, $object = null)
|
|
|
|
{
|
2013-06-07 13:29:11 +02:00
|
|
|
if (!$this->hasPermission($permission, $object)) {
|
2013-06-03 17:02:08 +02:00
|
|
|
// TODO: Log violation, create dedicated Exception class
|
|
|
|
throw new \Exception('Permission denied');
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Our benchmark wants to know when we started our dispatch loop
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function preDispatch()
|
|
|
|
{
|
|
|
|
Benchmark::measure('Action::preDispatch()');
|
2013-06-07 13:29:11 +02:00
|
|
|
if (!$this->allowAccess) {
|
2013-06-03 17:02:08 +02:00
|
|
|
$this->_request->setModuleName('default')
|
|
|
|
->setControllerName('authentication')
|
|
|
|
->setActionName('login')
|
|
|
|
->setDispatched(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->view->action_name = $this->action_name;
|
|
|
|
$this->view->controller_name = $this->controller_name;
|
|
|
|
$this->view->module_name = $this->module_name;
|
|
|
|
|
|
|
|
//$this->quickRedirect('/authentication/login?a=e');
|
|
|
|
}
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @param $url
|
|
|
|
* @param array $params
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
public function redirectNow($url, array $params = array())
|
|
|
|
{
|
|
|
|
$this->_helper->Redirector->gotoUrlAndExit($url);
|
|
|
|
}
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
public function handlesAuthentication()
|
|
|
|
{
|
|
|
|
return $this->handlesAuthentication;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render our benchmark
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function renderBenchmark()
|
|
|
|
{
|
|
|
|
return '<pre class="benchmark">'
|
2013-06-07 13:29:11 +02:00
|
|
|
. Benchmark::renderToHtml()
|
|
|
|
. '</pre>';
|
2013-06-03 17:02:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* After dispatch happend we are going to do some automagic stuff
|
|
|
|
*
|
|
|
|
* - Benchmark is completed and rendered
|
|
|
|
* - Notifications will be collected here
|
|
|
|
* - Layout is disabled for XHR requests
|
|
|
|
* - TODO: Headers with required JS and other things will be created
|
|
|
|
* for XHR requests
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function postDispatch()
|
|
|
|
{
|
|
|
|
Benchmark::measure('Action::postDispatch()');
|
2013-06-07 13:29:11 +02:00
|
|
|
|
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
// TODO: Move this elsewhere, this is just an ugly test:
|
|
|
|
if ($this->_request->getParam('filetype') === 'pdf') {
|
2013-06-07 13:29:11 +02:00
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
// Snippet stolen from less compiler in public/css.php:
|
|
|
|
|
|
|
|
require_once 'vendor/lessphp/lessc.inc.php';
|
|
|
|
$less = new \lessc;
|
|
|
|
$cssdir = dirname(ICINGA_LIBDIR) . '/public/css';
|
2013-06-07 13:29:11 +02:00
|
|
|
// TODO: We need a way to retrieve public dir, even if located elsewhere
|
2013-06-03 17:02:08 +02:00
|
|
|
|
|
|
|
$css = $less->compileFile($cssdir . '/pdfprint.less');
|
2013-06-07 13:29:11 +02:00
|
|
|
/*
|
|
|
|
foreach ($app->moduleManager()->getLoadedModules() as $name => $module) {
|
|
|
|
if ($module->hasCss()) {
|
|
|
|
$css .= $less->compile(
|
|
|
|
'.icinga-module.module-'
|
|
|
|
. $name
|
|
|
|
. " {\n"
|
|
|
|
. file_get_contents($module->getCssFilename())
|
|
|
|
. "}\n\n"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
|
|
|
|
// END of CSS test
|
2013-06-07 13:29:11 +02:00
|
|
|
|
|
|
|
$this->render(
|
2013-06-03 17:02:08 +02:00
|
|
|
null,
|
|
|
|
$this->_helper->viewRenderer->getResponseSegment(),
|
|
|
|
$this->_helper->viewRenderer->getNoController()
|
|
|
|
);
|
2013-06-07 13:29:11 +02:00
|
|
|
$html = '<style>' . $css . '</style>' . (string)$this->getResponse();
|
2013-06-03 17:02:08 +02:00
|
|
|
|
2013-06-07 13:35:03 +02:00
|
|
|
$pdf = new File();
|
2013-06-03 17:02:08 +02:00
|
|
|
$pdf->AddPage();
|
|
|
|
$pdf->writeHTMLCell(0, 0, '', '', $html, 0, 1, 0, true, '', true);
|
|
|
|
$pdf->Output('docs.pdf', 'I');
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
// END of PDF test
|
2013-06-07 13:29:11 +02:00
|
|
|
|
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
if ($this->_request->isXmlHttpRequest()) {
|
|
|
|
if ($this->replaceLayout || $this->_getParam('_render') === 'body') {
|
|
|
|
$this->_helper->layout()->setLayout('just-the-body');
|
|
|
|
header('X-Icinga-Target: body');
|
|
|
|
} else {
|
|
|
|
$this->_helper->layout()->setLayout('inline');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$notification = Notification::getInstance();
|
|
|
|
if ($notification->hasMessages()) {
|
|
|
|
$nhtml = '<ul class="notification">';
|
|
|
|
foreach ($notification->getMessages() as $msg) {
|
|
|
|
$nhtml .= '<li>['
|
2013-06-07 13:29:11 +02:00
|
|
|
. $msg->type
|
|
|
|
. '] '
|
|
|
|
. htmlspecialchars($msg->message);
|
2013-06-03 17:02:08 +02:00
|
|
|
}
|
|
|
|
$nhtml .= '</ul>';
|
|
|
|
$this->getResponse()->append('notification', $nhtml);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Session::getInstance()->show_benchmark) {
|
|
|
|
Benchmark::measure('Response ready');
|
|
|
|
$this->getResponse()->append('benchmark', $this->renderBenchmark());
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the token parameter is valid
|
|
|
|
*
|
|
|
|
* TODO: Could this make use of Icinga\Web\Session once done?
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @param int $maxAge Max allowed token age
|
2013-06-03 17:02:08 +02:00
|
|
|
* @param string $sessionId A specific session id (useful for tests?)
|
2013-06-07 13:29:11 +02:00
|
|
|
* @return bool
|
2013-06-03 17:02:08 +02:00
|
|
|
*/
|
|
|
|
public function hasValidToken($maxAge = 600, $sessionId = null)
|
|
|
|
{
|
|
|
|
$sessionId = $sessionId ? $sessionId : session_id();
|
|
|
|
$seed = $this->_getParam('seed');
|
2013-06-07 13:29:11 +02:00
|
|
|
if (!is_numeric($seed)) {
|
2013-06-03 17:02:08 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove quantitized timestamp portion so maxAge applies
|
|
|
|
$seed -= (intval(time() / $maxAge) * $maxAge);
|
|
|
|
$token = $this->_getParam('token');
|
|
|
|
return $token === hash('sha256', $sessionId . $seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a new seed/token pair
|
|
|
|
*
|
|
|
|
* TODO: Could this make use of Icinga\Web\Session once done?
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @param int $maxAge Max allowed token age
|
2013-06-03 17:02:08 +02:00
|
|
|
* @param string $sessionId A specific session id (useful for tests?)
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @return array
|
2013-06-03 17:02:08 +02:00
|
|
|
*/
|
|
|
|
public function getSeedTokenPair($maxAge = 600, $sessionId = null)
|
|
|
|
{
|
|
|
|
$sessionId = $sessionId ? $sessionId : session_id();
|
|
|
|
$seed = mt_rand();
|
|
|
|
$hash = hash('sha256', $sessionId . $seed);
|
|
|
|
|
|
|
|
// Add quantitized timestamp portion to apply maxAge
|
|
|
|
$seed += (intval(time() / $maxAge) * $maxAge);
|
|
|
|
return array($seed, $hash);
|
|
|
|
}
|
|
|
|
}
|