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 ) . '' . Html::element( 'caption', array(), $context->msg( 'templatedata-doc-params' ) ) . '' . 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' ) ) . '' . '' ; foreach ( $data->params as $paramName => $paramObj ) { $description = ''; $default = ''; $html .= '' . 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' ) ) . ''; } $html .= '
' . Html::closeElement( 'div' ); return $html; } private function __construct( stdClass $data = null ) { $this->data = $data; } }