2014-09-06 16:59:58 -06:00
|
|
|
<?php
|
|
|
|
|
namespace Grav\Plugin;
|
|
|
|
|
|
2014-10-01 22:28:16 +03:00
|
|
|
use Grav\Common\Config\Config;
|
2014-09-06 16:59:58 -06:00
|
|
|
use Grav\Common\Grav;
|
|
|
|
|
use Grav\Common\Page\Page;
|
|
|
|
|
use Grav\Common\Data;
|
|
|
|
|
|
2016-02-04 20:59:08 +01:00
|
|
|
/**
|
|
|
|
|
* Class Popularity
|
|
|
|
|
* @package Grav\Plugin
|
|
|
|
|
*/
|
2014-09-06 16:59:58 -06:00
|
|
|
class Popularity
|
|
|
|
|
{
|
2014-10-01 22:28:16 +03:00
|
|
|
/** @var Config */
|
2014-09-07 19:58:04 -06:00
|
|
|
protected $config;
|
2014-09-06 16:59:58 -06:00
|
|
|
protected $data_path;
|
2014-09-07 19:58:04 -06:00
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
protected $daily_file;
|
2014-09-07 19:58:04 -06:00
|
|
|
protected $monthly_file;
|
|
|
|
|
protected $totals_file;
|
|
|
|
|
protected $visitors_file;
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
protected $daily_data;
|
2014-09-07 19:58:04 -06:00
|
|
|
protected $monthly_data;
|
|
|
|
|
protected $totals_data;
|
|
|
|
|
protected $visitors_data;
|
2014-09-06 16:59:58 -06:00
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
const DAILY_FORMAT = 'd-m-Y';
|
|
|
|
|
const MONTHLY_FORMAT = 'm-Y';
|
|
|
|
|
const DAILY_FILE = 'daily.json';
|
2014-09-07 09:02:39 -06:00
|
|
|
const MONTHLY_FILE = 'monthly.json';
|
|
|
|
|
const TOTALS_FILE = 'totals.json';
|
2014-09-07 19:58:04 -06:00
|
|
|
const VISITORS_FILE = 'visitors.json';
|
2014-09-07 09:02:39 -06:00
|
|
|
|
2014-09-06 16:59:58 -06:00
|
|
|
public function __construct()
|
|
|
|
|
{
|
2016-02-04 20:59:08 +01:00
|
|
|
$this->config = Grav::instance()['config'];
|
2014-09-07 19:58:04 -06:00
|
|
|
|
2016-02-04 20:59:08 +01:00
|
|
|
$this->data_path = Grav::instance()['locator']->findResource('log://popularity', true, true);
|
2014-09-08 12:07:35 -06:00
|
|
|
$this->daily_file = $this->data_path.'/'.self::DAILY_FILE;
|
2014-09-07 19:58:04 -06:00
|
|
|
$this->monthly_file = $this->data_path.'/'.self::MONTHLY_FILE;
|
|
|
|
|
$this->totals_file = $this->data_path.'/'.self::TOTALS_FILE;
|
|
|
|
|
$this->visitors_file = $this->data_path.'/'.self::VISITORS_FILE;
|
|
|
|
|
|
2014-09-06 16:59:58 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function trackHit()
|
2015-10-06 14:56:26 +02:00
|
|
|
{
|
2015-09-30 18:46:53 +02:00
|
|
|
// Don't track bot or crawler requests
|
2016-02-04 20:59:08 +01:00
|
|
|
if (!Grav::instance()['browser']->isHuman()) {
|
2015-09-30 18:46:53 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-01 22:28:16 +03:00
|
|
|
/** @var Page $page */
|
2016-02-04 20:59:08 +01:00
|
|
|
$page = Grav::instance()['page'];
|
|
|
|
|
$relative_url = str_replace(Grav::instance()['base_url_relative'], '', $page->url());
|
2014-09-06 16:59:58 -06:00
|
|
|
|
2014-09-07 09:02:39 -06:00
|
|
|
// Don't track error pages or pages that have no route
|
|
|
|
|
if ($page->template() == 'error' || !$page->route()) {
|
2014-09-06 18:13:04 -06:00
|
|
|
return;
|
|
|
|
|
}
|
2014-09-06 16:59:58 -06:00
|
|
|
|
2014-09-07 09:02:39 -06:00
|
|
|
// Make sure no 'widcard-style' ignore matches this url
|
2014-09-07 19:58:04 -06:00
|
|
|
foreach ((array) $this->config->get('plugins.admin.popularity.ignore') as $ignore) {
|
2014-09-07 09:02:39 -06:00
|
|
|
if (fnmatch($ignore, $relative_url)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-06 16:59:58 -06:00
|
|
|
// initial creation if it doesn't exist
|
|
|
|
|
if (!file_exists($this->data_path)) {
|
|
|
|
|
mkdir($this->data_path);
|
2014-09-08 12:07:35 -06:00
|
|
|
$this->flushPopularity();
|
2014-09-06 16:59:58 -06:00
|
|
|
}
|
|
|
|
|
|
2014-09-07 09:02:39 -06:00
|
|
|
// Update the data we want to track
|
2014-09-08 12:07:35 -06:00
|
|
|
$this->updateDaily();
|
2014-09-07 19:58:04 -06:00
|
|
|
$this->updateMonthly();
|
|
|
|
|
$this->updateTotals($page->route());
|
2016-02-04 20:59:08 +01:00
|
|
|
$this->updateVisitors(Grav::instance()['uri']->ip());
|
2014-09-06 16:59:58 -06:00
|
|
|
|
2014-09-07 09:02:39 -06:00
|
|
|
}
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
protected function updateDaily()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
if (!$this->daily_data) {
|
|
|
|
|
$this->daily_data = $this->getData($this->daily_file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$day_month_year = date(self::DAILY_FORMAT);
|
|
|
|
|
|
|
|
|
|
// get the daily access count
|
|
|
|
|
if (array_key_exists($day_month_year, $this->daily_data)) {
|
|
|
|
|
$this->daily_data[$day_month_year] = intval($this->daily_data[$day_month_year]) + 1;
|
|
|
|
|
} else {
|
|
|
|
|
$this->daily_data[$day_month_year] = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// keep correct number as set by history
|
2015-07-24 17:31:29 -06:00
|
|
|
$count = intval($this->config->get('plugins.admin.popularity.history.daily', 30));
|
2014-09-08 12:07:35 -06:00
|
|
|
$total = count($this->daily_data);
|
2015-07-24 17:31:29 -06:00
|
|
|
|
|
|
|
|
if ($total > $count) {
|
|
|
|
|
$this->daily_data = array_slice($this->daily_data, -$count, $count, true);
|
|
|
|
|
}
|
2014-09-08 12:07:35 -06:00
|
|
|
|
|
|
|
|
file_put_contents($this->daily_file, json_encode($this->daily_data));
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
2014-09-08 18:32:13 -06:00
|
|
|
public function getDailyChartData()
|
|
|
|
|
{
|
|
|
|
|
if (!$this->daily_data) {
|
|
|
|
|
$this->daily_data = $this->getData($this->daily_file);
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-24 17:31:29 -06:00
|
|
|
$limit = intval($this->config->get('plugins.admin.popularity.dashboard.days_of_stats', 7));
|
|
|
|
|
$chart_data = array_slice($this->daily_data, -$limit, $limit);
|
|
|
|
|
|
2014-09-08 18:32:13 -06:00
|
|
|
$labels = array();
|
|
|
|
|
$data = array();
|
|
|
|
|
|
2015-07-24 17:31:29 -06:00
|
|
|
foreach ($chart_data as $date => $count) {
|
2016-02-04 20:59:08 +01:00
|
|
|
$labels[] = Grav::instance()['grav']['admin']->translate(['PLUGIN_ADMIN.' . strtoupper(date('D', strtotime($date)))]);
|
2014-09-08 18:32:13 -06:00
|
|
|
$data[] = $count;
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-01 11:27:34 -08:00
|
|
|
return array('labels' => $labels, 'data' => $data);
|
2014-09-08 18:32:13 -06:00
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @return int
|
|
|
|
|
*/
|
2014-09-08 18:32:13 -06:00
|
|
|
public function getDailyTotal()
|
|
|
|
|
{
|
|
|
|
|
if (!$this->daily_data) {
|
|
|
|
|
$this->daily_data = $this->getData($this->daily_file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isset($this->daily_data[date(self::DAILY_FORMAT)])) {
|
|
|
|
|
return $this->daily_data[date(self::DAILY_FORMAT)];
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @return int
|
|
|
|
|
*/
|
2014-09-08 18:32:13 -06:00
|
|
|
public function getWeeklyTotal()
|
|
|
|
|
{
|
|
|
|
|
if (!$this->daily_data) {
|
|
|
|
|
$this->daily_data = $this->getData($this->daily_file);
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-06 14:56:26 +02:00
|
|
|
$day = 0;
|
2014-09-08 18:32:13 -06:00
|
|
|
$total = 0;
|
2015-10-06 14:56:26 +02:00
|
|
|
foreach (array_reverse($this->daily_data) as $daily) {
|
2014-09-08 18:32:13 -06:00
|
|
|
$total += $daily;
|
2015-10-06 14:56:26 +02:00
|
|
|
$day++;
|
|
|
|
|
if ($day == 7) break;
|
2014-09-08 18:32:13 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $total;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @return int
|
|
|
|
|
*/
|
2014-09-08 18:32:13 -06:00
|
|
|
public function getMonthlyTotal()
|
|
|
|
|
{
|
|
|
|
|
if (!$this->monthly_data) {
|
|
|
|
|
$this->monthly_data = $this->getData($this->monthly_file);
|
|
|
|
|
}
|
|
|
|
|
if (isset($this->monthly_data[date(self::MONTHLY_FORMAT)])) {
|
|
|
|
|
return $this->monthly_data[date(self::MONTHLY_FORMAT)];
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
protected function updateMonthly()
|
2014-09-07 09:02:39 -06:00
|
|
|
{
|
|
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
if (!$this->monthly_data) {
|
|
|
|
|
$this->monthly_data = $this->getData($this->monthly_file);
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
$month_year = date(self::MONTHLY_FORMAT);
|
2014-09-07 09:02:39 -06:00
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
// get the monthly access count
|
|
|
|
|
if (array_key_exists($month_year, $this->monthly_data)) {
|
|
|
|
|
$this->monthly_data[$month_year] = intval($this->monthly_data[$month_year]) + 1;
|
|
|
|
|
} else {
|
|
|
|
|
$this->monthly_data[$month_year] = 1;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
// keep correct number as set by history
|
|
|
|
|
$count = intval($this->config->get('plugins.admin.popularity.history.monthly', 12));
|
|
|
|
|
$total = count($this->monthly_data);
|
|
|
|
|
$this->monthly_data = array_slice($this->monthly_data, $total - $count, $count);
|
|
|
|
|
|
|
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
file_put_contents($this->monthly_file, json_encode($this->monthly_data));
|
2014-09-07 09:02:39 -06:00
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
2014-09-08 18:32:13 -06:00
|
|
|
protected function getMonthyChartData()
|
|
|
|
|
{
|
|
|
|
|
if (!$this->monthly_data) {
|
|
|
|
|
$this->monthly_data = $this->getData($this->monthly_file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$labels = array();
|
|
|
|
|
$data = array();
|
|
|
|
|
|
|
|
|
|
foreach ($this->monthly_data as $date => $count) {
|
2015-07-24 17:31:29 -06:00
|
|
|
$labels[] = date('M', strtotime($date));
|
2014-09-08 18:32:13 -06:00
|
|
|
$data[] = $count;
|
|
|
|
|
}
|
|
|
|
|
return array('labels' => $labels, 'data' => $data);
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @param string $url
|
|
|
|
|
*/
|
2014-09-07 19:58:04 -06:00
|
|
|
protected function updateTotals($url)
|
2014-09-07 09:02:39 -06:00
|
|
|
{
|
2014-09-07 19:58:04 -06:00
|
|
|
if (!$this->totals_data) {
|
|
|
|
|
$this->totals_data = $this->getData($this->totals_file);
|
|
|
|
|
}
|
2014-09-07 09:02:39 -06:00
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
// get the totals for this url
|
|
|
|
|
if (array_key_exists($url, $this->totals_data)) {
|
|
|
|
|
$this->totals_data[$url] = intval($this->totals_data[$url]) + 1;
|
2014-09-06 16:59:58 -06:00
|
|
|
} else {
|
2014-09-07 19:58:04 -06:00
|
|
|
$this->totals_data[$url] = 1;
|
2014-09-06 16:59:58 -06:00
|
|
|
}
|
|
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
file_put_contents($this->totals_file, json_encode($this->totals_data));
|
2014-09-06 16:59:58 -06:00
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @param string $ip
|
|
|
|
|
*/
|
2014-09-07 19:58:04 -06:00
|
|
|
protected function updateVisitors($ip)
|
2014-09-06 16:59:58 -06:00
|
|
|
{
|
2014-09-07 19:58:04 -06:00
|
|
|
if (!$this->visitors_data) {
|
|
|
|
|
$this->visitors_data = $this->getData($this->visitors_file);
|
|
|
|
|
}
|
2014-09-06 16:59:58 -06:00
|
|
|
|
2014-09-07 19:58:04 -06:00
|
|
|
// update with current timestamp
|
|
|
|
|
$this->visitors_data[$ip] = time();
|
|
|
|
|
$visitors = $this->visitors_data;
|
|
|
|
|
arsort($visitors);
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
$count = intval($this->config->get('plugins.admin.popularity.history.visitors', 20));
|
2014-12-12 16:08:07 -07:00
|
|
|
$this->visitors_data = array_slice($visitors, 0, $count, true);
|
2014-09-07 19:58:04 -06:00
|
|
|
|
|
|
|
|
file_put_contents($this->visitors_file, json_encode($this->visitors_data));
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-21 09:46:38 +02:00
|
|
|
/**
|
|
|
|
|
* @param string $path
|
|
|
|
|
* @return array
|
|
|
|
|
*/
|
2014-09-07 19:58:04 -06:00
|
|
|
protected function getData($path)
|
|
|
|
|
{
|
2015-12-11 21:49:06 -07:00
|
|
|
if (file_exists($path)) {
|
|
|
|
|
return (array) json_decode(file_get_contents($path), true);
|
|
|
|
|
} else {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
2014-09-07 19:58:04 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-09-08 12:07:35 -06:00
|
|
|
public function flushPopularity()
|
2014-09-07 19:58:04 -06:00
|
|
|
{
|
2014-09-08 12:07:35 -06:00
|
|
|
file_put_contents($this->daily_file, array());
|
|
|
|
|
file_put_contents($this->monthly_file, array());
|
|
|
|
|
file_put_contents($this->totals_file, array());
|
|
|
|
|
file_put_contents($this->visitors_file, array());
|
2014-09-06 16:59:58 -06:00
|
|
|
}
|
|
|
|
|
}
|