PHP Classes

File: src/Sentry.php

Recommend this page to a friend!
  Classes of Thierry Feuzeu   Jaxon Sentry   src/Sentry.php   Download  
File: src/Sentry.php
Role: Class source
Content type: text/plain
Description: Class source
Class: Jaxon Sentry
Common classes for Jaxon based Ajax applications
Author: By
Last change: Allow third party integration in DI.
Added a DI container, to store Armada packages.
Date: 2 years ago
Size: 17,744 bytes
 

Contents

Class file image Download
<?php namespace Jaxon\Sentry; use Jaxon\Jaxon; use Jaxon\Utils\Traits\View; use Jaxon\Utils\Traits\Session; use Jaxon\Utils\Traits\Manager; use Jaxon\Utils\Traits\Event; use Jaxon\Utils\Traits\Validator; use Jaxon\Utils\Config\Config; use stdClass; use Exception; use Closure; class Sentry { use View, Session, Manager, Event, Validator; protected $aClassInitCallbacks = []; protected $xBeforeCallback = null; protected $xAfterCallback = null; protected $xInitCallback = null; protected $xInvalidCallback = null; protected $xErrorCallback = null; // Requested class and method private $xRequestObject = null; private $sRequestMethod = null; protected $aClassOptions = []; protected $appConfig = null; protected $xResponse = null; protected $aViewRenderers = array(); protected $aViewNamespaces = array(); // The Dependency Injection Container private $di = null; public function __construct() { $this->di = new \Pimple\Container(); } /** * Setup the library. * * @return void */ public function setup() { $jaxon = jaxon(); // Create the Jaxon response $this->xResponse = $jaxon->getResponse(); // Add the view renderer $this->addViewRenderer('sentry', function () { return new \Jaxon\Sentry\View\View(); }); // Set the pagination view namespace $this->addViewNamespace('pagination', '', '', 'sentry'); // Set the pagination renderer $jaxon->setPaginationRenderer(function () { return new \Jaxon\Sentry\Pagination\Renderer(); }); } /** * Get the Jaxon response. * * @return Jaxon\Response\Response */ public function ajaxResponse() { return $this->xResponse; } /** * Add a class namespace. * * @param string $sDirectory The path to the directory * @param string $sNamespace The associated namespace * @param string $sSeparator The character to use as separator in javascript class names * @param array $aProtected The functions that are not to be exported * * @return void */ public function addClassNamespace($sDirectory, $sNamespace, $sSeparator = '.', array $aProtected = array()) { // Valid separator values are '.' and '_'. Any other value is considered as '.'. $sSeparator = trim($sSeparator); if($sSeparator != '_') { $sSeparator = '.'; } jaxon()->addClassDir(trim($sDirectory), trim($sNamespace), $sSeparator, $aProtected); } /** * Read class namespaces from config values. * * @param Config $xAppConfig The application config options * * @return void */ public function addClassNamespaces($xAppConfig) { if($xAppConfig->hasOption('classes') && is_array($xAppConfig->getOption('classes'))) { $aNamespaces = $xAppConfig->getOption('classes'); // The public methods of the base class must not be exported to javascript $protected = array(); $baseClass = new \ReflectionClass('\\Jaxon\\Sentry\\Armada'); foreach ($baseClass->getMethods(\ReflectionMethod::IS_PUBLIC) as $xMethod) { $protected[] = $xMethod->getShortName(); } foreach($aNamespaces as $aNamespace) { // Check mandatory options if(!key_exists('directory', $aNamespace) || !key_exists('namespace', $aNamespace)) { continue; } // Set the default values for optional parameters if(!key_exists('separator', $aNamespace)) { $aNamespace['separator'] = '.'; } if(!key_exists('protected', $aNamespace)) { $aNamespace['protected'] = []; } $this->addClassNamespace($aNamespace['directory'], $aNamespace['namespace'], $aNamespace['separator'], array_merge($aNamespace['protected'], $protected)); } } } /** * Read class namespaces from config values. * * @param Config $xAppConfig The application config options * * @return void */ public function setClassNamespaces($xAppConfig) { $this->addClassNamespaces($xAppConfig); } /** * Read class options from config values. * * @param array $aOptions The application config options * * @return void */ public function mergeClassOptions($aOptions) { // The new options overwrite those already saved, and the merge // shall be done at option level, and not at class level. foreach($aOptions as $class => $aClassOption) { if(!array_key_exists($class, $this->aClassOptions)) { $this->aClassOptions[$class] = $aClassOption; } else { foreach($aClassOption as $key => $aOption) { // Merge the options $this->aClassOptions[$class][$key] = array_key_exists($key, $this->aClassOptions[$class]) ? array_merge($this->aClassOptions[$class][$key], $aOption) : $aOption; } } } } /** * Read class options from config values. * * @param Config $xAppConfig The application config options * * @return void */ public function addClassOptions($xAppConfig) { $aOptions = $xAppConfig->getOption('options.classes', []); $this->mergeClassOptions($aOptions); } /** * Add a view namespace, and set the corresponding renderer. * * @param string $sNamespace The namespace name * @param string $sDirectory The namespace directory * @param string $sExtension The extension to append to template names * @param string $sRenderer The corresponding renderer name * * @return void */ public function addViewNamespace($sNamespace, $sDirectory, $sExtension, $sRenderer) { $aNamespace = array( 'namespace' => $sNamespace, 'directory' => $sDirectory, 'extension' => $sExtension, ); if(key_exists($sRenderer, $this->aViewNamespaces)) { $this->aViewNamespaces[$sRenderer][] = $aNamespace; } else { $this->aViewNamespaces[$sRenderer] = array($aNamespace); } $this->aViewRenderers[$sNamespace] = $sRenderer; } /** * Set the view namespaces. * * @param Config $xAppConfig The application config options * * @return void */ public function addViewNamespaces($xAppConfig) { $sDefaultNamespace = $xAppConfig->getOption('options.views.default', false); if(is_array($namespaces = $xAppConfig->getOptionNames('views'))) { foreach($namespaces as $namespace => $option) { // If no default namespace is defined, use the first one as default. if($sDefaultNamespace == false) { $sDefaultNamespace = $namespace; } // Save the namespace $directory = $xAppConfig->getOption($option . '.directory'); $extension = $xAppConfig->getOption($option . '.extension', ''); $renderer = $xAppConfig->getOption($option . '.renderer', 'jaxon'); $this->addViewNamespace($namespace, $directory, $extension, $renderer); } } // Save the view renderers and namespaces in the DI container $this->initViewRenderers($this->aViewRenderers); $this->initViewNamespaces($this->aViewNamespaces, $sDefaultNamespace); } /** * Set the view namespaces. * * @param Config $xAppConfig The application config options * * @return void */ public function setViewNamespaces($xAppConfig) { $this->addViewNamespaces($xAppConfig); } /** * Set the Jaxon library default options. * * @return void */ public function setLibraryOptions($bExtern, $bMinify, $sJsUri, $sJsDir) { $jaxon = jaxon(); // Jaxon library settings if(!$jaxon->hasOption('js.app.extern')) { $jaxon->setOption('js.app.extern', $bExtern); } if(!$jaxon->hasOption('js.app.minify')) { $jaxon->setOption('js.app.minify', $bMinify); } if(!$jaxon->hasOption('js.app.uri')) { $jaxon->setOption('js.app.uri', $sJsUri); } if(!$jaxon->hasOption('js.app.dir')) { $jaxon->setOption('js.app.dir', $sJsDir); } // Set the request URI if(!$jaxon->hasOption('core.request.uri')) { $jaxon->setOption('core.request.uri', 'jaxon'); } } /** * Add a callback to initialise specific class instances. * * @param string $sNamespace The namespace name * @param callable $callable The callback function * * @return void */ public function addClassInitializer($sNamespace, $callable) { $sNamespace = trim($sNamespace, '\\ ') . '\\'; $this->aClassInitCallbacks[$sNamespace] = $callable; } /** * Set the init callback, used to initialise class instances. * * @param callable $callable The callback function * * @return void */ public function setInitCallback($callable) { $this->xInitCallback = $callable; } /** * Set the pre-request processing callback. * * @param callable $callable The callback function * * @return void */ public function setBeforeCallback($callable) { $this->xBeforeCallback = $callable; } /** * Set the post-request processing callback. * * @param callable $callable The callback function * * @return void */ public function setAfterCallback($callable) { $this->xAfterCallback = $callable; } /** * Set the processing error callback. * * @param callable $callable The callback function * * @return void */ public function setInvalidCallback($callable) { $this->xInvalidCallback = $callable; } /** * Set the processing exception callback. * * @param callable $callable The callback function * * @return void */ public function setErrorCallback($callable) { $this->xErrorCallback = $callable; } /** * Initialize a class instance. * * @return void */ protected function initInstance(Armada $instance) { // Return if the class instance has already been initialized. if(!($instance) || ($instance->response)) { return; } // Init the class instance $instance->response = $this->xResponse; $instance->rqFactory = new Factory\Request($instance); $instance->pgFactory = new Factory\Paginator($instance); if(($this->xInitCallback)) { call_user_func_array($this->xInitCallback, array($instance)); } // Apply class initializer foreach($this->aClassInitCallbacks as $namespace => $callback) { $classname = get_class($instance); if(strpos($classname, $namespace) !== false) { // Call the callback passing the class instance as parameter $callback($instance); } } // Call the init() function of the instance class $instance->init(); } /** * Get a class instance. * * @param string $classname the class name * * @return Jaxon\Sentry\Armada|null The registered instance of the class */ public function instance($classname) { // Find the class instance, and register the class if the instance is not found. if(!($instance = jaxon()->getPluginManager()->getRegisteredObject($classname))) { $instance = jaxon()->registerClass($classname, [], true); } if(($instance)) { $this->initInstance($instance); } return $instance; } /** * Get a Jaxon request to a given class. * * @param string $classname the class name * * @return Jaxon\Request\Request|null The request to the class */ public function request($classname) { $instance = $this->instance($classname); return ($instance != null ? $instance->request() : null); } /** * This is the pre-request processing callback passed to the Jaxon library. * * @param boolean &$bEndRequest if set to true, the request processing is interrupted. * * @return Jaxon\Response\Response the Jaxon response */ public function onEventBefore(&$bEndRequest) { // Validate the inputs $class = $_POST['jxncls']; $method = $_POST['jxnmthd']; if(!$this->validateClass($class) || !$this->validateMethod($method)) { // End the request processing if the input data are not valid. // Todo: write an error message in the response $bEndRequest = true; return $this->xResponse; } // Instanciate the class. This will include the required file. $this->xRequestObject = $this->instance($class); $this->sRequestMethod = $method; if(!$this->xRequestObject) { // End the request processing if the class cannot be found. // Todo: write an error message in the response $bEndRequest = true; return $this->xResponse; } // Call the user defined callback if(($this->xBeforeCallback)) { call_user_func_array($this->xBeforeCallback, array($this->xRequestObject, $this->sRequestMethod, &$bEndRequest)); } return $this->xResponse; } /** * This is the post-request processing callback passed to the Jaxon library. * * @return Jaxon\Response\Response the Jaxon response */ public function onEventAfter() { if(($this->xAfterCallback)) { call_user_func_array($this->xAfterCallback, array($this->xRequestObject, $this->sRequestMethod)); } return $this->xResponse; } /** * This callback is called whenever an invalid request is processed. * * @return Jaxon\Response\Response the Jaxon response */ public function onEventInvalid($sMessage) { if(($this->xInvalidCallback)) { call_user_func_array($this->xInvalidCallback, array($this->xResponse, $sMessage)); } return $this->xResponse; } /** * This callback is called whenever an invalid request is processed. * * @return Jaxon\Response\Response the Jaxon response */ public function onEventError(Exception $e) { if(($this->xErrorCallback)) { call_user_func_array($this->xErrorCallback, array($this->xResponse, $e)); } else { throw $e; } return $this->xResponse; } /** * Register the Jaxon classes. * * @return void */ public function register() { jaxon()->registerClasses($this->aClassOptions); } /** * Process the current Jaxon request. * * @return void */ public function processRequest() { // Process Jaxon Request $jaxon = jaxon(); if($jaxon->canProcessRequest()) { $jaxon->register(Jaxon::PROCESSING_EVENT, Jaxon::PROCESSING_EVENT_BEFORE, array($this, 'onEventBefore')); $jaxon->register(Jaxon::PROCESSING_EVENT, Jaxon::PROCESSING_EVENT_AFTER, array($this, 'onEventAfter')); $jaxon->register(Jaxon::PROCESSING_EVENT, Jaxon::PROCESSING_EVENT_INVALID, array($this, 'onEventInvalid')); $jaxon->register(Jaxon::PROCESSING_EVENT, Jaxon::PROCESSING_EVENT_ERROR, array($this, 'onEventError')); // Traiter la requete $jaxon->processRequest(); } } /** * Register a package in the DI container * * @param string $name The package name * @param Closure $closure The package instance, or a closure that returns the instance * * @return void */ public function registerPackage($name, Closure $closure) { $this->di['package.' . $name] = $closure; } /** * Get a package instance from the DI container * * @param string $name The package name * * @return mixed The package instance */ public function getPackage($name) { return $this->di['package.' . $name]; } }