mediawiki-extensions-Templa.../TemplateDataBlob.php
Timo Tijhof b7080c5f90 Initial TemplateData commit
Registers a parser tag <templatedata> that should have a JSON
blob as content. The blob is then validated and normalised when
MediaWiki parses the page (e.g. during save and preview).

If there are validation errors, the save is aborted from the
extension hook and an error is displayed.

If all goes well, the normalised blob is stored in the database
(which can be retrieved through the API). And an HTML
representation of the template parameters is returned to the
wikitext parser to show where the <templatedata> was in the page.

The blob format is specified in spec.templatedata.json and
is validated in TemplateDataBlob::parse.

Bug: 44444
Change-Id: Icf305892a9512545a63f5a5280cc0d340c61585f
2013-03-14 19:19:55 +00:00

215 lines
5.9 KiB
PHP

<?php
/**
* @file
* @ingroup Extensions
*/
/**
* Represents the information about a template,
* coming from the JSON blob in the <templatedata> tags
* on wiki pages.
*
* @class
*/
class TemplateDataBlob {
/**
* @var stdClass
*/
private $data;
/**
* @var Status: Cache of TemplateInfo::validate
*/
private $status;
/**
* @param string $json
* @return TemplateInfo
*/
public static function newFromJSON( $json ) {
$ti = new self( json_decode( $json ) );
$status = $ti->parse();
// TODO: Normalise `params.*.description` to a plain object.
if ( !$status->isOK() ) {
// Don't save invalid data, clear it.
$ti->data = new stdClass();
}
$ti->status = $status;
return $ti;
}
/**
* Parse the data, normalise it and validate it.
*
* See spec.templatedata.json for the expected format of the JSON object.
* @return Status
*/
private function parse() {
$data = $this->data;
if ( $data === null ) {
return Status::newFatal( 'templatedata-invalid-parse' );
}
if ( !is_object( $data ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'templatedata', 'object' );
}
foreach ( $data as $key => $value ) {
if ( !in_array( $key, array( 'params', 'description' ) ) ) {
return Status::newFatal( 'templatedata-invalid-unknown', $key );
}
}
if ( !isset( $data->params ) ) {
return Status::newFatal( 'templatedata-invalid-missing', 'params', 'object' );
}
if ( !is_object( $data->params ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params', 'object' );
}
if ( isset( $data->description ) ) {
if ( !is_object( $data->params ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params', 'object' );
}
} else {
$data->description = '';
}
foreach ( $data->params as $paramName => $paramObj ) {
if ( !is_object( $paramObj ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName, 'object' );
}
foreach ( $paramObj as $key => $value ) {
if ( !in_array( $key, array(
'required',
'description',
'deprecated',
'aliases',
'clones',
'default',
) ) ) {
return Status::newFatal( 'templatedata-invalid-unknown', $key );
}
}
if ( isset( $paramObj->required ) ) {
if ( !is_bool( $paramObj->required ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.required', 'boolean' );
}
} else {
$paramObj->required = false;
}
if ( isset( $paramObj->description ) ) {
if ( !is_object( $paramObj->description ) && !is_string( $paramObj->description ) ) {
// TODO: Also validate that if it is an object, the keys are valid lang codes
// and the values strings.
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.description', 'string|object' );
}
} else {
$paramObj->description = '';
}
if ( isset( $paramObj->deprecated ) ) {
if ( $paramObj->deprecated === false || is_string( $paramObj->deprecated ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.deprecated', 'boolean|string' );
}
} else {
$paramObj->deprecated = false;
}
if ( isset( $paramObj->aliases ) ) {
if ( !is_array( $paramObj->aliases ) ) {
// TODO: Validate the array values.
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.aliases', 'array' );
}
} else {
$paramObj->aliases = array();
}
if ( isset( $paramObj->clones ) ) {
if ( !is_array( $paramObj->clones ) ) {
// TODO: Validate the array values.
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.clones', 'array' );
}
} else {
$paramObj->clones = array();
}
if ( isset( $paramObj->default ) ) {
if ( !is_string( $paramObj->default ) ) {
return Status::newFatal( 'templatedata-invalid-type', 'params.' . $paramName . '.default', 'string' );
}
} else {
$paramObj->default = '';
}
}
return Status::newGood();
}
public function getStatus() {
return $this->status;
}
public function getJSON() {
return json_encode( $this->data );
}
public function getHtml( IContextSource $context ) {
$data = $this->data;
$html =
Html::openElement( 'div', array( 'class' => 'mw-templatedata-doc-wrap' ) )
. Html::element( 'p', array( 'class' => 'mw-templatedata-doc-desc' ), $data->description )
. '<table class="wikitable sortable mw-templatedata-doc-params">'
. Html::element( 'caption', array(), $context->msg( 'templatedata-doc-params' ) )
. '<thead><tr>'
. Html::element( 'th', array(), $context->msg( 'templatedata-doc-param-name' ) )
. Html::element( 'th', array(), $context->msg( 'templatedata-doc-param-desc' ) )
. Html::element( 'th', array(), $context->msg( 'templatedata-doc-param-default' ) )
. Html::element( 'th', array(), $context->msg( 'templatedata-doc-param-status' ) )
. '</tr></thead>'
. '<tbody>'
;
foreach ( $data->params as $paramName => $paramObj ) {
$description = '';
$default = '';
$html .= '<tr>'
. Html::element( 'th', array(), $paramName )
// Description
. Html::rawElement( 'td', array(
'class' => array(
'mw-templatedata-doc-param-empty' => $paramObj->description === '' && $paramObj->deprecated === false
)
), $paramObj->description !== '' ? $paramObj->description : 'no description' )
// Default
. Html::element( 'td', array(
'class' => array(
'mw-templatedata-doc-param-empty' => $paramObj->default === ''
)
), $paramObj->default !== '' ? $paramObj->default : 'empty' )
// Status
. Html::element( 'td', array(),
$paramObj->deprecated ? 'deprecated' : (
$paramObj->required ? 'required' : 'optional'
)
)
. '</tr>';
}
$html .= '</tbody></table>'
. Html::closeElement( 'div' );
return $html;
}
private function __construct( stdClass $data = null ) {
$this->data = $data;
}
}