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