API: Estratégia de Implementação
| Esta página faz parte da documentação da API de Ação do MediaWiki. |
Isto explica a implementação da máquina de API do MediaWiki no núcleo. Se quiser fornecer uma API no seu código para os clientes consumirem, leia API: Extensões.
Estrutura de Ficheiro/Módulo
api.phpé o ponto de entrada, localizado na raiz da wiki. Consulte API: Página principal#O ponto final.ApiEntryPointis the implementation of the entry point (factored out in Gerrit change 959047).includes/apiwill contain all files related to the API, but none of them will be allowed as entry points.- All API classes are derived from a common abstract class
ApiBase. The base class provides common functionality such as parameter parsing, profiling, and error handling. ApiMainis the main class instantiated byApiEntryPoint. It determines which module to execute based on theaction=XXXparameter.ApiMainalso creates an instance of theApiResultclass, which contains the output data array and related helper functions. Lastly,ApiMaininstantiates the formatting class that will output the data fromApiResultin XML/JSON/PHP or other format to the client.- Any module derived from
ApiBasewill receive a reference to an instance of theApiMainduring instantiation, so that during execution the module may get shared resources such as the result object.
Módulos de consulta
ApiQuerybehaves similar toApiMainin that it executes submodules. Each submodule derives fromApiQueryBase(exceptApiQueryitself, which is a top-level module). During instantiation, submodules receive a reference to the ApiQuery instance.- All extension query modules should use a 3 or more letter prefixes. The core modules use 2 letter prefixes.
ApiQueryexecution plan:- Get shared query parameters
list/prop/metato determine needed submodules. - Create an
ApiPageSetobject and populate it from thetitles/pageids/revidsparameters. Thepagesetobject contains the list of pages or revisions that query modules will work with. - If requested, a generator module is executed to create another
ApiPageSet. Similar to the piping streams in UNIX. Given pages are the input to generator that produces another set of pages for all other modules to work on.
- Get shared query parameters
- Requirements for query continuation:
- The SQL query must be totally ordered. In other words, the query must be using all columns of some unique key either as constants in the
WHEREclause or in theORDER BYclauses. - The SQL query must not filesort.
- The value given to
setContinueEnumParameter()must include all the columns in theORDER BYclause. - When continuing, a single compound condition should be added to the
WHEREclause. If the query hasORDER BY column_0, column_1, column_2, this condition should look something like this:
- The SQL query must be totally ordered. In other words, the query must be using all columns of some unique key either as constants in the
(column_0 > value_0 OR (column_0 = value_0 AND (column_1 > value_1 OR (column_1 = value_1 AND (column_2 >= value_2) )) ))
Of course, swap ">" for "<" if your ORDER BY columns are using DESC.
Be sure to avoid SQL injection in the values.
Estruturas de dados internas
- Query API has had very successful structure of one global nested
array()structure passed around. Various modules would add pieces of data to many different points of that array, until, finally, it would get rendered for the client by one of the printers (output modules). For the API, we suggest wrapping this array as a class with helper functions to append individual leaf nodes.
Reportar erros/estado
For now we decided to include error information inside the same structured output as normal result (option #2).
For the result, we may either use the standard HTTP error codes, or always return a properly formatted data:
- Utilizar código PHP
void header( string reason_phrase [, bool replace [, int http_response_code]] )
The header() can be used to set the return status of the operation.
We can define all possible values of the reason_phrase, so for the failed login we may return code=403 and phrase="BadPassword", whereas for any success we would simply return the response without altering the header.
"Prós": É um padrão. The client always has to deal with HTTP errors, so using HTTP code for result would remove any separate error handling the client would have to perform. Since the client may request data in multiple formats, an invalid format parameter would still be properly handled, as it will simply be another http error code.
"Contras": ...
- Include error information inside a proper response
This method would always return a properly formatted response object, but the error status/description will be the only values inside that object. This is similar to the way current Query API returns status codes.
"Prós": HTTP error codes are used only for the networking issues, not for the data (logical errors). We do not tied to the existing HTTP error codes.
"Contras": If the data format parameter is not properly specified, what is the format of the output data? Application has to parse the object to know of an error (perf?). Error checking code will have to be on both the connection and data parsing levels.
Boilerplate code
It has been suggested that this page or section be merged with API:Extensions#ApiSampleApiExtension.php.(Discuss) |
| Módulo de API simples |
|---|
<?php
class Api<nome do módulo> extends ApiBase {
public function __construct( $main, $action ) {
parent::__construct( $main, $action );
}
public function execute() {
}
public function getAllowedParams() {
return array(
'<nome do parâmetro>' => array(
ApiBase::PARAM_TYPE => array( 'foo', 'bar', 'baz' ),
),
);
}
public function getParamDescription() {
return array(
'<nome do parâmetro>' => '<Descrição do parâmetro>',
);
}
public function getDescription() {
return '<Descrição do módulo aqui>';
}
public function getExamples() {
return array(
'api.php?action=<nome do módulo>&<nome do parâmetro>=foo'
);
}
public function getHelpUrls() {
return '';
}
}
|