This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
symfony/src/Symfony/Component/HttpKernel/Profiler/MongoDbProfilerStorage.php

235 lines
6.4 KiB
PHP
Raw Normal View History

2011-07-23 19:05:48 +01:00
<?php
/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Symfony\Component\HttpKernel\Profiler;
class MongoDbProfilerStorage implements ProfilerStorageInterface
{
protected $dsn;
2011-09-19 21:37:12 +01:00
protected $lifetime;
private $mongo;
2011-07-23 19:05:48 +01:00
/**
* Constructor.
*
2012-01-11 14:52:51 +00:00
* @param string $dsn A data source name
* @param string $username Not used
* @param string $password Not used
* @param integer $lifetime The lifetime to use for the purge
2011-07-23 19:05:48 +01:00
*/
2011-09-19 21:37:12 +01:00
public function __construct($dsn, $username = '', $password = '', $lifetime = 86400)
2011-07-23 19:05:48 +01:00
{
$this->dsn = $dsn;
2011-09-19 21:37:12 +01:00
$this->lifetime = (int) $lifetime;
2011-07-23 19:05:48 +01:00
}
/**
* {@inheritdoc}
2011-07-23 19:05:48 +01:00
*/
public function find($ip, $url, $limit, $method, $start = null, $end = null)
2011-07-23 19:05:48 +01:00
{
$cursor = $this->getMongo()->find($this->buildQuery($ip, $url, $method, $start, $end), array('_id', 'parent', 'ip', 'method', 'url', 'time'))->sort(array('time' => -1))->limit($limit);
$tokens = array();
2011-07-23 19:05:48 +01:00
foreach ($cursor as $profile) {
$tokens[] = $this->getData($profile);
2011-07-23 19:05:48 +01:00
}
return $tokens;
2011-07-23 19:05:48 +01:00
}
/**
* {@inheritdoc}
2011-07-23 19:05:48 +01:00
*/
public function purge()
2011-07-23 19:05:48 +01:00
{
$this->getMongo()->remove(array());
2011-07-23 19:05:48 +01:00
}
/**
* {@inheritdoc}
2011-07-23 19:05:48 +01:00
*/
public function read($token)
2011-07-23 19:05:48 +01:00
{
$profile = $this->getMongo()->findOne(array('_id' => $token, 'data' => array('$exists' => true)));
if (null !== $profile) {
$profile = $this->createProfileFromData($this->getData($profile));
}
return $profile;
2011-07-23 19:05:48 +01:00
}
/**
* {@inheritdoc}
2011-07-23 19:05:48 +01:00
*/
public function write(Profile $profile)
2011-07-23 19:05:48 +01:00
{
2011-09-19 21:37:12 +01:00
$this->cleanup();
$record = array(
'_id' => $profile->getToken(),
'parent' => $profile->getParentToken(),
'data' => base64_encode(serialize($profile->getCollectors())),
'ip' => $profile->getIp(),
2011-09-24 10:20:46 +01:00
'method' => $profile->getMethod(),
'url' => $profile->getUrl(),
'time' => $profile->getTime()
);
$result = $this->getMongo()->update(array('_id' => $profile->getToken()), array_filter($record, function ($v) { return !empty($v); }), array('upsert' => true));
return (boolean) (isset($result['ok']) ? $result['ok'] : $result);
2011-07-23 19:05:48 +01:00
}
/**
* Internal convenience method that returns the instance of the MongoDB Collection
*
* @return \MongoCollection
*/
protected function getMongo()
{
if ($this->mongo === null) {
if (preg_match('#^(mongodb://.*)/(.*)/(.*)$#', $this->dsn, $matches)) {
$server = $matches[1] . (!empty($matches[2]) ? '/' . $matches[2] : '');
$database = $matches[2];
$collection = $matches[3];
$mongoClass = (version_compare(phpversion('mongo'), '1.3.0', '<')) ? '\Mongo' : '\MongoClient';
$mongo = new $mongoClass($server);
$this->mongo = $mongo->selectCollection($database, $collection);
} else {
throw new \RuntimeException(sprintf('Please check your configuration. You are trying to use MongoDB with an invalid dsn "%s". The expected format is "mongodb://[user:pass@]host/database/collection"', $this->dsn));
}
}
return $this->mongo;
}
/**
* @param array $data
2012-12-11 10:49:22 +00:00
*
* @return Profile
*/
protected function createProfileFromData(array $data)
{
$profile = $this->getProfile($data);
if ($data['parent']) {
$parent = $this->getMongo()->findOne(array('_id' => $data['parent'], 'data' => array('$exists' => true)));
if ($parent) {
$profile->setParent($this->getProfile($this->getData($parent)));
}
}
$profile->setChildren($this->readChildren($data['token']));
return $profile;
}
/**
* @param string $token
2012-12-11 10:49:22 +00:00
*
* @return Profile[] An array of Profile instances
*/
protected function readChildren($token)
{
$profiles = array();
$cursor = $this->getMongo()->find(array('parent' => $token, 'data' => array('$exists' => true)));
foreach ($cursor as $d) {
$profiles[] = $this->getProfile($this->getData($d));
}
return $profiles;
}
2011-09-19 21:37:12 +01:00
protected function cleanup()
{
$this->getMongo()->remove(array('time' => array('$lt' => time() - $this->lifetime)));
}
/**
* @param string $ip
* @param string $url
2011-09-24 10:20:46 +01:00
* @param string $method
* @param int $start
* @param int $end
2012-12-11 10:49:22 +00:00
*
* @return array
*/
private function buildQuery($ip, $url, $method, $start, $end)
{
$query = array();
if (!empty($ip)) {
$query['ip'] = $ip;
}
if (!empty($url)) {
$query['url'] = $url;
}
2011-09-24 10:20:46 +01:00
if (!empty($method)) {
$query['method'] = $method;
}
if (!empty($start) || !empty($end)) {
$query['time'] = array();
}
if (!empty($start)) {
$query['time']['$gte'] = $start;
}
if (!empty($end)) {
$query['time']['$lte'] = $end;
}
return $query;
}
/**
* @param array $data
2012-12-11 10:49:22 +00:00
*
* @return array
*/
private function getData(array $data)
{
return array(
'token' => $data['_id'],
'parent' => isset($data['parent']) ? $data['parent'] : null,
'ip' => isset($data['ip']) ? $data['ip'] : null,
2011-09-24 10:20:46 +01:00
'method' => isset($data['method']) ? $data['method'] : null,
'url' => isset($data['url']) ? $data['url'] : null,
'time' => isset($data['time']) ? $data['time'] : null,
'data' => isset($data['data']) ? $data['data'] : null,
);
}
/**
* @param array $data
2012-12-11 10:49:22 +00:00
*
* @return Profile
*/
private function getProfile(array $data)
{
$profile = new Profile($data['token']);
$profile->setIp($data['ip']);
2011-09-24 10:20:46 +01:00
$profile->setMethod($data['method']);
$profile->setUrl($data['url']);
$profile->setTime($data['time']);
$profile->setCollectors(unserialize(base64_decode($data['data'])));
return $profile;
}
}