File: VisitLastActionTime.php

package info (click to toggle)
matomo 5.5.1%2Bdfsg-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 73,596 kB
  • sloc: php: 231,041; javascript: 102,286; python: 202; xml: 189; sh: 172; makefile: 20; sql: 10
file content (118 lines) | stat: -rw-r--r-- 4,529 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
<?php

/**
 * Matomo - free/libre analytics platform
 *
 * @link    https://matomo.org
 * @license https://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
 */

namespace Piwik\Plugins\CoreHome\Columns;

use Piwik\Common;
use Piwik\Date;
use Piwik\Period;
use Piwik\Plugin\Dimension\VisitDimension;
use Piwik\Site;
use Piwik\Tracker\Action;
use Piwik\Tracker\Request;
use Piwik\Tracker\Visitor;
use Piwik\Metrics\Formatter;

require_once MATOMO_PLUGINS_PATH . '/plugins/VisitTime/functions.php';

/**
 * This dimension holds the best guess for a visit's end time. It is set the last action
 * time for each visit. `ping=1` requests can be sent to update the dimension value so
 * it can be a more accurate guess of the time the visitor spent on the site.
 *
 * Note: though it is named 'visit last action time' it actually refers to the visit's last action's
 * end time.
 */
class VisitLastActionTime extends VisitDimension
{
    protected $columnName = 'visit_last_action_time';
    protected $type = self::TYPE_DATETIME;
    protected $nameSingular = 'VisitTime_ColumnVisitEndSiteHour';
    protected $sqlSegment = 'HOUR(log_visit.visit_last_action_time)';
    protected $segmentName = 'visitServerHour';
    protected $acceptValues = '0, 1, 2, 3, ..., 20, 21, 22, 23';

    /**
     * Converts the hour to the hour depending on the configured site's timezone.
     * Only works correct if a date/period is present in the request. Otherwise the result may vary depending on the
     * day of the year as it can't know about daylight savings for example. Also if the currently selected date range
     * includes multiple days with daylight savings and some without, then it might not be 100% correct as we are
     * only looking at the start of the time.
     * @param int $hour
     * @param int $idSite
     * @return int
     * @throws \Exception
     */
    public static function convertHourToHourInSiteTimezone($hour, $idSite)
    {
        $date = Date::now();
        $dateString = Common::getRequestVar('date', '', 'string');
        $periodString = Common::getRequestVar('period', '', 'string');
        if (!empty($dateString) && !empty($periodString)) {
            try {
                $date = Period\Factory::build($periodString, $dateString)->getDateStart();
            } catch (\Exception $e) {
                // ignore any error eg if values are wrong...
            }
        }
        $timezone = Site::getTimezoneFor($idSite);
        $datetime = $date->toString() . ' ' . $hour . ':00:00';
        $hourInTz = (int)Date::factory($datetime, $timezone)->toString('H');
        return $hourInTz;
    }

    public function formatValue($value, $idSite, Formatter $formatter)
    {
        $hourInTz = self::convertHourToHourInSiteTimezone($value, $idSite);

        return \Piwik\Plugins\VisitTime\getTimeLabel($hourInTz);
    }

    // we do not install or define column definition here as we need to create this column when installing as there is
    // an index on it. Currently we do not define the index here... although we could overwrite the install() method
    // and add column 'visit_last_action_time' and add index. Problem is there is also an index
    // INDEX(idsite, config_id, visit_last_action_time) and we maybe not be sure whether idsite already exists at
    // installing point (we do not know whether idsite column will be added first).

    /**
     * @param Request $request
     * @param Visitor $visitor
     * @param Action|null $action
     * @return mixed
     */
    public function onNewVisit(Request $request, Visitor $visitor, $action)
    {
        return Date::getDatetimeFromTimestamp($request->getCurrentTimestamp());
    }

    /**
     * @param Request $request
     * @param Visitor $visitor
     * @param Action|null $action
     * @return mixed
     */
    public function onExistingVisit(Request $request, Visitor $visitor, $action)
    {
        if ($request->getParam('ping') == 1) {
            return false;
        }

        $originalVisitLastActionTime = $visitor->getPreviousVisitColumn('visit_last_action_time');

        if (
            !empty($originalVisitLastActionTime)
            && Date::factory($originalVisitLastActionTime)->getTimestamp() > $request->getCurrentTimestamp()
        ) {
            // make sure to not set visit_last_action_time to an earlier time eg if tracking requests aren't sent in order
            return $originalVisitLastActionTime;
        }

        return $this->onNewVisit($request, $visitor, $action);
    }
}