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