3v4l.org

run code in 300+ PHP versions simultaneously
<?php class Cache { public static function has( $name ){ return false; } public static function get( $name ){ return null; } public static function put( $name, $value, $time = 1440 ){ return true; } } class Collivery { protected $token; protected $client; protected $config; protected $errors = array(); protected $check_cache = 2; protected $default_address_id; protected $client_id; protected $user_id; /** * Setup class with basic Config * * @param Array $config Configuration Array */ function __construct( array $config = array() ) { $this->config = (object) array( 'app_name' => 'Default App Name', // Application Name 'app_version' => '0.0.1', // Application Version 'app_host' => '', // Framework/CMS name and version, eg 'Wordpress 3.8.1 WooCommerce 2.0.20' / 'Joomla! 2.5.17 VirtueMart 2.0.26d' 'app_url' => '', // URL your site is hosted on 'user_email' => 'demo@collivery.co.za', 'user_password' => 'demo', 'demo' => false, ); foreach ( $config as $key => $value ) { $this->config->$key = $value; } if ( $this->config->demo ){ $this->config->user_email = 'demo@collivery.co.za'; $this->config->user_password = 'demo'; } $this->authenticate(); } /** * Setup the Soap Object * * @return SoapClient MDS Collivery Soap Client */ protected function init () { if ( ! $this->client ){ try { $this->client = new SoapClient( // Setup the soap client 'http://www.collivery.co.za/wsdl/v2', // URL to WSDL File array( 'cache_wsdl' => WSDL_CACHE_NONE ) // Don't cache the WSDL file ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } } return true; } /** * Checks if the Soap Client has been set, and returns it. * * @return SoapClient Webserver Soap Client */ protected function client() { if ( ! $this->client ) { $this->init(); } if ( ! $this->token ) { $this->authenticate(); } return $this->client; } /** * Authenticate and set the token * * @return string */ protected function authenticate() { if ( ( $this->check_cache == 2 ) && Cache::has('collivery.auth') ) { $authenticate = Cache::get('collivery.auth'); $this->default_address_id = $authenticate['default_address_id']; $this->client_id = $authenticate['client_id']; $this->user_id = $authenticate['user_id']; $this->token = $authenticate['token']; return true; } else { if ( ! $this->init() ) return false; $user_email = $this->config->user_email; $user_password = $this->config->user_password; try { $authenticate = $this->client->authenticate($user_email, $user_password, $this->token, array( 'name' => $this->config->app_name . ' mds/collivery/class', 'version' => $this->config->app_version, 'host' => $this->config->app_host, 'url' => $this->config->app_url, 'lang' => 'PHP '. phpversion(), )); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( is_array( $authenticate ) && isset( $authenticate['token'] ) ){ if ( $this->check_cache != 0 ) Cache::put( 'collivery.auth', $authenticate, 50 ); $this->default_address_id = $authenticate['default_address_id']; $this->client_id = $authenticate['client_id']; $this->user_id = $authenticate['user_id']; $this->token = $authenticate['token']; return true; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Returns a list of avaibale parcel images for a given Waybill Number. * * @param int $collivery_id Collivery waybill number * @return array */ public function getParcelImageList( $collivery_id ) { if ( ( $this->check_cache == 2 ) && Cache::has( 'collivery.parcel_image_list.'. $this->client_id .'.'. $collivery_id ) ) { return Cache::get( 'collivery.parcel_image_list.'. $this->client_id .'.'. $collivery_id ); } else { try { $result = $this->client()->get_parcel_image_list( $collivery_id, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['images'] ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else if ( $this->check_cache != 0 ) Cache::put( 'collivery.parcel_image_list.'. $this->client_id .'.'. $collivery_id, $result['images'], 60*12 ); return $result['images']; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Returns the image of a given parcel-id of a waybill. * If the Waybill number is 54321 and there are 3 parcels, they would * be referenced by id's 54321-1, 54321-2 and 54321-3. * * @param string $parcel_id Parcel ID * @return array Array containing all the information * about the image including the image * itself in base64 */ public function getParcelImage( $parcel_id ) { if ( ( $this->check_cache == 2 ) && Cache::has( 'collivery.parcel_image.'. $this->client_id .'.'. $parcel_id ) ) { return Cache::get( 'collivery.parcel_image.'. $this->client_id .'.'. $parcel_id ); } else { try { $result = $this->client()->get_parcel_image( $parcel_id, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['image'] ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else if ( $this->check_cache != 0 ) Cache::put( 'collivery.parcel_image.'. $this->client_id .'.'. $parcel_id, $result['image'], 60*24 ); return $result['image']; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Returns the status tracking detail of a given Waybill number. * If the collivery is still active, the estimated time of delivery * will be provided. If delivered, the time and receivers name (if availble) * with returned. * * @param int $collivery_id Collivery ID * @return array Collivery Status Information */ public function getStatus( $collivery_id ) { if ( ( $this->check_cache == 2 ) && Cache::has( 'collivery.status.'. $this->client_id .'.'. $collivery_id ) ) { return Cache::get( 'collivery.status.'. $this->client_id .'.'. $collivery_id ); } else { try { $result = $this->client()->get_collivery_status( $collivery_id, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['status_id'] ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else if ( $this->check_cache != 0 ) Cache::put( 'collivery.status.'. $this->client_id .'.'. $collivery_id, $result, 60*12 ); return $result; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Create a new Address and Contact * * @param array $data Address and Contact Information * @return array Address ID and Contact ID */ public function addAddress( array $data ) { if ( ! isset( $data['location_type'] ) ) $this->setError( 'missing_data', 'location_type not set.' ); elseif ( ! isset( $this->getLocationTypes()[ $data['location_type'] ] ) ) $this->setError( 'invalid_data', 'Invalid location_type.' ); if ( ! isset( $data['town_id'] ) ) $this->setError( 'missing_data', 'town_id not set.' ); elseif ( ! isset( $this->getTowns()[ $data['town_id'] ] ) ) $this->setError( 'invalid_data', 'Invalid town_id.' ); if ( ! isset( $data['suburb_id'] ) ) $this->setError( 'missing_data', 'suburb_id not set.' ); elseif ( ! isset( $this->getSuburbs( $data['town_id'] )[ $data['suburb_id'] ] ) ) $this->setError( 'invalid_data', 'Invalid suburb_id.' ); if ( ! isset( $data['street'] ) ) $this->setError( 'missing_data', 'street not set.' ); if ( ! isset( $data['full_name'] ) ) $this->setError( 'missing_data', 'full_name not set.' ); if ( isset( $data['phone'] ) || isset( $data['cellphone'] ) ) $this->setError( 'missing_data', 'Please supply ether a phone or cellphone number...' ); if ( ! $this->hasErrors() ) { try { $result = $this->client()->add_address( $data, $this->token ); Cache::forget( 'collivery.addresses.'. $this->client_id ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['address_id'] ) ) { return $result; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Add's a contact person for a given Address ID * * @param array $data New Contact Data * @return int New Contact ID */ public function addContact( array $data ) { if ( ! isset( $data['address_id'] ) ) $this->setError( 'missing_data', 'address_id not set.' ); elseif ( ! is_array( $this->getAddress( $data['address_id'] ) ) ) $this->setError( 'invalid_data', 'Invalid address_id.' ); if ( ! isset( $data['street'] ) ) $this->setError( 'missing_data', 'street not set.' ); if ( ! isset( $data['full_name'] ) ) $this->setError( 'missing_data', 'full_name not set.' ); if ( isset( $data['phone'] ) || isset( $data['cellphone'] ) ) $this->setError( 'missing_data', 'Please supply ether a phone or cellphone number...' ); if ( ! $this->hasErrors() ) { try { $result = $this->client()->add_address( $data, $this->token ); Cache::forget( 'collivery.addresses.'. $this->client_id ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['address_id'] ) ) { return $result; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Returns the price based on the data provided. * * @param array $data Your Collivery Details * @return array Pricing for details supplied */ public function getPrice( array $data ) { if ( ! isset( $data['collivery_from'] ) && ! isset( $data['from_town_id'] ) ) $this->setError( 'missing_data', 'collivery_from/from_town_id not set.' ); elseif ( isset( $data['collivery_from'] ) && ! is_array( $this->getAddress( $data['collivery_from'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_from.' ); elseif ( isset( $data['from_town_id'] ) && ! isset( $this->getTowns()[ $data['from_town_id'] ] ) ) $this->setError( 'invalid_data', 'Invalid Town ID for: from_town_id.' ); if ( ! isset( $data['collivery_to'] ) && ! isset( $data['to_town_id'] ) ) $this->setError( 'missing_data', 'collivery_to/to_town_id not set.' ); elseif ( isset( $data['collivery_to'] ) && ! is_array( $this->getAddress( $data['collivery_to'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_to.' ); elseif ( isset( $data['to_town_id'] ) && ! isset( $this->getTowns()[ $data['to_town_id'] ] ) ) $this->setError( 'invalid_data', 'Invalid Town ID for: to_town_id.' ); if ( ! isset( $data['service'] ) ) $this->setError( 'missing_data', 'service not set.' ); if ( ! $this->hasErrors() ) { try { $result = $this->client()->get_price( $data, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( is_array( $result ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); return $result; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Validate Collivery * * Returns the validated data array of all details pertaining to a collivery. * This process validates the information based on services, time frames and parcel information. * Dates and times may be altered during this process based on the collection and delivery towns service parameters. * Certain towns are only serviced on specific days and between certain times. * This function automatically alters the values. * The parcels volumetric calculations are also done at this time. * It is important that the data is first validated before a collivery can be added. * * @param array $data Properties of the new Collivery * @return array The validated data */ public function validate( array $data ) { if ( ! isset( $data['collivery_from'] ) ) $this->setError( 'missing_data', 'collivery_from not set.' ); elseif ( ! is_array( $this->getAddress( $data['collivery_from'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_from.' ); if ( ! isset( $data['contact_from'] ) ) $this->setError( 'missing_data', 'contact_from not set.' ); elseif ( ! isset( $this->getContacts( $data['collivery_from'] )[ $data['contact_from'] ] ) ) $this->setError( 'invalid_data', 'Invalid Contact ID for: contact_from.' ); if ( ! isset( $data['collivery_to'] ) ) $this->setError( 'missing_data', 'collivery_to not set.' ); elseif ( ! is_array( $this->getAddress( $data['collivery_to'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_to.' ); if ( ! isset( $data['contact_to'] ) ) $this->setError( 'missing_data', 'contact_to not set.' ); elseif ( ! isset( $this->getContacts( $data['collivery_from'] )[ $data['contact_to'] ] ) ) $this->setError( 'invalid_data', 'Invalid Contact ID for: contact_to.' ); if ( ! isset( $data['collivery_type'] ) ) $this->setError( 'missing_data', 'collivery_type not set.' ); elseif ( ! isset( $this->getParcelTypes()[ $data['collivery_type'] ] ) ) $this->setError( 'invalid_data', 'Invalid collivery_type.' ); if ( ! isset( $data['service'] ) ) $this->setError( 'missing_data', 'service not set.' ); elseif ( ! isset( $this->getServices()[ $data['service'] ] ) ) $this->setError( 'invalid_data', 'Invalid service.' ); if ( ! $this->hasErrors() ) { try { $result = $this->client()->validate_collivery( $data, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( is_array( $result ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); return $result; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Creates a new Collivery based on the data array provided. * The array should first be validated before passing to this function. * The Waybill No is return apon successful creation of the collivery. * * @param array $data Properties of the new Collivery * @return int New Collivery ID */ public function addCollivery( array $data ) { if ( ! isset( $data['collivery_from'] ) ) $this->setError( 'missing_data', 'collivery_from not set.' ); elseif ( ! is_array( $this->getAddress( $data['collivery_from'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_from.' ); if ( ! isset( $data['contact_from'] ) ) $this->setError( 'missing_data', 'contact_from not set.' ); elseif ( ! isset( $this->getContacts( $data['collivery_from'] )[ $data['contact_from'] ] ) ) $this->setError( 'invalid_data', 'Invalid Contact ID for: contact_from.' ); if ( ! isset( $data['collivery_to'] ) ) $this->setError( 'missing_data', 'collivery_to not set.' ); elseif ( ! is_array( $this->getAddress( $data['collivery_to'] ) ) ) $this->setError( 'invalid_data', 'Invalid Address ID for: collivery_to.' ); if ( ! isset( $data['contact_to'] ) ) $this->setError( 'missing_data', 'contact_to not set.' ); elseif ( ! isset( $this->getContacts( $data['collivery_from'] )[ $data['contact_to'] ] ) ) $this->setError( 'invalid_data', 'Invalid Contact ID for: contact_to.' ); if ( ! isset( $data['collivery_type'] ) ) $this->setError( 'missing_data', 'collivery_type not set.' ); elseif ( ! isset( $this->getParcelTypes()[ $data['collivery_type'] ] ) ) $this->setError( 'invalid_data', 'Invalid collivery_type.' ); if ( ! isset( $data['service'] ) ) $this->setError( 'missing_data', 'service not set.' ); elseif ( ! isset( $this->getServices()[ $data['service'] ] ) ) $this->setError( 'invalid_data', 'Invalid service.' ); if ( ! $this->hasErrors() ) { try { $result = $this->client()->add_collivery( $data, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['collivery_id'] ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); return $result['collivery_id']; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } } /** * Accepts the newly created Collivery, moving it from Waiting Client Acceptance * to Accepted so that it can be processed. * * @param int $collivery_id ID of the Collivery you wish to accept * @return boolean Has the Collivery been accepted */ public function acceptCollivery( $collivery_id ) { try { $result = $this->client()->accept_collivery( $collivery_id, $this->token ); } catch (SoapFault $e) { $this->catchSoapFault( $e ); return false; } if ( isset( $result['result'] ) ) { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); return $result['result'] == 'Accepted'; } else { if ( isset( $result['error_id'] ) ) $this->setError( $result['error_id'], $result['error'] ); else $this->setError( 'result_unexpected', 'No address_id returned.' ); return false; } } /** * Handle error messages in SoapFault * * @param SoapFault $e SoapFault Object */ protected function catchSoapFault( $e ) { $this->setError( $e->faultcode, $e->faultstring ); } /** * Add a new error * * @param string $id Error ID * @param string $text Error text */ protected function setError( $id, $text ) { $this->errors[ $id ] = $text; } /** * Retrieve errors */ public function getErrors() { return $this->errors; } /** * Check if this instance has an error */ public function hasErrors() { return !empty($this->errors); } /** * Clears all the Errors */ public function clearErrors() { $this->errors = array(); } /** * Disable Cached completely and retrieve data directly from the webservice */ public function disableCache() { $this->check_cache = 0; } /** * Ignore Cached data and retrieve data directly from the webservice * Save returned data to Cache */ public function ignoreCache() { $this->check_cache = 1; } /** * Check if cache exists before querying the webservice * If webservice was queried, save returned data to Cache */ public function enableCache() { $this->check_cache = 2; } }

Here you find the average performance (time & memory) of each version. A grayed out version indicates it didn't complete successfully (based on exit-code).

VersionSystem time (s)User time (s)Memory (MiB)
7.4.00.0110.01115.17
7.3.120.0060.00914.74
7.3.110.0160.00614.69
7.3.100.0060.00914.86
7.3.90.0000.01315.08
7.3.80.0070.00314.86
7.3.70.0040.01114.64
7.3.60.0070.00714.91
7.3.50.0040.01114.84
7.3.40.0100.00014.90
7.3.30.0030.01214.93
7.3.20.0030.01016.61
7.3.10.0040.01216.46
7.3.00.0070.00716.51
7.2.250.0100.01314.79
7.2.240.0080.00815.01
7.2.230.0110.00314.93
7.2.220.0130.00315.17
7.2.210.0120.00615.01
7.2.200.0070.00714.99
7.2.190.0090.00914.95
7.2.180.0040.01115.00
7.2.170.0070.01014.73
7.2.00.0080.00518.92
7.1.330.0100.00315.23
7.1.320.0100.00715.39
7.1.310.0030.00715.61
7.1.300.0080.00315.41
7.1.290.0030.01015.66
7.1.280.0030.00915.30
7.1.270.0100.00315.61
7.1.260.0060.00915.38
7.1.100.0090.00018.00
7.1.70.0050.00517.05
7.1.60.0130.01319.43
7.1.50.0230.01334.45
7.1.00.0000.07022.32
7.0.200.0070.00416.37
7.0.140.0070.07022.01
7.0.100.0030.07320.20
7.0.90.0070.08020.20
7.0.80.0100.07720.09
7.0.70.0100.07720.18
7.0.60.0030.09320.23
7.0.50.0200.07020.48
7.0.40.0070.09020.14
7.0.30.0130.07720.13
7.0.20.0130.08020.13
7.0.10.0070.08720.12
7.0.00.0130.07720.13
5.6.280.0100.07020.91
5.6.250.0070.07720.95
5.6.240.0070.04720.71
5.6.230.0130.06720.64
5.6.220.0130.04020.78
5.6.210.0030.08720.76
5.6.200.0170.07721.16
5.6.190.0070.08321.30
5.6.180.0100.07321.25
5.6.170.0100.06321.08
5.6.160.0070.08721.17
5.6.150.0100.04721.25
5.6.140.0170.07321.16
5.6.130.0070.05021.15
5.6.120.0100.04021.15
5.6.110.0130.07721.16
5.6.100.0030.08721.21
5.6.90.0030.05321.14
5.6.80.0030.05320.52
5.6.70.0100.07320.62
5.6.60.0100.07720.53
5.6.50.0070.08320.63
5.6.40.0000.06020.52
5.6.30.0100.07720.57
5.6.20.0100.07020.49
5.6.10.0070.08320.46
5.6.00.0100.07320.49
5.5.380.0070.08020.70
5.5.370.0100.07020.58
5.5.360.0070.08320.43
5.5.350.0100.09720.59
5.5.340.0130.08020.83
5.5.330.0100.08021.00
5.5.320.0070.08720.97
5.5.310.0130.04321.00
5.5.300.0200.07020.99
5.5.290.0070.09020.93
5.5.280.0070.07020.90
5.5.270.0070.05021.01
5.5.260.0130.08020.99
5.5.250.0030.06020.85
5.5.240.0030.08320.32
5.5.230.0070.05320.24
5.5.220.0100.05720.31
5.5.210.0100.08020.31
5.5.200.0130.04320.39
5.5.190.0070.07320.41
5.5.180.0100.04720.13
5.5.160.0130.07320.30
5.5.150.0130.07320.41
5.5.140.0070.08320.40
5.5.130.0170.06720.28
5.5.120.0030.06720.11
5.5.110.0030.07720.39
5.5.100.0130.07320.21
5.5.90.0170.07320.18
5.5.80.0070.08320.24
5.5.70.0070.08320.20
5.5.60.0070.05020.20
5.5.50.0070.04720.29
5.5.40.0130.07320.28
5.5.30.0230.06320.18
5.5.20.0100.07720.19
5.5.10.0000.08320.23
5.5.00.0100.08020.16
5.4.450.0130.07719.41
5.4.440.0070.08019.49
5.4.430.0030.08719.29
5.4.420.0100.07719.42
5.4.410.0000.08719.16
5.4.400.0130.05319.13
5.4.390.0070.07018.96
5.4.380.0030.08319.23
5.4.370.0030.07719.06
5.4.360.0030.05319.12
5.4.350.0130.06718.93
5.4.340.0070.08319.09
5.4.320.0070.07319.13
5.4.310.0070.07719.10
5.4.300.0100.07019.28
5.4.290.0100.07319.18
5.4.280.0070.07019.12
5.4.270.0070.08019.10
5.4.260.0200.07019.15
5.4.250.0030.04319.18
5.4.240.0170.06719.17
5.4.230.0100.07718.91
5.4.220.0070.08019.09
5.4.210.0030.05019.17
5.4.200.0030.08718.96
5.4.190.0100.08019.01
5.4.180.0130.07018.91
5.4.170.0030.07719.09
5.4.160.0230.04319.08
5.4.150.0100.07019.16
5.4.140.0100.06716.53
5.4.130.0030.06016.31
5.4.120.0000.07716.42
5.4.110.0070.07316.48
5.4.100.0070.08016.49
5.4.90.0070.07016.59
5.4.80.0030.05016.48
5.4.70.0030.07316.43
5.4.60.0100.07016.45
5.4.50.0100.06016.36
5.4.40.0070.07716.36
5.4.30.0030.07716.51
5.4.20.0100.06716.45
5.4.10.0100.06316.41
5.4.00.0030.07716.01
5.3.290.0070.06014.80
5.3.280.0070.08014.81
5.3.270.0100.07314.75
5.3.260.0100.07714.80
5.3.250.0100.06714.66
5.3.240.0100.07314.63
5.3.230.0070.07314.81
5.3.220.0070.07314.62
5.3.210.0170.07314.65
5.3.200.0030.07314.64
5.3.190.0030.05714.79
5.3.180.0030.08014.63
5.3.170.0030.07714.57
5.3.160.0030.08014.78
5.3.150.0100.07314.57
5.3.140.0100.06314.69
5.3.130.0000.05014.56
5.3.120.0130.07314.69
5.3.110.0170.07014.56
5.3.100.0100.07014.09
5.3.90.0030.07314.18
5.3.80.0170.06314.13
5.3.70.0000.08314.21
5.3.60.0000.04314.19
5.3.50.0200.06014.14
5.3.40.0070.06013.91
5.3.30.0000.04014.05
5.3.20.0070.06013.76
5.3.10.0030.07313.78
5.3.00.0030.05313.73

preferences:
59.32 ms | 400 KiB | 5 Q