PluginProbe ʕ •ᴥ•ʔ
WooCommerce / 8.0.0-rc.2
WooCommerce v8.0.0-rc.2
10.8.1 10.8.0 10.8.0-rc.1 10.8.0-beta.2 10.8.0-beta.1 7.8.0-beta.1 7.8.0-beta.2 7.8.0-rc.1 7.8.0-rc.2 7.8.1 7.8.2 7.8.3 7.8.4 7.9.0 7.9.0-beta.1 7.9.0-beta.2 7.9.0-rc.2 7.9.0-rc.3 7.9.1 7.9.2 8.0.0 8.0.0-beta.1 8.0.0-beta.2 8.0.0-rc.1 8.0.0-rc.2 8.0.1 8.0.2 8.0.3 8.0.4 8.0.5 8.1.0 8.1.0-beta.1 8.1.0-rc.1 8.1.0-rc.2 8.1.1 8.1.2 8.1.3 8.1.4 8.2.0 8.2.0-beta.1 8.2.0-rc.1 8.2.0-rc.2 8.2.1 8.2.2 8.2.3 8.2.4 8.2.5 8.3.0 8.3.0-beta.1 8.3.0-rc.1 8.3.0-rc.2 8.3.1 8.3.2 8.3.3 8.3.4 8.4.0 8.4.0-beta.1 8.4.0-rc.1 8.4.1 8.4.2 8.4.3 8.5.0 8.5.0-beta.1 8.5.0-rc.1 8.5.1 8.5.2 8.5.3 8.5.4 8.5.5 8.6.0 8.6.0-beta.1 8.6.0-rc.1 8.6.1 8.6.2 8.6.3 8.6.4 8.7.0 8.7.0-beta.1 8.7.0-beta.2 8.7.0-rc.1 8.7.1 8.7.2 8.7.3 8.8.0 8.8.0-beta.1 8.8.0-rc.1 8.8.1 8.8.2 8.8.3 8.8.4 8.8.5 8.8.6 8.8.7 8.9.0 8.9.0-beta.1 8.9.0-rc.1 8.9.1 8.9.2 8.9.3 8.9.4 8.9.5 9.0.0 9.0.0-beta.1 9.0.0-beta.2 9.0.0-rc.1 9.0.1 9.0.2 9.0.3 9.0.4 9.1.0 9.1.0-beta.1 9.1.0-rc.1 9.1.1 9.1.2 9.1.3 9.1.4 9.1.5 9.1.6 9.2.0 9.2.0-beta.1 9.2.0-rc.1 9.2.1 9.2.2 9.2.3 9.2.4 9.2.5 9.3.0 9.3.0-beta.1 9.3.0-rc.1 9.3.1 9.3.2 9.3.3 9.3.4 9.3.5 9.3.6 9.4.0 9.4.0-beta.1 9.4.0-beta.2 9.4.0-rc.1 9.4.0-rc.2 9.4.0-rc.3 9.4.0-rc.4 9.4.1 9.4.2 9.4.3 9.4.4 9.4.5 9.5.0 9.5.0-beta.1 9.5.0-beta.2 9.5.0-rc.1 9.5.1 9.5.2 9.5.3 9.5.4 9.6.0 9.6.0-beta.1 9.6.0-beta.2 9.6.0-rc.1 9.6.1 9.6.2 9.6.3 9.6.4 9.7.0 9.7.0-beta.1 9.7.0-rc.1 9.7.1 9.7.2 9.7.3 9.8.0 9.8.0-beta.1 9.8.0-rc.1 9.8.1 9.8.2 9.8.3 9.8.4 9.8.5 9.8.6 9.8.7 9.9.0 9.9.0-beta.1 9.9.0-rc.1 9.9.1 9.9.2 9.9.3 9.9.4 9.9.5 9.9.6 9.9.7 3.7.3 7.1.2 3.8.0 7.2.0 3.8.0-beta.1 7.2.0-beta.1 3.8.0-rc.1 7.2.0-beta.2 3.8.0-rc.2 7.2.0-rc.1 3.8.1 7.2.0-rc.2 3.8.2 7.2.1 3.8.3 7.2.2 3.9.0 7.2.3 3.9.0-beta.1 7.2.4 3.9.0-beta.2 7.3.0 3.9.0-rc.1 7.3.0-beta.1 3.9.0-rc.2 7.3.0-beta.2 3.9.0-rc.3 7.3.0-rc.1 3.9.0-rc.4 7.3.0-rc.2 3.9.1 7.3.1 3.9.2 7.4.0 3.9.3 7.4.0-beta.1 3.9.4 7.4.0-beta.2 3.9.5 7.4.0-rc.1 4.0.0 7.4.0-rc.2 4.0.0-beta.1 7.4.1 4.0.0-rc.1 7.4.2 4.0.0-rc.2 7.5.0 4.0.1 7.5.0-beta.1 4.0.2 7.5.0-beta.2 4.0.3 7.5.0-rc.1 4.0.4 7.5.1 4.1.0 7.5.2 4.1.0-beta.1 7.6.0 4.1.0-beta.2 7.6.0-beta.1 4.1.0-rc.1 7.6.0-beta.2 4.1.0-rc.2 7.6.0-rc.1 4.1.1 7.6.0-rc.2 4.1.2 7.6.0-rc.3 4.1.3 7.6.1 4.1.4 7.6.2 4.2.0 7.7.0 4.2.0-RC.1 7.7.0-beta.1 4.2.0-RC.2 7.7.0-beta.2 4.2.0-beta.1 7.7.0-rc.1 4.2.1 7.7.1 4.2.2 7.7.2 4.2.3 7.7.3 4.2.4 7.8.0 4.2.5 4.3.0 4.3.0-beta.1 4.3.0-rc.1 4.3.0-rc.2 4.3.0-rc.3 4.3.1 4.3.2 4.3.3 4.3.4 4.3.5 4.3.6 4.4.0 4.4.0-beta.1 4.4.0-rc.1 4.4.1 4.4.2 4.4.3 4.4.4 4.5.0 4.5.0-beta.1 4.5.0-rc.1 4.5.0-rc.3 4.5.1 4.5.2 4.5.3 4.5.4 4.5.5 4.6.0 4.6.0-beta.1 4.6.0-rc.1 4.6.1 4.6.2 4.6.3 4.6.4 4.6.5 4.7.0 4.7.0-beta.1 4.7.0-beta.2 4.7.0-rc.1 4.7.1 4.7.1-beta.1 4.7.2 4.7.3 4.7.4 4.8.0 4.8.0-beta.1 4.8.0-rc.1 4.8.0-rc.2 4.8.1 4.8.2 4.8.3 4.9.0 4.9.0-beta.1 4.9.0-rc.1 4.9.0-rc.2 4.9.1 4.9.2 4.9.3 4.9.4 4.9.5 5.0.0 5.0.0-beta.1 5.0.0-beta.2 5.0.0-rc.1 5.0.0-rc.2 5.0.0-rc.3 5.0.1 5.0.2 5.0.3 5.1.0 5.1.0-beta.1 5.1.0-rc.1 trunk 5.1.1 10.0.0 5.1.2 10.0.0-rc.1 5.1.3 10.0.0-rc.2 5.2.0 10.0.1 5.2.0-beta.1 10.0.2 5.2.0-rc.1 10.0.3 5.2.0-rc.2 10.0.4 5.2.1 10.0.5 5.2.2 10.0.6 5.2.3 10.1.0 5.2.4 10.1.0-rc.1 5.2.5 10.1.0-rc.2 5.3.0 10.1.0-rc.3 5.3.0-beta.1 10.1.0-rc.4 5.3.0-rc.1 10.1.1 5.3.0-rc.2 10.1.2 5.3.1 10.1.3 5.3.2 10.1.4 5.3.3 10.2.0 5.4.0 10.2.0-beta.1 5.4.0-beta.1 10.2.0-beta.2 5.4.0-rc.1 10.2.0-rc.1 5.4.1 10.2.1 5.4.2 10.2.2 5.4.3 10.2.3 5.4.4 10.2.4 5.4.5 10.3.0 5.5.0 10.3.0-beta.1 5.5.0-beta.1 10.3.0-beta.2 5.5.0-rc.1 10.3.0-rc.1 5.5.0-rc.2 10.3.0-rc.2 5.5.1 10.3.1 5.5.2 10.3.2 5.5.3 10.3.3 5.5.4 10.3.4 5.5.5 10.3.5 5.6.0 10.3.6 5.6.0-beta.1 10.3.7 5.6.0-rc.1 10.3.8 5.6.0-rc.2 10.4.0 5.6.1 10.4.0-beta.1 5.6.2 10.4.0-beta.2 5.6.3 10.4.0-rc.1 5.7.0 10.4.1 5.7.0-beta.1 10.4.2 5.7.0-rc.1 10.4.3 5.7.1 10.4.4 5.7.2 10.5.0 5.7.3 10.5.0-beta.1 5.8.0 10.5.0-beta.2 5.8.0-beta.1 10.5.0-rc.1 5.8.0-beta.2 10.5.0-rc.2 5.8.0-rc.1 10.5.0-rc.3 5.8.1 10.5.1 5.8.2 10.5.2 5.9.0 10.5.3 5.9.0-beta.1 10.6.0 5.9.0-rc.1 10.6.0-beta.1 5.9.0-rc.2 10.6.0-beta.2 5.9.1 10.6.0-rc.1 5.9.2 10.6.1 6.0.0 10.6.2 6.0.0-beta.1 10.7.0 6.0.0-rc.1 10.7.0-beta.1 6.0.1 10.7.0-beta.2 6.0.2 10.7.0-rc.1 6.1.0 3.0.0 6.1.0-beta.1 3.0.1 6.1.0-rc.1 3.0.2 6.1.0-rc.2 3.0.3 6.1.1 3.0.4 6.1.2 3.0.5 6.1.3 3.0.6 6.2.0 3.0.7 6.2.0-beta.1 3.0.8 6.2.0-rc.1 3.0.9 6.2.0-rc.2 3.1.0 6.2.1 3.1.1 6.2.2 3.1.2 6.2.3 3.2.0 6.3.0 3.2.1 6.3.0-beta.1 3.2.2 6.3.0-rc.1 3.2.3 6.3.0-rc.2 3.2.4 6.3.1 3.2.5 6.3.2 3.2.6 6.4.0 3.3.0 6.4.0-beta.1 3.3.1 6.4.0-rc.1 3.3.2 6.4.1 3.3.2-rc.1 6.4.2 3.3.3 6.5.0 3.3.4 6.5.0-beta.1 3.3.5 6.5.0-rc.1 3.3.6 6.5.0-rc.2 3.4.0 6.5.1 3.4.0-beta.1 6.5.2 3.4.0-rc.2 6.6.0 3.4.1 6.6.0-beta.1 3.4.2 6.6.0-rc.1 3.4.3 6.6.0-rc.2 3.4.4 6.6.1 3.4.5 6.6.2 3.4.6 6.7.0 3.4.7 6.7.0-beta.1 3.4.8 6.7.0-beta.2 3.5.0 6.7.0-rc.1 3.5.0-beta.1 6.7.1 3.5.0-rc.1 6.8.0 3.5.0-rc.2 6.8.0-beta.1 3.5.1 6.8.0-beta.2 3.5.10 6.8.0-rc.1 3.5.2 6.8.1 3.5.3 6.8.2 3.5.4 6.8.3 3.5.5 6.9.0 3.5.6 6.9.0-beta.1 3.5.7 6.9.0-beta.2 3.5.8 6.9.0-rc.1 3.5.9 6.9.1 3.6.0 6.9.2 3.6.0-beta.1 6.9.3 3.6.0-rc.1 6.9.4 3.6.0-rc.2 6.9.5 3.6.0-rc.3 7.0.0 3.6.1 7.0.0-beta.1 3.6.2 7.0.0-beta.2 3.6.3 7.0.0-beta.3 3.6.4 7.0.0-rc.1 3.6.5 7.0.0-rc.2 3.6.6 7.0.1 3.6.7 7.0.2 3.7.0 7.1.0 3.7.0-beta.1 7.1.0-beta.1 3.7.0-rc.1 7.1.0-beta.2 3.7.0-rc.2 7.1.0-rc.1 3.7.1 7.1.0-rc.2 3.7.2 7.1.1
woocommerce / includes / class-wc-session-handler.php
woocommerce / includes Last commit date
abstracts 2 years ago admin 2 years ago blocks 5 years ago cli 3 years ago customizer 3 years ago data-stores 2 years ago emails 2 years ago export 2 years ago gateways 2 years ago import 3 years ago integrations 4 years ago interfaces 3 years ago legacy 2 years ago libraries 3 years ago log-handlers 4 years ago payment-tokens 5 years ago queue 4 years ago react-admin 2 years ago rest-api 3 years ago shipping 2 years ago shortcodes 2 years ago theme-support 3 years ago tracks 2 years ago traits 5 years ago walkers 5 years ago wccom-site 2 years ago widgets 3 years ago class-wc-ajax.php 2 years ago class-wc-api.php 4 years ago class-wc-auth.php 4 years ago class-wc-autoloader.php 5 years ago class-wc-background-emailer.php 5 years ago class-wc-background-updater.php 5 years ago class-wc-breadcrumb.php 5 years ago class-wc-cache-helper.php 3 years ago class-wc-cart-fees.php 5 years ago class-wc-cart-session.php 3 years ago class-wc-cart-totals.php 2 years ago class-wc-cart.php 3 years ago class-wc-checkout.php 3 years ago class-wc-cli.php 3 years ago class-wc-comments.php 3 years ago class-wc-countries.php 2 years ago class-wc-coupon.php 3 years ago class-wc-customer-download-log.php 5 years ago class-wc-customer-download.php 4 years ago class-wc-customer.php 3 years ago class-wc-data-exception.php 8 years ago class-wc-data-store.php 3 years ago class-wc-datetime.php 4 years ago class-wc-deprecated-action-hooks.php 8 years ago class-wc-deprecated-filter-hooks.php 3 years ago class-wc-discounts.php 3 years ago class-wc-download-handler.php 3 years ago class-wc-emails.php 3 years ago class-wc-embed.php 5 years ago class-wc-form-handler.php 3 years ago class-wc-frontend-scripts.php 3 years ago class-wc-geo-ip.php 4 years ago class-wc-geolite-integration.php 6 years ago class-wc-geolocation.php 3 years ago class-wc-https.php 2 years ago class-wc-install.php 2 years ago class-wc-integrations.php 5 years ago class-wc-log-levels.php 5 years ago class-wc-logger.php 4 years ago class-wc-meta-data.php 4 years ago class-wc-order-factory.php 3 years ago class-wc-order-item-coupon.php 4 years ago class-wc-order-item-fee.php 4 years ago class-wc-order-item-meta.php 4 years ago class-wc-order-item-product.php 4 years ago class-wc-order-item-shipping.php 4 years ago class-wc-order-item-tax.php 4 years ago class-wc-order-item.php 4 years ago class-wc-order-query.php 4 years ago class-wc-order-refund.php 2 years ago class-wc-order.php 2 years ago class-wc-payment-gateways.php 4 years ago class-wc-payment-tokens.php 3 years ago class-wc-post-data.php 3 years ago class-wc-post-types.php 3 years ago class-wc-privacy-background-process.php 5 years ago class-wc-privacy-erasers.php 4 years ago class-wc-privacy-exporters.php 4 years ago class-wc-privacy.php 2 years ago class-wc-product-attribute.php 4 years ago class-wc-product-download.php 4 years ago class-wc-product-external.php 5 years ago class-wc-product-factory.php 5 years ago class-wc-product-grouped.php 8 years ago class-wc-product-query.php 5 years ago class-wc-product-simple.php 3 years ago class-wc-product-variable.php 3 years ago class-wc-product-variation.php 4 years ago class-wc-query.php 3 years ago class-wc-rate-limiter.php 4 years ago class-wc-regenerate-images-request.php 3 years ago class-wc-regenerate-images.php 3 years ago class-wc-register-wp-admin-settings.php 4 years ago class-wc-rest-authentication.php 3 years ago class-wc-rest-exception.php 5 years ago class-wc-session-handler.php 2 years ago class-wc-shipping-rate.php 3 years ago class-wc-shipping-zone.php 5 years ago class-wc-shipping-zones.php 5 years ago class-wc-shipping.php 4 years ago class-wc-shortcodes.php 5 years ago class-wc-structured-data.php 3 years ago class-wc-tax.php 2 years ago class-wc-template-loader.php 2 years ago class-wc-tracker.php 2 years ago class-wc-validation.php 3 years ago class-wc-webhook.php 3 years ago class-woocommerce.php 2 years ago wc-account-functions.php 3 years ago wc-attribute-functions.php 3 years ago wc-cart-functions.php 3 years ago wc-conditional-functions.php 3 years ago wc-core-functions.php 2 years ago wc-coupon-functions.php 3 years ago wc-deprecated-functions.php 3 years ago wc-formatting-functions.php 2 years ago wc-notice-functions.php 3 years ago wc-order-functions.php 2 years ago wc-order-item-functions.php 3 years ago wc-page-functions.php 2 years ago wc-product-functions.php 3 years ago wc-rest-functions.php 3 years ago wc-stock-functions.php 3 years ago wc-template-functions.php 2 years ago wc-template-hooks.php 5 years ago wc-term-functions.php 3 years ago wc-update-functions.php 2 years ago wc-user-functions.php 2 years ago wc-webhook-functions.php 4 years ago wc-widget-functions.php 5 years ago
class-wc-session-handler.php
506 lines
1 <?php
2 /**
3 * Handle data for the current customers session.
4 * Implements the WC_Session abstract class.
5 *
6 * From 2.5 this uses a custom table for session storage. Based on https://github.com/kloon/woocommerce-large-sessions.
7 *
8 * @class WC_Session_Handler
9 * @version 2.5.0
10 * @package WooCommerce\Classes
11 */
12
13 use Automattic\Jetpack\Constants;
14
15 defined( 'ABSPATH' ) || exit;
16
17 /**
18 * Session handler class.
19 */
20 class WC_Session_Handler extends WC_Session {
21
22 /**
23 * Cookie name used for the session.
24 *
25 * @var string cookie name
26 */
27 protected $_cookie;
28
29 /**
30 * Stores session expiry.
31 *
32 * @var string session due to expire timestamp
33 */
34 protected $_session_expiring;
35
36 /**
37 * Stores session due to expire timestamp.
38 *
39 * @var string session expiration timestamp
40 */
41 protected $_session_expiration;
42
43 /**
44 * True when the cookie exists.
45 *
46 * @var bool Based on whether a cookie exists.
47 */
48 protected $_has_cookie = false;
49
50 /**
51 * Table name for session data.
52 *
53 * @var string Custom session table name
54 */
55 protected $_table;
56
57 /**
58 * Constructor for the session class.
59 */
60 public function __construct() {
61 $this->_cookie = apply_filters( 'woocommerce_cookie', 'wp_woocommerce_session_' . COOKIEHASH );
62 $this->_table = $GLOBALS['wpdb']->prefix . 'woocommerce_sessions';
63 }
64
65 /**
66 * Init hooks and session data.
67 *
68 * @since 3.3.0
69 */
70 public function init() {
71 $this->init_session_cookie();
72
73 add_action( 'woocommerce_set_cart_cookies', array( $this, 'set_customer_session_cookie' ), 10 );
74 add_action( 'shutdown', array( $this, 'save_data' ), 20 );
75 add_action( 'wp_logout', array( $this, 'destroy_session' ) );
76
77 if ( ! is_user_logged_in() ) {
78 add_filter( 'nonce_user_logged_out', array( $this, 'maybe_update_nonce_user_logged_out' ), 10, 2 );
79 }
80 }
81
82 /**
83 * Setup cookie and customer ID.
84 *
85 * @since 3.6.0
86 */
87 public function init_session_cookie() {
88 $cookie = $this->get_session_cookie();
89
90 if ( $cookie ) {
91 // Customer ID will be an MD5 hash id this is a guest session.
92 $this->_customer_id = $cookie[0];
93 $this->_session_expiration = $cookie[1];
94 $this->_session_expiring = $cookie[2];
95 $this->_has_cookie = true;
96 $this->_data = $this->get_session_data();
97
98 if ( ! $this->is_session_cookie_valid() ) {
99 $this->destroy_session();
100 $this->set_session_expiration();
101 }
102
103 // If the user logs in, update session.
104 if ( is_user_logged_in() && strval( get_current_user_id() ) !== $this->_customer_id ) {
105 $guest_session_id = $this->_customer_id;
106 $this->_customer_id = strval( get_current_user_id() );
107 $this->_dirty = true;
108 $this->save_data( $guest_session_id );
109 $this->set_customer_session_cookie( true );
110 }
111
112 // Update session if its close to expiring.
113 if ( time() > $this->_session_expiring ) {
114 $this->set_session_expiration();
115 $this->update_session_timestamp( $this->_customer_id, $this->_session_expiration );
116 }
117 } else {
118 $this->set_session_expiration();
119 $this->_customer_id = $this->generate_customer_id();
120 $this->_data = $this->get_session_data();
121 }
122 }
123
124 /**
125 * Checks if session cookie is expired, or belongs to a logged out user.
126 *
127 * @return bool Whether session cookie is valid.
128 */
129 private function is_session_cookie_valid() {
130 // If session is expired, session cookie is invalid.
131 if ( time() > $this->_session_expiration ) {
132 return false;
133 }
134
135 // If user has logged out, session cookie is invalid.
136 if ( ! is_user_logged_in() && ! $this->is_customer_guest( $this->_customer_id ) ) {
137 return false;
138 }
139
140 // Session from a different user is not valid. (Although from a guest user will be valid)
141 if ( is_user_logged_in() && ! $this->is_customer_guest( $this->_customer_id ) && strval( get_current_user_id() ) !== $this->_customer_id ) {
142 return false;
143 }
144
145 return true;
146 }
147
148 /**
149 * Sets the session cookie on-demand (usually after adding an item to the cart).
150 *
151 * Since the cookie name (as of 2.1) is prepended with wp, cache systems like batcache will not cache pages when set.
152 *
153 * Warning: Cookies will only be set if this is called before the headers are sent.
154 *
155 * @param bool $set Should the session cookie be set.
156 */
157 public function set_customer_session_cookie( $set ) {
158 if ( $set ) {
159 $to_hash = $this->_customer_id . '|' . $this->_session_expiration;
160 $cookie_hash = hash_hmac( 'md5', $to_hash, wp_hash( $to_hash ) );
161 $cookie_value = $this->_customer_id . '||' . $this->_session_expiration . '||' . $this->_session_expiring . '||' . $cookie_hash;
162 $this->_has_cookie = true;
163
164 if ( ! isset( $_COOKIE[ $this->_cookie ] ) || $_COOKIE[ $this->_cookie ] !== $cookie_value ) {
165 wc_setcookie( $this->_cookie, $cookie_value, $this->_session_expiration, $this->use_secure_cookie(), true );
166 }
167 }
168 }
169
170 /**
171 * Should the session cookie be secure?
172 *
173 * @since 3.6.0
174 * @return bool
175 */
176 protected function use_secure_cookie() {
177 return apply_filters( 'wc_session_use_secure_cookie', wc_site_is_https() && is_ssl() );
178 }
179
180 /**
181 * Return true if the current user has an active session, i.e. a cookie to retrieve values.
182 *
183 * @return bool
184 */
185 public function has_session() {
186 return isset( $_COOKIE[ $this->_cookie ] ) || $this->_has_cookie || is_user_logged_in(); // @codingStandardsIgnoreLine.
187 }
188
189 /**
190 * Set session expiration.
191 */
192 public function set_session_expiration() {
193 $this->_session_expiring = time() + intval( apply_filters( 'wc_session_expiring', 60 * 60 * 47 ) ); // 47 Hours.
194 $this->_session_expiration = time() + intval( apply_filters( 'wc_session_expiration', 60 * 60 * 48 ) ); // 48 Hours.
195 }
196
197 /**
198 * Generate a unique customer ID for guests, or return user ID if logged in.
199 *
200 * Uses Portable PHP password hashing framework to generate a unique cryptographically strong ID.
201 *
202 * @return string
203 */
204 public function generate_customer_id() {
205 $customer_id = '';
206
207 if ( is_user_logged_in() ) {
208 $customer_id = strval( get_current_user_id() );
209 }
210
211 if ( empty( $customer_id ) ) {
212 require_once ABSPATH . 'wp-includes/class-phpass.php';
213 $hasher = new PasswordHash( 8, false );
214 $customer_id = 't_' . substr( md5( $hasher->get_random_bytes( 32 ) ), 2 );
215 }
216
217 return $customer_id;
218 }
219
220 /**
221 * Checks if this is an auto-generated customer ID.
222 *
223 * @param string|int $customer_id Customer ID to check.
224 *
225 * @return bool Whether customer ID is randomly generated.
226 */
227 private function is_customer_guest( $customer_id ) {
228 $customer_id = strval( $customer_id );
229
230 if ( empty( $customer_id ) ) {
231 return true;
232 }
233
234 if ( 't_' === substr( $customer_id, 0, 2 ) ) {
235 return true;
236 }
237
238 /**
239 * Legacy checks. This is to handle sessions that were created from a previous release.
240 * Maybe we can get rid of them after a few releases.
241 */
242
243 // Almost all random $customer_ids will have some letters in it, while all actual ids will be integers.
244 if ( strval( (int) $customer_id ) !== $customer_id ) {
245 return true;
246 }
247
248 // Performance hack to potentially save a DB query, when same user as $customer_id is logged in.
249 if ( is_user_logged_in() && strval( get_current_user_id() ) === $customer_id ) {
250 return false;
251 } else {
252 $customer = new WC_Customer( $customer_id );
253
254 if ( 0 === $customer->get_id() ) {
255 return true;
256 }
257 }
258
259 return false;
260 }
261
262 /**
263 * Get session unique ID for requests if session is initialized or user ID if logged in.
264 * Introduced to help with unit tests.
265 *
266 * @since 5.3.0
267 * @return string
268 */
269 public function get_customer_unique_id() {
270 $customer_id = '';
271
272 if ( $this->has_session() && $this->_customer_id ) {
273 $customer_id = $this->_customer_id;
274 } elseif ( is_user_logged_in() ) {
275 $customer_id = (string) get_current_user_id();
276 }
277
278 return $customer_id;
279 }
280
281 /**
282 * Get the session cookie, if set. Otherwise return false.
283 *
284 * Session cookies without a customer ID are invalid.
285 *
286 * @return bool|array
287 */
288 public function get_session_cookie() {
289 $cookie_value = isset( $_COOKIE[ $this->_cookie ] ) ? wp_unslash( $_COOKIE[ $this->_cookie ] ) : false; // @codingStandardsIgnoreLine.
290
291 if ( empty( $cookie_value ) || ! is_string( $cookie_value ) ) {
292 return false;
293 }
294
295 $parsed_cookie = explode( '||', $cookie_value );
296
297 if ( count( $parsed_cookie ) < 4 ) {
298 return false;
299 }
300
301 list( $customer_id, $session_expiration, $session_expiring, $cookie_hash ) = $parsed_cookie;
302
303 if ( empty( $customer_id ) ) {
304 return false;
305 }
306
307 // Validate hash.
308 $to_hash = $customer_id . '|' . $session_expiration;
309 $hash = hash_hmac( 'md5', $to_hash, wp_hash( $to_hash ) );
310
311 if ( empty( $cookie_hash ) || ! hash_equals( $hash, $cookie_hash ) ) {
312 return false;
313 }
314
315 return array( $customer_id, $session_expiration, $session_expiring, $cookie_hash );
316 }
317
318 /**
319 * Get session data.
320 *
321 * @return array
322 */
323 public function get_session_data() {
324 return $this->has_session() ? (array) $this->get_session( $this->_customer_id, array() ) : array();
325 }
326
327 /**
328 * Gets a cache prefix. This is used in session names so the entire cache can be invalidated with 1 function call.
329 *
330 * @return string
331 */
332 private function get_cache_prefix() {
333 return WC_Cache_Helper::get_cache_prefix( WC_SESSION_CACHE_GROUP );
334 }
335
336 /**
337 * Save data and delete guest session.
338 *
339 * @param int $old_session_key session ID before user logs in.
340 */
341 public function save_data( $old_session_key = 0 ) {
342 // Dirty if something changed - prevents saving nothing new.
343 if ( $this->_dirty && $this->has_session() ) {
344 global $wpdb;
345
346 $wpdb->query(
347 $wpdb->prepare(
348 "INSERT INTO {$wpdb->prefix}woocommerce_sessions (`session_key`, `session_value`, `session_expiry`) VALUES (%s, %s, %d)
349 ON DUPLICATE KEY UPDATE `session_value` = VALUES(`session_value`), `session_expiry` = VALUES(`session_expiry`)",
350 $this->_customer_id,
351 maybe_serialize( $this->_data ),
352 $this->_session_expiration
353 )
354 );
355
356 wp_cache_set( $this->get_cache_prefix() . $this->_customer_id, $this->_data, WC_SESSION_CACHE_GROUP, $this->_session_expiration - time() );
357 $this->_dirty = false;
358 if ( get_current_user_id() != $old_session_key && ! is_object( get_user_by( 'id', $old_session_key ) ) ) {
359 $this->delete_session( $old_session_key );
360 }
361 }
362 }
363
364 /**
365 * Destroy all session data.
366 */
367 public function destroy_session() {
368 $this->delete_session( $this->_customer_id );
369 $this->forget_session();
370 }
371
372 /**
373 * Forget all session data without destroying it.
374 */
375 public function forget_session() {
376 wc_setcookie( $this->_cookie, '', time() - YEAR_IN_SECONDS, $this->use_secure_cookie(), true );
377
378 if ( ! is_admin() ) {
379 include_once WC_ABSPATH . 'includes/wc-cart-functions.php';
380
381 wc_empty_cart();
382 }
383
384 $this->_data = array();
385 $this->_dirty = false;
386 $this->_customer_id = $this->generate_customer_id();
387 }
388
389 /**
390 * When a user is logged out, ensure they have a unique nonce by using the customer/session ID.
391 *
392 * @deprecated 5.3.0
393 * @param int $uid User ID.
394 * @return int|string
395 */
396 public function nonce_user_logged_out( $uid ) {
397 wc_deprecated_function( 'WC_Session_Handler::nonce_user_logged_out', '5.3', 'WC_Session_Handler::maybe_update_nonce_user_logged_out' );
398
399 return $this->has_session() && $this->_customer_id ? $this->_customer_id : $uid;
400 }
401
402 /**
403 * When a user is logged out, ensure they have a unique nonce to manage cart and more using the customer/session ID.
404 * This filter runs everything `wp_verify_nonce()` and `wp_create_nonce()` gets called.
405 *
406 * @since 5.3.0
407 * @param int $uid User ID.
408 * @param string $action The nonce action.
409 * @return int|string
410 */
411 public function maybe_update_nonce_user_logged_out( $uid, $action ) {
412 if ( Automattic\WooCommerce\Utilities\StringUtil::starts_with( $action, 'woocommerce' ) ) {
413 return $this->has_session() && $this->_customer_id ? $this->_customer_id : $uid;
414 }
415
416 return $uid;
417 }
418
419 /**
420 * Cleanup session data from the database and clear caches.
421 */
422 public function cleanup_sessions() {
423 global $wpdb;
424
425 $wpdb->query( $wpdb->prepare( "DELETE FROM $this->_table WHERE session_expiry < %d", time() ) ); // @codingStandardsIgnoreLine.
426
427 if ( class_exists( 'WC_Cache_Helper' ) ) {
428 WC_Cache_Helper::invalidate_cache_group( WC_SESSION_CACHE_GROUP );
429 }
430 }
431
432 /**
433 * Returns the session.
434 *
435 * @param string $customer_id Customer ID.
436 * @param mixed $default Default session value.
437 * @return string|array
438 */
439 public function get_session( $customer_id, $default = false ) {
440 global $wpdb;
441
442 if ( Constants::is_defined( 'WP_SETUP_CONFIG' ) ) {
443 return false;
444 }
445
446 // Try to get it from the cache, it will return false if not present or if object cache not in use.
447 $value = wp_cache_get( $this->get_cache_prefix() . $customer_id, WC_SESSION_CACHE_GROUP );
448
449 if ( false === $value ) {
450 $value = $wpdb->get_var( $wpdb->prepare( "SELECT session_value FROM $this->_table WHERE session_key = %s", $customer_id ) ); // @codingStandardsIgnoreLine.
451
452 if ( is_null( $value ) ) {
453 $value = $default;
454 }
455
456 $cache_duration = $this->_session_expiration - time();
457 if ( 0 < $cache_duration ) {
458 wp_cache_add( $this->get_cache_prefix() . $customer_id, $value, WC_SESSION_CACHE_GROUP, $cache_duration );
459 }
460 }
461
462 return maybe_unserialize( $value );
463 }
464
465 /**
466 * Delete the session from the cache and database.
467 *
468 * @param int $customer_id Customer ID.
469 */
470 public function delete_session( $customer_id ) {
471 global $wpdb;
472
473 wp_cache_delete( $this->get_cache_prefix() . $customer_id, WC_SESSION_CACHE_GROUP );
474
475 $wpdb->delete(
476 $this->_table,
477 array(
478 'session_key' => $customer_id,
479 )
480 );
481 }
482
483 /**
484 * Update the session expiry timestamp.
485 *
486 * @param string $customer_id Customer ID.
487 * @param int $timestamp Timestamp to expire the cookie.
488 */
489 public function update_session_timestamp( $customer_id, $timestamp ) {
490 global $wpdb;
491
492 $wpdb->update(
493 $this->_table,
494 array(
495 'session_expiry' => $timestamp,
496 ),
497 array(
498 'session_key' => $customer_id,
499 ),
500 array(
501 '%d',
502 )
503 );
504 }
505 }
506