woocommerce/includes/admin/reports/class-wc-admin-report.php

671 lines
20 KiB
PHP
Raw Normal View History

<?php
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly
}
2014-02-14 13:02:37 +00:00
/**
* Admin Report.
*
* Extended by reports to show charts and stats in admin.
*
* @author WooThemes
* @category Admin
* @package WooCommerce/Admin/Reports
* @version 2.1.0
*/
class WC_Admin_Report {
/**
* The chart interval.
*
* @var int
*/
2014-02-12 12:26:32 +00:00
public $chart_interval;
/**
* Group by SQL query.
*
* @var string
*/
2014-02-12 12:26:32 +00:00
public $group_by_query;
/**
* The bar width.
*
* @var int
*/
2014-02-12 12:26:32 +00:00
public $barwidth;
/**
* Group chart item by day or month.
*
* @var string
*/
2014-02-12 12:26:32 +00:00
public $chart_groupby;
/**
* The start date of the report.
*
2016-06-06 18:15:40 +00:00
* @var int timestamp
*/
public $start_date;
/**
* The end date of the report.
*
2016-06-06 18:15:40 +00:00
* @var int timestamp
*/
public $end_date;
/**
* Get report totals such as order totals and discount amounts.
*
2015-11-03 13:31:20 +00:00
* Data example:
*
* '_order_total' => array(
* 'type' => 'meta',
* 'function' => 'SUM',
* 'name' => 'total_sales'
* )
*
* @param array $args
2015-04-01 13:33:56 +00:00
* @return mixed depending on query_type
*/
public function get_order_report_data( $args = array() ) {
global $wpdb;
2014-10-07 09:48:44 +00:00
$default_args = array(
'data' => array(),
'where' => array(),
'where_meta' => array(),
'query_type' => 'get_row',
'group_by' => '',
'order_by' => '',
'limit' => '',
'filter_range' => false,
2014-10-14 16:01:15 +00:00
'nocache' => false,
'debug' => false,
'order_types' => wc_get_order_types( 'reports' ),
'order_status' => array( 'completed', 'processing', 'on-hold' ),
'parent_order_status' => false,
);
2014-10-07 09:48:44 +00:00
$args = apply_filters( 'woocommerce_reports_get_order_report_data_args', $args );
$args = wp_parse_args( $args, $default_args );
extract( $args );
2014-05-28 14:19:16 +00:00
if ( empty( $data ) ) {
return '';
2014-05-28 14:19:16 +00:00
}
$order_status = apply_filters( 'woocommerce_reports_order_statuses', $order_status );
2014-10-07 09:31:07 +00:00
$query = array();
$select = array();
2017-07-12 09:53:39 +00:00
foreach ( $data as $raw_key => $value ) {
$key = sanitize_key( $raw_key );
$distinct = '';
2014-10-07 09:48:44 +00:00
if ( isset( $value['distinct'] ) ) {
$distinct = 'DISTINCT';
2014-10-07 09:48:44 +00:00
}
switch ( $value['type'] ) {
case 'meta' :
$get_key = "meta_{$key}.meta_value";
break;
case 'parent_meta' :
$get_key = "parent_meta_{$key}.meta_value";
break;
case 'post_data' :
$get_key = "posts.{$key}";
break;
case 'order_item_meta' :
$get_key = "order_item_meta_{$key}.meta_value";
break;
case 'order_item' :
$get_key = "order_items.{$key}";
break;
default :
continue;
}
2013-06-28 16:33:37 +00:00
if ( $value['function'] ) {
2013-06-28 16:33:37 +00:00
$get = "{$value['function']}({$distinct} {$get_key})";
} else {
2013-06-28 16:33:37 +00:00
$get = "{$distinct} {$get_key}";
}
2013-06-28 16:33:37 +00:00
$select[] = "{$get} as {$value['name']}";
}
$query['select'] = "SELECT " . implode( ',', $select );
$query['from'] = "FROM {$wpdb->posts} AS posts";
// Joins
$joins = array();
2017-07-12 09:53:39 +00:00
foreach ( ( $data + $where ) as $raw_key => $value ) {
$join_type = isset( $value['join_type'] ) ? $value['join_type'] : 'INNER';
$type = isset( $value['type'] ) ? $value['type'] : false;
2017-07-12 09:53:39 +00:00
$key = sanitize_key( $raw_key );
switch ( $type ) {
case 'meta' :
2017-07-12 09:53:39 +00:00
$joins[ "meta_{$key}" ] = "{$join_type} JOIN {$wpdb->postmeta} AS meta_{$key} ON ( posts.ID = meta_{$key}.post_id AND meta_{$key}.meta_key = '{$raw_key}' )";
break;
case 'parent_meta' :
2017-07-12 09:53:39 +00:00
$joins[ "parent_meta_{$key}" ] = "{$join_type} JOIN {$wpdb->postmeta} AS parent_meta_{$key} ON (posts.post_parent = parent_meta_{$key}.post_id) AND (parent_meta_{$key}.meta_key = '{$raw_key}')";
break;
case 'order_item_meta' :
$joins["order_items"] = "{$join_type} JOIN {$wpdb->prefix}woocommerce_order_items AS order_items ON (posts.ID = order_items.order_id)";
if ( ! empty( $value['order_item_type'] ) ) {
$joins["order_items"] .= " AND (order_items.order_item_type = '{$value['order_item_type']}')";
}
$joins[ "order_item_meta_{$key}" ] = "{$join_type} JOIN {$wpdb->prefix}woocommerce_order_itemmeta AS order_item_meta_{$key} ON " .
"(order_items.order_item_id = order_item_meta_{$key}.order_item_id) " .
2017-07-12 09:53:39 +00:00
" AND (order_item_meta_{$key}.meta_key = '{$raw_key}')";
break;
case 'order_item' :
$joins["order_items"] = "{$join_type} JOIN {$wpdb->prefix}woocommerce_order_items AS order_items ON posts.ID = order_items.order_id";
break;
}
}
if ( ! empty( $where_meta ) ) {
foreach ( $where_meta as $value ) {
if ( ! is_array( $value ) ) {
continue;
}
$join_type = isset( $value['join_type'] ) ? $value['join_type'] : 'INNER';
$type = isset( $value['type'] ) ? $value['type'] : false;
2017-07-12 09:53:39 +00:00
$key = sanitize_key( is_array( $value['meta_key'] ) ? $value['meta_key'][0] . '_array' : $value['meta_key'] );
2013-07-25 14:00:23 +00:00
if ( 'order_item_meta' === $type ) {
$joins["order_items"] = "{$join_type} JOIN {$wpdb->prefix}woocommerce_order_items AS order_items ON posts.ID = order_items.order_id";
$joins[ "order_item_meta_{$key}" ] = "{$join_type} JOIN {$wpdb->prefix}woocommerce_order_itemmeta AS order_item_meta_{$key} ON order_items.order_item_id = order_item_meta_{$key}.order_item_id";
} else {
// If we have a where clause for meta, join the postmeta table
$joins[ "meta_{$key}" ] = "{$join_type} JOIN {$wpdb->postmeta} AS meta_{$key} ON posts.ID = meta_{$key}.post_id";
}
}
}
if ( ! empty( $parent_order_status ) ) {
$joins["parent"] = "LEFT JOIN {$wpdb->posts} AS parent ON posts.post_parent = parent.ID";
}
$query['join'] = implode( ' ', $joins );
$query['where'] = "
WHERE posts.post_type IN ( '" . implode( "','", $order_types ) . "' )
";
if ( ! empty( $order_status ) ) {
$query['where'] .= "
AND posts.post_status IN ( 'wc-" . implode( "','wc-", $order_status ) . "')
";
}
if ( ! empty( $parent_order_status ) ) {
2015-04-24 14:58:13 +00:00
if ( ! empty( $order_status ) ) {
$query['where'] .= " AND ( parent.post_status IN ( 'wc-" . implode( "','wc-", $parent_order_status ) . "') OR parent.ID IS NULL ) ";
} else {
$query['where'] .= " AND parent.post_status IN ( 'wc-" . implode( "','wc-", $parent_order_status ) . "') ";
}
}
if ( $filter_range ) {
$query['where'] .= "
2017-01-10 15:50:03 +00:00
AND posts.post_date >= '" . date( 'Y-m-d H:i:s', $this->start_date ) . "'
AND posts.post_date < '" . date( 'Y-m-d H:i:s', strtotime( '+1 DAY', $this->end_date ) ) . "'
";
}
if ( ! empty( $where_meta ) ) {
$relation = isset( $where_meta['relation'] ) ? $where_meta['relation'] : 'AND';
$query['where'] .= " AND (";
foreach ( $where_meta as $index => $value ) {
if ( ! is_array( $value ) ) {
continue;
}
2017-07-12 09:53:39 +00:00
$key = sanitize_key( is_array( $value['meta_key'] ) ? $value['meta_key'][0] . '_array' : $value['meta_key'] );
2013-07-25 14:00:23 +00:00
if ( strtolower( $value['operator'] ) == 'in' || strtolower( $value['operator'] ) == 'not in' ) {
if ( is_array( $value['meta_value'] ) ) {
$value['meta_value'] = implode( "','", $value['meta_value'] );
}
if ( ! empty( $value['meta_value'] ) ) {
$where_value = "{$value['operator']} ('{$value['meta_value']}')";
}
} else {
$where_value = "{$value['operator']} '{$value['meta_value']}'";
}
2013-07-08 15:36:09 +00:00
if ( ! empty( $where_value ) ) {
if ( $index > 0 ) {
$query['where'] .= ' ' . $relation;
}
2016-09-09 00:14:28 +00:00
if ( isset( $value['type'] ) && 'order_item_meta' === $value['type'] ) {
if ( is_array( $value['meta_key'] ) ) {
2013-07-25 14:00:23 +00:00
$query['where'] .= " ( order_item_meta_{$key}.meta_key IN ('" . implode( "','", $value['meta_key'] ) . "')";
} else {
2013-07-25 14:00:23 +00:00
$query['where'] .= " ( order_item_meta_{$key}.meta_key = '{$value['meta_key']}'";
}
2013-07-25 14:00:23 +00:00
$query['where'] .= " AND order_item_meta_{$key}.meta_value {$where_value} )";
2013-07-08 15:36:09 +00:00
} else {
if ( is_array( $value['meta_key'] ) ) {
2013-07-25 14:00:23 +00:00
$query['where'] .= " ( meta_{$key}.meta_key IN ('" . implode( "','", $value['meta_key'] ) . "')";
} else {
2013-07-25 14:00:23 +00:00
$query['where'] .= " ( meta_{$key}.meta_key = '{$value['meta_key']}'";
}
2013-07-25 14:00:23 +00:00
$query['where'] .= " AND meta_{$key}.meta_value {$where_value} )";
2013-07-08 15:36:09 +00:00
}
}
}
$query['where'] .= ")";
}
if ( ! empty( $where ) ) {
foreach ( $where as $value ) {
if ( strtolower( $value['operator'] ) == 'in' || strtolower( $value['operator'] ) == 'not in' ) {
if ( is_array( $value['value'] ) ) {
$value['value'] = implode( "','", $value['value'] );
}
if ( ! empty( $value['value'] ) ) {
$where_value = "{$value['operator']} ('{$value['value']}')";
}
} else {
$where_value = "{$value['operator']} '{$value['value']}'";
}
if ( ! empty( $where_value ) ) {
2013-07-08 15:36:09 +00:00
$query['where'] .= " AND {$value['key']} {$where_value}";
}
}
}
if ( $group_by ) {
$query['group_by'] = "GROUP BY {$group_by}";
}
if ( $order_by ) {
$query['order_by'] = "ORDER BY {$order_by}";
}
if ( $limit ) {
$query['limit'] = "LIMIT {$limit}";
}
$query = apply_filters( 'woocommerce_reports_get_order_report_query', $query );
$query = implode( ' ', $query );
$query_hash = md5( $query_type . $query );
$cached_results = get_transient( strtolower( get_class( $this ) ) );
if ( $debug ) {
2014-10-07 09:30:17 +00:00
echo '<pre>';
wc_print_r( $query );
2014-10-07 09:30:17 +00:00
echo '</pre>';
}
2013-07-08 15:36:09 +00:00
if ( $debug || $nocache || false === $cached_results || ! isset( $cached_results[ $query_hash ] ) ) {
// Enable big selects for reports
$wpdb->query( 'SET SESSION SQL_BIG_SELECTS=1' );
$cached_results[ $query_hash ] = apply_filters( 'woocommerce_reports_get_order_report_data', $wpdb->$query_type( $query ), $data );
set_transient( strtolower( get_class( $this ) ), $cached_results, DAY_IN_SECONDS );
}
$result = $cached_results[ $query_hash ];
return $result;
}
/**
* Put data with post_date's into an array of times.
*
* @param array $data array of your data
* @param string $date_key key for the 'date' field. e.g. 'post_date'
* @param string $data_key key for the data you are charting
* @param int $interval
* @param string $start_date
* @param string $group_by
* @return array
*/
public function prepare_chart_data( $data, $date_key, $data_key, $interval, $start_date, $group_by ) {
$prepared_data = array();
2014-07-23 17:17:01 +00:00
2016-12-16 12:35:33 +00:00
// Ensure all days (or months) have values in this range.
if ( 'day' === $group_by ) {
for ( $i = 0; $i <= $interval; $i ++ ) {
$time = strtotime( date( 'Ymd', strtotime( "+{$i} DAY", $start_date ) ) ) . '000';
if ( ! isset( $prepared_data[ $time ] ) ) {
$prepared_data[ $time ] = array( esc_js( $time ), 0 );
}
}
2016-12-16 12:35:33 +00:00
} else {
$current_yearnum = date( 'Y', $start_date );
$current_monthnum = date( 'm', $start_date );
2014-07-23 17:17:01 +00:00
2016-12-16 12:35:33 +00:00
for ( $i = 0; $i <= $interval; $i ++ ) {
$time = strtotime( $current_yearnum . str_pad( $current_monthnum, 2, '0', STR_PAD_LEFT ) . '01' ) . '000';
if ( ! isset( $prepared_data[ $time ] ) ) {
$prepared_data[ $time ] = array( esc_js( $time ), 0 );
}
$current_monthnum ++;
if ( $current_monthnum > 12 ) {
$current_monthnum = 1;
$current_yearnum ++;
}
}
}
foreach ( $data as $d ) {
switch ( $group_by ) {
case 'day' :
$time = strtotime( date( 'Ymd', strtotime( $d->$date_key ) ) ) . '000';
break;
case 'month' :
2015-01-20 13:50:25 +00:00
default :
$time = strtotime( date( 'Ym', strtotime( $d->$date_key ) ) . '01' ) . '000';
break;
}
if ( ! isset( $prepared_data[ $time ] ) ) {
continue;
}
if ( $data_key ) {
2013-07-09 14:31:22 +00:00
$prepared_data[ $time ][1] += $d->$data_key;
} else {
2013-07-09 14:31:22 +00:00
$prepared_data[ $time ][1] ++;
}
}
return $prepared_data;
}
/**
* Prepares a sparkline to show sales in the last X days.
*
2013-07-25 14:00:23 +00:00
* @param int $id ID of the product to show. Blank to get all orders.
* @param int $days Days of stats to get.
* @param string $type Type of sparkline to get. Ignored if ID is not set.
* @return string
*/
2013-07-25 14:00:23 +00:00
public function sales_sparkline( $id = '', $days = 7, $type = 'sales' ) {
if ( $id ) {
2016-09-09 00:14:28 +00:00
$meta_key = ( 'sales' === $type ) ? '_line_total' : '_qty';
2013-07-25 14:00:23 +00:00
$data = $this->get_order_report_data( array(
'data' => array(
'_product_id' => array(
'type' => 'order_item_meta',
'order_item_type' => 'line_item',
'function' => '',
'name' => 'product_id',
2013-07-25 14:00:23 +00:00
),
$meta_key => array(
'type' => 'order_item_meta',
'order_item_type' => 'line_item',
'function' => 'SUM',
'name' => 'sparkline_value',
2013-07-25 14:00:23 +00:00
),
'post_date' => array(
'type' => 'post_data',
'function' => '',
'name' => 'post_date',
2013-07-25 14:00:23 +00:00
),
),
2013-07-25 14:00:23 +00:00
'where' => array(
array(
'key' => 'post_date',
'value' => date( 'Y-m-d', strtotime( 'midnight -' . ( $days - 1 ) . ' days', current_time( 'timestamp' ) ) ),
'operator' => '>',
2013-07-25 14:00:23 +00:00
),
array(
'key' => 'order_item_meta__product_id.meta_value',
'value' => $id,
'operator' => '=',
),
),
'group_by' => 'YEAR(posts.post_date), MONTH(posts.post_date), DAY(posts.post_date)',
2013-07-25 14:00:23 +00:00
'query_type' => 'get_results',
'filter_range' => false,
2013-07-25 14:00:23 +00:00
) );
} else {
2013-07-25 14:00:23 +00:00
$data = $this->get_order_report_data( array(
'data' => array(
'_order_total' => array(
'type' => 'meta',
'function' => 'SUM',
'name' => 'sparkline_value',
2013-07-25 14:00:23 +00:00
),
'post_date' => array(
'type' => 'post_data',
'function' => '',
'name' => 'post_date',
2013-07-25 14:00:23 +00:00
),
),
2013-07-25 14:00:23 +00:00
'where' => array(
array(
'key' => 'post_date',
'value' => date( 'Y-m-d', strtotime( 'midnight -' . ( $days - 1 ) . ' days', current_time( 'timestamp' ) ) ),
'operator' => '>',
),
),
'group_by' => 'YEAR(posts.post_date), MONTH(posts.post_date), DAY(posts.post_date)',
2013-07-25 14:00:23 +00:00
'query_type' => 'get_results',
'filter_range' => false,
2013-07-25 14:00:23 +00:00
) );
}
$total = 0;
foreach ( $data as $d ) {
2013-07-25 14:00:23 +00:00
$total += $d->sparkline_value;
}
2016-09-09 00:14:28 +00:00
if ( 'sales' === $type ) {
2016-10-29 20:03:28 +00:00
/* translators: 1: total income 2: days */
2016-09-01 20:50:14 +00:00
$tooltip = sprintf( __( 'Sold %1$s worth in the last %2$d days', 'woocommerce' ), strip_tags( wc_price( $total ) ), $days );
} else {
2016-10-29 20:03:28 +00:00
/* translators: 1: total items sold 2: days */
2016-09-01 20:50:14 +00:00
$tooltip = sprintf( _n( 'Sold 1 item in the last %2$d days', 'Sold %1$d items in the last %2$d days', $total, 'woocommerce' ), $total, $days );
}
2013-07-25 14:00:23 +00:00
$sparkline_data = array_values( $this->prepare_chart_data( $data, 'post_date', 'sparkline_value', $days - 1, strtotime( 'midnight -' . ( $days - 1 ) . ' days', current_time( 'timestamp' ) ), 'day' ) );
2016-09-09 00:14:28 +00:00
return '<span class="wc_sparkline ' . ( ( 'sales' === $type ) ? 'lines' : 'bars' ) . ' tips" data-color="#777" data-tip="' . esc_attr( $tooltip ) . '" data-barwidth="' . 60 * 60 * 16 * 1000 . '" data-sparkline="' . esc_attr( json_encode( $sparkline_data ) ) . '"></span>';
}
/**
* Get the current range and calculate the start and end dates.
*
* @param string $current_range
*/
public function calculate_current_range( $current_range ) {
switch ( $current_range ) {
case 'custom' :
2017-04-03 09:46:37 +00:00
$this->start_date = max( strtotime( '-20 years' ), strtotime( sanitize_text_field( $_GET['start_date'] ) ) );
2017-03-27 11:11:08 +00:00
if ( empty( $_GET['end_date'] ) ) {
$this->end_date = strtotime( 'midnight', current_time( 'timestamp' ) );
} else {
$this->end_date = strtotime( 'midnight', strtotime( sanitize_text_field( $_GET['end_date'] ) ) );
}
$interval = 0;
$min_date = $this->start_date;
while ( ( $min_date = strtotime( "+1 MONTH", $min_date ) ) <= $this->end_date ) {
$interval ++;
}
// 3 months max for day view
if ( $interval > 3 ) {
$this->chart_groupby = 'month';
} else {
$this->chart_groupby = 'day';
}
break;
case 'year' :
$this->start_date = strtotime( date( 'Y-01-01', current_time( 'timestamp' ) ) );
$this->end_date = strtotime( 'midnight', current_time( 'timestamp' ) );
$this->chart_groupby = 'month';
break;
case 'last_month' :
$first_day_current_month = strtotime( date( 'Y-m-01', current_time( 'timestamp' ) ) );
$this->start_date = strtotime( date( 'Y-m-01', strtotime( '-1 DAY', $first_day_current_month ) ) );
$this->end_date = strtotime( date( 'Y-m-t', strtotime( '-1 DAY', $first_day_current_month ) ) );
$this->chart_groupby = 'day';
break;
case 'month' :
$this->start_date = strtotime( date( 'Y-m-01', current_time( 'timestamp' ) ) );
$this->end_date = strtotime( 'midnight', current_time( 'timestamp' ) );
$this->chart_groupby = 'day';
break;
case '7day' :
$this->start_date = strtotime( '-6 days', strtotime( 'midnight', current_time( 'timestamp' ) ) );
$this->end_date = strtotime( 'midnight', current_time( 'timestamp' ) );
$this->chart_groupby = 'day';
break;
}
// Group by
switch ( $this->chart_groupby ) {
case 'day' :
$this->group_by_query = 'YEAR(posts.post_date), MONTH(posts.post_date), DAY(posts.post_date)';
2016-06-07 13:02:02 +00:00
$this->chart_interval = absint( ceil( max( 0, ( $this->end_date - $this->start_date ) / ( 60 * 60 * 24 ) ) ) );
$this->barwidth = 60 * 60 * 24 * 1000;
break;
case 'month' :
$this->group_by_query = 'YEAR(posts.post_date), MONTH(posts.post_date)';
$this->chart_interval = 0;
2016-12-16 12:35:33 +00:00
$min_date = strtotime( date( 'Y-m-01', $this->start_date ) );
2016-12-16 12:35:33 +00:00
while ( ( $min_date = strtotime( "+1 MONTH", $min_date ) ) <= $this->end_date ) {
$this->chart_interval ++;
}
$this->barwidth = 60 * 60 * 24 * 7 * 4 * 1000;
break;
}
}
2014-09-19 07:23:58 +00:00
/**
* Return currency tooltip JS based on WooCommerce currency position settings.
*
* @return string
*/
public function get_currency_tooltip() {
switch ( get_option( 'woocommerce_currency_pos' ) ) {
2014-09-19 07:23:58 +00:00
case 'right':
$currency_tooltip = 'append_tooltip: "' . get_woocommerce_currency_symbol() . '"';
break;
2014-09-19 07:23:58 +00:00
case 'right_space':
$currency_tooltip = 'append_tooltip: "&nbsp;' . get_woocommerce_currency_symbol() . '"';
break;
case 'left':
$currency_tooltip = 'prepend_tooltip: "' . get_woocommerce_currency_symbol() . '"';
break;
case 'left_space':
default:
$currency_tooltip = 'prepend_tooltip: "' . get_woocommerce_currency_symbol() . '&nbsp;"';
break;
2014-09-19 07:23:58 +00:00
}
return $currency_tooltip;
}
/**
* Get the main chart.
*/
public function get_main_chart() {}
/**
* Get the legend for the main chart sidebar.
*
* @return array
*/
public function get_chart_legend() {
return array();
}
/**
* Get chart widgets.
*
* @return array
*/
public function get_chart_widgets() {
return array();
}
2013-07-18 11:56:12 +00:00
/**
* Get an export link if needed.
2013-07-18 11:56:12 +00:00
*/
public function get_export_button() {}
/**
* Output the report.
*/
public function output_report() {}
/**
* Check nonce for current range.
*
* @since 3.0.4
* @param string $current_range Current range.
*/
public function check_current_range_nonce( $current_range ) {
if ( 'custom' !== $current_range ) {
return;
}
if ( ! isset( $_GET['wc_reports_nonce'] ) || ! wp_verify_nonce( sanitize_key( $_GET['wc_reports_nonce'] ), 'custom_range' ) ) { // WPCS: input var ok, CSRF ok.
wp_die(
/* translators: %1$s: open link, %2$s: close link */
sprintf( esc_html__( 'This report link has expired. %1$sClick here to view the filtered report%2$s.', 'woocommerce' ), '<a href="' . esc_url( wp_nonce_url( esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ), 'custom_range', 'wc_reports_nonce' ) ) . '">', '</a>' ), // @codingStandardsIgnoreLine.
esc_attr__( 'Confirm navigation', 'woocommerce' )
);
exit;
}
}
2013-11-21 13:28:09 +00:00
}