195 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			PHP
		
	
	
		
		
			
		
	
	
			195 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			PHP
		
	
	
| 
								 | 
							
								<?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\Cache\DataCollector;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Cache\Adapter\TraceableAdapter;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Cache\Adapter\TraceableAdapterEvent;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\HttpFoundation\Request;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\HttpFoundation\Response;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\HttpKernel\DataCollector\DataCollector;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\HttpKernel\DataCollector\LateDataCollectorInterface;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * @author Aaron Scherer <aequasi@gmail.com>
							 | 
						||
| 
								 | 
							
								 * @author Tobias Nyholm <tobias.nyholm@gmail.com>
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @final since Symfony 4.4
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class CacheDataCollector extends DataCollector implements LateDataCollectorInterface
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @var TraceableAdapter[]
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    private $instances = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @param string $name
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function addInstance($name, TraceableAdapter $instance)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $this->instances[$name] = $instance;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * {@inheritdoc}
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @param \Throwable|null $exception
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function collect(Request $request, Response $response/*, \Throwable $exception = null*/)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $empty = ['calls' => [], 'config' => [], 'options' => [], 'statistics' => []];
							 | 
						||
| 
								 | 
							
								        $this->data = ['instances' => $empty, 'total' => $empty];
							 | 
						||
| 
								 | 
							
								        foreach ($this->instances as $name => $instance) {
							 | 
						||
| 
								 | 
							
								            $this->data['instances']['calls'][$name] = $instance->getCalls();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->data['instances']['statistics'] = $this->calculateStatistics();
							 | 
						||
| 
								 | 
							
								        $this->data['total']['statistics'] = $this->calculateTotalStatistics();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function reset()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $this->data = [];
							 | 
						||
| 
								 | 
							
								        foreach ($this->instances as $instance) {
							 | 
						||
| 
								 | 
							
								            $instance->clearCalls();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function lateCollect()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $this->data['instances']['calls'] = $this->cloneVar($this->data['instances']['calls']);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * {@inheritdoc}
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function getName()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return 'cache';
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Method returns amount of logged Cache reads: "get" calls.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @return array
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function getStatistics()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return $this->data['instances']['statistics'];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Method returns the statistic totals.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @return array
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function getTotals()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return $this->data['total']['statistics'];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Method returns all logged Cache call objects.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @return mixed
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function getCalls()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return $this->data['instances']['calls'];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function calculateStatistics(): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $statistics = [];
							 | 
						||
| 
								 | 
							
								        foreach ($this->data['instances']['calls'] as $name => $calls) {
							 | 
						||
| 
								 | 
							
								            $statistics[$name] = [
							 | 
						||
| 
								 | 
							
								                'calls' => 0,
							 | 
						||
| 
								 | 
							
								                'time' => 0,
							 | 
						||
| 
								 | 
							
								                'reads' => 0,
							 | 
						||
| 
								 | 
							
								                'writes' => 0,
							 | 
						||
| 
								 | 
							
								                'deletes' => 0,
							 | 
						||
| 
								 | 
							
								                'hits' => 0,
							 | 
						||
| 
								 | 
							
								                'misses' => 0,
							 | 
						||
| 
								 | 
							
								            ];
							 | 
						||
| 
								 | 
							
								            /** @var TraceableAdapterEvent $call */
							 | 
						||
| 
								 | 
							
								            foreach ($calls as $call) {
							 | 
						||
| 
								 | 
							
								                ++$statistics[$name]['calls'];
							 | 
						||
| 
								 | 
							
								                $statistics[$name]['time'] += $call->end - $call->start;
							 | 
						||
| 
								 | 
							
								                if ('get' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    ++$statistics[$name]['reads'];
							 | 
						||
| 
								 | 
							
								                    if ($call->hits) {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['hits'];
							 | 
						||
| 
								 | 
							
								                    } else {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['misses'];
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['writes'];
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ('getItem' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    ++$statistics[$name]['reads'];
							 | 
						||
| 
								 | 
							
								                    if ($call->hits) {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['hits'];
							 | 
						||
| 
								 | 
							
								                    } else {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['misses'];
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ('getItems' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    $statistics[$name]['reads'] += $call->hits + $call->misses;
							 | 
						||
| 
								 | 
							
								                    $statistics[$name]['hits'] += $call->hits;
							 | 
						||
| 
								 | 
							
								                    $statistics[$name]['misses'] += $call->misses;
							 | 
						||
| 
								 | 
							
								                } elseif ('hasItem' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    ++$statistics[$name]['reads'];
							 | 
						||
| 
								 | 
							
								                    if (false === $call->result) {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['misses'];
							 | 
						||
| 
								 | 
							
								                    } else {
							 | 
						||
| 
								 | 
							
								                        ++$statistics[$name]['hits'];
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ('save' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    ++$statistics[$name]['writes'];
							 | 
						||
| 
								 | 
							
								                } elseif ('deleteItem' === $call->name) {
							 | 
						||
| 
								 | 
							
								                    ++$statistics[$name]['deletes'];
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            if ($statistics[$name]['reads']) {
							 | 
						||
| 
								 | 
							
								                $statistics[$name]['hit_read_ratio'] = round(100 * $statistics[$name]['hits'] / $statistics[$name]['reads'], 2);
							 | 
						||
| 
								 | 
							
								            } else {
							 | 
						||
| 
								 | 
							
								                $statistics[$name]['hit_read_ratio'] = null;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $statistics;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function calculateTotalStatistics(): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $statistics = $this->getStatistics();
							 | 
						||
| 
								 | 
							
								        $totals = [
							 | 
						||
| 
								 | 
							
								            'calls' => 0,
							 | 
						||
| 
								 | 
							
								            'time' => 0,
							 | 
						||
| 
								 | 
							
								            'reads' => 0,
							 | 
						||
| 
								 | 
							
								            'writes' => 0,
							 | 
						||
| 
								 | 
							
								            'deletes' => 0,
							 | 
						||
| 
								 | 
							
								            'hits' => 0,
							 | 
						||
| 
								 | 
							
								            'misses' => 0,
							 | 
						||
| 
								 | 
							
								        ];
							 | 
						||
| 
								 | 
							
								        foreach ($statistics as $name => $values) {
							 | 
						||
| 
								 | 
							
								            foreach ($totals as $key => $value) {
							 | 
						||
| 
								 | 
							
								                $totals[$key] += $statistics[$name][$key];
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        if ($totals['reads']) {
							 | 
						||
| 
								 | 
							
								            $totals['hit_read_ratio'] = round(100 * $totals['hits'] / $totals['reads'], 2);
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $totals['hit_read_ratio'] = null;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $totals;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |