report_data ) ) { $this->query_report_data(); } return $this->report_data; } /** * Get the number of periods each subscription has between sign-up and end. * * This function uses a new "living" and "age" terminology to refer to the time between when a subscription * is created and when it ends (i.e. expires or is cancelled). The function can't use "active" because the * subscription may not have been active all of that time. Instead, it may have been on-hold for part of it. * * @since 2.1 * @return void */ private function query_report_data() { $this->report_data = new stdClass; // First, let's find the age of the longest living subscription in days $oldest_subscription_age_in_days = $this->get_max_subscription_age_in_days(); // Now determine what interval to use based on that length if ( $oldest_subscription_age_in_days > 365 ) { $this->report_data->interval_period = 'month'; } elseif ( $oldest_subscription_age_in_days > 182 ) { $this->report_data->interval_period = 'week'; } else { $this->report_data->interval_period = 'day'; } // Use the number of days in the chosen interval period to determine how many periods between each start/end date $days_in_interval_period = wcs_get_days_in_cycle( $this->report_data->interval_period, 1 ); // Find the number of these periods in the longest living subscription $oldest_subscription_age = floor( $oldest_subscription_age_in_days / $days_in_interval_period ); // Now get all subscriptions, not just those that have ended, and find out how long they have lived (or if they haven't ended yet, consider them as being alive for one period longer than the longest living subsription) $subscription_ages = $this->fetch_subscriptions_ages( $days_in_interval_period, $oldest_subscription_age ); // Set initial values for the report data. $this->report_data->total_subscriptions = absint( array_sum( wp_list_pluck( $subscription_ages, 'count' ) ) ); $this->report_data->unended_subscriptions = $this->report_data->total_subscriptions; $this->report_data->living_subscriptions = array(); // At day zero, no subscriptions have ended $this->report_data->living_subscriptions[0] = $this->report_data->total_subscriptions; // Fill out the report data to provide a smooth curve for ( $i = 0; $i <= $oldest_subscription_age; $i++ ) { // We want to push the the array keys ahead by one to make sure out the 0 index represents the total subscriptions $periods_after_sign_up = $i + 1; // Only reduce the number of living subscriptions when we have a new number for a given period as that indicates a new set of subscriptions have ended if ( isset( $subscription_ages[ $i ] ) ) { $this->report_data->living_subscriptions[ $periods_after_sign_up ] = $this->report_data->living_subscriptions[ $i ] - $subscription_ages[ $i ]->count; $this->report_data->unended_subscriptions -= $subscription_ages[ $i ]->count; } else { $this->report_data->living_subscriptions[ $periods_after_sign_up ] = $this->report_data->living_subscriptions[ $i ]; } } } /** * Get the age of the longest living subscription in days. * * @return int */ private function get_max_subscription_age_in_days() { global $wpdb; $end_date_meta_key = wcs_get_date_meta_key( 'end' ); if ( wcs_is_custom_order_tables_usage_enabled() ) { $query = $wpdb->prepare( "SELECT MAX(DATEDIFF(CAST(meta.meta_value AS DATETIME),orders.date_created_gmt)) as age_in_days FROM {$wpdb->prefix}wc_orders orders LEFT JOIN {$wpdb->prefix}wc_orders_meta meta ON orders.ID = meta.order_id WHERE orders.type = 'shop_subscription' AND meta.meta_key = %s AND meta.meta_value <> '0' ORDER BY age_in_days DESC LIMIT 1", $end_date_meta_key ); } else { $query = $wpdb->prepare( "SELECT MAX(DATEDIFF(CAST(postmeta.meta_value AS DATETIME),posts.post_date_gmt)) as age_in_days FROM {$wpdb->prefix}posts posts LEFT JOIN {$wpdb->prefix}postmeta postmeta ON posts.ID = postmeta.post_id WHERE posts.post_type = 'shop_subscription' AND postmeta.meta_key = %s AND postmeta.meta_value <> '0' ORDER BY age_in_days DESC LIMIT 1", $end_date_meta_key ); } return $wpdb->get_var( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- The query is prepared above. } /** * Fetch the number of periods each subscription has between creating and ending. * * @param int $days_in_interval_period * @param int $oldest_subscription_age * @return array */ private function fetch_subscriptions_ages( $days_in_interval_period, $oldest_subscription_age ) { global $wpdb; $end_date_meta_key = wcs_get_date_meta_key( 'end' ); $cancelled_date_meta_key = wcs_get_date_meta_key( 'cancelled' ); if ( wcs_is_custom_order_tables_usage_enabled() ) { $query = $wpdb->prepare( "SELECT IF(COALESCE(cancelled_date.meta_value,end_date.meta_value) <> '0',CEIL(DATEDIFF(CAST(COALESCE(cancelled_date.meta_value,end_date.meta_value) AS DATETIME),orders.date_created_gmt)/%d),%d) as periods_active, COUNT(orders.ID) as count FROM {$wpdb->prefix}wc_orders orders LEFT JOIN {$wpdb->prefix}wc_orders_meta cancelled_date ON orders.ID = cancelled_date.order_id AND cancelled_date.meta_key = %s AND cancelled_date.meta_value <> '0' LEFT JOIN {$wpdb->prefix}wc_orders_meta end_date ON orders.ID = end_date.order_id AND end_date.meta_key = %s WHERE orders.type = 'shop_subscription' AND orders.status NOT IN( 'wc-pending', 'trash' ) GROUP BY periods_active ORDER BY periods_active ASC", $days_in_interval_period, ( $oldest_subscription_age + 1 ), // Consider living subscriptions as being alive for one period longer than the longest living subscription $cancelled_date_meta_key, // If a subscription has a cancelled date, use that to determine a more accurate lifetime $end_date_meta_key // Otherwise, we want to use the end date for subscriptions that have expired ); } else { $query = $wpdb->prepare( "SELECT IF(COALESCE(cancelled_date.meta_value,end_date.meta_value) <> '0',CEIL(DATEDIFF(CAST(COALESCE(cancelled_date.meta_value,end_date.meta_value) AS DATETIME),posts.post_date_gmt)/%d),%d) as periods_active, COUNT(posts.ID) as count FROM {$wpdb->prefix}posts posts LEFT JOIN {$wpdb->prefix}postmeta cancelled_date ON posts.ID = cancelled_date.post_id AND cancelled_date.meta_key = %s AND cancelled_date.meta_value <> '0' LEFT JOIN {$wpdb->prefix}postmeta end_date ON posts.ID = end_date.post_id AND end_date.meta_key = %s WHERE posts.post_type = 'shop_subscription' AND posts.post_status NOT IN( 'wc-pending', 'trash' ) GROUP BY periods_active ORDER BY periods_active ASC", $days_in_interval_period, ( $oldest_subscription_age + 1 ), // Consider living subscriptions as being alive for one period longer than the longest living subscription $cancelled_date_meta_key, // If a subscription has a cancelled date, use that to determine a more accurate lifetime $end_date_meta_key // Otherwise, we want to use the end date for subscriptions that have expired ); } return $wpdb->get_results( $query, OBJECT_K ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- The query is prepared above. } /** * Output the report * * Use a custom report as we don't need the date filters provided by the WooCommerce html-report-by-date.php template. * * @since 2.1 */ public function output_report() { include( WC_Subscriptions_Plugin::instance()->get_plugin_directory( 'includes/admin/views/html-report-by-period.php' ) ); } /** * Output the HTML and JavaScript to plot the chart * * @since 2.1 */ public function get_main_chart() { $this->get_report_data(); $data_to_plot = array(); foreach ( $this->report_data->living_subscriptions as $periods_since_sign_up => $living_subscription_count ) { $data_to_plot[] = array( absint( $periods_since_sign_up ), absint( $living_subscription_count ), ); } $x_axes_label = ''; switch ( $this->report_data->interval_period ) { case 'day': $x_axes_label = _x( 'Number of days after sign-up', 'X axis label on retention rate graph', 'woocommerce-subscriptions' ); break; case 'week': $x_axes_label = _x( 'Number of weeks after sign-up', 'X axis label on retention rate graph', 'woocommerce-subscriptions' ); break; case 'month': $x_axes_label = _x( 'Number of months after sign-up', 'X axis label on retention rate graph', 'woocommerce-subscriptions' ); break; } ?>