1da0ae2c0SAndreas Gohr<?php 2da0ae2c0SAndreas Gohr 3a646d519SAndreas Gohr/** 4a646d519SAndreas Gohr * Core Manager for the Farm functionality 5a646d519SAndreas Gohr * 6a646d519SAndreas Gohr * This class is initialized before any other DokuWiki code runs. Therefore it is 7a646d519SAndreas Gohr * completely selfcontained and does not use any of DokuWiki's utility functions. 8a646d519SAndreas Gohr * 9b96c66ccSAndreas Gohr * It's registered as a global $FARMCORE variable but you should not interact with 10b96c66ccSAndreas Gohr * it directly. Instead use the Farmer plugin's helper component. 110a5d2da2SAndreas Gohr * 120a5d2da2SAndreas Gohr * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 130a5d2da2SAndreas Gohr * @author Andreas Gohr <gohr@cosmocode.de> 14a646d519SAndreas Gohr */ 15da0ae2c0SAndreas Gohrclass DokuWikiFarmCore { 16da0ae2c0SAndreas Gohr /** 17da0ae2c0SAndreas Gohr * @var array The default config - changed by loadConfig 18da0ae2c0SAndreas Gohr */ 19da0ae2c0SAndreas Gohr protected $config = array( 20a646d519SAndreas Gohr 'base' => array( 21a646d519SAndreas Gohr 'farmdir' => '', 22c4c8e953SAndreas Gohr 'farmhost' => '', 23c4c8e953SAndreas Gohr 'basedomain' => '', 24a646d519SAndreas Gohr ), 25da0ae2c0SAndreas Gohr 'notfound' => array( 26da0ae2c0SAndreas Gohr 'show' => 'farmer', 27da0ae2c0SAndreas Gohr 'url' => '' 28da0ae2c0SAndreas Gohr ), 29da0ae2c0SAndreas Gohr 'inherit' => array( 30da0ae2c0SAndreas Gohr 'main' => 1, 31da0ae2c0SAndreas Gohr 'acronyms' => 1, 32da0ae2c0SAndreas Gohr 'entities' => 1, 33da0ae2c0SAndreas Gohr 'interwiki' => 1, 34da0ae2c0SAndreas Gohr 'license' => 1, 35da0ae2c0SAndreas Gohr 'mime' => 1, 36da0ae2c0SAndreas Gohr 'scheme' => 1, 37da0ae2c0SAndreas Gohr 'smileys' => 1, 38da0ae2c0SAndreas Gohr 'wordblock' => 1, 391272da0cSAndreas Gohr 'users' => 0, 40da0ae2c0SAndreas Gohr 'userstyle' => 0, 41da0ae2c0SAndreas Gohr 'userscript' => 0 42da0ae2c0SAndreas Gohr ) 43da0ae2c0SAndreas Gohr ); 44da0ae2c0SAndreas Gohr 45a646d519SAndreas Gohr /** @var string|false The current animal, false for farmer */ 46a646d519SAndreas Gohr protected $animal = false; 47a646d519SAndreas Gohr /** @var bool true if an animal was requested but was not found */ 48a646d519SAndreas Gohr protected $notfound = false; 49a646d519SAndreas Gohr /** @var bool true if the current animal was requested by host */ 50a646d519SAndreas Gohr protected $hostbased = false; 51a646d519SAndreas Gohr 52da0ae2c0SAndreas Gohr /** 53da0ae2c0SAndreas Gohr * DokuWikiFarmCore constructor. 54da0ae2c0SAndreas Gohr * 55a646d519SAndreas Gohr * This initializes the whole farm by loading the configuration and setting 56a646d519SAndreas Gohr * DOKU_CONF depending on the requested animal 57da0ae2c0SAndreas Gohr */ 58da0ae2c0SAndreas Gohr public function __construct() { 59da0ae2c0SAndreas Gohr $this->loadConfig(); 60a646d519SAndreas Gohr if($this->config['base']['farmdir'] === '') return; // farm setup not complete 61*8262a4cbSAndreas Gohr $this->config['base']['farmdir'] = rtrim($this->config['base']['farmdir'], '/').'/'; // trailing slash always 6249f2871cSAndreas Gohr define('DOKU_FARMDIR', $this->config['base']['farmdir']); 63a646d519SAndreas Gohr 64a646d519SAndreas Gohr // animal? 65a646d519SAndreas Gohr $this->detectAnimal(); 66a646d519SAndreas Gohr 67a646d519SAndreas Gohr // setup defines 68a646d519SAndreas Gohr define('DOKU_FARM_ANIMAL', $this->animal); 69a646d519SAndreas Gohr if($this->animal) { 70*8262a4cbSAndreas Gohr define('DOKU_CONF', DOKU_FARMDIR . $this->animal . '/conf/'); 71a646d519SAndreas Gohr } else { 72a646d519SAndreas Gohr define('DOKU_CONF', DOKU_INC . '/conf/'); 73a646d519SAndreas Gohr } 74a646d519SAndreas Gohr 75a646d519SAndreas Gohr $this->setupCascade(); 76a646d519SAndreas Gohr $this->adjustCascade(); 77da0ae2c0SAndreas Gohr } 78da0ae2c0SAndreas Gohr 79da0ae2c0SAndreas Gohr /** 80da0ae2c0SAndreas Gohr * @return array the current farm configuration 81da0ae2c0SAndreas Gohr */ 82da0ae2c0SAndreas Gohr public function getConfig() { 83da0ae2c0SAndreas Gohr return $this->config; 84da0ae2c0SAndreas Gohr } 85da0ae2c0SAndreas Gohr 86da0ae2c0SAndreas Gohr /** 87a646d519SAndreas Gohr * @return false|string 88a646d519SAndreas Gohr */ 89a646d519SAndreas Gohr public function getAnimal() { 90a646d519SAndreas Gohr return $this->animal; 91a646d519SAndreas Gohr } 92a646d519SAndreas Gohr 93a646d519SAndreas Gohr /** 94a646d519SAndreas Gohr * @return boolean 95a646d519SAndreas Gohr */ 96a646d519SAndreas Gohr public function isHostbased() { 97a646d519SAndreas Gohr return $this->hostbased; 98a646d519SAndreas Gohr } 99a646d519SAndreas Gohr 100a646d519SAndreas Gohr /** 101a646d519SAndreas Gohr * @return boolean 102a646d519SAndreas Gohr */ 103a646d519SAndreas Gohr public function wasNotfound() { 104a646d519SAndreas Gohr return $this->notfound; 105a646d519SAndreas Gohr } 106a646d519SAndreas Gohr 107a646d519SAndreas Gohr /** 108b330074aSAndreas Gohr * @return string 109b330074aSAndreas Gohr */ 110b330074aSAndreas Gohr public function getAnimalDataDir() { 111*8262a4cbSAndreas Gohr return DOKU_FARMDIR . $this->getAnimal() . '/data/'; 112b330074aSAndreas Gohr } 113b330074aSAndreas Gohr 114b330074aSAndreas Gohr /** 115b330074aSAndreas Gohr * @return string 116b330074aSAndreas Gohr */ 117b330074aSAndreas Gohr public function getAnimalBaseDir() { 118b330074aSAndreas Gohr if($this->isHostbased()) return ''; 119b330074aSAndreas Gohr return getBaseURL() . '!' . $this->getAnimal(); 120b330074aSAndreas Gohr } 121b330074aSAndreas Gohr 122b330074aSAndreas Gohr /** 123a646d519SAndreas Gohr * Detect the current animal 124a646d519SAndreas Gohr * 125a646d519SAndreas Gohr * Sets internal members $animal, $notfound and $hostbased 126a646d519SAndreas Gohr * 127a646d519SAndreas Gohr * This borrows form DokuWiki's inc/farm.php but does not support a default conf dir 128a646d519SAndreas Gohr */ 129a646d519SAndreas Gohr protected function detectAnimal() { 130a646d519SAndreas Gohr $farmdir = $this->config['base']['farmdir']; 131a646d519SAndreas Gohr $farmhost = $this->config['base']['farmhost']; 132a646d519SAndreas Gohr 133a646d519SAndreas Gohr // check if animal was set via parameter (rewrite or CLI) 134a646d519SAndreas Gohr $animal = ''; 135a646d519SAndreas Gohr if(isset($_REQUEST['animal'])) $animal = $_REQUEST['animal']; 136a646d519SAndreas Gohr if('cli' == php_sapi_name() && isset($_SERVER['animal'])) $animal = $_SERVER['animal']; 137a646d519SAndreas Gohr if($animal) { 138a646d519SAndreas Gohr // check that $animal is a string and just a directory name and not a path 139a646d519SAndreas Gohr if(!is_string($animal) || strpbrk($animal, '\\/') !== false) { 140a646d519SAndreas Gohr $this->notfound = true; 141a646d519SAndreas Gohr return; 142a646d519SAndreas Gohr }; 143a646d519SAndreas Gohr $animal = strtolower($animal); 144a646d519SAndreas Gohr 145a646d519SAndreas Gohr // check if animal exists 146a646d519SAndreas Gohr if(is_dir("$farmdir/$animal/conf")) { 147a646d519SAndreas Gohr $this->animal = $animal; 148a646d519SAndreas Gohr return; 149a646d519SAndreas Gohr } else { 150a646d519SAndreas Gohr $this->notfound = true; 151a646d519SAndreas Gohr return; 152a646d519SAndreas Gohr } 153a646d519SAndreas Gohr } 154a646d519SAndreas Gohr 155a646d519SAndreas Gohr // is this the farmer? 156a646d519SAndreas Gohr if(strtolower($_SERVER['HTTP_HOST']) == $farmhost) { 157a646d519SAndreas Gohr return; 158a646d519SAndreas Gohr } 159a646d519SAndreas Gohr 16085becf1bSAndreas Gohr // still here? check for host based 161a646d519SAndreas Gohr $this->hostbased = true; 16285becf1bSAndreas Gohr $possible = $this->getAnimalNamesForHost($_SERVER['HTTP_HOST']); 16385becf1bSAndreas Gohr foreach($possible as $animal) { 164a646d519SAndreas Gohr if(is_dir("$farmdir/$animal/conf/")) { 165a646d519SAndreas Gohr $this->animal = $animal; 166a646d519SAndreas Gohr return; 167a646d519SAndreas Gohr } 168a646d519SAndreas Gohr } 169a646d519SAndreas Gohr 170a646d519SAndreas Gohr // no hit 171a646d519SAndreas Gohr $this->notfound = true; 172a646d519SAndreas Gohr return; 173a646d519SAndreas Gohr } 174a646d519SAndreas Gohr 175a646d519SAndreas Gohr /** 17685becf1bSAndreas Gohr * Return a list of possible animal names for the given host 17785becf1bSAndreas Gohr * 17885becf1bSAndreas Gohr * @param string $host the HTTP_HOST header 17985becf1bSAndreas Gohr * @return array 18085becf1bSAndreas Gohr */ 18185becf1bSAndreas Gohr protected function getAnimalNamesForHost($host) { 18285becf1bSAndreas Gohr $animals = array(); 18385becf1bSAndreas Gohr $parts = explode('.', implode('.', explode(':', rtrim($host, '.')))); 18485becf1bSAndreas Gohr for($j = count($parts); $j > 0; $j--) { 18585becf1bSAndreas Gohr // strip from the end 18685becf1bSAndreas Gohr $animals[] = implode('.', array_slice($parts, 0, $j)); 18785becf1bSAndreas Gohr // strip from the end without host part 18885becf1bSAndreas Gohr $animals[] = implode('.', array_slice($parts, 1, $j)); 18985becf1bSAndreas Gohr } 19085becf1bSAndreas Gohr $animals = array_unique($animals); 19185becf1bSAndreas Gohr $animals = array_filter($animals); 1920a5d2da2SAndreas Gohr usort( 193bfecda9bSAndreas Gohr $animals, 194bfecda9bSAndreas Gohr // compare by length, then alphabet 195bfecda9bSAndreas Gohr function ($a, $b) { 196bfecda9bSAndreas Gohr $ret = strlen($b) - strlen($a); 197bfecda9bSAndreas Gohr if($ret != 0) return $ret; 198bfecda9bSAndreas Gohr return $a > $b; 1990a5d2da2SAndreas Gohr } 2000a5d2da2SAndreas Gohr ); 20185becf1bSAndreas Gohr return $animals; 20285becf1bSAndreas Gohr } 20385becf1bSAndreas Gohr 20485becf1bSAndreas Gohr /** 205a646d519SAndreas Gohr * This sets up the default farming config cascade 206a646d519SAndreas Gohr */ 207a646d519SAndreas Gohr protected function setupCascade() { 208a646d519SAndreas Gohr global $config_cascade; 209a646d519SAndreas Gohr $config_cascade = array( 210a646d519SAndreas Gohr 'main' => array( 211a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/dokuwiki.php',), 212a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'local.php',), 213a646d519SAndreas Gohr 'protected' => array(DOKU_CONF . 'local.protected.php',), 214a646d519SAndreas Gohr ), 215a646d519SAndreas Gohr 'acronyms' => array( 216a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/acronyms.conf',), 217a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'acronyms.local.conf',), 218a646d519SAndreas Gohr ), 219a646d519SAndreas Gohr 'entities' => array( 220a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/entities.conf',), 221a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'entities.local.conf',), 222a646d519SAndreas Gohr ), 223a646d519SAndreas Gohr 'interwiki' => array( 224a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/interwiki.conf',), 225a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'interwiki.local.conf',), 226a646d519SAndreas Gohr ), 227a646d519SAndreas Gohr 'license' => array( 228a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/license.php',), 229a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'license.local.php',), 230a646d519SAndreas Gohr ), 231a646d519SAndreas Gohr 'mediameta' => array( 232a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/mediameta.php',), 233a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'mediameta.local.php',), 234a646d519SAndreas Gohr ), 235a646d519SAndreas Gohr 'mime' => array( 236a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/mime.conf',), 237a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'mime.local.conf',), 238a646d519SAndreas Gohr ), 239a646d519SAndreas Gohr 'scheme' => array( 240a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/scheme.conf',), 241a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'scheme.local.conf',), 242a646d519SAndreas Gohr ), 243a646d519SAndreas Gohr 'smileys' => array( 244a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/smileys.conf',), 245a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'smileys.local.conf',), 246a646d519SAndreas Gohr ), 247a646d519SAndreas Gohr 'wordblock' => array( 248a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/wordblock.conf',), 249a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'wordblock.local.conf',), 250a646d519SAndreas Gohr ), 251a646d519SAndreas Gohr 'acl' => array( 252a646d519SAndreas Gohr 'default' => DOKU_CONF . 'acl.auth.php', 253a646d519SAndreas Gohr ), 254a646d519SAndreas Gohr 'plainauth.users' => array( 255a646d519SAndreas Gohr 'default' => DOKU_CONF . 'users.auth.php', 256a646d519SAndreas Gohr ), 257de156015SAndreas Gohr 'plugins' => array( 258a646d519SAndreas Gohr 'default' => array(DOKU_INC . 'conf/plugins.php',), 259a646d519SAndreas Gohr 'local' => array(DOKU_CONF . 'plugins.local.php',), 260a646d519SAndreas Gohr 'protected' => array( 261a646d519SAndreas Gohr DOKU_INC . 'conf/plugins.required.php', 262a646d519SAndreas Gohr DOKU_CONF . 'plugins.protected.php', 263a646d519SAndreas Gohr ), 264a646d519SAndreas Gohr ), 265a646d519SAndreas Gohr 'userstyle' => array( 266a646d519SAndreas Gohr 'screen' => array(DOKU_CONF . 'userstyle.css', DOKU_CONF . 'userstyle.less',), 267a646d519SAndreas Gohr 'print' => array(DOKU_CONF . 'userprint.css', DOKU_CONF . 'userprint.less',), 268a646d519SAndreas Gohr 'feed' => array(DOKU_CONF . 'userfeed.css', DOKU_CONF . 'userfeed.less',), 269a646d519SAndreas Gohr 'all' => array(DOKU_CONF . 'userall.css', DOKU_CONF . 'userall.less',), 270a646d519SAndreas Gohr ), 271a646d519SAndreas Gohr 'userscript' => array( 272a646d519SAndreas Gohr 'default' => array(DOKU_CONF . 'userscript.js',), 273a646d519SAndreas Gohr ), 274a646d519SAndreas Gohr ); 275a646d519SAndreas Gohr } 276a646d519SAndreas Gohr 277a646d519SAndreas Gohr /** 278a646d519SAndreas Gohr * This adds additional files to the config cascade based on the inheritence settings 279a646d519SAndreas Gohr * 280a646d519SAndreas Gohr * These are only added for animals, not the farmer 281a646d519SAndreas Gohr */ 282a646d519SAndreas Gohr protected function adjustCascade() { 283b330074aSAndreas Gohr // nothing to do when on the farmer: 284b330074aSAndreas Gohr if(!$this->animal) return; 285a646d519SAndreas Gohr 286b330074aSAndreas Gohr global $config_cascade; 287a646d519SAndreas Gohr foreach($this->config['inherit'] as $key => $val) { 288a646d519SAndreas Gohr if(!$val) continue; 289a646d519SAndreas Gohr 290a646d519SAndreas Gohr // prepare what is to append or prepend 291a646d519SAndreas Gohr $append = array(); 292a646d519SAndreas Gohr $prepend = array(); 293a646d519SAndreas Gohr if($key == 'main') { 294b330074aSAndreas Gohr $append = array( 295b330074aSAndreas Gohr 'default' => array(DOKU_INC . 'conf/local.php'), 296b330074aSAndreas Gohr 'protected' => array(DOKU_INC . 'lib/plugins/farmer/includes/config.php') 297b330074aSAndreas Gohr ); 298a646d519SAndreas Gohr } elseif($key == 'license') { 299a646d519SAndreas Gohr $append = array('default' => array(DOKU_INC . 'conf/' . $key . '.local.php')); 300a646d519SAndreas Gohr } elseif($key == 'userscript') { 301a646d519SAndreas Gohr $prepend = array('default' => array(DOKU_INC . 'conf/userscript.js')); 302a646d519SAndreas Gohr } elseif($key == 'userstyle') { 303a646d519SAndreas Gohr $prepend = array( 304a646d519SAndreas Gohr 'screen' => array(DOKU_INC . 'conf/userstyle.css', DOKU_INC . 'conf/userstyle.less',), 305a646d519SAndreas Gohr 'print' => array(DOKU_INC . 'conf/userprint.css', DOKU_INC . 'conf/userprint.less',), 306a646d519SAndreas Gohr 'feed' => array(DOKU_INC . 'conf/userfeed.css', DOKU_INC . 'conf/userfeed.less',), 307a646d519SAndreas Gohr 'all' => array(DOKU_INC . 'conf/userall.css', DOKU_INC . 'conf/userall.less',), 308a646d519SAndreas Gohr ); 3091272da0cSAndreas Gohr } elseif($key == 'users') { 3101272da0cSAndreas Gohr $config_cascade['plainauth.users']['protected'] = DOKU_INC . 'conf/users.auth.php'; 311a646d519SAndreas Gohr } else { 312a646d519SAndreas Gohr $append = array('default' => array(DOKU_INC . 'conf/' . $key . '.local.conf')); 313a646d519SAndreas Gohr } 314a646d519SAndreas Gohr 315a646d519SAndreas Gohr // add to cascade 316a646d519SAndreas Gohr foreach($prepend as $section => $data) { 317a646d519SAndreas Gohr $config_cascade[$key][$section] = array_merge($data, $config_cascade[$key][$section]); 318a646d519SAndreas Gohr } 319a646d519SAndreas Gohr foreach($append as $section => $data) { 320a646d519SAndreas Gohr $config_cascade[$key][$section] = array_merge($config_cascade[$key][$section], $data); 321a646d519SAndreas Gohr } 322a646d519SAndreas Gohr } 323de156015SAndreas Gohr 324de156015SAndreas Gohr // add plugin overrides 325de156015SAndreas Gohr $config_cascade['plugins']['protected'][] = DOKU_INC . 'lib/plugins/farmer/includes/plugins.php'; 326a646d519SAndreas Gohr } 327a646d519SAndreas Gohr 328a646d519SAndreas Gohr /** 329da0ae2c0SAndreas Gohr * Loads the farm config 330da0ae2c0SAndreas Gohr */ 331da0ae2c0SAndreas Gohr protected function loadConfig() { 332da0ae2c0SAndreas Gohr $ini = DOKU_INC . 'conf/farm.ini'; 333da0ae2c0SAndreas Gohr if(!file_exists($ini)) return; 334da0ae2c0SAndreas Gohr $config = parse_ini_file($ini, true); 335da0ae2c0SAndreas Gohr foreach(array_keys($this->config) as $section) { 336da0ae2c0SAndreas Gohr if(isset($config[$section])) { 337da0ae2c0SAndreas Gohr $this->config[$section] = array_merge( 338da0ae2c0SAndreas Gohr $this->config[$section], 339da0ae2c0SAndreas Gohr $config[$section] 340da0ae2c0SAndreas Gohr ); 341da0ae2c0SAndreas Gohr } 342da0ae2c0SAndreas Gohr } 343da0ae2c0SAndreas Gohr 344a646d519SAndreas Gohr $this->config['base']['farmdir'] = trim($this->config['base']['farmdir']); 345a646d519SAndreas Gohr $this->config['base']['farmhost'] = strtolower(trim($this->config['base']['farmhost'])); 346a646d519SAndreas Gohr } 347da0ae2c0SAndreas Gohr 348da0ae2c0SAndreas Gohr} 349da0ae2c0SAndreas Gohr 350da0ae2c0SAndreas Gohr// initialize it globally 35185becf1bSAndreas Gohrif(!defined('DOKU_UNITTEST')) { 352da0ae2c0SAndreas Gohr global $FARMCORE; 353da0ae2c0SAndreas Gohr $FARMCORE = new DokuWikiFarmCore(); 35485becf1bSAndreas Gohr} 355