mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
5c9044a007
API Introduce HTMLFragment as casting helper for HTMLText with shortcodes disabled API Introduce DBField::CDATA for XML file value encoding API RSSFeed now casts from the underlying model rather than by override API Introduce CustomMethods::getExtraMethodConfig() to allow metadata to be queried BUG Remove _call hack from VirtualPage API Remove FormField::$dontEscape API Introduce HTMLReadonlyField for non-editable readonly HTML API FormField::Field() now returns string in many cases rather than DBField instance. API Remove redundant *_val methods from ViewableData API ViewableData::obj() no longer has a $forceReturnObject parameter as it always returns an object BUG Fix issue with ViewableData caching incorrect field values after being modified. API Remove deprecated DB class methods API Enforce plain text left/right formfield titles
164 lines
4.9 KiB
PHP
164 lines
4.9 KiB
PHP
<?php
|
|
|
|
use SilverStripe\ORM\DataObjectInterface;
|
|
use SilverStripe\ORM\DataObject;
|
|
use SilverStripe\ORM\SS_List;
|
|
use SilverStripe\ORM\FieldType\DBHTMLText;
|
|
|
|
/**
|
|
* @package framework
|
|
* @subpackage formatters
|
|
*/
|
|
class XMLDataFormatter extends DataFormatter {
|
|
|
|
/**
|
|
* @config
|
|
* @todo pass this from the API to the data formatter somehow
|
|
*/
|
|
private static $api_base = "api/v1/";
|
|
|
|
protected $outputContentType = 'text/xml';
|
|
|
|
public function supportedExtensions() {
|
|
return array(
|
|
'xml'
|
|
);
|
|
}
|
|
|
|
public function supportedMimeTypes() {
|
|
return array(
|
|
'text/xml',
|
|
'application/xml',
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Generate an XML representation of the given {@link DataObject}.
|
|
*
|
|
* @param DataObjectInterface|DataObject $obj
|
|
* @param array $fields
|
|
* @return string XML
|
|
*/
|
|
public function convertDataObject(DataObjectInterface $obj, $fields = null) {
|
|
$response = Controller::curr()->getResponse();
|
|
if($response) {
|
|
$response->addHeader("Content-Type", "text/xml");
|
|
}
|
|
|
|
return "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" . $this->convertDataObjectWithoutHeader($obj, $fields);
|
|
}
|
|
|
|
public function convertDataObjectWithoutHeader(DataObject $obj, $fields = null, $relations = null) {
|
|
$className = $obj->class;
|
|
$id = $obj->ID;
|
|
$objHref = Director::absoluteURL($this->config()->api_base . "$obj->class/$obj->ID");
|
|
|
|
$xml = "<$className href=\"$objHref.xml\">\n";
|
|
foreach($this->getFieldsForObj($obj) as $fieldName => $fieldType) {
|
|
// Field filtering
|
|
if($fields && !in_array($fieldName, $fields)) {
|
|
continue;
|
|
}
|
|
$fieldObject = $obj->obj($fieldName);
|
|
$fieldValue = $fieldObject->forTemplate();
|
|
if(!mb_check_encoding($fieldValue, 'utf-8')) {
|
|
$fieldValue = "(data is badly encoded)";
|
|
}
|
|
|
|
if(is_object($fieldValue) && is_subclass_of($fieldValue, 'Object') && $fieldValue->hasMethod('toXML')) {
|
|
$xml .= $fieldValue->toXML();
|
|
} else {
|
|
if($fieldObject instanceof DBHTMLText) {
|
|
// Escape HTML values using CDATA
|
|
$fieldValue = sprintf('<![CDATA[%s]]>', str_replace(']]>', ']]]]><![CDATA[>', $fieldValue));
|
|
}
|
|
$xml .= "<$fieldName>$fieldValue</$fieldName>\n";
|
|
}
|
|
}
|
|
|
|
if($this->relationDepth > 0) {
|
|
foreach($obj->hasOne() as $relName => $relClass) {
|
|
if(!singleton($relClass)->stat('api_access')) continue;
|
|
|
|
// Field filtering
|
|
if($fields && !in_array($relName, $fields)) continue;
|
|
if($this->customRelations && !in_array($relName, $this->customRelations)) continue;
|
|
|
|
$fieldName = $relName . 'ID';
|
|
if($obj->$fieldName) {
|
|
$href = Director::absoluteURL($this->config()->api_base . "$relClass/" . $obj->$fieldName);
|
|
} else {
|
|
$href = Director::absoluteURL($this->config()->api_base . "$className/$id/$relName");
|
|
}
|
|
$xml .= "<$relName linktype=\"has_one\" href=\"$href.xml\" id=\"" . $obj->$fieldName
|
|
. "\"></$relName>\n";
|
|
}
|
|
|
|
foreach($obj->hasMany() as $relName => $relClass) {
|
|
if(!singleton($relClass)->stat('api_access')) continue;
|
|
|
|
// Field filtering
|
|
if($fields && !in_array($relName, $fields)) continue;
|
|
if($this->customRelations && !in_array($relName, $this->customRelations)) continue;
|
|
|
|
$xml .= "<$relName linktype=\"has_many\" href=\"$objHref/$relName.xml\">\n";
|
|
$items = $obj->$relName();
|
|
if ($items) {
|
|
foreach($items as $item) {
|
|
$href = Director::absoluteURL($this->config()->api_base . "$relClass/$item->ID");
|
|
$xml .= "<$relClass href=\"$href.xml\" id=\"{$item->ID}\"></$relClass>\n";
|
|
}
|
|
}
|
|
$xml .= "</$relName>\n";
|
|
}
|
|
|
|
foreach($obj->manyMany() as $relName => $relClass) {
|
|
if(!singleton($relClass)->stat('api_access')) continue;
|
|
|
|
// Field filtering
|
|
if($fields && !in_array($relName, $fields)) continue;
|
|
if($this->customRelations && !in_array($relName, $this->customRelations)) continue;
|
|
|
|
$xml .= "<$relName linktype=\"many_many\" href=\"$objHref/$relName.xml\">\n";
|
|
$items = $obj->$relName();
|
|
if ($items) {
|
|
foreach($items as $item) {
|
|
$href = Director::absoluteURL($this->config()->api_base . "$relClass/$item->ID");
|
|
$xml .= "<$relClass href=\"$href.xml\" id=\"{$item->ID}\"></$relClass>\n";
|
|
}
|
|
}
|
|
$xml .= "</$relName>\n";
|
|
}
|
|
}
|
|
|
|
$xml .= "</$className>";
|
|
|
|
return $xml;
|
|
}
|
|
|
|
/**
|
|
* Generate an XML representation of the given {@link SS_List}.
|
|
*
|
|
* @param SS_List $set
|
|
* @param array $fields
|
|
* @return String XML
|
|
*/
|
|
public function convertDataObjectSet(SS_List $set, $fields = null) {
|
|
Controller::curr()->getResponse()->addHeader("Content-Type", "text/xml");
|
|
$className = get_class($set);
|
|
|
|
$xml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";
|
|
$xml .= (is_numeric($this->totalSize)) ? "<$className totalSize=\"{$this->totalSize}\">\n" : "<$className>\n";
|
|
foreach($set as $item) {
|
|
$xml .= $this->convertDataObjectWithoutHeader($item, $fields);
|
|
}
|
|
$xml .= "</$className>";
|
|
|
|
return $xml;
|
|
}
|
|
|
|
public function convertStringToArray($strData) {
|
|
return Convert::xml2array($strData);
|
|
}
|
|
}
|