xref: /plugin/farmer/DokuWikiFarmCore.php (revision 36282384980ddf0e8c163acad5205d76724ba521)
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,
40af1c6dd8SAndreas 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
156*36282384SAndreas Gohr        // no host - no host based setup. if we're still here then it's the farmer
157*36282384SAndreas Gohr        if(!isset($_SERVER['HTTP_HOST'])) return;
158*36282384SAndreas Gohr
159a646d519SAndreas Gohr        // is this the farmer?
160a646d519SAndreas Gohr        if(strtolower($_SERVER['HTTP_HOST']) == $farmhost) {
161a646d519SAndreas Gohr            return;
162a646d519SAndreas Gohr        }
163a646d519SAndreas Gohr
16485becf1bSAndreas Gohr        // still here? check for host based
165a646d519SAndreas Gohr        $this->hostbased = true;
16685becf1bSAndreas Gohr        $possible = $this->getAnimalNamesForHost($_SERVER['HTTP_HOST']);
16785becf1bSAndreas Gohr        foreach($possible as $animal) {
168a646d519SAndreas Gohr            if(is_dir("$farmdir/$animal/conf/")) {
169a646d519SAndreas Gohr                $this->animal = $animal;
170a646d519SAndreas Gohr                return;
171a646d519SAndreas Gohr            }
172a646d519SAndreas Gohr        }
173a646d519SAndreas Gohr
174a646d519SAndreas Gohr        // no hit
175a646d519SAndreas Gohr        $this->notfound = true;
176a646d519SAndreas Gohr        return;
177a646d519SAndreas Gohr    }
178a646d519SAndreas Gohr
179a646d519SAndreas Gohr    /**
18085becf1bSAndreas Gohr     * Return a list of possible animal names for the given host
18185becf1bSAndreas Gohr     *
18285becf1bSAndreas Gohr     * @param string $host the HTTP_HOST header
18385becf1bSAndreas Gohr     * @return array
18485becf1bSAndreas Gohr     */
18585becf1bSAndreas Gohr    protected function getAnimalNamesForHost($host) {
18685becf1bSAndreas Gohr        $animals = array();
18785becf1bSAndreas Gohr        $parts = explode('.', implode('.', explode(':', rtrim($host, '.'))));
18885becf1bSAndreas Gohr        for($j = count($parts); $j > 0; $j--) {
18985becf1bSAndreas Gohr            // strip from the end
19085becf1bSAndreas Gohr            $animals[] = implode('.', array_slice($parts, 0, $j));
19185becf1bSAndreas Gohr            // strip from the end without host part
19285becf1bSAndreas Gohr            $animals[] = implode('.', array_slice($parts, 1, $j));
19385becf1bSAndreas Gohr        }
19485becf1bSAndreas Gohr        $animals = array_unique($animals);
19585becf1bSAndreas Gohr        $animals = array_filter($animals);
1960a5d2da2SAndreas Gohr        usort(
197bfecda9bSAndreas Gohr            $animals,
198bfecda9bSAndreas Gohr            // compare by length, then alphabet
199bfecda9bSAndreas Gohr            function ($a, $b) {
200bfecda9bSAndreas Gohr                $ret = strlen($b) - strlen($a);
201bfecda9bSAndreas Gohr                if($ret != 0) return $ret;
202bfecda9bSAndreas Gohr                return $a > $b;
2030a5d2da2SAndreas Gohr            }
2040a5d2da2SAndreas Gohr        );
20585becf1bSAndreas Gohr        return $animals;
20685becf1bSAndreas Gohr    }
20785becf1bSAndreas Gohr
20885becf1bSAndreas Gohr    /**
209a646d519SAndreas Gohr     * This sets up the default farming config cascade
210a646d519SAndreas Gohr     */
211a646d519SAndreas Gohr    protected function setupCascade() {
212a646d519SAndreas Gohr        global $config_cascade;
213a646d519SAndreas Gohr        $config_cascade = array(
214a646d519SAndreas Gohr            'main' => array(
215a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/dokuwiki.php',),
216a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'local.php',),
217a646d519SAndreas Gohr                'protected' => array(DOKU_CONF . 'local.protected.php',),
218a646d519SAndreas Gohr            ),
219a646d519SAndreas Gohr            'acronyms' => array(
220a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/acronyms.conf',),
221a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'acronyms.local.conf',),
222a646d519SAndreas Gohr            ),
223a646d519SAndreas Gohr            'entities' => array(
224a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/entities.conf',),
225a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'entities.local.conf',),
226a646d519SAndreas Gohr            ),
227a646d519SAndreas Gohr            'interwiki' => array(
228a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/interwiki.conf',),
229a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'interwiki.local.conf',),
230a646d519SAndreas Gohr            ),
231a646d519SAndreas Gohr            'license' => array(
232a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/license.php',),
233a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'license.local.php',),
234a646d519SAndreas Gohr            ),
235a646d519SAndreas Gohr            'mediameta' => array(
236a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/mediameta.php',),
237a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'mediameta.local.php',),
238a646d519SAndreas Gohr            ),
239a646d519SAndreas Gohr            'mime' => array(
240a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/mime.conf',),
241a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'mime.local.conf',),
242a646d519SAndreas Gohr            ),
243a646d519SAndreas Gohr            'scheme' => array(
244a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/scheme.conf',),
245a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'scheme.local.conf',),
246a646d519SAndreas Gohr            ),
247a646d519SAndreas Gohr            'smileys' => array(
248a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/smileys.conf',),
249a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'smileys.local.conf',),
250a646d519SAndreas Gohr            ),
251a646d519SAndreas Gohr            'wordblock' => array(
252a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/wordblock.conf',),
253a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'wordblock.local.conf',),
254a646d519SAndreas Gohr            ),
255a646d519SAndreas Gohr            'acl' => array(
256a646d519SAndreas Gohr                'default' => DOKU_CONF . 'acl.auth.php',
257a646d519SAndreas Gohr            ),
258a646d519SAndreas Gohr            'plainauth.users' => array(
259a646d519SAndreas Gohr                'default' => DOKU_CONF . 'users.auth.php',
260a646d519SAndreas Gohr            ),
261de156015SAndreas Gohr            'plugins' => array(
262a646d519SAndreas Gohr                'default' => array(DOKU_INC . 'conf/plugins.php',),
263a646d519SAndreas Gohr                'local' => array(DOKU_CONF . 'plugins.local.php',),
264a646d519SAndreas Gohr                'protected' => array(
265a646d519SAndreas Gohr                    DOKU_INC . 'conf/plugins.required.php',
266a646d519SAndreas Gohr                    DOKU_CONF . 'plugins.protected.php',
267a646d519SAndreas Gohr                ),
268a646d519SAndreas Gohr            ),
269a646d519SAndreas Gohr            'userstyle' => array(
270a646d519SAndreas Gohr                'screen' => array(DOKU_CONF . 'userstyle.css', DOKU_CONF . 'userstyle.less',),
271a646d519SAndreas Gohr                'print' => array(DOKU_CONF . 'userprint.css', DOKU_CONF . 'userprint.less',),
272a646d519SAndreas Gohr                'feed' => array(DOKU_CONF . 'userfeed.css', DOKU_CONF . 'userfeed.less',),
273a646d519SAndreas Gohr                'all' => array(DOKU_CONF . 'userall.css', DOKU_CONF . 'userall.less',),
274a646d519SAndreas Gohr            ),
275a646d519SAndreas Gohr            'userscript' => array(
276a646d519SAndreas Gohr                'default' => array(DOKU_CONF . 'userscript.js',),
277a646d519SAndreas Gohr            ),
278a646d519SAndreas Gohr        );
279a646d519SAndreas Gohr    }
280a646d519SAndreas Gohr
281a646d519SAndreas Gohr    /**
282a646d519SAndreas Gohr     * This adds additional files to the config cascade based on the inheritence settings
283a646d519SAndreas Gohr     *
284a646d519SAndreas Gohr     * These are only added for animals, not the farmer
285a646d519SAndreas Gohr     */
286a646d519SAndreas Gohr    protected function adjustCascade() {
287b330074aSAndreas Gohr        // nothing to do when on the farmer:
288b330074aSAndreas Gohr        if(!$this->animal) return;
289a646d519SAndreas Gohr
290b330074aSAndreas Gohr        global $config_cascade;
291a646d519SAndreas Gohr        foreach($this->config['inherit'] as $key => $val) {
292a646d519SAndreas Gohr            if(!$val) continue;
293a646d519SAndreas Gohr
294a646d519SAndreas Gohr            // prepare what is to append or prepend
295a646d519SAndreas Gohr            $append = array();
296a646d519SAndreas Gohr            $prepend = array();
297a646d519SAndreas Gohr            if($key == 'main') {
298b330074aSAndreas Gohr                $append = array(
299b330074aSAndreas Gohr                    'default' => array(DOKU_INC . 'conf/local.php'),
300b330074aSAndreas Gohr                    'protected' => array(DOKU_INC . 'lib/plugins/farmer/includes/config.php')
301b330074aSAndreas Gohr                );
302a646d519SAndreas Gohr            } elseif($key == 'license') {
303a646d519SAndreas Gohr                $append = array('default' => array(DOKU_INC . 'conf/' . $key . '.local.php'));
304a646d519SAndreas Gohr            } elseif($key == 'userscript') {
305a646d519SAndreas Gohr                $prepend = array('default' => array(DOKU_INC . 'conf/userscript.js'));
306a646d519SAndreas Gohr            } elseif($key == 'userstyle') {
307a646d519SAndreas Gohr                $prepend = array(
308a646d519SAndreas Gohr                    'screen' => array(DOKU_INC . 'conf/userstyle.css', DOKU_INC . 'conf/userstyle.less',),
309a646d519SAndreas Gohr                    'print' => array(DOKU_INC . 'conf/userprint.css', DOKU_INC . 'conf/userprint.less',),
310a646d519SAndreas Gohr                    'feed' => array(DOKU_INC . 'conf/userfeed.css', DOKU_INC . 'conf/userfeed.less',),
311a646d519SAndreas Gohr                    'all' => array(DOKU_INC . 'conf/userall.css', DOKU_INC . 'conf/userall.less',),
312a646d519SAndreas Gohr                );
3131272da0cSAndreas Gohr            } elseif($key == 'users') {
3141272da0cSAndreas Gohr                $config_cascade['plainauth.users']['protected'] = DOKU_INC . 'conf/users.auth.php';
315af1c6dd8SAndreas Gohr            } elseif($key == 'plugins') {
316af1c6dd8SAndreas Gohr                $append = array('default' => array(DOKU_INC . 'conf/plugins.local.php'));
317a646d519SAndreas Gohr            } else {
318a646d519SAndreas Gohr                $append = array('default' => array(DOKU_INC . 'conf/' . $key . '.local.conf'));
319a646d519SAndreas Gohr            }
320a646d519SAndreas Gohr
321a646d519SAndreas Gohr            // add to cascade
322a646d519SAndreas Gohr            foreach($prepend as $section => $data) {
323a646d519SAndreas Gohr                $config_cascade[$key][$section] = array_merge($data, $config_cascade[$key][$section]);
324a646d519SAndreas Gohr            }
325a646d519SAndreas Gohr            foreach($append as $section => $data) {
326a646d519SAndreas Gohr                $config_cascade[$key][$section] = array_merge($config_cascade[$key][$section], $data);
327a646d519SAndreas Gohr            }
328a646d519SAndreas Gohr        }
329de156015SAndreas Gohr
330de156015SAndreas Gohr        // add plugin overrides
331de156015SAndreas Gohr        $config_cascade['plugins']['protected'][] = DOKU_INC . 'lib/plugins/farmer/includes/plugins.php';
332a646d519SAndreas Gohr    }
333a646d519SAndreas Gohr
334a646d519SAndreas Gohr    /**
335da0ae2c0SAndreas Gohr     * Loads the farm config
336da0ae2c0SAndreas Gohr     */
337da0ae2c0SAndreas Gohr    protected function loadConfig() {
338da0ae2c0SAndreas Gohr        $ini = DOKU_INC . 'conf/farm.ini';
339da0ae2c0SAndreas Gohr        if(!file_exists($ini)) return;
340da0ae2c0SAndreas Gohr        $config = parse_ini_file($ini, true);
341da0ae2c0SAndreas Gohr        foreach(array_keys($this->config) as $section) {
342da0ae2c0SAndreas Gohr            if(isset($config[$section])) {
343da0ae2c0SAndreas Gohr                $this->config[$section] = array_merge(
344da0ae2c0SAndreas Gohr                    $this->config[$section],
345da0ae2c0SAndreas Gohr                    $config[$section]
346da0ae2c0SAndreas Gohr                );
347da0ae2c0SAndreas Gohr            }
348da0ae2c0SAndreas Gohr        }
349da0ae2c0SAndreas Gohr
350a646d519SAndreas Gohr        $this->config['base']['farmdir'] = trim($this->config['base']['farmdir']);
351a646d519SAndreas Gohr        $this->config['base']['farmhost'] = strtolower(trim($this->config['base']['farmhost']));
352a646d519SAndreas Gohr    }
353da0ae2c0SAndreas Gohr
354da0ae2c0SAndreas Gohr}
355da0ae2c0SAndreas Gohr
356da0ae2c0SAndreas Gohr// initialize it globally
35785becf1bSAndreas Gohrif(!defined('DOKU_UNITTEST')) {
358da0ae2c0SAndreas Gohr    global $FARMCORE;
359da0ae2c0SAndreas Gohr    $FARMCORE = new DokuWikiFarmCore();
36085becf1bSAndreas Gohr}
361