forked from GNUsocial/gnu-social
		
	
		
			
				
	
	
		
			281 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			281 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * StatusNet, the distributed open-source microblogging tool
 | |
|  *
 | |
|  * Low-level generator for XML
 | |
|  *
 | |
|  * PHP version 5
 | |
|  *
 | |
|  * LICENCE: This program is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Affero General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Affero General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Affero General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * @category  Output
 | |
|  * @package   StatusNet
 | |
|  * @author    Evan Prodromou <evan@status.net>
 | |
|  * @author    Sarven Capadisli <csarven@status.net>
 | |
|  * @copyright 2008 StatusNet, Inc.
 | |
|  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
 | |
|  * @link      http://status.net/
 | |
|  */
 | |
| 
 | |
| if (!defined('STATUSNET') && !defined('LACONICA')) {
 | |
|     exit(1);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Low-level generator for XML
 | |
|  *
 | |
|  * This is a thin wrapper around PHP's XMLWriter. The main
 | |
|  * advantage is the element() method, which simplifies outputting
 | |
|  * an element.
 | |
|  *
 | |
|  * @category Output
 | |
|  * @package  StatusNet
 | |
|  * @author   Evan Prodromou <evan@status.net>
 | |
|  * @author   Sarven Capadisli <csarven@status.net>
 | |
|  * @license  http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
 | |
|  * @link     http://status.net/
 | |
|  * @see      Action
 | |
|  * @see      HTMLOutputter
 | |
|  */
 | |
| class XMLOutputter
 | |
| {
 | |
|     /**
 | |
|      * Wrapped XMLWriter object, which does most of the heavy lifting
 | |
|      * for output.
 | |
|      */
 | |
| 
 | |
|     public $xw = null;
 | |
| 
 | |
|     /**
 | |
|      * Constructor
 | |
|      *
 | |
|      * Initializes the wrapped XMLWriter.
 | |
|      *
 | |
|      * @param string $output URL for outputting, if null it defaults to stdout ('php://output')
 | |
|      * @param boolean $indent Whether to indent output, default true
 | |
|      */
 | |
| 
 | |
|     public function __construct($output = null, $indent = null)
 | |
|     {
 | |
|         if (is_null($output)) {
 | |
|             $output = 'php://output';
 | |
|         }
 | |
|         $this->xw = new XMLWriter();
 | |
|         $this->xw->openURI($output);
 | |
|         if (is_null($indent)) {
 | |
|             $indent = common_config('site', 'indent');
 | |
|         }
 | |
|         $this->xw->setIndent($indent);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Start a new XML document
 | |
|      *
 | |
|      * @param string $doc document element
 | |
|      * @param string $public public identifier
 | |
|      * @param string $system system identifier
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function startXML($doc = null, $public = null, $system = null)
 | |
|     {
 | |
|         $this->xw->startDocument('1.0', 'UTF-8');
 | |
|         if ($doc) {
 | |
|             $this->xw->writeDTD($doc, $public, $system);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * finish an XML document
 | |
|      *
 | |
|      * It's probably a bad idea to continue to use this object
 | |
|      * after calling endXML().
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function endXML()
 | |
|     {
 | |
|         $this->xw->endDocument();
 | |
|         $this->xw->flush();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output an XML element
 | |
|      *
 | |
|      * Utility for outputting an XML element. A convenient wrapper
 | |
|      * for a bunch of longer XMLWriter calls. This is best for
 | |
|      * when an element doesn't have any sub-elements; if that's the
 | |
|      * case, use elementStart() and elementEnd() instead.
 | |
|      *
 | |
|      * The $content element will be escaped for XML. If you need
 | |
|      * raw output, use elementStart() and elementEnd() with a call
 | |
|      * to raw() in the middle.
 | |
|      *
 | |
|      * If $attrs is a string instead of an array, it will be treated
 | |
|      * as the class attribute of the element.
 | |
|      *
 | |
|      * @param string $tag Element type or tagname
 | |
|      * @param array|string|null $attrs Array of element attributes, as key-value pairs
 | |
|      * @param string|null $content string content of the element
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function element(string $tag, $attrs = null, $content = null)
 | |
|     {
 | |
|         $this->elementStart($tag, $attrs);
 | |
|         if (!is_null($content)) {
 | |
|             $this->xw->text(strval($content));
 | |
|         }
 | |
|         $this->elementEnd($tag);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output a start tag for an element
 | |
|      *
 | |
|      * Mostly used for when an element has content that's
 | |
|      * not a simple string.
 | |
|      *
 | |
|      * If $attrs is a string instead of an array, it will be treated
 | |
|      * as the class attribute of the element.
 | |
|      *
 | |
|      * @param string $tag Element type or tagname
 | |
|      * @param array|string|null $attrs Attributes
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function elementStart(string $tag, $attrs = null)
 | |
|     {
 | |
|         $this->xw->startElement($tag);
 | |
|         if (is_array($attrs)) {
 | |
|             foreach ($attrs as $name => $value) {
 | |
|                 $this->xw->writeAttribute($name, $value);
 | |
|             }
 | |
|         } elseif (is_string($attrs)) {
 | |
|             $this->xw->writeAttribute('class', $attrs);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output an end tag for an element
 | |
|      *
 | |
|      * Used in conjunction with elementStart(). $tag param
 | |
|      * should match the elementStart() param.
 | |
|      *
 | |
|      * For HTML 4 compatibility, this method will force
 | |
|      * a full end element (</tag>) even if the element is
 | |
|      * empty, except for a handful of exception tagnames.
 | |
|      * This is a hack.
 | |
|      *
 | |
|      * @param string $tag Element type or tagname.
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function elementEnd(string $tag)
 | |
|     {
 | |
|         static $empty_tag = ['base', 'meta', 'link', 'hr',
 | |
|                              'br', 'param', 'img', 'area',
 | |
|                              'input', 'col', 'source'];
 | |
|         // XXX: check namespace
 | |
|         if (in_array($tag, $empty_tag)) {
 | |
|             $this->xw->endElement();
 | |
|         } else {
 | |
|             $this->xw->fullEndElement();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function elementNS(array $ns, $tag, $attrs = null, $content = null)
 | |
|     {
 | |
|         $this->elementStartNS($ns, $tag, $attrs);
 | |
|         if (!is_null($content)) {
 | |
|             $this->xw->text($content);
 | |
|         }
 | |
|         $this->elementEnd($tag);
 | |
|     }
 | |
| 
 | |
|     public function elementStartNS(array $ns, $tag, $attrs = null)
 | |
|     {
 | |
|         reset($ns); // array pointer to 0
 | |
|         $uri = key($ns);
 | |
|         $this->xw->startElementNS($ns[$uri], $tag, $uri);
 | |
|         if (is_array($attrs)) {
 | |
|             foreach ($attrs as $name => $value) {
 | |
|                 $this->xw->writeAttribute($name, $value);
 | |
|             }
 | |
|         } elseif (is_string($attrs)) {
 | |
|             $this->xw->writeAttribute('class', $attrs);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output plain text
 | |
|      *
 | |
|      * Text will be escaped. If you need it not to be,
 | |
|      * use raw() instead.
 | |
|      *
 | |
|      * @param string $txt Text to output.
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function text($txt)
 | |
|     {
 | |
|         $this->xw->text($txt);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output raw xml
 | |
|      *
 | |
|      * This will spit out its argument verbatim -- no escaping is
 | |
|      * done.
 | |
|      *
 | |
|      * @param string $xml XML to output.
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function raw($xml)
 | |
|     {
 | |
|         $this->xw->writeRaw($xml);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * output a comment
 | |
|      *
 | |
|      * @param string $txt text of the comment
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function comment($txt)
 | |
|     {
 | |
|         $this->xw->writeComment($txt);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Flush output buffers
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
| 
 | |
|     public function flush()
 | |
|     {
 | |
|         $this->xw->flush();
 | |
|     }
 | |
| }
 |