mirror of
https://github.com/OpenXE-org/OpenXE.git
synced 2025-01-25 04:01:14 +01:00
299 lines
9.0 KiB
PHP
299 lines
9.0 KiB
PHP
|
<?php
|
||
|
|
||
|
namespace Sabre\Xml;
|
||
|
|
||
|
/**
|
||
|
* XML parsing and writing service.
|
||
|
*
|
||
|
* You are encouraged to make a instance of this for your application and
|
||
|
* potentially extend it, as a central API point for dealing with xml and
|
||
|
* configuring the reader and writer.
|
||
|
*
|
||
|
* @copyright Copyright (C) 2009-2015 fruux GmbH (https://fruux.com/).
|
||
|
* @author Evert Pot (http://evertpot.com/)
|
||
|
* @license http://sabre.io/license/ Modified BSD License
|
||
|
*/
|
||
|
class Service {
|
||
|
|
||
|
/**
|
||
|
* This is the element map. It contains a list of XML elements (in clark
|
||
|
* notation) as keys and PHP class names as values.
|
||
|
*
|
||
|
* The PHP class names must implement Sabre\Xml\Element.
|
||
|
*
|
||
|
* Values may also be a callable. In that case the function will be called
|
||
|
* directly.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
public $elementMap = [];
|
||
|
|
||
|
/**
|
||
|
* This is a list of namespaces that you want to give default prefixes.
|
||
|
*
|
||
|
* You must make sure you create this entire list before starting to write.
|
||
|
* They should be registered on the root element.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
public $namespaceMap = [];
|
||
|
|
||
|
/**
|
||
|
* This is a list of custom serializers for specific classes.
|
||
|
*
|
||
|
* The writer may use this if you attempt to serialize an object with a
|
||
|
* class that does not implement XmlSerializable.
|
||
|
*
|
||
|
* Instead it will look at this classmap to see if there is a custom
|
||
|
* serializer here. This is useful if you don't want your value objects
|
||
|
* to be responsible for serializing themselves.
|
||
|
*
|
||
|
* The keys in this classmap need to be fully qualified PHP class names,
|
||
|
* the values must be callbacks. The callbacks take two arguments. The
|
||
|
* writer class, and the value that must be written.
|
||
|
*
|
||
|
* function (Writer $writer, object $value)
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
public $classMap = [];
|
||
|
|
||
|
/**
|
||
|
* Returns a fresh XML Reader
|
||
|
*
|
||
|
* @return Reader
|
||
|
*/
|
||
|
function getReader() {
|
||
|
|
||
|
$r = new Reader();
|
||
|
$r->elementMap = $this->elementMap;
|
||
|
return $r;
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a fresh xml writer
|
||
|
*
|
||
|
* @return Writer
|
||
|
*/
|
||
|
function getWriter() {
|
||
|
|
||
|
$w = new Writer();
|
||
|
$w->namespaceMap = $this->namespaceMap;
|
||
|
$w->classMap = $this->classMap;
|
||
|
return $w;
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses a document in full.
|
||
|
*
|
||
|
* Input may be specified as a string or readable stream resource.
|
||
|
* The returned value is the value of the root document.
|
||
|
*
|
||
|
* Specifying the $contextUri allows the parser to figure out what the URI
|
||
|
* of the document was. This allows relative URIs within the document to be
|
||
|
* expanded easily.
|
||
|
*
|
||
|
* The $rootElementName is specified by reference and will be populated
|
||
|
* with the root element name of the document.
|
||
|
*
|
||
|
* @param string|resource $input
|
||
|
* @param string|null $contextUri
|
||
|
* @param string|null $rootElementName
|
||
|
* @throws ParseException
|
||
|
* @return array|object|string
|
||
|
*/
|
||
|
function parse($input, $contextUri = null, &$rootElementName = null) {
|
||
|
|
||
|
if (is_resource($input)) {
|
||
|
// Unfortunately the XMLReader doesn't support streams. When it
|
||
|
// does, we can optimize this.
|
||
|
$input = stream_get_contents($input);
|
||
|
}
|
||
|
$r = $this->getReader();
|
||
|
$r->contextUri = $contextUri;
|
||
|
$r->xml($input);
|
||
|
|
||
|
$result = $r->parse();
|
||
|
$rootElementName = $result['name'];
|
||
|
return $result['value'];
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses a document in full, and specify what the expected root element
|
||
|
* name is.
|
||
|
*
|
||
|
* This function works similar to parse, but the difference is that the
|
||
|
* user can specify what the expected name of the root element should be,
|
||
|
* in clark notation.
|
||
|
*
|
||
|
* This is useful in cases where you expected a specific document to be
|
||
|
* passed, and reduces the amount of if statements.
|
||
|
*
|
||
|
* It's also possible to pass an array of expected rootElements if your
|
||
|
* code may expect more than one document type.
|
||
|
*
|
||
|
* @param string|string[] $rootElementName
|
||
|
* @param string|resource $input
|
||
|
* @param string|null $contextUri
|
||
|
* @throws ParseException
|
||
|
* @return array|object|string
|
||
|
*/
|
||
|
function expect($rootElementName, $input, $contextUri = null) {
|
||
|
|
||
|
if (is_resource($input)) {
|
||
|
// Unfortunately the XMLReader doesn't support streams. When it
|
||
|
// does, we can optimize this.
|
||
|
$input = stream_get_contents($input);
|
||
|
}
|
||
|
$r = $this->getReader();
|
||
|
$r->contextUri = $contextUri;
|
||
|
$r->xml($input);
|
||
|
|
||
|
$rootElementName = (array)$rootElementName;
|
||
|
|
||
|
foreach ($rootElementName as &$rEl) {
|
||
|
if ($rEl[0] !== '{') $rEl = '{}' . $rEl;
|
||
|
}
|
||
|
|
||
|
$result = $r->parse();
|
||
|
if (!in_array($result['name'], $rootElementName, true)) {
|
||
|
throw new ParseException('Expected ' . implode(' or ', (array)$rootElementName) . ' but received ' . $result['name'] . ' as the root element');
|
||
|
}
|
||
|
return $result['value'];
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Generates an XML document in one go.
|
||
|
*
|
||
|
* The $rootElement must be specified in clark notation.
|
||
|
* The value must be a string, an array or an object implementing
|
||
|
* XmlSerializable. Basically, anything that's supported by the Writer
|
||
|
* object.
|
||
|
*
|
||
|
* $contextUri can be used to specify a sort of 'root' of the PHP application,
|
||
|
* in case the xml document is used as a http response.
|
||
|
*
|
||
|
* This allows an implementor to easily create URI's relative to the root
|
||
|
* of the domain.
|
||
|
*
|
||
|
* @param string $rootElementName
|
||
|
* @param string|array|XmlSerializable $value
|
||
|
* @param string|null $contextUri
|
||
|
*/
|
||
|
function write($rootElementName, $value, $contextUri = null) {
|
||
|
|
||
|
$w = $this->getWriter();
|
||
|
$w->openMemory();
|
||
|
$w->contextUri = $contextUri;
|
||
|
$w->setIndent(true);
|
||
|
$w->startDocument();
|
||
|
$w->writeElement($rootElementName, $value);
|
||
|
return $w->outputMemory();
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Map an xml element to a PHP class.
|
||
|
*
|
||
|
* Calling this function will automatically setup the Reader and Writer
|
||
|
* classes to turn a specific XML element to a PHP class.
|
||
|
*
|
||
|
* For example, given a class such as :
|
||
|
*
|
||
|
* class Author {
|
||
|
* public $firstName;
|
||
|
* public $lastName;
|
||
|
* }
|
||
|
*
|
||
|
* and an XML element such as:
|
||
|
*
|
||
|
* <author xmlns="http://example.org/ns">
|
||
|
* <firstName>...</firstName>
|
||
|
* <lastName>...</lastName>
|
||
|
* </author>
|
||
|
*
|
||
|
* These can easily be mapped by calling:
|
||
|
*
|
||
|
* $service->mapValueObject('{http://example.org}author', 'Author');
|
||
|
*
|
||
|
* @param string $elementName
|
||
|
* @param object $className
|
||
|
* @return void
|
||
|
*/
|
||
|
function mapValueObject($elementName, $className) {
|
||
|
list($namespace) = self::parseClarkNotation($elementName);
|
||
|
|
||
|
$this->elementMap[$elementName] = function(Reader $reader) use ($className, $namespace) {
|
||
|
return \Sabre\Xml\Deserializer\valueObject($reader, $className, $namespace);
|
||
|
};
|
||
|
$this->classMap[$className] = function(Writer $writer, $valueObject) use ($namespace) {
|
||
|
return \Sabre\Xml\Serializer\valueObject($writer, $valueObject, $namespace);
|
||
|
};
|
||
|
$this->valueObjectMap[$className] = $elementName;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Writes a value object.
|
||
|
*
|
||
|
* This function largely behaves similar to write(), except that it's
|
||
|
* intended specifically to serialize a Value Object into an XML document.
|
||
|
*
|
||
|
* The ValueObject must have been previously registered using
|
||
|
* mapValueObject().
|
||
|
*
|
||
|
* @param object $object
|
||
|
* @param string $contextUri
|
||
|
* @return void
|
||
|
*/
|
||
|
function writeValueObject($object, $contextUri = null) {
|
||
|
|
||
|
if (!isset($this->valueObjectMap[get_class($object)])) {
|
||
|
throw new \InvalidArgumentException('"' . get_class($object) . '" is not a registered value object class. Register your class with mapValueObject.');
|
||
|
}
|
||
|
return $this->write(
|
||
|
$this->valueObjectMap[get_class($object)],
|
||
|
$object,
|
||
|
$contextUri
|
||
|
);
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses a clark-notation string, and returns the namespace and element
|
||
|
* name components.
|
||
|
*
|
||
|
* If the string was invalid, it will throw an InvalidArgumentException.
|
||
|
*
|
||
|
* @param string $str
|
||
|
* @throws InvalidArgumentException
|
||
|
* @return array
|
||
|
*/
|
||
|
static function parseClarkNotation($str) {
|
||
|
static $cache = [];
|
||
|
|
||
|
if (!isset($cache[$str])) {
|
||
|
|
||
|
if (!preg_match('/^{([^}]*)}(.*)$/', $str, $matches)) {
|
||
|
throw new \InvalidArgumentException('\'' . $str . '\' is not a valid clark-notation formatted string');
|
||
|
}
|
||
|
|
||
|
$cache[$str] = [
|
||
|
$matches[1],
|
||
|
$matches[2]
|
||
|
];
|
||
|
}
|
||
|
|
||
|
return $cache[$str];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* A list of classes and which XML elements they map to.
|
||
|
*/
|
||
|
protected $valueObjectMap = [];
|
||
|
|
||
|
}
|