forked from rebillar/site-accueil-insa
		
	
		
			
				
	
	
		
			381 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			381 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * Matomo - free/libre analytics platform
 | |
|  *
 | |
|  * @link https://matomo.org
 | |
|  * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
 | |
|  *
 | |
|  */
 | |
| 
 | |
| namespace Piwik;
 | |
| 
 | |
| use Exception;
 | |
| use Piwik\Container\StaticContainer;
 | |
| use Piwik\Plugins\BulkTracking\Tracker\Requests;
 | |
| use Piwik\Plugins\PrivacyManager\Config as PrivacyManagerConfig;
 | |
| use Piwik\Tracker\Db as TrackerDb;
 | |
| use Piwik\Tracker\Db\DbException;
 | |
| use Piwik\Tracker\Handler;
 | |
| use Piwik\Tracker\Request;
 | |
| use Piwik\Tracker\RequestSet;
 | |
| use Piwik\Tracker\TrackerConfig;
 | |
| use Piwik\Tracker\Visit;
 | |
| use Piwik\Plugin\Manager as PluginManager;
 | |
| use Psr\Log\LoggerInterface;
 | |
| 
 | |
| /**
 | |
|  * Class used by the logging script piwik.php called by the javascript tag.
 | |
|  * Handles the visitor and their actions on the website, saves the data in the DB,
 | |
|  * saves information in the cookie, etc.
 | |
|  *
 | |
|  * We try to include as little files as possible (no dependency on 3rd party modules).
 | |
|  */
 | |
| class Tracker
 | |
| {
 | |
|     /**
 | |
|      * @var Db
 | |
|      */
 | |
|     private static $db = null;
 | |
| 
 | |
|     // We use hex ID that are 16 chars in length, ie. 64 bits IDs
 | |
|     const LENGTH_HEX_ID_STRING = 16;
 | |
|     const LENGTH_BINARY_ID = 8;
 | |
| 
 | |
|     public static $initTrackerMode = false;
 | |
| 
 | |
|     private $countOfLoggedRequests = 0;
 | |
|     protected $isInstalled = null;
 | |
| 
 | |
|     /**
 | |
|      * @var LoggerInterface
 | |
|      */
 | |
|     private $logger;
 | |
| 
 | |
|     public function __construct()
 | |
|     {
 | |
|         $this->logger = StaticContainer::get(LoggerInterface::class);
 | |
|     }
 | |
| 
 | |
|     public function isDebugModeEnabled()
 | |
|     {
 | |
|         return array_key_exists('PIWIK_TRACKER_DEBUG', $GLOBALS) && $GLOBALS['PIWIK_TRACKER_DEBUG'] === true;
 | |
|     }
 | |
| 
 | |
|     public function shouldRecordStatistics()
 | |
|     {
 | |
|         $record = TrackerConfig::getConfigValue('record_statistics') != 0;
 | |
| 
 | |
|         if (!$record) {
 | |
|             $this->logger->debug('Tracking is disabled in the config.ini.php via record_statistics=0');
 | |
|         }
 | |
| 
 | |
|         return $record && $this->isInstalled();
 | |
|     }
 | |
| 
 | |
|     public static function loadTrackerEnvironment()
 | |
|     {
 | |
|         SettingsServer::setIsTrackerApiRequest();
 | |
|         if (empty($GLOBALS['PIWIK_TRACKER_DEBUG'])) {
 | |
|             $GLOBALS['PIWIK_TRACKER_DEBUG'] = self::isDebugEnabled();
 | |
|         }
 | |
|         if (!empty($GLOBALS['PIWIK_TRACKER_DEBUG']) && !Common::isPhpCliMode()) {
 | |
|             Common::sendHeader('Content-Type: text/plain');
 | |
|         }
 | |
|         PluginManager::getInstance()->loadTrackerPlugins();
 | |
|     }
 | |
| 
 | |
|     private function init()
 | |
|     {
 | |
|         $this->handleFatalErrors();
 | |
| 
 | |
|         if ($this->isDebugModeEnabled()) {
 | |
|             ErrorHandler::registerErrorHandler();
 | |
|             ExceptionHandler::setUp();
 | |
| 
 | |
|             $this->logger->debug("Debug enabled - Input parameters: {params}", [
 | |
|                 'params' => var_export($_GET + $_POST, true),
 | |
|             ]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function isInstalled()
 | |
|     {
 | |
|         if (is_null($this->isInstalled)) {
 | |
|             $this->isInstalled = SettingsPiwik::isMatomoInstalled();
 | |
|         }
 | |
| 
 | |
|         return $this->isInstalled;
 | |
|     }
 | |
| 
 | |
|     public function main(Handler $handler, RequestSet $requestSet)
 | |
|     {
 | |
|         try {
 | |
|             $this->init();
 | |
| 
 | |
|             if ($this->isPreFlightCorsRequest()) {
 | |
|                 Common::sendHeader('Access-Control-Allow-Methods: GET, POST');
 | |
|                 Common::sendHeader('Access-Control-Allow-Headers: *');
 | |
|                 Common::sendHeader('Access-Control-Allow-Origin: *');
 | |
|                 Common::sendResponseCode(204);
 | |
|                 $this->logger->debug("Tracker detected preflight CORS request. Skipping...");
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             $handler->init($this, $requestSet);
 | |
| 
 | |
|             $this->track($handler, $requestSet);
 | |
|         } catch (Exception $e) {
 | |
|             $this->logger->debug("Tracker encountered an exception: {ex}", [$e]);
 | |
| 
 | |
|             $handler->onException($this, $requestSet, $e);
 | |
|         }
 | |
| 
 | |
|         Piwik::postEvent('Tracker.end');
 | |
|         $response = $handler->finish($this, $requestSet);
 | |
| 
 | |
|         $this->disconnectDatabase();
 | |
| 
 | |
|         return $response;
 | |
|     }
 | |
| 
 | |
|     public function track(Handler $handler, RequestSet $requestSet)
 | |
|     {
 | |
|         if (!$this->shouldRecordStatistics()) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $requestSet->initRequestsAndTokenAuth();
 | |
| 
 | |
|         if ($requestSet->hasRequests()) {
 | |
|             $handler->onStartTrackRequests($this, $requestSet);
 | |
|             $handler->process($this, $requestSet);
 | |
|             $handler->onAllRequestsTracked($this, $requestSet);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Request $request
 | |
|      * @return array
 | |
|      */
 | |
|     public function trackRequest(Request $request)
 | |
|     {
 | |
|         if ($request->isEmptyRequest()) {
 | |
|             $this->logger->debug('The request is empty');
 | |
|         } else {
 | |
|             $this->logger->debug('Current datetime: {date}', [
 | |
|                 'date' => date("Y-m-d H:i:s", $request->getCurrentTimestamp()),
 | |
|             ]);
 | |
| 
 | |
|             $visit = Visit\Factory::make();
 | |
|             $visit->setRequest($request);
 | |
|             $visit->handle();
 | |
|         }
 | |
| 
 | |
|         // increment successfully logged request count. make sure to do this after try-catch,
 | |
|         // since an excluded visit is considered 'successfully logged'
 | |
|         ++$this->countOfLoggedRequests;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Used to initialize core Piwik components on a piwik.php request
 | |
|      * Eg. when cache is missed and we will be calling some APIs to generate cache
 | |
|      */
 | |
|     public static function initCorePiwikInTrackerMode()
 | |
|     {
 | |
|         if (
 | |
|             SettingsServer::isTrackerApiRequest()
 | |
|             && self::$initTrackerMode === false
 | |
|         ) {
 | |
|             self::$initTrackerMode = true;
 | |
|             require_once PIWIK_INCLUDE_PATH . '/core/Option.php';
 | |
| 
 | |
|             Access::getInstance();
 | |
|             Config::getInstance();
 | |
| 
 | |
|             try {
 | |
|                 Db::get();
 | |
|             } catch (Exception $e) {
 | |
|                 Db::createDatabaseObject();
 | |
|             }
 | |
| 
 | |
|             PluginManager::getInstance()->loadCorePluginsDuringTracker();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public static function restoreTrackerPlugins()
 | |
|     {
 | |
|         if (SettingsServer::isTrackerApiRequest() && Tracker::$initTrackerMode) {
 | |
|             Plugin\Manager::getInstance()->loadTrackerPlugins();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function getCountOfLoggedRequests()
 | |
|     {
 | |
|         return $this->countOfLoggedRequests;
 | |
|     }
 | |
| 
 | |
|     public function setCountOfLoggedRequests($numLoggedRequests)
 | |
|     {
 | |
|         $this->countOfLoggedRequests = $numLoggedRequests;
 | |
|     }
 | |
| 
 | |
|     public function hasLoggedRequests()
 | |
|     {
 | |
|         return 0 !== $this->countOfLoggedRequests;
 | |
|     }
 | |
| 
 | |
|     public function isDatabaseConnected()
 | |
|     {
 | |
|         return !is_null(self::$db);
 | |
|     }
 | |
| 
 | |
|     public static function getDatabase()
 | |
|     {
 | |
|         if (is_null(self::$db)) {
 | |
|             try {
 | |
|                 self::$db = TrackerDb::connectPiwikTrackerDb();
 | |
|             } catch (Exception $e) {
 | |
|                 $code = $e->getCode();
 | |
|                 // Note: PDOException might return a string as code, but we can't use this for DbException
 | |
|                 throw new DbException($e->getMessage(), is_int($code) ? $code : 0);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return self::$db;
 | |
|     }
 | |
| 
 | |
|     protected function disconnectDatabase()
 | |
|     {
 | |
|         if ($this->isDatabaseConnected()) { // note: I think we do this only for the tests
 | |
|             self::$db->disconnect();
 | |
|             self::$db = null;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // for tests
 | |
|     public static function disconnectCachedDbConnection()
 | |
|     {
 | |
|         // code redundancy w/ above is on purpose; above disconnectDatabase depends on method that can potentially be overridden
 | |
|         if (!is_null(self::$db)) {
 | |
|             self::$db->disconnect();
 | |
|             self::$db = null;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public static function setTestEnvironment($args = null, $requestMethod = null)
 | |
|     {
 | |
|         if (is_null($args)) {
 | |
|             $requests = new Requests();
 | |
|             $args     = $requests->getRequestsArrayFromBulkRequest($requests->getRawBulkRequest());
 | |
|             $args = $_GET + $args;
 | |
|         }
 | |
| 
 | |
|         if (is_null($requestMethod) && array_key_exists('REQUEST_METHOD', $_SERVER)) {
 | |
|             $requestMethod = $_SERVER['REQUEST_METHOD'];
 | |
|         } elseif (is_null($requestMethod)) {
 | |
|             $requestMethod = 'GET';
 | |
|         }
 | |
| 
 | |
|         // Do not run scheduled tasks during tests
 | |
|         if (!defined('DEBUG_FORCE_SCHEDULED_TASKS')) {
 | |
|             TrackerConfig::setConfigValue('scheduled_tasks_min_interval', 0);
 | |
|         }
 | |
| 
 | |
|         // if nothing found in _GET/_POST and we're doing a POST, assume bulk request. in which case,
 | |
|         // we have to bypass authentication
 | |
|         if (empty($args) && $requestMethod == 'POST') {
 | |
|             TrackerConfig::setConfigValue('tracking_requests_require_authentication', 0);
 | |
|         }
 | |
| 
 | |
|         // Tests can force the use of 3rd party cookie for ID visitor
 | |
|         if (Common::getRequestVar('forceEnableFingerprintingAcrossWebsites', false, null, $args) == 1) {
 | |
|             TrackerConfig::setConfigValue('enable_fingerprinting_across_websites', 1);
 | |
|         }
 | |
| 
 | |
|         // Tests can simulate the tracker API maintenance mode
 | |
|         if (Common::getRequestVar('forceEnableTrackerMaintenanceMode', false, null, $args) == 1) {
 | |
|             TrackerConfig::setConfigValue('record_statistics', 0);
 | |
|         }
 | |
| 
 | |
|         // Tests can force the use of 3rd party cookie for ID visitor
 | |
|         if (Common::getRequestVar('forceUseThirdPartyCookie', false, null, $args) == 1) {
 | |
|             TrackerConfig::setConfigValue('use_third_party_id_cookie', 1);
 | |
|         }
 | |
| 
 | |
|         // Tests using window_look_back_for_visitor
 | |
|         if (
 | |
|             Common::getRequestVar('forceLargeWindowLookBackForVisitor', false, null, $args) == 1
 | |
|             // also look for this in bulk requests (see fake_logs_replay.log)
 | |
|             || strpos(json_encode($args, true), '"forceLargeWindowLookBackForVisitor":"1"') !== false
 | |
|         ) {
 | |
|             TrackerConfig::setConfigValue('window_look_back_for_visitor', 2678400);
 | |
|         }
 | |
| 
 | |
|         // Tests can force the enabling of IP anonymization
 | |
|         if (Common::getRequestVar('forceIpAnonymization', false, null, $args) == 1) {
 | |
|             self::getDatabase(); // make sure db is initialized
 | |
| 
 | |
|             $privacyConfig = new PrivacyManagerConfig();
 | |
|             $privacyConfig->ipAddressMaskLength = 2;
 | |
| 
 | |
|             \Piwik\Plugins\PrivacyManager\IPAnonymizer::activate();
 | |
| 
 | |
|             \Piwik\Tracker\Cache::deleteTrackerCache();
 | |
|             Filesystem::clearPhpCaches();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     protected function loadTrackerPlugins()
 | |
|     {
 | |
|         try {
 | |
|             $pluginManager  = PluginManager::getInstance();
 | |
|             $pluginsTracker = $pluginManager->loadTrackerPlugins();
 | |
| 
 | |
|             $this->logger->debug("Loading plugins: { {plugins} }", [
 | |
|                 'plugins' => implode(", ", $pluginsTracker),
 | |
|             ]);
 | |
|         } catch (Exception $e) {
 | |
|             $this->logger->error('Error loading tracker plugins: {exception}', [
 | |
|                 'exception' => $e,
 | |
|             ]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private function handleFatalErrors()
 | |
|     {
 | |
|         register_shutdown_function(function () {
 | |
|             // TODO: add a log here
 | |
|             $lastError = error_get_last();
 | |
|             if (!empty($lastError) && $lastError['type'] == E_ERROR) {
 | |
|                 Common::sendResponseCode(500);
 | |
|             }
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     private static function isDebugEnabled()
 | |
|     {
 | |
|         try {
 | |
|             $debug = (bool) TrackerConfig::getConfigValue('debug');
 | |
|             if ($debug) {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             $debugOnDemand = (bool) TrackerConfig::getConfigValue('debug_on_demand');
 | |
|             if ($debugOnDemand) {
 | |
|                 return (bool) Common::getRequestVar('debug', false);
 | |
|             }
 | |
|         } catch (Exception $e) {
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     public function isPreFlightCorsRequest(): bool
 | |
|     {
 | |
|         if (isset($_SERVER['REQUEST_METHOD']) && strtoupper($_SERVER['REQUEST_METHOD']) === 'OPTIONS') {
 | |
|             return !empty($_SERVER['HTTP_ACCESS_CONTROL_REQUEST_HEADERS']) || !empty($_SERVER['HTTP_ACCESS_CONTROL_REQUEST_METHOD']);
 | |
|         }
 | |
|         return false;
 | |
|     }
 | |
| }
 |