HEX
Server: Apache/2.4.41 (Ubuntu)
System: Linux wordpress-ubuntu-s-2vcpu-4gb-fra1-01 5.4.0-169-generic #187-Ubuntu SMP Thu Nov 23 14:52:28 UTC 2023 x86_64
User: root (0)
PHP: 7.4.33
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: /var/www/shoetique/wp-content/plugins/googleanalytics/class/class-ga-stats.php
<?php
/**
 * Google Analytics hook class.
 *
 * @package GoogleAnalytics
 */

/**
 * Ga_Stats class
 *
 * Preparing request and parsing response from Google Analytics Reporting Api
 *
 * @author wle@adips.com
 * @version 1.0
 */
class Ga_Stats {

	/**
	 * Profile object.
	 *
	 * @var array Profile.
	 */
	private $profile = array();

	/**
	 * Primary class constructor.
	 *
	 * @access public
	 * @since 7.0.0
	 */
	public function __construct() {
	}

	/**
	 * Preparing query to get Analytics data
	 *
	 * @param string $query      Query type.
	 * @param int    $id_view    The Analytics view ID from which to retrieve data.
	 * @param string $date_range The start date for the query in the format YYYY-MM-DD or '7daysAgo'.
	 * @param string $metric     A metric expression.
	 * @param bool   $old        Use old query style.
	 *
	 * @return array Request query
	 */
	public static function get_query( $query, $id_view, $date_range = null, $metric = null, $old = false ) {
		if ( 'main_chart' === $query ) {
			return $old ? self::main_chart_query_old( $id_view, $date_range, $metric ) : self::main_chart_query(
				$id_view,
				$date_range,
				$metric
			);
		} elseif ( 'gender' === $query ) {
			return self::gender_chart_query( $id_view, $date_range, $metric );
		} elseif ( 'device' === $query ) {
			return self::device_chart_query( $id_view, $date_range, $metric );
		} elseif ( 'age' === $query ) {
			return self::age_chart_query( $id_view, $date_range, $metric );
		} elseif ( 'boxes' === $query ) {
			return self::boxes_query( $id_view );
		} elseif ( 'dashboard_boxes' === $query ) {
			return $old ? self::dashboard_boxes_query_old( $id_view, $date_range ) :
				self::dashboard_boxes_query( $id_view, $date_range );
		} elseif ( 'sources' === $query ) {
			return self::sources_query( $id_view, $date_range );
		} else {
			return array();
		}
	}

	/**
	 * Preparing query for top traffic sources table
	 *
	 * @param int   $id_view       The Analytics view ID from which to retrieve data.
	 * @param array $date_ranges An array representing the date ranges that will be passed to chart query.
	 *
	 * @return array Sources query
	 */
	public static function sources_query( $id_view, $date_ranges ) {
		$reports_requests = array();

		$ts = filter_input( INPUT_GET, 'ts', FILTER_UNSAFE_RAW );

		if ( false === empty( $ts ) ) {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => $date_ranges,
				'metrics'          => self::set_metrics( array( 'ga:pageviews' ) ),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:sourceMedium' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		} else {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => $date_ranges,
				'metrics'          => self::set_metrics(
					array(
						'ga:pageviews',
						'ga:uniquePageviews',
						'ga:timeOnPage',
						'ga:bounces',
						'ga:entrances',
						'ga:exits',
					)
				),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:pagePath' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		}

		$query = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for dashboard boxes
	 *
	 * @param int   $id_view       The Analytics view ID from which to retrieve data.
	 * @param array $date_ranges An array representing the date ranges that will be passed to chart query.
	 *
	 * @return array Dashboard boxes query
	 */
	public static function dashboard_boxes_query( $id_view, $date_ranges ) {
		$reports_requests = array();

		$ts = filter_input( INPUT_GET, 'ts', FILTER_SANITIZE_STRING );

		if ( false === empty( $ts ) ) {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => $date_ranges,
				'metrics'          => self::set_metrics( array( 'ga:pageviews' ) ),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:sourceMedium' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		} else {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => $date_ranges,
				'metrics'          => self::set_metrics(
					array(
						'ga:pageviews',
						'ga:uniquePageviews',
						'ga:timeOnPage',
						'ga:bounces',
						'ga:entrances',
						'ga:exits',
					)
				),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:pagePath' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		}
		$query = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for dashboard boxes
	 *
	 * @param int    $id_view    The Analytics view ID from which to retrieve data.
	 * @param string $date_range The start date for the query in the format YYYY-MM-DD or '7daysAgo'.
	 *
	 * @return array Dashboard boxes query
	 * @deprecated
	 */
	public static function dashboard_boxes_query_old( $id_view, $date_range ) {
		$reports_requests = array();

		$th = filter_input( INPUT_GET, 'th', FILTER_SANITIZE_STRING );
		$ts = filter_input( INPUT_GET, 'ts', FILTER_SANITIZE_STRING );

		$days_ago = false === empty( $th ) ? '30daysAgo' : '7daysAgo';

		if ( false === empty( $ts ) ) {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => self::set_date_ranges( $days_ago, 'yesterday' ),
				'metrics'          => self::set_metrics( array( 'ga:pageviews' ) ),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:sourceMedium' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		} else {
			$reports_requests[] = array(
				'viewId'           => $id_view,
				'dateRanges'       => self::set_date_ranges( $days_ago, 'yesterday' ),
				'metrics'          => self::set_metrics(
					array(
						'ga:pageviews',
						'ga:uniquePageviews',
						'ga:timeOnPage',
						'ga:bounces',
						'ga:entrances',
						'ga:exits',
					)
				),
				'includeEmptyRows' => true,
				'pageSize'         => 10,
				'dimensions'       => self::set_dimensions( 'ga:pagePath' ),
				'orderBys'         => self::set_order_bys( 'ga:pageviews', 'DESCENDING' ),
			);
		}
		$query = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for stats boxes
	 *
	 * @param int $id_view The Analytics view ID from which to retrieve data.
	 *
	 * @return array Boxes query
	 */
	public static function boxes_query( $id_view ) {
		$th = filter_input( INPUT_GET, 'th', FILTER_UNSAFE_RAW );

		$range              = false === empty( $th ) ? '30daysAgo' : '7daysAgo';
		$range_s_prev       = false === empty( $th ) ? '60daysAgo' : '14daysAgo';
		$range_e_prev       = false === empty( $th ) ? '31daysAgo' : '8daysAgo';
		$reports_requests   = array();
		$reports_requests[] = array(
			'viewId'           => $id_view,
			'dateRanges'       => self::set_date_ranges( $range, 'yesterday', $range_s_prev, $range_e_prev ),
			'metrics'          => self::set_metrics(
				array(
					'ga:users',
					'ga:pageviews',
					'ga:pageviewsPerSession',
					'ga:BounceRate',
				)
			),
			'includeEmptyRows' => true,
			'dimensions'       => self::set_dimensions( 'ga:date' ),
		);
		$query              = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for chart
	 *
	 * @param int    $id_view     The Analytics view ID from which to retrieve data.
	 * @param array  $date_ranges An array representing the date ranges that will be passed to chart query.
	 * @param string $metric      A metric expression.
	 *
	 * @return array Chart query
	 */
	public static function main_chart_query( $id_view, $date_ranges = null, $metric = null ) {
		if ( true === empty( $metric ) ) {
			$metric = 'ga:pageviews';
		} else {
			$metric = 'ga:' . $metric;
		}

		$reports_requests   = array();
		$reports_requests[] = array(
			'viewId'           => $id_view,
			'dateRanges'       => $date_ranges,
			'metrics'          => self::set_metrics( $metric ),
			'includeEmptyRows' => true,
			'dimensions'       => self::set_dimensions( 'ga:date' ),
		);
		$query              = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for chart
	 *
	 * @param int    $id_view    The Analytics view ID from which to retrieve data.
	 * @param string $date_range The start date for the query in the format YYYY-MM-DD or '7daysAgo'.
	 * @param string $metric     A metric expression.
	 *
	 * @return array Chart query
	 * @deprecated
	 */
	public static function main_chart_query_old( $id_view, $date_range = null, $metric = null ) {
		if ( empty( $date_range ) ) {
			$date_ranges = self::set_date_ranges( '7daysAgo', 'yesterday', '14daysAgo', '8daysAgo' );
		} else {
			$date_ranges = self::set_date_ranges( $date_range, 'yesterday', '14daysAgo', '8daysAgo' );
		}

		if ( empty( $metric ) ) {
			$metric = 'ga:pageviews';
		} else {
			$metric = 'ga:' . $metric;
		}

		$reports_requests   = array();
		$reports_requests[] = array(
			'viewId'           => $id_view,
			'dateRanges'       => $date_ranges,
			'metrics'          => self::set_metrics( $metric ),
			'includeEmptyRows' => true,
			'dimensions'       => self::set_dimensions( 'ga:date' ),
		);
		$query              = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for gender chart
	 *
	 * @param int    $id_view     The Analytics view ID from which to retrieve data.
	 * @param array  $date_ranges An array representing the date ranges that will be passed to chart query.
	 * @param string $metric      A metric expression.
	 *
	 * @return array Chart query
	 */
	public static function gender_chart_query( $id_view, $date_ranges = null, $metric = null ) {
		if ( true === empty( $date_ranges ) ) {
			$date_ranges = self::set_date_ranges( '7daysAgo', 'yesterday', '14daysAgo', '8daysAgo' );
		}

		$reports_requests   = array();
		$reports_requests[] = array(
			'viewId'           => $id_view,
			'dateRanges'       => $date_ranges,
			'metrics'          => self::set_metrics( 'ga:sessions' ),
			'includeEmptyRows' => true,
			'dimensions'       => self::set_dimensions( 'ga:userGender' ),
		);
		$query              = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Preparing query for device chart.
	 *
	 * @param int    $id_view     The Analytics view ID from which to retrieve data.
	 * @param array  $date_ranges An array representing the date ranges that will be passed to chart query.
	 * @param string $metric      A metric expression.
	 *
	 * @return array Chart query
	 * @since 2.5.2
	 */
	public static function device_chart_query( $id_view, $date_ranges = null, $metric = null ) {
		return array(
			'reportRequests' => array(
				array(
					'viewId'           => $id_view,
					'dateRanges'       => $date_ranges,
					'metrics'          => self::set_metrics( 'ga:sessions' ),
					'includeEmptyRows' => true,
					'dimensions'       => self::set_dimensions( 'ga:deviceCategory' ),
				),
			),
		);
	}

	/**
	 * Preparing query for age chart
	 *
	 * @param int    $id_view     The Analytics view ID from which to retrieve data.
	 * @param array  $date_ranges An array representing the date ranges that will be passed to chart query.
	 * @param string $metric      A metric expression.
	 *
	 * @return array Chart query
	 */
	public static function age_chart_query( $id_view, $date_ranges = null, $metric = null ) {
		if ( true === empty( $date_ranges ) ) {
			$date_ranges = self::set_date_ranges( '7daysAgo', 'yesterday', '14daysAgo', '8daysAgo' );
		}

		$reports_requests   = array();
		$reports_requests[] = array(
			'viewId'           => $id_view,
			'dateRanges'       => $date_ranges,
			'metrics'          => self::set_metrics( 'ga:sessions' ),
			'includeEmptyRows' => true,
			'dimensions'       => self::set_dimensions( 'ga:userAgeBracket' ),
		);
		$query              = array(
			'reportRequests' => $reports_requests,
		);

		return $query;
	}

	/**
	 * Setting order for requests
	 *
	 * @param string $name The field which to sort by. The default sort order is ascending. Example: ga:browser.
	 * @param string $sort The sorting order for the field. 'ASCENDING' or 'DESCENDING'.
	 *
	 * @return array OrderBys
	 */
	public static function set_order_bys( $name, $sort ) {
		$order   = array();
		$order[] = array(
			'fieldName' => $name,
			'sortOrder' => $sort,
		);

		return $order;
	}

	/**
	 * Setting metrics for requests
	 *
	 * @param mixed $expression A metric expression or array of expressions.
	 *
	 * @return array Metrics
	 */
	public static function set_metrics( $expression ) {
		$metrics = array();
		if ( is_array( $expression ) ) {
			foreach ( $expression as $exp ) {
				$metrics[] = array(
					'expression' => $exp,
				);
			}
		} else {
			$metrics[] = array(
				'expression' => $expression,
			);
		}

		return $metrics;
	}

	/**
	 * Setting dimensions for requests
	 *
	 * @param string $name Name of the dimension to fetch, for example ga:browser.
	 *
	 * @return array Dimensions
	 */
	public static function set_dimensions( $name ) {
		$dimensions   = array();
		$dimensions[] = array(
			'name' => $name,
		);

		return $dimensions;
	}

	/**
	 * Setting date ranges for requests
	 *
	 * @param string $start_date The start date for the query in the format YYYY-MM-DD.
	 * @param string $end_date The end date for the query in the format YYYY-MM-DD.
	 * @param string $prev_start_date The start date (second range) for the query in the format YYYY-MM-DD.
	 * @param string $prev_end_date The start date (second range) for the query in the format YYYY-MM-DD.
	 *
	 * @return array Date ranges
	 */
	public static function set_date_ranges( $start_date, $end_date, $prev_start_date = '', $prev_end_date = '' ) {
		$date_danges   = array();
		$date_danges[] = array(
			'startDate' => $start_date,
			'endDate'   => $end_date,
		);
		if ( false === empty( $prev_start_date ) && false === empty( $prev_end_date ) ) {
			$date_danges[] = array(
				'startDate' => $prev_start_date,
				'endDate'   => $prev_end_date,
			);
		}

		return $date_danges;
	}

	/**
	 * Preparing response for data received from analytics
	 *
	 * @param array $data Analytics response.
	 *
	 * @return array Response rows
	 */
	public static function prepare_response( $data ) {
		$data = self::get_reports_from_response( $data );
		self::handle_more_reports( $data );
		$report = self::get_single_report( $data );
		self::get_report_column_header( $report );
		$report_data = self::get_report_data( $report );
		self::get_totals( $report_data );
		self::get_row_count( $report_data );
		$rows = self::get_rows( $report_data );

		return $rows;
	}

	/**
	 * Get dimensions from response row
	 *
	 * @param array $row Analytics response row.
	 *
	 * @return array|bool Dimensions
	 */
	public static function get_dimensions( $row ) {
		if ( false === empty( $row['dimensions'] ) ) {
			return $row['dimensions'];
		}

		return false;
	}

	/**
	 * Get metrics from response row
	 *
	 * @param array $row Analytics response row.
	 *
	 * @return array|bool Metrics
	 */
	public static function get_metrics( $row ) {
		if ( false === empty( $row['metrics'] ) ) {
			return $row['metrics'];
		}

		return false;
	}

	/**
	 * Get row from response report data
	 *
	 * @param array $report_data Analytics response report data.
	 *
	 * @return array|bool Rows
	 */
	public static function get_rows( $report_data ) {
		if ( false === empty( $report_data['rows'] ) ) {
			return $report_data['rows'];
		}

		return false;
	}

	/**
	 * Get row count from response report data
	 *
	 * @param array $report_data Analytics response report data.
	 *
	 * @return array|bool Row count
	 */
	public static function get_row_count( $report_data ) {
		if ( false === empty( $report_data['rowCount'] ) ) {
			return $report_data['rowCount'];
		}

		return false;
	}

	/**
	 * Get totals from response report data
	 *
	 * @param array $report_data Analytics response report data.
	 *
	 * @return array|bool Totals
	 */
	public static function get_totals( $report_data ) {
		if ( false === empty( $report_data['totals'] ) ) {
			return $report_data['totals'];
		}

		return false;
	}

	/**
	 * Get reports from response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array|bool Reports
	 */
	public static function get_reports_from_response( $data ) {
		if ( false === empty( $data['reports'] ) ) {
			return $data['reports'];
		}

		return false;
	}

	/**
	 * Show info for multiple data
	 *
	 * @param array $data Analytics response data.
	 */
	public static function handle_more_reports( $data ) {
		if ( count( $data ) > 1 ) {
			echo 'more than one report';
		}
	}

	/**
	 * Show info for multiple rows
	 *
	 * @param array $rows Analytics response rows.
	 */
	public static function handle_more_rows( $rows ) {
		if ( count( $rows ) > 1 ) {
			echo 'more than one row';
		}
	}

	/**
	 * Get single report from response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array|bool Report
	 */
	public static function get_single_report( $data ) {
		if ( false === empty( $data ) ) {
			foreach ( $data as $report ) {
				if ( false === empty( $report ) ) {
					return $report;
				}
			}
		}

		return false;
	}

	/**
	 * Get single row from response data rows
	 *
	 * @param array $rows Analytics response data rows.
	 *
	 * @return array|bool Row
	 */
	public static function get_single_row( $rows ) {
		if ( false === empty( $rows ) ) {
			foreach ( $rows as $row ) {
				if ( false === empty( $row ) ) {
					return $row;
				}
			}
		}

		return false;
	}

	/**
	 * Get column header from response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array Column header
	 */
	public static function get_report_column_header( $data ) {
		if ( false === empty( $data['columnHeader'] ) ) {
			return $data['columnHeader'];
		}

		return false;
	}

	/**
	 * Get report data from response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array|bool data
	 */
	public static function get_report_data( $data ) {
		if ( false === empty( $data['data'] ) ) {
			return $data['data'];
		}

		return false;
	}

	/**
	 * Get chart from response data
	 *
	 * @param array $response_data  Analytics response data.
	 * @param int   $period_in_days Period in days (default = 7).
	 *
	 * @return array chart data
	 */
	public static function get_chart( $response_data, $period_in_days = 7 ) {
		$chart_data = array();
		if ( false === empty( $response_data ) ) {
			$data = (
				false === empty( $response_data['reports'] )
				&& false === empty( $response_data['reports'][0] )
				&& false === empty( $response_data['reports'][0]['data'] )
			)
				? $response_data['reports'][0]['data'] : array();
			$rows = ( false === empty( $data['rows'] ) ) ? $data['rows'] : array();
			if ( false === empty( $rows ) ) {
				foreach ( $rows as $key => $row ) {
					if ( $key < $period_in_days ) {
						$chart_data[ $key ]['previous']     = false === empty( $row['metrics'][1]['values'][0] ) ? $row['metrics'][1]['values'][0] : 0;
						$chart_data[ $key ]['previous-day'] = gmdate( 'M j', strtotime( $row['dimensions'][0] ) );
					} else {
						$chart_data[ $key - $period_in_days ]['day']     = gmdate( 'M j', strtotime( $row['dimensions'][0] ) );
						$chart_data[ $key - $period_in_days ]['current'] = false === empty( $row['metrics'][0]['values'][0] ) ? $row['metrics'][0]['values'][0] : 0;

						if ( 0 === $chart_data[ $key - $period_in_days ]['current'] ) {
							$chart_data[ $key - $period_in_days ]['current'] = false === empty( $row['metrics'][1]['values'][0] ) ? $row['metrics'][1]['values'][0] : 0;
						}

						$chart_data['date'] = strtotime( $row['dimensions'][0] );
					}
				}
			}
		}

		return $chart_data;
	}

	/**
	 * Get gender chart from response data
	 *
	 * @param array $response_data Analytics response data.
	 *
	 * @return array chart data
	 */
	public static function get_gender_chart( $response_data ) {
		$chart_data = array();
		if ( false === empty( $response_data ) ) {
			$data = ( false === empty( $response_data['reports'] ) && false === empty( $response_data['reports'][0] ) && false === empty( $response_data['reports'][0]['data'] ) ) ? $response_data['reports'][0]['data'] : array();
			$rows = ( false === empty( $data['rows'] ) ) ? $data['rows'] : array();
			if ( false === empty( $rows ) ) {
				foreach ( $rows as $key => $row ) {
					$chart_data[ $row['dimensions'][0] ] = self::get_metric_value( $row['metrics'] );
				}
			}
		}

		return $chart_data;
	}

	/**
	 * Get device chart from response data.
	 *
	 * @param array $response_data Analytics response data array.
	 *
	 * @return array Chart data array.
	 */
	public static function get_device_chart( $response_data ) {
		$chart_data = array();
		if ( false === empty( $response_data ) ) {
			$data = ( false === empty( $response_data['reports'] ) && false === empty( $response_data['reports'][0] ) && false === empty( $response_data['reports'][0]['data'] ) ) ? $response_data['reports'][0]['data'] : array();
			$rows = ( false === empty( $data['rows'] ) ) ? $data['rows'] : array();
			if ( false === empty( $rows ) ) {
				foreach ( $rows as $row ) {
					$chart_data[ $row['dimensions'][0] ] = self::get_metric_value( $row['metrics'] );
				}
			}
		}

		return $chart_data;
	}

	/**
	 * Get the value of metric data response.
	 *
	 * @param array $metrics Metrics array.
	 *
	 * @return mixed
	 */
	private static function get_metric_value( $metrics ) {
		if ( is_array( $metrics ) ) {
			foreach ( $metrics as $metric ) {
				$values[] = $metric['values'][0];
			}
		}

		return $values[0];
	}

	/**
	 * Get gender chart from response data
	 *
	 * @param array $response_data Analytics response data.
	 *
	 * @return array chart data
	 */
	public static function get_age_chart( $response_data ) {
		$chart_data = array();
		if ( false === empty( $response_data ) ) {
			$data = ( false === empty( $response_data['reports'] ) && false === empty( $response_data['reports'][0] ) && false === empty( $response_data['reports'][0]['data'] ) ) ? $response_data['reports'][0]['data'] : array();
			$rows = ( false === empty( $data['rows'] ) ) ? $data['rows'] : array();
			if ( false === empty( $rows ) ) {
				foreach ( $rows as $key => $row ) {
					$chart_data[ $row['dimensions'][0] ] = self::get_metric_value( $row['metrics'] );
				}
			}
		}

		return $chart_data;
	}

	/**
	 * Get dasboard chart from response data
	 *
	 * @param array $response_data Analytics response data.
	 *
	 * @return array dashboard chart data
	 */
	public static function get_dashboard_chart( $response_data ) {
		$chart_data = array();
		if ( false === empty( $response_data ) ) {
			$data = (
				false === empty( $response_data['reports'] )
				&& false === empty( $response_data['reports'][0] )
				&& false === empty( $response_data['reports'][0]['data'] )
			) ? $response_data['reports'][0]['data'] : array();

			$rows = ( false === empty( $data['rows'] ) ) ? $data['rows'] : array();
			if ( false === empty( $rows ) ) {
				foreach ( $rows as $row ) {
					$chart_data[] = array(
						'day'     => gmdate( 'M j', strtotime( $row['dimensions'][0] ) ),
						'current' => false === empty( $row['metrics'][0]['values'][0] ) ? $row['metrics'][0]['values'][0] : 0,
					);
				}
			}
		}

		return $chart_data;
	}

	/**
	 * Get boxes from response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array boxes data
	 */
	public static function get_boxes( $data ) {
		if ( false === empty( $data ) ) {
			$data = self::get_reports_from_response( $data );
			self::handle_more_reports( $data );
			$report = self::get_single_report( $data );
			self::get_report_column_header( $report );
			$report_data = self::get_report_data( $report );
			$totals      = self::get_totals( $report_data );

			return self::get_boxes_from_totals( $totals );
		}
	}

	/**
	 * Get boxes from totals
	 *
	 * @param array $totals Analytics response totals.
	 *
	 * @return array|bool boxes data
	 */
	public static function get_boxes_from_totals( $totals ) {
		if ( false === empty( $totals ) ) {
			$boxes_data = array();
			foreach ( $totals as $key => $total ) {
				if ( 0 === $key ) {
					$boxes_data['Users']['current']               = $total['values'][0];
					$boxes_data['Pageviews']['current']           = $total['values'][1];
					$boxes_data['PageviewsPerSession']['current'] = $total['values'][2];
					$boxes_data['BounceRate']['current']          = round( $total['values'][3], 2 );
				} else {
					$boxes_data['Users']['previous']               = $total['values'][0];
					$boxes_data['Pageviews']['previous']           = $total['values'][1];
					$boxes_data['PageviewsPerSession']['previous'] = $total['values'][2];
					$boxes_data['BounceRate']['previous']          = round( $total['values'][3], 2 );
				}
			}

			return self::prepare_boxes( $boxes_data );
		}

		return false;
	}

	/**
	 * Prepare boxes data
	 *
	 * @param array $boxes_data Boxes data.
	 *
	 * @return array boxes data
	 */
	public static function prepare_boxes( $boxes_data ) {
		$boxes_data['Users']['diff']                     = ( $boxes_data['Users']['previous'] > 0 ) ? round( ( $boxes_data['Users']['current'] - $boxes_data['Users']['previous'] ) / $boxes_data['Users']['previous'] * 100, 2 ) : 100;
		$boxes_data['Pageviews']['diff']                 = ( $boxes_data['Pageviews']['previous'] > 0 ) ? round( ( $boxes_data['Pageviews']['current'] - $boxes_data['Pageviews']['previous'] ) / $boxes_data['Pageviews']['previous'] * 100, 2 ) : 100;
		$boxes_data['PageviewsPerSession']['diff']       = ( $boxes_data['PageviewsPerSession']['previous'] > 0 ) ? round( ( $boxes_data['PageviewsPerSession']['current'] - $boxes_data['PageviewsPerSession']['previous'] ) / $boxes_data['PageviewsPerSession']['previous'] * 100, 2 ) : 100;
		$boxes_data['BounceRate']['diff']                = ( $boxes_data['BounceRate']['previous'] > 0 ) ? round( ( $boxes_data['BounceRate']['current'] - $boxes_data['BounceRate']['previous'] ) / $boxes_data['BounceRate']['previous'] * 100, 2 ) : 100;
		$boxes_data['Users']['diff']                     = ( 0 === $boxes_data['Users']['previous'] && 0 === $boxes_data['Users']['current'] ) ? 0 : $boxes_data['Users']['diff'];
		$boxes_data['Pageviews']['diff']                 = ( 0 === $boxes_data['Pageviews']['previous'] && 0 === $boxes_data['Pageviews']['current'] ) ? 0 : $boxes_data['Pageviews']['diff'];
		$boxes_data['PageviewsPerSession']['diff']       = ( 0 === $boxes_data['PageviewsPerSession']['previous'] && 0 === $boxes_data['PageviewsPerSession']['current'] ) ? 0 : $boxes_data['PageviewsPerSession']['diff'];
		$boxes_data['BounceRate']['diff']                = ( 0 === $boxes_data['BounceRate']['previous'] && 0 === $boxes_data['BounceRate']['current'] ) ? 0 : $boxes_data['BounceRate']['diff'];
		$boxes_data['Users']['label']                    = 'Users';
		$boxes_data['Pageviews']['label']                = 'Pageviews';
		$boxes_data['PageviewsPerSession']['label']      = 'Pages / Session';
		$boxes_data['BounceRate']['label']               = 'Bounce Rate';
		$boxes_data['Users']['comparison']               = $boxes_data['Users']['current'] . ' vs ' . $boxes_data['Users']['previous'];
		$boxes_data['Pageviews']['comparison']           = $boxes_data['Pageviews']['current'] . ' vs ' . $boxes_data['Pageviews']['previous'];
		$boxes_data['PageviewsPerSession']['comparison'] = self::number_format_clean( $boxes_data['PageviewsPerSession']['current'], 2, '.', ',' ) . ' vs ' . self::number_format_clean( $boxes_data['PageviewsPerSession']['previous'], 2, '.', ',' );
		$boxes_data['BounceRate']['comparison']          = self::number_format_clean( $boxes_data['BounceRate']['current'], 2, '.', ',' ) . '% vs ' . self::number_format_clean( $boxes_data['BounceRate']['previous'], 2, '.', ',' ) . '%';
		$boxes_data['Users']['color']                    = ( $boxes_data['Users']['diff'] > 0 ) ? 'green' : 'red';
		$boxes_data['Pageviews']['color']                = ( $boxes_data['Pageviews']['diff'] > 0 ) ? 'green' : 'red';
		$boxes_data['PageviewsPerSession']['color']      = ( $boxes_data['PageviewsPerSession']['diff'] > 0 ) ? 'green' : 'red';
		$boxes_data['BounceRate']['color']               = ( $boxes_data['BounceRate']['diff'] > 0 ) ? 'red' : 'green';
		$boxes_data['Users']['color']                    = ( 0 !== $boxes_data['Users']['diff'] ) ? $boxes_data['Users']['color'] : 'black';
		$boxes_data['Pageviews']['color']                = ( 0 !== $boxes_data['Pageviews']['diff'] ) ? $boxes_data['Pageviews']['color'] : 'black';
		$boxes_data['PageviewsPerSession']['color']      = ( 0 !== $boxes_data['PageviewsPerSession']['diff'] ) ? $boxes_data['PageviewsPerSession']['color'] : 'black';
		$boxes_data['BounceRate']['color']               = ( 0 !== $boxes_data['BounceRate']['diff'] ) ? $boxes_data['BounceRate']['color'] : 'black';

		return $boxes_data;
	}

	/**
	 * Number format for boxes
	 *
	 * @param float  $number        Number to format.
	 * @param int    $precision     Precision.
	 * @param string $dec_point     Decimal point.
	 * @param string $thousands_sep Thousands Separator.
	 *
	 * @return string clean number format
	 */
	public static function number_format_clean( $number, $precision = 0, $dec_point = '.', $thousands_sep = ',' ) {
		if ( 0 === $number ) {
			return 0;
		} else {
			$format = number_format( $number, $precision, $dec_point, $thousands_sep );
			if ( '.00' === substr( $format, 2 ) ) {
				return substr( $format, 0, - 3 );
			}

			return $format;
		}
	}

	/**
	 * Get sources from analytics response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array|bool sources data
	 */
	public static function get_sources( $data ) {
		if ( false === empty( $data ) ) {
			$data = self::get_reports_from_response( $data );
			self::handle_more_reports( $data );
			$report = self::get_single_report( $data );
			self::get_report_column_header( $report );
			$report_data = self::get_report_data( $report );
			$rows        = self::get_rows( $report_data );
			$totals      = self::get_totals( $report_data );
			$total_count = array();
			if ( false === empty( $totals ) ) {
				foreach ( $totals as $key => $total ) {
					$total_count = $total['values'][0];
				}
			}
			$sources = array(
				'total' => $total_count,
				'sum'   => 0,
				'rows'  => array(),
			);
			if ( false === empty( $rows ) ) {
				$i = 1;
				foreach ( $rows as $row ) {
					if ( false === empty( $row ) ) {
						foreach ( $row as $key => $value ) {
							if ( 'dimensions' === $key ) {
								$sources['rows'][ $i ]['name'] = $value[0];
								$sources['rows'][ $i ]['url']  = $value[0];
							} elseif ( 'metrics' === $key ) {
								$sources['rows'][ $i ]['number']  = $value[0]['values'][0];
								$sources['rows'][ $i ]['percent'] = ( false === empty( $total_count ) ) ? round( $value[0]['values'][0] / $total_count * 100, 2 ) : 0;
								$sources['sum']                  += $value[0]['values'][0];
							}
						}
						$i ++;
					}
				}
			}

			return $sources;
		}

		return false;
	}

	/**
	 * Get dashboard boxes data from analytics response data
	 *
	 * @param array $data Analytics response data.
	 *
	 * @return array dashboard boxes data
	 */
	public static function get_dashboard_boxes_data( $data ) {
		if ( false === empty( $data ) ) {
			$data = self::get_reports_from_response( $data );
			self::handle_more_reports( $data );
			$report = self::get_single_report( $data );
			self::get_report_column_header( $report );
			$report_data                       = self::get_report_data( $report );
			$totals                            = self::get_totals( $report_data );
			$boxes_data                        = array();
			$boxes_data['Sessions']            = array(
				'label' => 'Visits',
				'value' => $totals[0]['values'][0],
			);
			$boxes_data['Pageviews']           = array(
				'label' => 'Pageviews',
				'value' => $totals[0]['values'][1],
			);
			$boxes_data['pageviewsPerSession'] = array(
				'label' => 'Pages / Visit',
				'value' => self::number_format_clean( $totals[0]['values'][2], 2, '.', ',' ),
			);
			$boxes_data['BounceRate']          = array(
				'label' => 'Bounce Rate',
				'value' => self::number_format_clean( $totals[0]['values'][3], 2, '.', ',' ) . '%',
			);
			$boxes_data['avgTimeOnPage']       = array(
				'label' => 'Avg. Time on Site',
				'value' => gmdate( 'H:i:s', $totals[0]['values'][4] ),
			);
			$boxes_data['percentNewSessions']  = array(
				'label' => '% of New Visits',
				'value' => self::number_format_clean( $totals[0]['values'][5], 2, '.', ',' ),
			);

			return $boxes_data;
		}
	}

	/**
	 * Get Empty Boxes Structure.
	 *
	 * @return array Array of empty boxes structure values.
	 */
	public static function get_empty_boxes_structure() {
		$boxes_data                        = array();
		$boxes_data['Sessions']            = array(
			'label' => 'Visits',
			'value' => 0,
		);
		$boxes_data['Pageviews']           = array(
			'label' => 'Pageviews',
			'value' => 0,
		);
		$boxes_data['pageviewsPerSession'] = array(
			'label' => 'Pages / Visit',
			'value' => self::number_format_clean( 0, 2, '.', ',' ),
		);
		$boxes_data['BounceRate']          = array(
			'label' => 'Bounce Rate',
			'value' => self::number_format_clean( 0, 2, '.', ',' ) . '%',
		);
		$boxes_data['avgTimeOnPage']       = array(
			'label' => 'Avg. Time on Site',
			'value' => gmdate( 'H:i:s', 0 ),
		);
		$boxes_data['percentNewSessions']  = array(
			'label' => '% of New Visits',
			'value' => self::number_format_clean( 0, 2, '.', ',' ),
		);

		return $boxes_data;
	}
}