| 
						 | 
						- <?php
 - namespace App\Jobs;
 - 
 - use DB;
 - use Log;
 - use Quartz;
 - use p3k\Multipart;
 - use App\Jobs\Job;
 - use Illuminate\Contracts\Bus\SelfHandling;
 - use Illuminate\Contracts\Queue\ShouldQueue;
 - use DateTime, DateTimeZone;
 - 
 - class TripComplete extends Job implements SelfHandling, ShouldQueue
 - {
 -   private $_dbid;
 -   private $_data;
 - 
 -   public function __construct($dbid, $data) {
 -     $this->_dbid = $dbid;
 -     $this->_data = $data;
 -   }
 - 
 -   public function handle() {
 -     // echo "Job Data\n";
 -     // echo json_encode($this->_data)."\n";
 -     if(!is_array($this->_data)) return;
 - 
 -     $db = DB::table('databases')->where('id','=',$this->_dbid)->first();
 - 
 -     Log::info("Starting job for ".$db->name);
 - 
 -     Log::debug(json_encode($this->_data));
 - 
 -     if(!$db->micropub_endpoint) {
 -       Log::info('No micropub endpoint configured for database ' . $db->name);
 -       return;
 -     }
 - 
 -     $qz = new Quartz\DB(env('STORAGE_DIR').$db->name, 'r');
 - 
 -     // Load the data from the start and end times
 -     $start = new DateTime($this->_data['properties']['start']);
 -     $end = new DateTime($this->_data['properties']['end']);
 -     $results = $qz->queryRange($start, $end);
 -     $features = [];
 -     foreach($results as $id=>$record) {
 -       // Don't include app action tracking data
 -       if(!property_exists($record->data->properties, 'action')) {
 - 	    // Ignore locations with accuracy worse than 5000m
 - 	    if(property_exists($record->data->properties, 'horizontal_accuracy') && $record->data->properties->horizontal_accuracy <= 5000) {
 - 	      $record->data->properties = array_filter((array)$record->data->properties, function($k){
 - 	        // Remove some of the app-specific tracking keys from each record
 - 	        return !in_array($k, ['locations_in_payload','desired_accuracy','significant_change','pauses','deferred']);
 - 	      }, ARRAY_FILTER_USE_KEY);
 - 	      $features[] = $record->data;
 - 	    }
 -       }
 -     }
 - 
 -     // Build the GeoJSON for this trip
 -     $geojson = [
 -       'type' => 'FeatureCollection',
 -       'features' => $features
 -     ];
 -     $file_path = tempnam(sys_get_temp_dir(), 'compass');
 -     file_put_contents($file_path, json_encode($geojson));
 - 
 -     // If there are no start/end coordinates in the request, use the first and last coordinates
 -     if(count($features)) {
 -       if(!array_key_exists('start-coordinates', $this->_data['properties'])) {
 -         $this->_data['properties']['start-coordinates'] = $features[0]->geometry->coordinates;
 -       }
 -       if(!array_key_exists('end-coordinates', $this->_data['properties'])) {
 -         $this->_data['properties']['end-coordinates'] = $features[count($features)-1]->geometry->coordinates;
 -       }
 -     }
 - 
 -     $startAdr = false;
 -     if(array_key_exists('start-coordinates', $this->_data['properties'])) {
 -       // Reverse geocode the start and end location to get an h-adr
 -       $startAdr = [
 -         'type' => 'h-adr',
 -         'properties' => [
 -           'latitude' => $this->_data['properties']['start-coordinates'][1],
 -           'longitude' => $this->_data['properties']['start-coordinates'][0],
 -         ]
 -       ];
 -       Log::info('Looking up start location');
 -       $start = self::geocode($this->_data['properties']['start-coordinates'][1], $this->_data['properties']['start-coordinates'][0]);
 -       if($start) {
 -         $startAdr['properties']['locality'] = $start->locality;
 -         $startAdr['properties']['region'] = $start->region;
 -         $startAdr['properties']['country'] = $start->country;
 -         Log::info('Found start: '.$start->full_name.' '.$start->timezone);
 -       }
 -     } else {
 -       $start = false;
 -     }
 - 
 -     $endAdr = false;
 -     if(array_key_exists('end-coordinates', $this->_data['properties'])) {
 -       $endAdr = [
 -         'type' => 'h-adr',
 -         'properties' => [
 -           'latitude' => $this->_data['properties']['end-coordinates'][1],
 -           'longitude' => $this->_data['properties']['end-coordinates'][0],
 -         ]
 -       ];
 -       Log::info('Looking up end location');
 -       $end = self::geocode($this->_data['properties']['end-coordinates'][1], $this->_data['properties']['end-coordinates'][0]);
 -       if($end) {
 -         $endAdr['properties']['locality'] = $end->locality;
 -         $endAdr['properties']['region'] = $end->region;
 -         $endAdr['properties']['country'] = $end->country;
 -         Log::info('Found end: '.$end->full_name.' '.$end->timezone);
 -       }
 -     } else {
 -       $end = false;
 -     }
 - 
 -     // Set the timezone of the dates based on the location
 -     $startDate = new DateTime($this->_data['properties']['start']);
 -     if($start && $start->timezone) {
 -       $startDate->setTimeZone(new DateTimeZone($start->timezone));
 -     }
 - 
 -     $endDate = new DateTime($this->_data['properties']['end']);
 -     if($end && $end->timezone) {
 -       $endDate->setTimeZone(new DateTimeZone($end->timezone));
 -     }
 - 
 -     if($endDate->format('U') - $startDate->format('U') < 15) {
 -       Log::info("Skipping trip since it was too short");
 -       return;
 -     }
 - 
 -     $params = [
 -       'h' => 'entry',
 -       'published' => $endDate->format('c'),
 -       'trip' => [
 -         'type' => 'h-trip',
 -         'properties' => [
 -           'mode-of-transport' => $this->_data['properties']['mode'],
 -           'start' => $startDate->format('c'),
 -           'end' => $endDate->format('c'),
 -           'route' => 'route.json'
 -           // TODO: avgpace for runs
 -           // TODO: avgspeed for bike rides
 -           // TODO: avg heart rate if available
 -         ]
 -       ]
 -     ];
 - 
 -     if($startAdr) {
 -       $params['trip']['properties']['start-location'] = $startAdr;
 -     }
 -     if($endAdr) {
 -       $params['trip']['properties']['end-location'] = $endAdr;
 -     }
 -     if(array_key_exists('distance', $this->_data['properties'])) {
 -       $params['trip']['properties']['distance'] = [
 -         'type' => 'h-measure',
 -         'properties' => [
 -           'num' => round($this->_data['properties']['distance']),
 -           'unit' => 'meter'
 -         ]
 -       ];
 -     }
 -     if(array_key_exists('duration', $this->_data['properties'])) {
 -       $params['trip']['properties']['duration'] = [
 -         'type' => 'h-measure',
 -         'properties' => [
 -           'num' => round($this->_data['properties']['duration']),
 -           'unit' => 'second'
 -         ]
 -       ];
 -     }
 -     if(array_key_exists('cost', $this->_data['properties'])) {
 -       $params['trip']['properties']['cost'] = [
 -         'type' => 'h-measure',
 -         'properties' => [
 -           'num' => round($this->_data['properties']['cost'], 2),
 -           'unit' => 'USD'
 -         ]
 -       ];
 -     }
 - 
 -     // If there is trip data, recalculate the distance and duration based on the actual data
 -     if(count($features)) {
 -       $startTime = strtotime($features[0]->properties['timestamp']);
 -       $endTime = strtotime($features[count($features)-1]->properties['timestamp']);
 -       $duration = $endTime - $startTime;
 -       $params['trip']['properties']['duration']['type'] = 'h-measure';
 -       $params['trip']['properties']['duration']['properties']['num'] = $duration;
 -       $params['trip']['properties']['duration']['properties']['unit'] = 'second';
 -       Log::debug("Overriding duration to $duration");
 - 
 -       $points = array_map(function($f){
 -         return $f->geometry->coordinates;
 -       }, $features);
 -       $simple = $this->_ramerDouglasPeucker($points, 0.0001);
 -       $last = false;
 -       $distance = 0;
 -       foreach($simple as $p) {
 -         if($last) {
 -           $distance += $this->_gc_distance($p[1], $p[0], $last[1], $last[0]);
 -         }
 -         $last = $p;
 -       }
 -       if($distance) {
 -         $params['trip']['properties']['distance']['type'] = 'h-measure';
 -         $params['trip']['properties']['distance']['properties']['num'] = $distance;
 -         $params['trip']['properties']['distance']['properties']['unit'] = 'meter';
 -         Log::debug("Overriding distance to $distance");
 -       }
 -     }
 - 
 -     // echo "Micropub Params\n";
 -     // print_r($params);
 - 
 -     $multipart = new Multipart();
 -     $multipart->addArray($params);
 -     $multipart->addFile('route.json', $file_path, 'application/json');
 - 
 -     $httpheaders = [
 -       'Authorization: Bearer ' . $db->micropub_token,
 -       'Content-type: ' . $multipart->contentType()
 -     ];
 - 
 -     Log::info('Sending to the Micropub endpoint: '.$db->micropub_endpoint);
 -     // Post to the Micropub endpoint
 -     $ch = curl_init($db->micropub_endpoint);
 -     curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
 -     curl_setopt($ch, CURLOPT_POST, true);
 -     curl_setopt($ch, CURLOPT_HTTPHEADER, $httpheaders);
 -     curl_setopt($ch, CURLOPT_POSTFIELDS, $multipart->data());
 -     curl_setopt($ch, CURLOPT_HEADER, true);
 -     $response = curl_exec($ch);
 - 
 -     Log::info("Done!");
 -     Log::info($response);
 - 
 -     // echo "========\n";
 -     // echo $response."\n========\n";
 -     //
 -     // echo "\n";
 -   }
 - 
 -   public static function geocode($lat, $lng) {
 -     $ch = curl_init(env('ATLAS_BASE').'api/geocode?latitude='.$lat.'&longitude='.$lng);
 -     curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
 -     curl_setopt($ch, CURLOPT_TIMEOUT, 8);
 -     $response = curl_exec($ch);
 -     if($response) {
 -       return json_decode($response);
 -     }
 -   }
 - 
 - 
 - 
 - 
 - 
 - 
 -   // TODO: move this to a library p3k/Geo
 - 
 - 
 -   // http://www.loughrigg.org/rdp/
 - 
 -   //The author has placed this work in the Public Domain, thereby relinquishing all copyrights.
 -   //You may use, modify, republish, sell or give away this work without prior consent.
 -   //This implementation comes with no warranty or guarantee of fitness for any purpose.
 - 
 -   //=========================================================================
 -   //An implementation of the Ramer-Douglas-Peucker algorithm for reducing
 -   //the number of points on a polyline
 -   //see http://en.wikipedia.org/wiki/Ramer%E2%80%93Douglas%E2%80%93Peucker_algorithm
 -   //=========================================================================
 - 
 -   //Finds the perpendicular distance from a point to a straight line.
 -   //The coordinates of the point are specified as $ptX and $ptY.
 -   //The line passes through points l1 and l2, specified respectively with their
 -   //coordinates $l1x and $l1y, and $l2x and $l2y
 -   public function _perpendicularDistance($ptX, $ptY, $l1x, $l1y, $l2x, $l2y)
 -   {
 -       $result = 0;
 -       if ($l2x == $l1x)
 -       {
 -           //vertical lines - treat this case specially to avoid divide by zero
 -           $result = abs($ptX - $l2x);
 -       }
 -       else
 -       {
 -           $slope = (($l2y-$l1y) / ($l2x-$l1x));
 -           $passThroughY = (0-$l1x)*$slope + $l1y;
 -           $result = (abs(($slope * $ptX) - $ptY + $passThroughY)) / (sqrt($slope*$slope + 1));
 -       }
 -       return $result;
 -   }
 - 
 -   //RamerDouglasPeucker
 -   //Reduces the number of points on a polyline by removing those that are closer to the line
 -   //than the distance $epsilon.
 -   //The polyline is provided as an array of arrays, where each internal array is one point on the polyline,
 -   //specified by easting (x-coordinate) with key "0" and northing (y-coordinate) with key "1".
 -   //It is assumed that the coordinates and distance $epsilon are given in the same units.
 -   //The result is returned as an array in a similar format.
 -   //Each point returned in the result array will retain all its original data, including its E and N
 -   //values along with any others.
 -   public function _ramerDouglasPeucker($pointList, $epsilon)
 -   {
 -       if(count($pointList) == 0)
 -         return array();
 - 
 -       // Find the point with the maximum distance
 -       $dmax = 0;
 -       $index = 0;
 -       $totalPoints = count($pointList);
 -       for ($i = 1; $i < ($totalPoints - 1); $i++)
 -       {
 -           $d = $this->_perpendicularDistance($pointList[$i][0], $pointList[$i][1],
 -                                      $pointList[0][0], $pointList[0][1],
 -                                      $pointList[$totalPoints-1][0], $pointList[$totalPoints-1][1]);
 - 
 -           if ($d > $dmax)
 -           {
 -               $index = $i;
 -               $dmax = $d;
 -           }
 -       }
 - 
 -       $resultList = array();
 - 
 -       // If max distance is greater than epsilon, recursively simplify
 -       if ($dmax >= $epsilon)
 -       {
 -           // Recursive call
 -           $recResults1 = $this->_ramerDouglasPeucker(array_slice($pointList, 0, $index + 1), $epsilon);
 -           $recResults2 = $this->_ramerDouglasPeucker(array_slice($pointList, $index, $totalPoints - $index), $epsilon);
 - 
 -           // Build the result list
 -           $resultList = array_merge(array_slice($recResults1, 0, count($recResults1) - 1),
 -                                     array_slice($recResults2, 0, count($recResults2)));
 -       }
 -       else
 -       {
 -           $resultList = array($pointList[0], $pointList[$totalPoints-1]);
 -       }
 -       // Return the result
 -       return $resultList;
 -   }
 - 
 -   function _gc_distance($lat1, $lng1, $lat2, $lng2) {
 -     return ( 6378100 * acos( cos( deg2rad($lat1) ) * cos( deg2rad($lat2) ) * cos( deg2rad($lng2) - deg2rad($lng1) ) + sin( deg2rad($lat1) ) * sin( deg2rad($lat2) ) ) );
 -   }
 - 
 - 
 - }
 
 
  |