logger = $logger; $idsite = $settings->get_option( 'site_id' ); if ( ! $idsite ) { return; } $api_endpoint = rtrim( $settings->get_matomo_url(), '/' ) . '/matomo.php'; parent::__construct( (int) $idsite, $api_endpoint ); $this->ip = false; if ( ! $settings->get_global_option( 'disable_cookies' ) ) { $cookie_domain = $this->get_tracking_cookie_domain( $settings ); $this->enableCookies( $cookie_domain ); } else { $this->disableCookieSupport(); } if ( $this->loadVisitorIdCookie() ) { if ( ! empty( $this->cookieVisitorId ) ) { $this->has_cookie = true; $this->set_visitor_id_safe( $this->cookieVisitorId ); } } } public function set_visitor_id_safe( $visitor_id ) { try { $this->setVisitorId( $visitor_id ); } catch ( \Exception $ex ) { // do not fatal if the visitor ID is invalid for some reason if ( ! $this->is_invalid_visitor_id_error( $ex ) ) { throw $ex; } } } protected function setCookie( $cookieName, $cookieValue, $cookieTTL ) { if ( ! $this->has_cookie ) { // we only set / overwrite cookies if it is a visitor that has eg no JS enabled or ad blocker enabled etc. // this way we will track all cart updates and orders into the same visitor on following requests. // If we recognized the visitor before via cookie we want in our case to make sure to not overwrite // any cookie return parent::setCookie( $cookieName, $cookieValue, $cookieTTL ); } return $this; } protected function sendRequest( string $url, string $method = 'GET', $data = null, bool $force = false ): string { if ( ! $this->idSite ) { $this->logger->log( 'ecommerce tracking could not find idSite, cannot send request' ); return ''; // not installed or synced yet } if ( $this->is_prerender() ) { // do not track if for some reason we are prerendering return ''; } $args = array( 'method' => $method, 'headers' => array( 'User-Agent' => $this->userAgent, ), 'blocking' => false, ); if ( ! empty( $data ) ) { $args['body'] = $data; } $url = $url . '&bots=1'; try { $response = $this->wp_remote_request( $url, $args ); } catch ( \Exception $ex ) { $this->logger->log( 'ajax_tracker: ' . $ex->getMessage() ); return ''; } if ( is_wp_error( $response ) ) { $this->logger->log( 'ajax_tracker: ' . new \Exception( $response->get_error_message() ) ); return ''; } return $response['body']; } private function is_invalid_visitor_id_error( \Exception $ex ) { return strpos( $ex->getMessage(), 'setVisitorId() expects' ) === 0; } /** * See https://developer.chrome.com/docs/web-platform/prerender-pages * * @return bool */ private function is_prerender() { // phpcs:disable WordPress.Security.ValidatedSanitizedInput.InputNotSanitized $purpose = strtolower( isset( $_SERVER['HTTP_SEC_PURPOSE'] ) ? wp_unslash( $_SERVER['HTTP_SEC_PURPOSE'] ) : '' ); return strpos( $purpose, 'prefetch' ) !== false || strpos( $purpose, 'prerender' ) !== false; } /** * for tests to override * * @param string $url * @param array $args * @return array|\WP_Error */ protected function wp_remote_request( $url, $args ) { return wp_remote_request( $url, $args ); } /** * In Connect Matomo we want to rely entirely on JavaScript tracker * for creating cookies. * * @return self */ protected function setFirstPartyCookies() { // disabled return $this; } public static function getCurrentUrl(): string { return parent::getCurrentUrl(); } public function get_tracking_cookie_domain( Settings $settings ) { if ( $settings->get_global_option( 'track_across' ) || $settings->get_global_option( 'track_crossdomain_linking' ) ) { $host = wp_parse_url( home_url(), PHP_URL_HOST ); if ( ! empty( $host ) ) { return '*.' . $host; } } return ''; } }