request = $request;
$this->outputFormat = $outputFormat;
}
/**
* This method processes the data resulting from the API call.
*
* - If the data resulted from the API call is a Piwik_DataTable then
* - we apply the standard filters if the parameters have been found
* in the URL. For example to offset,limit the Table you can add the following parameters to any API
* call that returns a DataTable: filter_limit=10&filter_offset=20
* - we apply the filters that have been previously queued on the DataTable
* @see Piwik_DataTable::queueFilter()
* - we apply the renderer that generate the DataTable in a given format (XML, PHP, HTML, JSON, etc.)
* the format can be changed using the 'format' parameter in the request.
* Example: format=xml
*
* - If there is nothing returned (void) we display a standard success message
*
* - If there is a PHP array returned, we try to convert it to a dataTable
* It is then possible to convert this datatable to any requested format (xml/etc)
*
* - If a bool is returned we convert to a string (true is displayed as 'true' false as 'false')
*
* - If an integer / float is returned, we simply return it
*
* @throws Exception If an object/resource is returned, if any of conversion fails, etc.
*
* @param mixed The initial returned value, before post process
* @return mixed Usually a string, but can still be a PHP data structure if the format requested is 'original'
*/
public function getResponse($value)
{
// If the returned value is an object DataTable we
// apply the set of generic filters if asked in the URL
// and we render the DataTable according to the format specified in the URL
if($value instanceof Piwik_DataTable
|| $value instanceof Piwik_DataTable_Array)
{
return $this->handleDataTable($value);
}
// Case an array is returned from the API call, we convert it to the requested format
// - if calling from inside the application (format = original)
// => the data stays unchanged (ie. a standard php array or whatever data structure)
// - if any other format is requested, we have to convert this data structure (which we assume
// to be an array) to a DataTable in order to apply the requested DataTable_Renderer (for example XML)
if(is_array($value))
{
return $this->handleArray($value);
}
// when null or void is returned from the api call, we handle it as a successful operation
if(!isset($value))
{
return $this->handleSuccess();
}
// original data structure requested, we return without process
if( $this->outputFormat == 'original' )
{
return $value;
}
if( is_object($value)
|| is_resource($value))
{
return $this->getResponseException(new Exception('The API cannot handle this data structure.'));
}
// bool // integer // float // serialized object
return $this->handleScalar($value);
}
/**
* Returns an error $message in the requested $format
*
* @param string $message
* @param string $format xml/json/php/csv
* @return string
*/
public function getResponseException(Exception $e)
{
$message = htmlentities($e->getMessage(), ENT_COMPAT, "UTF-8");
switch($this->outputFormat)
{
case 'original':
throw $e;
break;
case 'xml':
@header("Content-Type: text/xml;charset=utf-8");
$return =
"\n" .
"\n".
"\t\n".
"";
break;
case 'json':
@header( "Content-type: application/json" );
// we remove the \n from the resulting string as this is not allowed in json
$message = str_replace("\n","",$message);
$return = '{"result":"error", "message":"'.$message.'"}';
break;
case 'php':
$return = array('result' => 'error', 'message' => $message);
if($this->caseRendererPHPSerialize())
{
$return = serialize($return);
}
break;
case 'html':
$return = nl2br($message);
break;
default:
$return = 'Error: '.$message;
break;
}
return $return;
}
/**
* Returns true if the user requested to serialize the output data (&serialize=1 in the request)
*
* @param $defaultSerializeValue Default value in case the user hasn't specified a value
* @return bool
*/
protected function caseRendererPHPSerialize($defaultSerializeValue = 1)
{
$serialize = Piwik_Common::getRequestVar('serialize', $defaultSerializeValue, 'int', $this->request);
if($serialize)
{
return true;
}
return false;
}
/**
* Apply the specified renderer to the DataTable
*
* @param Piwik_DataTable
* @return string
*/
protected function getRenderedDataTable($dataTable)
{
$format = strtolower($this->outputFormat);
// if asked for original dataStructure
if($format == 'original')
{
// if the original dataStructure is a simpleDataTable
// and has only one column, we return the value
if($dataTable instanceof Piwik_DataTable_Simple)
{
$columns = $dataTable->getFirstRow()->getColumns();
if(count($columns) == 1)
{
$values = array_values($columns);
return $values[0];
}
}
// by default "original" data is not serialized
if($this->caseRendererPHPSerialize( $defaultSerialize = 0))
{
$dataTable = serialize($dataTable);
}
return $dataTable;
}
$renderer = Piwik_DataTable_Renderer::factory($format);
$renderer->setTable($dataTable);
$renderer->setRenderSubTables(Piwik_Common::getRequestVar('expanded', false, 'int', $this->request));
if($format == 'php')
{
$renderer->setSerialize( $this->caseRendererPHPSerialize());
$renderer->setPrettyDisplay(Piwik_Common::getRequestVar('prettyDisplay', false, 'int', $this->request));
}
else if($format == 'html')
{
$renderer->setTableId($this->request['method']);
}
return $renderer->render();
}
/**
* Returns a success $message in the requested $format
*
* @param string $format xml/json/php/csv
* @param string $message
* @return string
*/
protected function handleSuccess( $message = 'ok' )
{
switch($this->outputFormat)
{
case 'xml':
@header("Content-Type: text/xml;charset=utf-8");
$return =
"\n" .
"\n".
"\t\n".
"";
break;
case 'json':
@header( "Content-type: application/json" );
$return = '{"result":"success", "message":"'.$message.'"}';
break;
case 'php':
$return = array('result' => 'success', 'message' => $message);
if($this->caseRendererPHPSerialize())
{
$return = serialize($return);
}
break;
case 'csv':
header("Content-type: application/vnd.ms-excel");
header("Content-Disposition: attachment; filename=piwik-report-export.csv");
$return = "message\n".$message;
break;
default:
$return = 'Success:'.$message;
break;
}
return $return;
}
protected function handleScalar($scalar)
{
$dataTable = new Piwik_DataTable_Simple();
$dataTable->addRowsFromArray( array($scalar) );
return $this->getRenderedDataTable($dataTable);
}
protected function handleDataTable($datatable)
{
// if the flag disable_generic_filters is defined we skip the generic filters
if('false' == Piwik_Common::getRequestVar('disable_generic_filters', 'false', 'string', $this->request))
{
$genericFilter = new Piwik_API_DataTableGenericFilter($datatable, $this->request);
$genericFilter->filter();
}
// we automatically safe decode all datatable labels (against xss)
$datatable->queueFilter('SafeDecodeLabel');
// if the flag disable_queued_filters is defined we skip the filters that were queued
if(Piwik_Common::getRequestVar('disable_queued_filters', 'false', 'string', $this->request) == 'false')
{
$datatable->applyQueuedFilters();
}
return $this->getRenderedDataTable($datatable);
}
protected function handleArray($array)
{
if($this->outputFormat == 'original')
{
// we handle the serialization. Because some php array have a very special structure that
// couldn't be converted with the automatic DataTable->addRowsFromSimpleArray
// the user may want to request the original PHP data structure serialized by the API
// in case he has to setup serialize=1 in the URL
if($this->caseRendererPHPSerialize( $defaultSerialize = 0))
{
return serialize($array);
}
}
else
{
$dataTable = new Piwik_DataTable();
$dataTable->addRowsFromSimpleArray($array);
return $this->getRenderedDataTable($dataTable);
}
}
}