2 /* vim: set expandtab tabstop=4 shiftwidth=4: */
3 // +----------------------------------------------------------------------+
5 // +----------------------------------------------------------------------+
6 // | Copyright (c) 1997-2002 The PHP Group |
7 // +----------------------------------------------------------------------+
8 // | This source file is subject to version 2.02 of the PHP license, |
9 // | that is bundled with this package in the file LICENSE, and is |
10 // | available at through the world-wide-web at |
11 // | http://www.php.net/license/2_02.txt. |
12 // | If you did not receive a copy of the PHP license and are unable to |
13 // | obtain it through the world-wide-web, please send a note to |
14 // | license@php.net so we can mail you a copy immediately. |
15 // +----------------------------------------------------------------------+
16 // | Authors: Alan Knowles <alan@akbkhome.com> |
17 // +----------------------------------------------------------------------+
19 // $Id: FlexyFramework.php,v 1.8 2003/02/22 01:52:50 alan Exp $
22 // A Page (URL) to Object Mapper
23 // Cleaned up version.. - for use on new projects -- not BC!! beware!!!
26 //-----------------------------------------------------------
27 // Debian APACHE - some idiot disabled AcceptPathInfo - it needs adding back.
28 //-----------------------------------------------------------
32 // Initialize Static Options
33 require_once 'HTML/FlexyFramework2/Page.php';
36 // To be removed ?? or made optional or something..
41 error_reporting(E_ALL & ~E_STRICT );
42 //ini_set('display_errors','off');
43 //ini_set('log_errors','off');
51 * The URL to Object Mapper
54 * Create a index.php and add these lines.
56 * ini_set("include_path", "/path/to/application");
57 * require_once 'HTML/FlexyFramework.php';
58 * HTML_FlexyFramework::factory(array("dir"=>"/where/my/config/dir/is");
61 * the path could include pear's path, if you dont install all the pear
62 * packages into the development directory.
64 * It attempts to load a the config file from the includepath,
65 * looks for ConfigData/default.ini
66 * or ConfigData/{hostname}.ini
67 * if your file is called staging.php rather than index.php
68 * it will try staging.ini
72 class HTML_FlexyFramework2 {
75 * Confirgurable items..
76 * If we set them to 'true', they must be set, otherwise they are optional.
78 var $project; // base class name
79 var $database; // set to true even if nodatabase=true
83 var $enable = false; // modules
84 var $disable = false; // modules or permissions
86 var $appNameShort = false; // appname (which has templates)
87 var $version = false; // give it a version name. (appended to compile dir)
88 var $nodatabase = false; // set to true to block db config and testing.
89 var $fatalAction = false; // page to redirct to on failure. (eg. databse down etc.)
90 var $charset = false; // default UTF8
91 var $dataObjectsCache = true; // use dataobjects ini cache.. - let's try this as the default behaviour...
92 var $dataObjectsCacheExpires = 72000; // 20 hours..
93 var $languages = false; // language settings -- see _handlelanguage
94 var $projectExtends = false; // if this is an array, it's a fallback of 'Projects' that can be called
95 var $initOnly = false; // use to prevent run() being called...
96 // use $this->run($_SERVER['REQUEST_URI'],false); to manually boot it..
100 var $cli = false; // from cli
101 var $run = false; // from cli
102 var $enableArray = false; // from enable.
103 var $classPrefix = false; // from prject.
104 var $baseDir = false ; // (directory+project)
105 var $rootDir = false ; // (directory that index.php is in!)
107 var $baseURL = false;
108 var $rootURL = false ; // basename($baseURL)
110 var $page = false; // active page..
111 var $timer = false; // the debug timer
112 var $calls = false; // the number of calls made to run!
113 var $start = false; // the start tiem.
115 var $baseRequest = '';
116 var $ext; // the striped extention.
118 var $dataObjectsOriginalIni = ''; // 1 houre..
120 // used to be $_GLOBALS[__CLASS__]
127 * Constructor - with assoc. array of props as option
128 * called by index.php usually, and runs the app code,
130 * uses 'universal construcor' format, so the argument relates directly to properties of this object.
135 function __construct($config)
137 if (isset(self::$singleton)) {
138 trigger_error("FlexyFramework Construct called twice!", E_ERROR);
141 self::$singleton = $this;
145 $m = explode(' ',microtime());
146 $this->start = $m[0] + $m[1];
148 $config = $this->loadModuleConfig($config);
150 foreach($config as $k=>$v) {
153 $this->_parseConfig();
155 // echo '<PRE>'; print_r($this);exit;
157 $args = $_SERVER['argv'];
160 $this->_run($this->run,false,$args);
164 // handle apache mod_rewrite..
165 // it looks like this might not work anymore..
167 if ($this->initOnly) {
172 if (!empty($_SERVER['REDIRECT_URL'])) {
174 $this->_run($_SERVER['SCRIPT_NAME'] . $_SERVER['REQUEST_URI'],false);
179 $this->_run($_SERVER['REQUEST_URI'],false);
184 * This is the standard way to get information about the application settings.
185 * $ff = HTML_FlexyFramework::get();
186 * if ($ff->SomeVar[...])....
189 static function get()
191 return self::$singleton;
194 * looks for files in the path and load up the default values for config?
196 function loadModuleConfig($cfg)
200 $proj = $cfg['project'];
201 $rootDir = realpath(dirname($_SERVER["SCRIPT_FILENAME"]));
203 $cls = $proj.'_Config';
204 if (file_exists($rootDir . '/'.str_replace('_','/', $cls). '.php')) {
205 require_once str_replace('_','/', $cls). '.php';
207 if (method_exists($c,'init')) {
208 $cfg = $c->init($this,$cfg);
212 $mods = empty($cfg['enable']) ? array() : explode(',',$cfg['enable']);
213 array_unshift($mods,'');
215 foreach($mods as $m) {
216 $cls = $proj. (strlen($m) ? '_'. $m : '' ) . '_Config';
218 if (!file_exists($rootDir . '/'.str_replace('_','/', $cls). '.php')) {
221 require_once str_replace('_','/', $cls). '.php';
223 if (method_exists($c,'init')) {
224 $cfg = $c->init($this,$cfg);
232 * parse the configuration set by the constructor.
237 function _parseConfig()
240 // make sure required values are set.. (anything that is not defaulted to false..)
241 foreach(get_class_vars(__CLASS__) as $k =>$v) {
242 if ($v === false && !isset($this->$k)) {
243 die("$k is not set");
247 $this->_handleLanguages();
250 if (!empty($this->enable)) {
251 $this->enableArray = explode(',', $this->enable);
253 if (!in_array('Core',$this->enableArray ) &&
254 !in_array('Core', explode(',', $this->disable ? $this->disable : '')))
256 $this->enable = 'Core,'. $this->enable ;
257 $this->enableArray = explode(',', $this->enable);
260 // are we running cli?
261 $this->cli = php_sapi_name() == 'cli';
263 // will these work ok with cli?
264 $bits = explode(basename($_SERVER["SCRIPT_FILENAME"]), $_SERVER["SCRIPT_NAME"]);
266 $bits[0] = str_replace('%2F','/',urlencode($bits[0]));
267 $this->baseURL = $bits[0] . basename($_SERVER["SCRIPT_FILENAME"]);
269 if (empty($_SERVER['SCRIPT_NAME'])) {
270 $this->baseURL = ''; // ??? this is if we replace top level...
273 // if cli - you have to have set baseURL...
276 $this->rootDir = realpath(dirname($_SERVER["SCRIPT_FILENAME"]));
277 $this->baseDir = $this->rootDir .'/'. $this->project;
278 $this->rootURL = dirname($this->baseURL);
279 $this->rootURL = ($this->rootURL == '/') ? '' : $this->rootURL;
282 //var_dump($this->baseURL);
284 if (!isset($this->database) && isset($this->PDO_DataObject['database'])) {
285 $this->database = $this->PDO_DataObject['database'];
288 $this->classPrefix = str_replace('/', '_', $this->project) . '_';
290 // list the available options..
291 if ($this->cli && empty($_SERVER['argv'][1])) {
292 require_once 'HTML/FlexyFramework2/Cli.php';
293 $fcli = new HTML_FlexyFramework2_Cli($this);
299 // see if it's a framework assignment.
302 require_once 'HTML/FlexyFramework2/Cli.php';
303 $fcli = new HTML_FlexyFramework2_Cli($this);
304 $res = $fcli->parseDefaultOpts();
312 $this->run = $this->cli ? $_SERVER['argv'][1] : false;
315 $this->_parseConfigDataObjects();
316 if ($this->dataObjectsCache && !$this->nodatabase) {
317 $this->_configDataObjectsCache();
320 $this->_parseConfigTemplate();
321 $this->_parseConfigMail();
323 //echo '<PRE>';print_r($this);exit;
325 //$this->_exposeToPear();
328 $this->_validateEnv();
334 $this->_validateDatabase();
336 $this->_validateTemplate();
342 *'languages' => array(
344 'avail' => array('en','zh_HK', 'zh_CN'),
346 'cookie' => 'TalentPricing_lang',
347 'localemap' => array(
348 'en' => 'en_US.utf8',
349 'zh_HK' => 'zh_TW.utf8',
350 'zh_CN' => 'zh_CN.utf8',
355 function _handleLanguages()
358 empty($this->languages) ||
360 !isset($this->languages['cookie']) && !isset($this->languages['default'])
366 $cfg = $this->languages;
368 $default = $cfg['default'];
370 if(!empty($_SERVER["HTTP_ACCEPT_LANGUAGE"])){
372 $brower_langs = explode(",", $_SERVER["HTTP_ACCEPT_LANGUAGE"]);
374 foreach ($brower_langs as $bl) {
375 $l = preg_replace('/;(.*)/', '', $bl);
377 $l = str_replace('-', '_', $l);
379 if(!in_array($l, $cfg['avail'])){
388 $lang = isset($_COOKIE[$cfg['cookie']]) ? $_COOKIE[$cfg['cookie']] : $default;
390 if (isset($_REQUEST[$cfg['param']])) {
391 $lang = $_REQUEST[$cfg['param']];
394 if (!in_array($lang, $cfg['avail'])) {
395 $lang = $cfg['default'];
397 if (isset($cfg['localemap'][$lang])) {
398 setlocale(LC_ALL, $cfg['localemap'][$lang]);
400 setcookie($cfg['cookie'], $lang, 0, '/');
402 $this->locale = $lang;
404 if (!empty($this->HTML_Template_Flexy)) {
405 $this->HTML_Template_Flexy['locale'] = $lang; //set a language for template engine
411 * overlay array properties..
414 function applyIf($prop, $ar) {
415 if (!isset($this->$prop)) {
419 // add only things that where not set!!!.
420 $this->$prop = array_merge($ar,$this->$prop);
423 //foreach($ar as $k=>$v) {
424 // if (!isset($this->$prop->$k)) {
425 // $this->$prop->$k = $v;
432 * - if turned on (dataObjectsCache = true) then
433 * a) ini file points to a parsed version of the structure.
434 * b) links.ini is a merged version of the configured link files.
436 * This only will force a generation if no file exists at all.. - after that it has to be called manually
437 * from the core page.. - which uses the Expires time to determine if regeneration is needed..
442 function _configDataObjectsCache()
444 // cli works under different users... it may cause problems..
445 $this->debug(__METHOD__);
446 if (function_exists('posix_getpwuid')) {
447 $uinfo = posix_getpwuid( posix_getuid () );
448 $user = $uinfo['name'];
450 $user = getenv('USERNAME'); // windows.
455 $iniCache = ini_get('session.save_path') .'/' .
456 'pdocfg-' . $user . '/'. str_replace('/', '_', $this->project) ;
459 if ($this->appNameShort) {
460 $iniCache .= '_' . $this->appNameShort;
462 if ($this->version) {
463 $iniCache .= '.' . $this->version;
465 if ($this->database === false) {
469 $dburl = parse_url($this->database);
470 if (!empty($dburl['path'])) {
471 $iniCache .= '-'.ltrim($dburl['path'],'/');
475 $this->debug(__METHOD__ . " : ini cache : $iniCache");
477 $dburl = parse_url($this->database);
478 //override ini setting... - store original..
480 if (isset($this->PDO_DataObject['schema_location'])) {
481 $this->dataObjectsOriginalIni = $this->PDO_DataObject['schema_location'];
482 ///print_r($this->DB_DataObject);exit;
485 $this->PDO_DataObject['schema_location'] = $iniCache;
486 PDO_DataObject::config($this->PDO_DataObject);
488 // we now have the configuration file name..
490 if (!file_exists($iniCache) || empty( $this->dataObjectsCacheExpires)) {
491 $this->generateDataobjectsCache(true);
499 * generateDataobjectsCache:
501 * if the 'cache file' does not exist, then
502 * create it using HTML_FlexyFramework2_Generator
504 * create xxx.ini and xxx.links.ini
506 * @arg force (boolean) force generation - default false;
510 function generateDataobjectsCache($force = false)
512 //$this->debug('generateDataobjectsCache: force=' . ($force ? 'yes' : 'no'));
513 if (!$this->dataObjectsCache) { // does not use dataObjects Caching..
514 PDO_DataObject::config('schema_location', $this->dataObjectsOriginalIni );
516 $this->debug('generateDataobjectsCache', 'dataObjectsCache - empty');
520 $dburl = parse_url($this->database);
521 $dbnick = basename($dburl['path']);
525 $iniCache = $this->PDO_DataObject['schema_location'];
530 if (file_exists($iniCache)) {
531 $files = glob(dirname($iniCache).'/*.ini');
532 foreach($files as $f) {
533 $replace[$f] = md5(file_get_contents($f)); // hash it..
540 $iniCacheTmp = $iniCache . '.tmp' .md5(rand()); // random to stop two processes using the same file.
542 $force = ($force ? $force : !file_exists($iniCache)) || !$this->dataObjectsCacheExpires;
543 // $this->debug('generateDataobjectsCache: after check : force=' . ($force ? 'yes' : 'no'));
544 // not force or not expired, do not bother..
546 if ((filemtime($iniCache) + $this->dataObjectsCacheExpires) >time()) {
552 $this->debug('generateDataobjectsCache', 'dataObjectsCache generating');
555 // force quoting of column names..
556 // unless it forced off..
557 if (!isset($this->PDO_DataObject['quote_identifiers'] )) {
558 $this->PDO_DataObject['quote_identifiers'] = true;
560 if (!file_exists(dirname($iniCache))) {
561 mkdir(dirname($iniCache),0700, true);
564 $this->PDO_DataObject['schema_location'] = $iniCacheTmp;
566 PDO_DataObject::reset();
568 PDO_DataObject::config($this->PDO_DataObject);
572 // DB_DataObject::debugLevel(1);
573 require_once 'HTML/FlexyFramework2/Generator.php';
575 PDO_DataObject::config('database', $this->database);
576 $generator = new HTML_FlexyFramework2_Generator();
579 $this->debug('generateDataobjectsCache', 'dataObjectsCache writing');
580 HTML_FlexyFramework2_Generator::writeCache($iniCacheTmp, $iniCache, $replace);
581 // reset the cache to the correct lcoation.
582 PDO_DataObject::config('schema_location', $iniCache);
583 $this->PDO_DataObject['schema_location'] = $iniCache;
584 //$this->_exposeToPear();
586 //$GLOBALS['_DB_DATAOBJECT']['INI'][$this->database] = parse_ini_file($iniCache, true);
587 //$GLOBALS['_DB_DATAOBJECT']['SEQUENCE']
588 // clear any dataobject cache..
595 * DataObject Configuration:
596 * Always in Project/DataObjects
597 * unless enableArray is available...
602 function _parseConfigDataObjects()
604 if ($this->nodatabase && !$this->database) {
608 // better done here.. ?? only include if we have database configuration?
609 require_once 'PDO/DataObject.php';
611 $dburl = parse_url($this->database);
612 $dbini = 'ini_'. basename($dburl['path']);
614 $dbinis = array(); //array(dirname(__FILE__) . '/Pman/DataObjects/pman.ini');
615 $dbreq = array(); //array( dirname(__FILE__) . '/Pman/DataObjects/');
616 $dbcls = array(); //array('Pman_DataObjects_');
618 $project = explode('/',$this->project)[0];
620 if (!empty($this->enableArray)) {
622 $tops = array_merge( array($project), empty($this->projectExtends) ? array() : $this->projectExtends);
624 foreach($tops as $td) {
625 $bd = $this->rootDir .'/'.$td;
626 foreach($this->enableArray as $m) {
627 // look in Pman/MODULE/DataObjects/*
628 if (file_exists($bd.'/'.$m.'/DataObjects')) {
629 $dbinis[] = $bd.'/'.$m.'/DataObjects/'. strtolower($td).'.ini';
630 $dbcls[] = $td.'_'. $m . '_DataObjects_';
631 $dbreq[] = $bd.'/'.$m.'/DataObjects';
634 // look in MODULE/DataObjects ?? DO WE SUPPORT THIS ANYMORE???
635 if (file_exists($bd.'/../'.$m.'/DataObjects')) {
636 $dbinis[] = $bd.'/../'.$m.'/DataObjects/'. strtolower($td).'.ini';
637 $dbcls[] = $td. '_DataObjects_';
638 $dbreq[] = $bd.'/../'.$m.'/DataObjects';
647 if (isset($this->PDO_DataObject['schema_location'])) {
648 $dbinis[] = $this->PDO_DataObject['schema_location'] .'/'.basename($dburl['path']).'.ini';
650 $dbinis[] = $this->baseDir.'/DataObjects/'.basename($dburl['path']).'.ini';
654 $dbcls[] = $project .'_DataObjects_';
655 $dbreq[] = $this->baseDir.'/DataObjects';
659 $this->applyIf('PDO_DataObject', array(
661 'class_location' => implode(PATH_SEPARATOR,$dbreq),
662 'class_prefix' => implode(PATH_SEPARATOR,$dbcls),
663 'database' => $this->database,
664 ///'require_prefix' =>
665 // 'schema_location' => dirname(__FILE__) . '/Pman/DataObjects/',
666 'schema_location' => implode(PATH_SEPARATOR,$dbinis),
670 PDO_DataObject::config($this->PDO_DataObject);
677 function _parseConfigTemplate()
681 if (function_exists('posix_getpwuid')) {
682 $uinfo = posix_getpwuid( posix_getuid () );
684 $user = $uinfo['name'];
686 $user = getenv('USERNAME'); // windows.
689 $compileDir = ini_get('session.save_path') .'/' .
690 $user . '_compiled_templates_' . $this->project;
692 if ($this->appNameShort) {
693 $compileDir .= '_' . $this->appNameShort;
695 if ($this->version) {
696 $compileDir .= '.' . $this->version;
699 // templates. -- all this should be cached!!!
703 if ($this->appNameShort && !in_array('Core', explode(',', $this->disable ? $this->disable : ''))) {
704 // in app based version, template directory is in Core
707 $this->baseDir . '/Core/templates'
711 if(!empty($this->projectExtends)){
712 foreach ($this->projectExtends as $e){
713 $add = $this->rootDir . '/' . $e .'/templates';
714 if (!in_array($add,$src) && file_exists($add)) {
720 $src[] = $this->baseDir . '/templates';
724 if (!empty($this->enableArray)) {
727 foreach($this->enableArray as $m) {
728 $add = $this->baseDir . '/' . $m .'/templates';
729 if (!in_array($add,$src) && file_exists($add) && $this->appNameShort != $m) {
734 if (!empty($this->projectExtends) ) {
735 foreach ($this->projectExtends as $extend){
736 foreach($this->enableArray as $m) {
737 $add = $this->rootDir . '/' . $extend . '/' . $m .'/templates';
738 if (!in_array($add,$src) && file_exists($add) && $this->appNameShort != $m) {
748 if ($this->appNameShort) {
749 $src[] = $this->baseDir . '/'. $this->appNameShort. '/templates';
752 // images may come from multiple places: - if we have multiple template directories.?
753 // how do we deal with this..?
754 // images/ << should always be mapped to master!
755 // for overridden appdir ones we will have to se rootURL etc.
757 $url_rewrite = 'images/:'. $this->rootURL . '/'. $this->project. '/templates/images/';
759 $this->applyIf('HTML_Template_Flexy', array(
760 'templateDir' => implode(PATH_SEPARATOR, $src),
761 'compileDir' => $compileDir,
762 'multiSource' => true,
764 'url_rewrite' => $url_rewrite,
765 'filters' => 'Php,SimpleTags', /// for non-tokenizer version?
766 'debug' => $this->debug ? 1 : 0,
775 function _parseConfigMail()
777 $this->applyIf('HTML_Template_Flexy', array(
780 'host' => 'localhost',
786 * exposes to PEAR::getStaticProperty..
787 * at present, only FlexyFramework uses this..
788 * // we should really stop it..
789 * ctor for Flexy - should read from config...
792 function _exposeToPear()
794 $cls = array_keys(get_class_vars(__CLASS__));
797 // anything that get's set, that's not in our default properties
798 // is assumed to be an option set .
799 foreach(get_object_vars($this) as $k=>$v) {
800 if (in_array($k,$cls)) {
804 $options = &PEAR::getStaticProperty($k,'options');
807 // $options = &PEAR::getStaticProperty('HTML_FlexyFramework2','options');
814 function _validateEnv()
816 /* have I been initialized */
819 if (get_magic_quotes_gpc() && !$this->cli) {
821 "magic quotes is enabled add the line<BR>
822 php_value magic_quotes_gpc 0<BR>
823 to your .htaccess file <BR>
824 (Apache has to be configured to "AllowOverride Options AuthConfig" for the directory)
829 // set up error handling -
830 //$this->error = new HTML_FlexyFramework2_Error();
832 /// fudge work around bugs in PEAR::setErrorHandling(,)
833 //$GLOBALS['_PEAR_default_error_mode'] = PEAR_ERROR_CALLBACK;
834 //$GLOBALS['_PEAR_default_error_options'] = array($this->error,'raiseError');
839 require_once 'Benchmark/Timer.php';
840 $this->timer = new BenchMark_Timer(true);
841 register_shutdown_function(function() { echo $this->timer->getOutput(); });
846 function _validateDatabase()
848 //echo "<PRE>"; print_r($this);
850 if ($this->nodatabase) {
854 // database is the only setting - we dont support mult databses?
857 $x = new PDO_Dataobject();
858 PDO_Dataobject::config('database', $this->database);
861 } catch (Exception $e) {
864 $this->fatalError("Configuration or Database Error: could not connect to Database, <BR>
865 Please check the value given to HTML_FlexyFramework2, or run with debug on!<BR>
866 <BR> ".$e->getMessage());
872 function _validateTemplate()
874 // check that we have a writeable directory for flexy's compiled templates.
876 if (empty($this->HTML_Template_Flexy['compileDir'])) {
880 if ( !file_exists($this->HTML_Template_Flexy['compileDir'])) {
881 mkdir($this->HTML_Template_Flexy['compileDir'], 0700, true);
884 if ( !file_exists($this->HTML_Template_Flexy['compileDir'])) {
886 $this->fatalError("Configuration Error: you specified a directory that does not exist for<BR>
887 HTML_Template_Flexy => compileDir {$this->HTML_Template_Flexy['compileDir']}<BR>\n"
892 if (!is_writeable($this->HTML_Template_Flexy['compileDir'])) {
893 $this->fatalError("Configuration Error: Please make sure the template cache directory is writeable<BR>
895 chmod 700 {$this->HTML_Template_Flexy['compileDir']}<BR>
896 chgrp apache_user {$this->HTML_Template_Flexy['compileDir']}<BR>\n"
899 //echo "<PRE>";print_R($config);
917 * HTML_FlexyFramework::run('someurl/someother',array('somearg'=>'xxx'));
919 * exit; <- dont cary on!!!!
923 * @param string redirect to url
924 * @param array Args Optional any data you want to send to the next page..
933 static function run($request,$args=array())
935 self::$singleton->_run($request,true,$args);
940 * initPage - load the page up, and set the variables, but do not actually run it.
941 * - used where we want features from this framework, buy may actually be running another one..
943 * call with ($_SERVER["SCRIPT_NAME"]) for root page...
946 public function initPage($request, $isRedirect=false, $args=array())
948 $newRequest = $this->_getRequest($request,$isRedirect);
950 // find the class/file to load
951 list($classname,$subRequest) = $this->requestToClassName($newRequest,FALSE);
954 $this->debug("requestToClassName return = CLASSNAME: $classname SUB REQUEST: $subRequest");
956 // assume that this was handled by getclassname ?????
961 // make page data/object accessable at anypoint in time using this
962 // not sure if this is used anymore - or even works..?
964 $classobj = new $classname(); // normally do not have constructors.
966 $classobj->baseURL = $this->baseURL;
967 $classobj->rootURL = $this->rootURL;
968 $classobj->rootDir = $this->rootDir;
969 $classobj->bootLoader = $this;
970 $classobj->request = $newRequest;
971 $classobj->timer = &$this->timer;
972 $classobj->subrequest = $subRequest;
974 $this->page = $classobj;
980 * The main execution loop
982 * recursivly self called if redirects (eg. return values from page start methods)
984 * @param string from $_REQUEST or redirect from it'self.
985 * @param boolean isRedirect = is the request a redirect TRUE = always handle as 'GET', 1 = handle as POST/GET depending on method, 0 = handle as initial request.
988 * @return false || other false indicates no page was served!
989 * @access public|private
990 * @see see also methods.....
993 public function _run($request,$isRedirect = false,$args = array())
996 // clean the request up.
999 if ($this->calls > 5) {
1000 // to many redirections...
1001 trigger_error("FlexyFramework:: too many redirects - backtrace me!",E_USER_ERROR);
1005 $classobj = $this->initPage($request,$isRedirect,$args);
1006 $classobj->cli = $this->cli;
1008 if ($this->cli && !$isRedirect ) { // redirect always just takes redirect args..
1009 require_once 'HTML/FlexyFramework2/Cli.php';
1010 $fcli = new HTML_FlexyFramework2_Cli($this);
1011 $nargs = $fcli->cliParse(get_class($classobj));
1012 $args = $nargs === false ? $args : $nargs; /// replace if found.
1013 $classobj->cli_args = $nargs;
1016 // echo '<PRE>'; print_r($this);exit;
1017 // echo "CHECK GET AUTH?";
1018 if (!method_exists($classobj, 'getAuth')) {
1019 // echo "NO GET AUTH?";
1020 $this->fatalError("class ". get_class($classobj) . " does not have a getAuth Method");
1024 /* check auth on the page */
1025 if (is_string($redirect = $classobj->getAuth())) {
1026 $this->debug("GOT AUTH REDIRECT".$redirect);
1027 return $this->_run($redirect,TRUE);
1029 // used HTML_FlexyFramework::run();
1032 if ($redirect === false) {
1033 $this->debug("GOT AUTH FALSE");
1034 return false; /// Access deined!!! -
1037 // allow the page to implement caching (for the full page..)
1038 // although normally it should implement caching on the outputBody() method.
1040 if (method_exists($classobj,"getCache")) {
1041 if ($result = $classobj->getCache()) {
1045 /* allow redirect from start */
1046 if (method_exists($classobj,"start")) {
1047 if (is_string($redirect = $classobj->start($classobj->subrequest,$isRedirect,$args))) {
1048 $this->debug("REDIRECT $redirect <BR>");
1049 return $this->_run($redirect,TRUE);
1051 if ($redirect === false) {
1057 // used HTML_FlexyFramework::run();
1060 * Modules are common page components like navigation headers etc.
1061 * that can have dynamic code.
1062 * Code has been removed now..
1067 $this->timer->setMarker("After $request loadModules Modules");
1070 /* output it - (our base page does not implement output for cli. */
1072 if ( method_exists($classobj,'output')) {
1073 $classobj->output();
1078 $this->timer->setMarker("After $request output");
1079 $this->timer->stop(); //?? really - yes...
1089 exit; /// die here...
1094 * map the request into an object and run the page.
1096 * The core of the work is done here.
1099 * @param request the request string
1100 * @param boolean isRedirect - indicates that it should not attempt to strip the .../index.php from the request.
1105 function _getRequest($request, $isRedirect)
1114 $startRequest = $request;
1115 $rq = explode('?', $request);
1116 $request = array_shift($rq);
1117 $this->debug("INPUT REQUEST $request<BR>");
1119 // check that request forms contains baseurl????
1121 $subreq = substr($request,0,strlen($this->baseURL));
1122 if ($subreq != substr($this->baseURL,0,strlen($subreq))) {
1124 "Configuration error: Got base of $subreq which does not
1125 match configuration of: $this->baseURL} ");
1127 $request = substr($request,strlen($this->baseURL));
1131 // echo "REQUEST WAS: $request<BR>";
1132 // $request = preg_replace('/^'.preg_quote($base_url,'/').'/','',trim($request));
1133 // echo "IS NOW: $request<BR>";
1135 // strip valid html stuff
1136 //$request = preg_replace('/\/[.]+/','',$request);
1139 $request = preg_replace('/^[\/]*/','',$request);
1140 $request = preg_replace('/\?.*$/','',$request);
1141 $request = preg_replace('/[\/]*$/','',$request);
1142 $this->baseRequest = $request;
1143 $request = str_replace('&','',$request); // any other invalid characters???
1144 $request = preg_replace('/\.([a-z]+)$/','',$request);
1145 $this->ext = substr($this->baseRequest , strlen($request));
1147 // REDIRECT ROO to index.php! for example..
1149 // this has a slight issue when we are using 'hidden index.php' mod_rewrite..
1152 normal (non-mod_rewrite)
1153 $_SERVER['REQUEST_URI'] start with baseURL..
1155 if (isset($_SERVER['REQUEST_URI']) && substr($_SERVER['REQUEST_URI'], 0, strlen($this->baseURL)) != $this->baseURL) {
1156 // then we are rewriting?
1157 $this->baseURL = $this->rootURL;
1161 //var_dump(array($this->baseURL, $this->rootURL, $request ,$isRedirect, $_SERVER['REQUEST_URI'])); phpinfo();exit;
1162 if (!$request && !$isRedirect) {
1165 if ($this->baseURL && (strlen($startRequest) < strlen($this->baseURL))) {
1167 // needs to handle https + port
1168 $http = ((!empty($_SERVER["HTTPS"]) && $_SERVER["HTTPS"] == 'on')) ? 'https' : 'http';
1170 if (!empty($_SERVER['SERVER_PORT'])) {
1171 if ((($http == 'http') && ($_SERVER['SERVER_PORT'] == 80)) || (($http == 'https') && ($_SERVER['SERVER_PORT'] == 443))) {
1174 $sp .= ':'.((int) $_SERVER['SERVER_PORT']);
1177 //var_dump($startRequest); echo '<PRE>'; print_R($this); phpinfo();exit;
1178 $host = !empty($_SERVER["HTTP_X_FORWARDED_HOST"]) ? $_SERVER["HTTP_X_FORWARDED_HOST"] : $_SERVER["HTTP_HOST"];
1179 header('Location: '.$http.'://'.$host .$sp . $this->baseURL);
1185 $this->debug("OUTPUT REQUEST $request<BR>");
1193 * get the Class name and filename to load
1195 * Parses the request and converts that into a File + Classname
1196 * if the class doesnt exist it will attempt to find a file below it, and
1197 * call that one with the data.
1198 * Used by the module loader to determine the location of the modules
1200 * @param request the request string
1201 * @param boolean showError - if false, allows you to continue if the class doesnt exist.
1204 * @return array classname, filepath
1209 function requestToClassName($request,$showError=TRUE)
1211 // if ($request == "error") {
1212 // return array("HTML_FlexyFramework_Error","");
1215 // special classes ::
1216 if ($this->cli && in_array($request, array('DataObjects'))) {
1217 require_once 'HTML/FlexyFramework2/'. $request . '.php';
1218 return array('HTML_FlexyFramework2_'. $request,'');
1222 $request_array=explode("/",$request);
1223 $original_request_array = $request_array;
1224 $sub_request_array = array();
1225 $l = count($request_array)-1;
1226 if ($l > 10) { // ?? configurable?
1227 //PEAR::raiseError("Request To Long");
1228 $this->fatalError("Request To Long - " . $request);
1233 // tidy up request array
1235 if ($request_array) {
1236 foreach(array_keys($request_array) as $i) {
1237 $request_array[$i] = preg_replace('/[^a-z0-9]/i','_',urldecode($request_array[$i]));
1240 //echo "<PRE>"; print_r($request_array);
1241 // technically each module should do a check here... similar to this..
1244 for ($i=$l;$i >-1;$i--) {
1245 $location = implode('/',$request_array) . ".php";
1246 if ($location == '.php') {
1247 $this->debug("SKIP first path check, as request str is empty");
1251 $this->debug("baseDIR = {$this->baseDir}");
1253 $floc = "{$this->baseDir}/$location";
1254 $this->debug("CHECK LOCATION = $location");
1258 if (!empty($location) && $location != '.php' && @file_exists($floc )) { // hide? error???
1259 require_once $floc ;
1260 $classname = $this->classPrefix . implode('_',$request_array);
1261 $this->debug("FOUND FILE - SET CLASS = $classname <BR>");
1265 // in here check the 'projectExtends' versions..?
1267 if(!empty($this->projectExtends)){
1268 $this->debug("Trying project Extends<BR>");
1269 $has_extend_class = false;
1271 foreach ($this->projectExtends as $e){
1272 $floc = "{$this->rootDir}/{$e}/$location";
1273 $this->debug("Trying file: $floc");
1274 if (!empty($location) && @file_exists($floc)) { // hide? error???
1275 require_once $floc ;
1276 $classname = $e . '_' . implode('_',$request_array);
1277 $has_extend_class = true;
1278 $this->debug("FOUND FILE - SET CLASS = $classname <BR>");
1283 if(!empty($has_extend_class)){
1290 $this->debug("$floc - !!FOUND NOT FILE!!");
1292 $sub_request_array[] = $original_request_array[$i];
1293 unset($request_array[$i]);
1294 unset($original_request_array[$i]);
1297 // is this really needed here!
1299 $classname = preg_replace('/[^a-z0-9]/i','_',$classname);
1300 $this->debug("CLASSNAME is '$classname'");
1302 if ($classname && class_exists($classname)) {
1303 $this->debug("using $classname");
1304 //print_r($sub_request_array);
1305 return array($classname,implode('/',array_reverse($sub_request_array)));
1309 $this->fatalError("INVALID REQUEST: \n $request FILE:".$this->baseDir. "/{$location} CLASS:{$classname}");
1314 $this->debug("Try base {$this->baseDir}.php");
1315 // try {project name}.php
1316 // this used to be silenced @ - if this fails we are usually pretty fried..
1318 if (file_exists($this->baseDir.'.php')) {
1321 $classname = str_replace('/', '_', $this->project); // basename($this->baseDir);
1323 $this->debug("FOUND {$this->baseDir} requring and checking class $classname");
1324 require_once $this->baseDir.'.php';
1325 $this->debug("require success");
1327 if (!class_exists($classname)) {
1328 $this->fatalError( "{$this->baseDir}.php did not contain class $classname");
1331 // got projectname.php
1332 if ($classname && class_exists($classname)) {
1333 $this->debug("using $classname");
1334 //print_r($sub_request_array);
1336 return array($classname,implode('/',array_reverse($sub_request_array)));
1340 $this->fatalError( "can not find {$this->baseDir}.php"); // dies..
1346 * ensure Single CLi process
1348 * HTML_FlexyFramework2::ensureSingle(__FILE__, $this);
1349 * @param string filename of running class
1350 * @param object class
1353 static function ensureSingle($sig, $class)
1355 //echo "check single: $sig / ". get_class($class) ."\n";
1356 $ff = HTML_FlexyFramework2::get();
1357 if (function_exists('posix_getpwuid')) {
1358 $uinfo = posix_getpwuid( posix_getuid () );
1359 $user = $uinfo['name'];
1361 $user = getenv('USERNAME'); // windows.
1363 $fdir = ini_get('session.save_path') .'/' .
1364 $user . '_cli_' . $ff->project ;
1367 if (!file_exists($fdir)) {
1371 $lock = $fdir.'/'. md5($sig) . '.' . get_class($class);
1372 //echo "check single: lock : $lock\n";
1373 if (!file_exists($lock)) {
1374 file_put_contents($lock, getmypid());
1375 //echo "check single: lock : DOES NOT EXIST\n";
1378 $oldpid = file_get_contents($lock);
1379 if (!file_exists('/proc/' . $oldpid)) {
1381 file_put_contents($lock, getmypid());
1382 // echo "check single: lock : PROC NOT EXIST\n";
1385 // file exists, but process might not be the same..
1386 $ca = explode('_', get_class($class));
1387 $name = array_pop($ca);
1388 $cmd = file_get_contents('/proc/' . $oldpid.'/cmdline');
1389 if (!preg_match('/php/i',$cmd) || !preg_match('/'.$name.'/i',$cmd)) {
1390 file_put_contents($lock, getmypid());
1391 //echo "check single: lock : CMDLINE !have PHP \n";
1394 die("process " . $sig . " already running\n");
1398 * removes the lock for the applicaiton - use with care...
1402 static function ensureSingleClear($sig, $class)
1404 $ff = HTML_FlexyFramework2::get();
1405 if (function_exists('posix_getpwuid')) {
1406 $uinfo = posix_getpwuid( posix_getuid () );
1407 $user = $uinfo['name'];
1409 $user = getenv('USERNAME'); // windows.
1411 $fdir = ini_get('session.save_path') .'/' .
1412 $user . '_cli_' . $ff->project ;
1415 if (!file_exists($fdir)) {
1418 $lock = $fdir.'/'. md5($sig);
1419 if (!file_exists($lock)) {
1430 * @param string text to output.
1434 function debug($output) {
1436 if (empty($this->debug)) {
1440 "HTML_FlexyFramework2::debug - ".$output."\n"
1441 : "<B>HTML_FlexyFramework2::debug</B> - ".$output."<BR>\n";
1445 * Raises a fatal error. - normally only used when setting up to help get the config right.
1447 * can redirect to fatal Action page.. - hoepfully not issued before basic vars are set up..
1449 * @param string text to output.
1453 function fatalError($msg,$showConfig = 0)
1457 if ($this->fatalAction) {
1458 HTML_FlexyFramework2::run($this->fatalAction,$msg);
1461 header('HTTP/1.1 503 Service Temporarily Unavailable');
1462 header('Status: 503 Service Temporarily Unavailable');
1463 header('Retry-After: 300');
1464 echo $this->cli ? $msg ."\n" : "<H1>$msg</H1>configuration information<PRE>";
1469 $ff = HTML_FlexyFramework2::get();