Location: PHPKode > projects > DotClear > inc/clearbricks/rest/class.rest.php
<?php
# -- BEGIN LICENSE BLOCK ---------------------------------------
#
# This file is part of Clearbricks.
#
# Copyright (c) 2003-2010 Olivier Meunier & Association Dotclear
# Licensed under the GPL version 2.0 license.
# See LICENSE file or
# http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
#
# -- END LICENSE BLOCK -----------------------------------------

/**
* REST Server
*
* A very simple REST server implementation
*
* @package Clearbricks
* @subpackage Rest
*/
class restServer
{
	/** @var xmlTag Server response */		public $rsp;
	/** @var array Server's functions */	public $functions = array();
	
	/**
	* Constructor
	*/
	public function __construct()
	{
		$this->rsp = new xmlTag('rsp');
	}
	
	/**
	* Add Function
	*
	* This adds a new function to the server. <var>$callback</var> should be
	* a valid PHP callback. Callback function takes two arguments: GET and
	* POST values.
	*
	* @param string	$name		Function name
	* @param callback	$callback		Callback function
	*/
	public function addFunction($name, $callback)
	{
		if (is_callable($callback)) {
			$this->functions[$name] = $callback;
		}
	}
	
	/**
	* Call Function
	*
	* This method calls callback named <var>$name</var>.
	*
	* @param string	$name		Function name
	* @param array		$get			GET values
	* @param array		$post		POST values
	* @return mixed
	*/
	protected function callFunction($name,$get,$post)
	{
		if (isset($this->functions[$name])) {
			return call_user_func($this->functions[$name],$get,$post);
		}
	}
	
	/**
	* Main server
	*
	* This method creates the main server.
	*
	* @param string	$encoding		Server charset
	*/
	public function serve($encoding='UTF-8')
	{
		$get = array();
		if (isset($_GET)) {
			$get = $_GET;
		}
		
		$post = array();
		if (isset($_POST)) {
			$post = $_POST;
		}
		
		if (!isset($_REQUEST['f'])) {
			$this->rsp->status = 'failed';
			$this->rsp->message('No function given');
			$this->getXML($encoding);
			return false;
		}
		
		if (!isset($this->functions[$_REQUEST['f']])) {
			$this->rsp->status = 'failed';
			$this->rsp->message('Function does not exist');
			$this->getXML($encoding);
			return false;
		}
		
		try {
			$res = $this->callFunction($_REQUEST['f'],$get,$post);
		} catch (Exception $e) {
			$this->rsp->status = 'failed';
			$this->rsp->message($e->getMessage());
			$this->getXML($encoding);
			return false;
		}
		
		$this->rsp->status = 'ok';
		
		$this->rsp->insertNode($res);
		
		$this->getXML($encoding);
		return true;
	}
	
	private function getXML($encoding='UTF-8')
	{
		header('Content-Type: text/xml; charset='.$encoding);
		echo $this->rsp->toXML(1,$encoding);
	}
}

/**
* XML Tree
*
* @package Clearbricks
* @subpackage XML
*/
class xmlTag
{
	private $_name;
	private $_attr = array();
	private $_nodes = array();
	
	/**
	* Constructor
	*
	* Creates the root XML tag named <var>$name</var>. If content is given,
	* it will be appended to root tag with {@link insertNode()}
	*
	* @param string	$name		Tag name
	* @param mixed		$content		Tag content
	*/
	public function __construct($name=null, $content=null)
	{
		$this->_name = $name;
		
		if ($content !== null) {
			$this->insertNode($content);
		}
	}
	
	/**
	* Add Attribute
	*
	* Magic __set method to add an attribute.
	*
	* @param string	$name		Attribute name
	* @param string	$value		Attribute value
	* @see insertAttr()
	*/
	public function __set($name, $value)
	{
		$this->insertAttr($name, $value);
	}
	
	/**
	* Add a tag
	*
	* This magic __call method appends a tag to XML tree.
	*
	* @param string	$name		Tag name
	* @param array		$args		Function arguments, the first one would be tag content
	*/
	public function __call($name, $args)
	{
		if (!preg_match('#^[a-z_]#',$name)) {
			return false;
		}
		
		if (!isset($args[0])) {
			$args[0] = null;
		}
		
		$this->insertNode(new self($name,$args[0]));
	}
	
	/**
	* Add CDTA
	*
	* Appends CDATA to current tag.
	*
	* @param string	$value		Tag CDATA content
	*/
	public function CDATA($value)
	{
		$this->insertNode($value);
	}
	
	/**
	* Add Attribute
	*
	* This method adds an attribute to current tag.
	*
	* @param string	$name		Attribute name
	* @param string	$value		Attribute value
	* @see insertAttr()
	*/
	public function insertAttr($name, $value)
	{
		$this->_attr[$name] = $value;
	}
	
	/**
	* Insert Node
	*
	* This method adds a new XML node. Node could be a instance of xmlTag, an
	* array of valid values, a boolean or a string.
	*
	* @param xmlTag|array|boolean|string	$node	Node value
	*/
	public function insertNode($node=null)
	{
		if ($node instanceof self)
		{
			$this->_nodes[] = $node;
		}
		elseif (is_array($node))
		{
			$child = new self(null);
			foreach ($node as $tag => $n) {
				$child->insertNode(new self($tag,$n));
			}
			$this->_nodes[] = $child;
		}
		elseif (is_bool($node))
		{
			$this->_nodes[] = $node ? '1' : '0';
		}
		else
		{
			$this->_nodes[] = (string) $node;
		}
	}
	
	/**
	* XML Result
	*
	* Returns a string with XML content.
	*
	* @param boolean	$prolog		Append prolog to result
	* @param string	$encoding		Result charset
	* @return string
	*/
	public function toXML($prolog=false,$encoding='UTF-8')
	{
		if ($this->_name && count($this->_nodes) > 0) {
			$p = '<%1$s%2$s>%3$s</%1$s>';
		} elseif ($this->_name && count($this->_nodes) == 0) {
			$p = '<%1$s%2$s/>';
		} else {
			$p = '%3$s';
		}
		
		$res = $attr = $content = '';
		
		
		foreach ($this->_attr as $k => $v) {
			$attr .= ' '.$k.'="'.htmlspecialchars($v,ENT_QUOTES,$encoding).'"';
		}
		
		foreach ($this->_nodes as $node)
		{
			if ($node instanceof self) {
				$content .= $node->toXML();
			} else {
				$content .= htmlspecialchars($node,ENT_QUOTES,$encoding);
			}
		}
		
		$res = sprintf($p,$this->_name,$attr,$content);
		
		if ($prolog && $this->_name) {
			$res = '<?xml version="1.0" encoding="'.$encoding.'" ?>'."\n".$res;
		}
		
		return $res;
	}
}
?>
Return current item: DotClear