xref: /plugin/ditaa/syntax.php (revision 400b2bf32de242a59e3d73d976b51be8d8895045)
1a34ed36bSDennis Ploeger<?php
2a34ed36bSDennis Ploeger/**
3a34ed36bSDennis Ploeger * Ditaa-Plugin: Converts Ascii-Flowcharts into a png-File
4a34ed36bSDennis Ploeger *
5a34ed36bSDennis Ploeger * @license     GPL 2 (http://www.gnu.org/licenses/gpl.html)
6a34ed36bSDennis Ploeger * @author      Dennis Ploeger <develop [at] dieploegers [dot] de>
7a34ed36bSDennis Ploeger * @author      Christoph Mertins <c [dot] mertins [at] gmail [dot] com>
8c5eb4ae0SGerry Weißbach * @author      Gerry Weißbach / i-net software <tools [at] inetsoftware [dot] de>
9c9ea14c6Seinhirn * @author      Christian Marg <marg@rz.tu-clausthal.de>
10d402f512SAndreas Gohr * @author      Andreas Gohr <andi@splitbrain.org>
11a34ed36bSDennis Ploeger */
12a34ed36bSDennis Ploeger
13a34ed36bSDennis Ploegerif(!defined('DOKU_INC')) define('DOKU_INC', realpath(dirname(__FILE__) . '/../../') . '/');
14a34ed36bSDennis Ploegerif(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN', DOKU_INC . 'lib/plugins/');
15a34ed36bSDennis Ploegerrequire_once(DOKU_PLUGIN . 'syntax.php');
16a34ed36bSDennis Ploeger
1735cad792SAndreas Gohr/**
1835cad792SAndreas Gohr * Class syntax_plugin_ditaa
1935cad792SAndreas Gohr */
20a34ed36bSDennis Ploegerclass syntax_plugin_ditaa extends DokuWiki_Syntax_Plugin {
21a34ed36bSDennis Ploeger
22a34ed36bSDennis Ploeger    /**
23a34ed36bSDennis Ploeger     * What about paragraphs?
24a34ed36bSDennis Ploeger     */
2535cad792SAndreas Gohr    public function getPType() {
26a34ed36bSDennis Ploeger        return 'normal';
27a34ed36bSDennis Ploeger    }
28a34ed36bSDennis Ploeger
29a34ed36bSDennis Ploeger    /**
30a34ed36bSDennis Ploeger     * What kind of syntax are we?
31a34ed36bSDennis Ploeger     */
3235cad792SAndreas Gohr    public function getType() {
33a34ed36bSDennis Ploeger        return 'substition';
34a34ed36bSDennis Ploeger    }
35a34ed36bSDennis Ploeger
36a34ed36bSDennis Ploeger    /**
37a34ed36bSDennis Ploeger     * Where to sort in?
38a34ed36bSDennis Ploeger     */
3935cad792SAndreas Gohr    public function getSort() {
40a34ed36bSDennis Ploeger        return 200;
41a34ed36bSDennis Ploeger    }
42a34ed36bSDennis Ploeger
43a34ed36bSDennis Ploeger    /**
4472a2bf1dSAndreas Gohr     * Connect pattern to lexer
4535cad792SAndreas Gohr     *
4635cad792SAndreas Gohr     * @param string $mode
47a34ed36bSDennis Ploeger     */
4835cad792SAndreas Gohr    public function connectTo($mode) {
4972a2bf1dSAndreas Gohr        $this->Lexer->addSpecialPattern('<ditaa.*?>\n.*?\n</ditaa>', $mode, 'plugin_ditaa');
50a34ed36bSDennis Ploeger    }
51a34ed36bSDennis Ploeger
52a34ed36bSDennis Ploeger    /**
5335cad792SAndreas Gohr     * Stores all infor about the diagram in two files. One is the actual ditaa data, the other
5435cad792SAndreas Gohr     * contains the options.
5535cad792SAndreas Gohr     *
5635cad792SAndreas Gohr     * @param   string $match The text matched by the patterns
5735cad792SAndreas Gohr     * @param   int $state The lexer state for the match
5835cad792SAndreas Gohr     * @param   int $pos The character position of the matched text
5935cad792SAndreas Gohr     * @param   Doku_Handler $handler The Doku_Handler object
6035cad792SAndreas Gohr     * @return  bool|array Return an array with all data you want to use in render, false don't add an instruction
61a34ed36bSDennis Ploeger     */
6235cad792SAndreas Gohr    public function handle($match, $state, $pos, Doku_Handler $handler) {
632a956e66SAndreas Gohr        $info = $this->getInfo();
642a956e66SAndreas Gohr
6572a2bf1dSAndreas Gohr        // prepare default data
6672a2bf1dSAndreas Gohr        $return = array(
6772a2bf1dSAndreas Gohr            'width' => 0,
6872a2bf1dSAndreas Gohr            'height' => 0,
6972a2bf1dSAndreas Gohr            'antialias' => true,
7072a2bf1dSAndreas Gohr            'edgesep' => true,
7172a2bf1dSAndreas Gohr            'round' => false,
7272a2bf1dSAndreas Gohr            'shadow' => true,
7372a2bf1dSAndreas Gohr            'scale' => 1,
7472a2bf1dSAndreas Gohr            'align' => '',
7538c92790SGerry Weißbach            'version' => $info['date'], //force rebuild of images on update
7672a2bf1dSAndreas Gohr        );
77a34ed36bSDennis Ploeger
7872a2bf1dSAndreas Gohr        // prepare input
7972a2bf1dSAndreas Gohr        $lines = explode("\n", $match);
8072a2bf1dSAndreas Gohr        $conf = array_shift($lines);
8172a2bf1dSAndreas Gohr        array_pop($lines);
8272a2bf1dSAndreas Gohr
8372a2bf1dSAndreas Gohr        // match config options
8472a2bf1dSAndreas Gohr        if(preg_match('/\b(left|center|right)\b/i', $conf, $match)) $return['align'] = $match[1];
8572a2bf1dSAndreas Gohr        if(preg_match('/\b(\d+)x(\d+)\b/', $conf, $match)) {
8672a2bf1dSAndreas Gohr            $return['width'] = $match[1];
8772a2bf1dSAndreas Gohr            $return['height'] = $match[2];
88a34ed36bSDennis Ploeger        }
892a956e66SAndreas Gohr        if(preg_match('/\b(\d+(\.\d+)?)X\b/', $conf, $match)) $return['scale'] = $match[1];
9072a2bf1dSAndreas Gohr        if(preg_match('/\bwidth=([0-9]+)\b/i', $conf, $match)) $return['width'] = $match[1];
9172a2bf1dSAndreas Gohr        if(preg_match('/\bheight=([0-9]+)\b/i', $conf, $match)) $return['height'] = $match[1];
9272a2bf1dSAndreas Gohr        // match boolean toggles
9372a2bf1dSAndreas Gohr        if(preg_match_all('/\b(no)?(antialias|edgesep|round|shadow)\b/i', $conf, $matches, PREG_SET_ORDER)) {
9472a2bf1dSAndreas Gohr            foreach($matches as $match) {
9572a2bf1dSAndreas Gohr                $return[$match[2]] = !$match[1];
9672a2bf1dSAndreas Gohr            }
9772a2bf1dSAndreas Gohr        }
9872a2bf1dSAndreas Gohr
9905cbae88SAndreas Gohr        $input = join("\n", $lines);
10005cbae88SAndreas Gohr        $return['md5'] = md5($input); // we only pass a hash around
10105cbae88SAndreas Gohr
10235cad792SAndreas Gohr        // store input for later use in _imagefile()
10335cad792SAndreas Gohr        io_saveFile(getCacheName($return['md5'], '.ditaa.txt'), $input);
10435cad792SAndreas Gohr        io_saveFile(getCacheName($return['md5'], '.ditaa.cfg'), serialize($return));
10572a2bf1dSAndreas Gohr
10672a2bf1dSAndreas Gohr        return $return;
107a34ed36bSDennis Ploeger    }
108a34ed36bSDennis Ploeger
109a34ed36bSDennis Ploeger    /**
11035cad792SAndreas Gohr     * Output the image
11135cad792SAndreas Gohr     *
11235cad792SAndreas Gohr     * @param string $format output format being rendered
11335cad792SAndreas Gohr     * @param Doku_Renderer $R the current renderer object
11435cad792SAndreas Gohr     * @param array $data data created by handler()
11535cad792SAndreas Gohr     * @return  boolean                 rendered correctly?
116c5eb4ae0SGerry Weißbach     */
11735cad792SAndreas Gohr    public function render($format, Doku_Renderer $R, $data) {
11838c92790SGerry Weißbach        global $ID;
11995615d6cSAndreas Gohr        if($format == 'xhtml') {
12038c92790SGerry Weißbach            // Only use the md5 key
12138c92790SGerry Weißbach            $img = ml($ID, array('ditaa' => $data['md5']));
12295615d6cSAndreas Gohr            $R->doc .= '<img src="' . $img . '" class="media' . $data['align'] . '" alt=""';
12395615d6cSAndreas Gohr            if($data['width']) $R->doc .= ' width="' . $data['width'] . '"';
12495615d6cSAndreas Gohr            if($data['height']) $R->doc .= ' height="' . $data['height'] . '"';
125c77baa73SWilli Schönborn            if($data['align'] == 'right') $R->doc .= ' align="right"';
126c77baa73SWilli Schönborn            if($data['align'] == 'left') $R->doc .= ' align="left"';
12795615d6cSAndreas Gohr            $R->doc .= '/>';
12895615d6cSAndreas Gohr            return true;
12995615d6cSAndreas Gohr        } else if($format == 'odt') {
13035cad792SAndreas Gohr            $src = $this->_imgfile($data['md5']);
13135cad792SAndreas Gohr            /** @var  renderer_plugin_odt $R */
13295615d6cSAndreas Gohr            $R->_odtAddImage($src, $data['width'], $data['height'], $data['align']);
13395615d6cSAndreas Gohr            return true;
13495615d6cSAndreas Gohr        }
13595615d6cSAndreas Gohr        return false;
13695615d6cSAndreas Gohr    }
13795615d6cSAndreas Gohr
13895615d6cSAndreas Gohr    /**
13995615d6cSAndreas Gohr     * Return path to the rendered image on our local system
14035cad792SAndreas Gohr     *
14135cad792SAndreas Gohr     * @param string $md5 MD5 of the input data, used to identify the cache files
14235cad792SAndreas Gohr     * @return false|string path to file or fals on error
14395615d6cSAndreas Gohr     */
14435cad792SAndreas Gohr    public function _imgfile($md5) {
14535cad792SAndreas Gohr        $file_cfg = getCacheName($md5, '.ditaa.cfg'); // configs
14635cad792SAndreas Gohr        $file_txt = getCacheName($md5, '.ditaa.txt'); // input
14735cad792SAndreas Gohr        $file_png = getCacheName($md5, '.ditaa.png'); // ouput
14838c92790SGerry Weißbach
14935cad792SAndreas Gohr        if(!file_exists($file_cfg) || !file_exists($file_txt)) {
15035cad792SAndreas Gohr            return false;
15138c92790SGerry Weißbach        }
15235cad792SAndreas Gohr        $data = unserialize(io_readFile($file_cfg, false));
15335cad792SAndreas Gohr
15435cad792SAndreas Gohr        // file does not exist or is outdated
15535cad792SAndreas Gohr        if(@filemtime($file_png) < filemtime($file_cfg)) {
15638c92790SGerry Weißbach
15795615d6cSAndreas Gohr            if($this->getConf('java')) {
15835cad792SAndreas Gohr                $ok = $this->_runJava($data, $file_txt, $file_png);
15995615d6cSAndreas Gohr            } else {
16035cad792SAndreas Gohr                $ok = $this->_runGo($data, $file_txt, $file_png);
161967610bbSAndreas Gohr                #$ok = $this->_remote($data, $in, $cache);
16295615d6cSAndreas Gohr            }
16395615d6cSAndreas Gohr            if(!$ok) return false;
16435cad792SAndreas Gohr
16535cad792SAndreas Gohr            clearstatcache($file_png);
16695615d6cSAndreas Gohr        }
16795615d6cSAndreas Gohr
16895615d6cSAndreas Gohr        // resized version
16995615d6cSAndreas Gohr        if($data['width']) {
17035cad792SAndreas Gohr            $file_png = media_resize_image($file_png, 'png', $data['width'], $data['height']);
17195615d6cSAndreas Gohr        }
17295615d6cSAndreas Gohr
17395615d6cSAndreas Gohr        // something went wrong, we're missing the file
17435cad792SAndreas Gohr        if(!file_exists($file_png)) return false;
17595615d6cSAndreas Gohr
17635cad792SAndreas Gohr        return $file_png;
17795615d6cSAndreas Gohr    }
17895615d6cSAndreas Gohr
17995615d6cSAndreas Gohr    /**
18005cbae88SAndreas Gohr     * Render the output remotely at ditaa.org
18135cad792SAndreas Gohr     *
18235cad792SAndreas Gohr     * @deprecated ditaa.org is no longer available, so this defunct
18335cad792SAndreas Gohr     * @param array $data The config settings
18435cad792SAndreas Gohr     * @param string $in Path to the ditaa input file (txt)
18535cad792SAndreas Gohr     * @param string $out Path to the output file (PNG)
18635cad792SAndreas Gohr     * @return bool true if the image was created, false otherwise
18705cbae88SAndreas Gohr     */
18835cad792SAndreas Gohr    protected function _remote($data, $in, $out) {
189967610bbSAndreas Gohr        global $conf;
190967610bbSAndreas Gohr
19195615d6cSAndreas Gohr        if(!file_exists($in)) {
19295615d6cSAndreas Gohr            if($conf['debug']) {
19395615d6cSAndreas Gohr                dbglog($in, 'no such ditaa input file');
19495615d6cSAndreas Gohr            }
19595615d6cSAndreas Gohr            return false;
19695615d6cSAndreas Gohr        }
19795615d6cSAndreas Gohr
19805cbae88SAndreas Gohr        $http = new DokuHTTPClient();
19905cbae88SAndreas Gohr        $http->timeout = 30;
20005cbae88SAndreas Gohr
20105cbae88SAndreas Gohr        $pass = array();
20205cbae88SAndreas Gohr        $pass['scale'] = $data['scale'];
20305cbae88SAndreas Gohr        $pass['timeout'] = 25;
20405cbae88SAndreas Gohr        $pass['grid'] = io_readFile($in);
20505cbae88SAndreas Gohr        if(!$data['antialias']) $pass['A'] = 'on';
20605cbae88SAndreas Gohr        if(!$data['shadow']) $pass['S'] = 'on';
20705cbae88SAndreas Gohr        if($data['round']) $pass['r'] = 'on';
20805cbae88SAndreas Gohr        if(!$data['edgesep']) $pass['E'] = 'on';
20905cbae88SAndreas Gohr
21005cbae88SAndreas Gohr        $img = $http->post('http://ditaa.org/ditaa/render', $pass);
21105cbae88SAndreas Gohr        if(!$img) return false;
21205cbae88SAndreas Gohr
21395615d6cSAndreas Gohr        return io_saveFile($out, $img);
21405cbae88SAndreas Gohr    }
21505cbae88SAndreas Gohr
216a34ed36bSDennis Ploeger    /**
2172a956e66SAndreas Gohr     * Run the ditaa Java program
21835cad792SAndreas Gohr     *
21935cad792SAndreas Gohr     * @param array $data The config settings
22035cad792SAndreas Gohr     * @param string $in Path to the ditaa input file (txt)
22135cad792SAndreas Gohr     * @param string $out Path to the output file (PNG)
22235cad792SAndreas Gohr     * @return bool true if the image was created, false otherwise
223a34ed36bSDennis Ploeger     */
22435cad792SAndreas Gohr    protected function _runJava($data, $in, $out) {
2252a956e66SAndreas Gohr        global $conf;
226a34ed36bSDennis Ploeger
22705cbae88SAndreas Gohr        if(!file_exists($in)) {
22805cbae88SAndreas Gohr            if($conf['debug']) {
22905cbae88SAndreas Gohr                dbglog($in, 'no such ditaa input file');
23005cbae88SAndreas Gohr            }
23105cbae88SAndreas Gohr            return false;
23205cbae88SAndreas Gohr        }
233a34ed36bSDennis Ploeger
2342a956e66SAndreas Gohr        $cmd = $this->getConf('java');
23536721e14SAndreas Gohr        $cmd .= ' -Djava.awt.headless=true -Dfile.encoding=UTF-8 -jar';
2362a956e66SAndreas Gohr        $cmd .= ' ' . escapeshellarg(dirname(__FILE__) . '/ditaa/ditaa0_9.jar'); //ditaa jar
23736721e14SAndreas Gohr        $cmd .= ' --encoding UTF-8';
23805cbae88SAndreas Gohr        $cmd .= ' ' . escapeshellarg($in); //input
23905cbae88SAndreas Gohr        $cmd .= ' ' . escapeshellarg($out); //output
2402a956e66SAndreas Gohr        $cmd .= ' -s ' . escapeshellarg($data['scale']);
2412a956e66SAndreas Gohr        if(!$data['antialias']) $cmd .= ' -A';
2422a956e66SAndreas Gohr        if(!$data['shadow']) $cmd .= ' -S';
2432a956e66SAndreas Gohr        if($data['round']) $cmd .= ' -r';
2442a956e66SAndreas Gohr        if(!$data['edgesep']) $cmd .= ' -E';
245a34ed36bSDennis Ploeger
246a34ed36bSDennis Ploeger        exec($cmd, $output, $error);
247a34ed36bSDennis Ploeger
2483c74ed32SAndreas Gohr        if($error != 0) {
2493c74ed32SAndreas Gohr            if($conf['debug']) {
2503c74ed32SAndreas Gohr                dbglog(join("\n", $output), 'ditaa command failed: ' . $cmd);
2513c74ed32SAndreas Gohr            }
2523c74ed32SAndreas Gohr            return false;
2533c74ed32SAndreas Gohr        }
25405cbae88SAndreas Gohr
255a34ed36bSDennis Ploeger        return true;
256a34ed36bSDennis Ploeger    }
257a34ed36bSDennis Ploeger
258967610bbSAndreas Gohr    /**
259967610bbSAndreas Gohr     * Run the ditaa Go program
26035cad792SAndreas Gohr     *
26135cad792SAndreas Gohr     * @param array $data The config settings - currently not used because the Go relase supports no options
26235cad792SAndreas Gohr     * @param string $in Path to the ditaa input file (txt)
26335cad792SAndreas Gohr     * @param string $out Path to the output file (PNG)
26435cad792SAndreas Gohr     * @return bool true if the image was created, false otherwise
265967610bbSAndreas Gohr     */
26635cad792SAndreas Gohr    protected function _runGo($data, $in, $out) {
267967610bbSAndreas Gohr        global $conf;
268967610bbSAndreas Gohr
269967610bbSAndreas Gohr        if(!file_exists($in)) {
270967610bbSAndreas Gohr            if($conf['debug']) {
271967610bbSAndreas Gohr                dbglog($in, 'no such ditaa input file');
272967610bbSAndreas Gohr            }
273967610bbSAndreas Gohr            return false;
274967610bbSAndreas Gohr        }
275967610bbSAndreas Gohr
276967610bbSAndreas Gohr        $cmd = $this->getLocalBinary();
277967610bbSAndreas Gohr        if(!$cmd) return false;
278967610bbSAndreas Gohr        $cmd .= ' ' . escapeshellarg($in); //input
279967610bbSAndreas Gohr        $cmd .= ' ' . escapeshellarg($out); //output
280967610bbSAndreas Gohr
281967610bbSAndreas Gohr        exec($cmd, $output, $error);
282967610bbSAndreas Gohr
283967610bbSAndreas Gohr        if($error != 0) {
284967610bbSAndreas Gohr            if($conf['debug']) {
285967610bbSAndreas Gohr                dbglog(join("\n", $output), 'ditaa command failed: ' . $cmd);
286967610bbSAndreas Gohr            }
287967610bbSAndreas Gohr            return false;
288967610bbSAndreas Gohr        }
289967610bbSAndreas Gohr
290967610bbSAndreas Gohr        return true;
291967610bbSAndreas Gohr    }
292967610bbSAndreas Gohr
293967610bbSAndreas Gohr    /**
294967610bbSAndreas Gohr     * Detects the platform of the PHP host and constructs the appropriate binary name
295967610bbSAndreas Gohr     *
296967610bbSAndreas Gohr     * @return false|string
297967610bbSAndreas Gohr     */
298967610bbSAndreas Gohr    protected function getBinaryName() {
299967610bbSAndreas Gohr        $ext = '';
300967610bbSAndreas Gohr
301967610bbSAndreas Gohr        $os = php_uname('s');
302967610bbSAndreas Gohr        if(preg_match('/darwin/i', $os)) {
303967610bbSAndreas Gohr            $os = 'darwin';
304967610bbSAndreas Gohr        } elseif(preg_match('/win/i', $os)) {
305967610bbSAndreas Gohr            $os = 'windows';
306967610bbSAndreas Gohr            $ext = '.exe';
307967610bbSAndreas Gohr        } elseif(preg_match('/linux/i', $os)) {
308967610bbSAndreas Gohr            $os = 'linux';
309967610bbSAndreas Gohr        } elseif(preg_match('/freebsd/i', $os)) {
310967610bbSAndreas Gohr            $os = 'freebsd';
311967610bbSAndreas Gohr        } elseif(preg_match('/openbsd/i', $os)) {
312967610bbSAndreas Gohr            $os = 'openbsd';
313967610bbSAndreas Gohr        } elseif(preg_match('/netbsd/i', $os)) {
314967610bbSAndreas Gohr            $os = 'netbsd';
315967610bbSAndreas Gohr        } elseif(preg_match('/(solaris|netbsd)/i', $os)) {
316967610bbSAndreas Gohr            $os = 'freebsd';
317967610bbSAndreas Gohr        } else {
318967610bbSAndreas Gohr            return false;
319967610bbSAndreas Gohr        }
320967610bbSAndreas Gohr
321967610bbSAndreas Gohr        $arch = php_uname('m');
322967610bbSAndreas Gohr        if($arch == 'x86_64') {
323967610bbSAndreas Gohr            $arch = 'amd64';
324967610bbSAndreas Gohr        } elseif(preg_match('/arm/i', $arch)) {
325967610bbSAndreas Gohr            $arch = 'amd';
326967610bbSAndreas Gohr        } else {
327967610bbSAndreas Gohr            $arch = '386';
328967610bbSAndreas Gohr        }
329967610bbSAndreas Gohr
330967610bbSAndreas Gohr        return "ditaa-$os-$arch$ext";
331967610bbSAndreas Gohr    }
332967610bbSAndreas Gohr
333967610bbSAndreas Gohr    /**
334967610bbSAndreas Gohr     * Returns the local binary to use
335967610bbSAndreas Gohr     *
33635cad792SAndreas Gohr     * Downloads it if necessary
33735cad792SAndreas Gohr     *
338967610bbSAndreas Gohr     * @return bool|string
339967610bbSAndreas Gohr     */
340967610bbSAndreas Gohr    protected function getLocalBinary() {
341967610bbSAndreas Gohr        global $conf;
342967610bbSAndreas Gohr
343967610bbSAndreas Gohr        $bin = $this->getBinaryName();
344967610bbSAndreas Gohr        if(!$bin) return false;
345967610bbSAndreas Gohr
346967610bbSAndreas Gohr        // check distributed files first
347967610bbSAndreas Gohr        if(file_exists(__DIR__ . '/ditaa/' . $bin)) {
348967610bbSAndreas Gohr            return __DIR__ . '/ditaa/' . $bin;
349967610bbSAndreas Gohr        }
350967610bbSAndreas Gohr
351967610bbSAndreas Gohr        $info = $this->getInfo();
352967610bbSAndreas Gohr        $cache = getCacheName($info['date'], ".$bin");
353967610bbSAndreas Gohr
354967610bbSAndreas Gohr        if(file_exists($cache)) return $cache;
355967610bbSAndreas Gohr
356*400b2bf3SAndreas Gohr        $url = 'https://github.com/akavel/ditaa/releases/download/g1.0.0/' . $bin;
357967610bbSAndreas Gohr        if(io_download($url, $cache, false, '', 0)) {
358967610bbSAndreas Gohr            @chmod($cache, $conf['dmode']);
359967610bbSAndreas Gohr            return $cache;
360967610bbSAndreas Gohr        }
361967610bbSAndreas Gohr
362967610bbSAndreas Gohr        return false;
363967610bbSAndreas Gohr    }
364a34ed36bSDennis Ploeger}
365a34ed36bSDennis Ploeger
366