sql2xml.php
来自「视频监控网络部分的协议ddns,的模块的实现代码,请大家大胆指正.」· PHP 代码 · 共 929 行 · 第 1/2 页
PHP
929 行
<?php// +----------------------------------------------------------------------+// | PHP Version 4 |// +----------------------------------------------------------------------+// | Copyright (c) 1997-2003 The PHP Group |// +----------------------------------------------------------------------+// | This source file is subject to version 2.0 of the PHP license, |// | that is bundled with this package in the file LICENSE, and is |// | available at through the world-wide-web at |// | http://www.php.net/license/2_02.txt. |// | If you did not receive a copy of the PHP license and are unable to |// | obtain it through the world-wide-web, please send a note to |// | license@php.net so we can mail you a copy immediately. |// +----------------------------------------------------------------------+// | Authors: Christian Stocker <chregu@phant.ch> |// +----------------------------------------------------------------------+//// $Id: sql2xml.php,v 1.13 2004/04/01 07:17:12 chregu Exp $/*** This class takes a PEAR::DB-Result Object, a sql-query-string or an array* and returns a xml-representation of it.** TODO* -encoding etc, options for header* -ERROR CHECKING** Usage example** include_once ("DB.php");* include_once("XML/sql2xml.php");* $db = DB::connect("mysql://root@localhost/xmltest");* $sql2xml = new xml_sql2xml();* //the next one is only needed, if you need others than the default* $sql2xml->setEncoding("ISO-8859-1","UTF-8");* $result = $db->query("select * from bands");* $xmlstring = $sql2xml->getXML($result);** or** include_once ("DB.php");* include_once("XML/sql2xml.php");* $sql2xml = new xml_sql2xml("mysql://root@localhost/xmltest");* $sql2xml->Add("select * from bands");* $xmlstring = $sql2xml->getXML();** More documentation and a tutorial/how-to can be found at* http://php.chregu.tv/sql2xml** @author Christian Stocker <chregu@bitflux.ch>* @version $Id: sql2xml.php,v 1.13 2004/04/01 07:17:12 chregu Exp $* @package XML*/class XML_sql2xml { /** * If joined-tables should be output nested. * Means, if you have joined two or more queries, the later * specified tables will be nested within the result of the former * table. * Works at the moment only with mysql automagically. For other RDBMS * you have to provide your table-relations by hand (see user_tableinfo) * * @var boolean * @see $user_tableinfo, doSql2Xml(), doArray2Xml(); */ var $nested = True; /** * Name of the tag element for resultsets * * @var string * @see insertNewResult() */ var $tagNameResult = "result"; /** * Name of the tag element for rows * * @var string * @see insertNewRow() */ var $tagNameRow = "row"; /** * * @var object PEAR::DB * @access private */ var $db = Null; /** * Options to be used in extended Classes (for example in sql2xml_ext). * They are passed with SetOptions as an array (arrary("user_options" = array()); * and can then be accessed with $this->user_options["bla"] from your * extended classes for additional features. * This array is not use in this base class, it's only for passing easy parameters * to extended classes. * * @var array */ var $user_options = array(); /** * The DomDocument Object to be used in the whole class * * @var object DomDocument * @access private */ var $xmldoc; /** * The Root of the domxml object * I'm not sure, if we need this as a class variable.... * could be replaced by domxml_root($this->xmldoc); * * @var object DomNode * @access private */ var $xmlroot; /** * This array is used to give the structure of your database to the class. * It's especially useful, if you don't use mysql, since other RDBMS than * mysql are not able at the moment to provide the right information about * your database structure within the query. And if you have more than 2 * tables joined in the sql it's also not possible for mysql to find out * your real relations. * The parameters are the same as in fieldInfo from the PEAR::DB and some * additional ones. Here they come: * From PEAR::DB->fieldinfo: * * $tableInfo[$i]["table"] : the table, which field #$i belongs to. * for some rdbms/comples queries and with arrays, it's impossible * to find out to which table the field actually belongs. You can * specify it here more accurate. Or if you want, that one fields * belongs to another table, than it actually says (yes, there's * use for that, see the upcoming tutorial ...) * * $tableInfo[$i]["name"] : the name of field #$i. if you want another * name for the tag, than the query or your array provides, assign * it here. * * Additional info * $tableInfo["parent_key"][$table] : index of the parent key for $table. * this is the field, where the programm looks for changes, if this * field changes, it assumes, that we need a new "rowset" in the * parent table. * * $tableInfo["parent_table"][$table]: name of the parent table for $table. * * @var array * @access private */ var $user_tableInfo = array(); /** * the encoding type, the input from the db has */ var $encoding_from = "ISO-8859-1"; /** * the encoding type, the output in the xml should have * (note that domxml at the moment only support UTF-8, or at least it looks like) */ var $encoding_to = "UTF-8"; var $tagname = "tagname"; /** * Constructor * The Constructor can take a Pear::DB "data source name" (eg. * "mysql://user:passwd@localhost/dbname") and will then connect * to the DB, or a PEAR::DB object link, if you already connected * the db before. " If you provide nothing as $dsn, you only can later add stuff with * a pear::db-resultset or as an array. providing sql-strings will * not work. * the $root param is used, if you want to provide another name for your * root-tag than "root". if you give an empty string (""), there will be no * root element created here, but only when you add a resultset/array/sql-string. * And the first tag of this result is used as the root tag. * * @param mixed $dsn PEAR::DB "data source name" or object DB object * @param string $root the name of the xml-doc root element. * @access public */ function XML_sql2xml ($dsn = Null, $root = "root") { // if it's a string, then it must be a dsn-identifier; if (is_string($dsn)) { include_once ("DB.php"); $this->db = DB::Connect($dsn); if (DB::isError($this->db)) { print "The given dsn for XML_sql2xml was not valid in file ".__FILE__." at line ".__LINE__."<br>\n"; return new DB_Error($this->db->code,PEAR_ERROR_DIE); } } elseif (is_object($dsn) && DB::isError($dsn)) { print "The given param for XML_sql2xml was not valid in file ".__FILE__." at line ".__LINE__."<br>\n"; return new DB_Error($dsn->code,PEAR_ERROR_DIE); } // if parent class is db_common, then it's already a connected identifier elseif (get_parent_class($dsn) == "db_common") { $this->db = $dsn; } $this->xmldoc = domxml_new_xmldoc('1.0'); //oehm, seems not to work, unfortunately.... does anybody know a solution? $this->xmldoc->encoding = $this->encoding_to; if ($root) { $this->xmlroot = $this->xmldoc->add_root($root); //PHP 4.0.6 had $root->name as tagname, check for that here... if (!isset($this->xmlroot->{$this->tagname})) { $this->tagname = "name"; } } } /** * General method for adding new resultsets to the xml-object * Give a sql-query-string, a pear::db_result object or an array as * input parameter, and the method calls the appropriate method for this * input and adds this to $this->xmldoc * * @param string sql-string, or object db_result, or array * @param mixed additional parameters for the following functions * @access public * @see addResult(), addSql(), addArray(), addXmlFile() */ function add ($resultset, $params = Null) { // if string, then it's a query, a xml-file or a xml-string... if (is_string($resultset)) { if (preg_match("/\.xml$/",$resultset)) { $this->AddXmlFile($resultset,$params); } elseif (preg_match("/.*select.*from.*/i" , $resultset)) { $this->AddSql($resultset); } else { $this->AddXmlString($resultset); } } // if array, then it's an array... elseif (is_array($resultset)) { $this->AddArray($resultset); } if (get_class($resultset) == "db_result") { $this->AddResult($resultset); } } /** * Adds the content of a xml-file to $this->xmldoc, on the same level * as a normal resultset (mostly just below <root>) * * @param string filename * @param mixed xpath either a string with the xpath expression or an array with "xpath"=>xpath expression and "root"=tag/subtag/etc, which are the tags to be inserted before the result * @access public * @see doXmlString2Xml() */ function addXmlFile($file,$xpath = Null) { $fd = fopen( $file, "r" ); $content = fread( $fd, filesize( $file ) ); fclose( $fd ); $this->doXmlString2Xml($content,$xpath); } /** * Adds the content of a xml-string to $this->xmldoc, on the same level * as a normal resultset (mostly just below <root>) * * @param string xml * @param mixed xpath either a string with the xpath expression or an array with "xpath"=>xpath expression and "root"=tag/subtag/etc, which are the tags to be inserted before the result * @access public * @see doXmlString2Xml() */ function addXmlString($string,$xpath = Null) { $this->doXmlString2Xml($string,$xpath); } /** * Adds an additional pear::db_result resultset to $this->xmldoc * * @param Object db_result result from a DB-query * @see doSql2Xml() * @access public */ function addResult($result) { $this->doSql2Xml($result); } /** * Adds an aditional resultset generated from an sql-statement * to $this->xmldoc * * @param string sql a string containing an sql-statement. * @access public * @see doSql2Xml() */ function addSql($sql) { /* if there are {} expressions in the sql query, we assume it's an xpath expression to * be evaluated. */ if (preg_match_all ("/\{([^\}]+)\}/i",$sql,$matches)) { foreach ($matches[1] as $match) { $sql = preg_replace("#\{".preg_quote($match)."\}# ", $this->getXpathValue($match),$sql); } } $result = $this->db->query($sql); //very strange if (PEAR::isError($result->result)) { print "You have an SQL-Error:<br>".$result->result->userinfo; print "<br>"; new DB_Error($result->result->code,PEAR_ERROR_DIE); } $this->doSql2Xml($result); } /** * Adds an aditional resultset generated from an Array * to $this->xmldoc * TODO: more explanation, how arrays are transferred * * @param array multidimensional array. * @access public * @see doArray2Xml() */ function addArray ($array) { $parent_row = $this->insertNewResult($metadata); $this->DoArray2Xml($array,$parent_row); } /** * Returns an xml-string with a xml-representation of the resultsets. * * The resultset can be directly provided here, or if you need more than one * in your xml, then you have to provide each of them with add() before you * call getXML, but the last one can also be provided here. * * @param mixed $result result Object from a DB-query * @return string xml * @access public */ function getXML($result = Null) { $xmldoc = $this->getXMLObject($result); return $xmldoc->dumpmem(); } /** * Returns an xml DomDocument Object with a xml-representation of the resultsets. * * The resultset can be directly provided here, or if you need more than one * in your xml, then you have to provide each of them with add() before you * call getXMLObject, but the last one can also be provided here. * * @param mixed $result result Object from a DB-query * @return Object DomDocument * @access public */ function getXMLObject($result = Null) { if ($result) { $this->add ($result); } return $this->xmldoc; } /** * For adding db_result-"trees" to $this->xmldoc * @param Object db_result * @access private * @see addResult(),addSql() */ function doSql2Xml($result) { if (DB::IsError($result)) { print "Error in file ".__FILE__." at line ".__LINE__."<br>\n"; print $result->userinfo."<br>\n"; new DB_Error($result->code,PEAR_ERROR_DIE); } // the method_exists is here, cause tableInfo is only in the cvs at the moment // BE CAREFUL: if you have fields with the same name in different tables, you will get errors // later, since DB_FETCHMODE_ASSOC doesn't differentiate that stuff. $this->LastResult = &$result; if (!method_exists($result,"tableInfo") || ! ($tableInfo = $result->tableInfo(False))) { //emulate tableInfo. this can go away, if every db supports tableInfo $fetchmode = DB_FETCHMODE_ASSOC; $res = $result->FetchRow($fetchmode); $this->nested = False; $i = 0; while (list($key, $val) = each($res)) { $tableInfo[$i]["table"]= $this->tagNameResult; $tableInfo[$i]["name"] = $key; $resFirstRow[$i] = $val; $i++; } $res = $resFirstRow; $FirstFetchDone = True; $fetchmode = DB_FETCHMODE_ORDERED; } else { $FirstFetchDone = False; $fetchmode = DB_FETCHMODE_ORDERED; } // initialize db hierarchy... $parenttable = "root"; $tableInfo["parent_key"]["root"] = 0; foreach ($tableInfo as $key => $value) { if (is_int($key)) { // if the sql-query had a function the table starts with a # (only in mysql i think....), then give the field the name of the table before... if (preg_match ("/^#/",$value["table"]) || strlen($value["table"]) == 0) { $value["table"] = $tableInfo[($key - 1)]["table"] ; $tableInfo[$key]["table"] = $value["table"]; } if (!isset($tableInfo["parent_table"]) || !isset($tableInfo["parent_table"][$value["table"]]) || is_null($tableInfo["parent_table"][$value["table"]])) { $tableInfo["parent_key"][$value["table"]] = $key; $tableInfo["parent_table"][$value["table"]] = $parenttable; $parenttable = $value["table"] ; }
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?