You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

221 lines
7.2 KiB

  1. <?php
  2. namespace App\Jobs;
  3. use DB;
  4. use Log;
  5. use Quartz;
  6. use p3k\Multipart;
  7. use App\Jobs\Job;
  8. use Illuminate\Contracts\Bus\SelfHandling;
  9. use Illuminate\Contracts\Queue\ShouldQueue;
  10. use DateTime, DateTimeZone;
  11. class TripComplete extends Job implements SelfHandling, ShouldQueue
  12. {
  13. private $_dbid;
  14. private $_data;
  15. public function __construct($dbid, $data) {
  16. $this->_dbid = $dbid;
  17. $this->_data = $data;
  18. }
  19. public function handle() {
  20. // echo "Job Data\n";
  21. // echo json_encode($this->_data)."\n";
  22. if(!is_array($this->_data)) return;
  23. $db = DB::table('databases')->where('id','=',$this->_dbid)->first();
  24. Log::info("Starting job for ".$db->name);
  25. Log::debug(json_encode($this->_data));
  26. if(!$db->micropub_endpoint) {
  27. Log::info('No micropub endpoint configured for database ' . $db->name);
  28. return;
  29. }
  30. $qz = new Quartz\DB(env('STORAGE_DIR').$db->name, 'r');
  31. // Load the data from the start and end times
  32. $start = new DateTime($this->_data['properties']['start']);
  33. $end = new DateTime($this->_data['properties']['end']);
  34. $results = $qz->queryRange($start, $end);
  35. $features = [];
  36. foreach($results as $id=>$record) {
  37. if(!property_exists($record->data->properties, 'action')) {
  38. $record->data->properties = array_filter((array)$record->data->properties, function($k){
  39. // Remove some of the app-specific tracking keys
  40. return !in_array($k, ['locations_in_payload','desired_accuracy','significant_change','pauses','deferred']);
  41. }, ARRAY_FILTER_USE_KEY);
  42. $features[] = $record->data;
  43. }
  44. }
  45. // Build the GeoJSON for this trip
  46. $geojson = [
  47. 'type' => 'FeatureCollection',
  48. 'features' => $features
  49. ];
  50. $file_path = tempnam(sys_get_temp_dir(), 'compass');
  51. file_put_contents($file_path, json_encode($geojson));
  52. // If there are no start/end coordinates in the request, use the first and last coordinates
  53. if(count($features)) {
  54. if(!array_key_exists('start-coordinates', $this->_data['properties'])) {
  55. $this->_data['properties']['start-coordinates'] = $features[0]->geometry->coordinates;
  56. }
  57. if(!array_key_exists('end-coordinates', $this->_data['properties'])) {
  58. $this->_data['properties']['end-coordinates'] = $features[count($features)-1]->geometry->coordinates;
  59. }
  60. }
  61. $startAdr = false;
  62. if(array_key_exists('start-coordinates', $this->_data['properties'])) {
  63. // Reverse geocode the start and end location to get an h-adr
  64. $startAdr = [
  65. 'type' => 'h-adr',
  66. 'properties' => [
  67. 'latitude' => $this->_data['properties']['start-coordinates'][1],
  68. 'longitude' => $this->_data['properties']['start-coordinates'][0],
  69. ]
  70. ];
  71. Log::info('Looking up start location');
  72. $start = self::geocode($this->_data['properties']['start-coordinates'][1], $this->_data['properties']['start-coordinates'][0]);
  73. if($start) {
  74. $startAdr['properties']['locality'] = $start->locality;
  75. $startAdr['properties']['region'] = $start->region;
  76. $startAdr['properties']['country'] = $start->country;
  77. Log::info('Found start: '.$start->full_name.' '.$start->timezone);
  78. }
  79. } else {
  80. $start = false;
  81. }
  82. $endAdr = false;
  83. if(array_key_exists('end-coordinates', $this->_data['properties'])) {
  84. $endAdr = [
  85. 'type' => 'h-adr',
  86. 'properties' => [
  87. 'latitude' => $this->_data['properties']['end-coordinates'][1],
  88. 'longitude' => $this->_data['properties']['end-coordinates'][0],
  89. ]
  90. ];
  91. Log::info('Looking up end location');
  92. $end = self::geocode($this->_data['properties']['end-coordinates'][1], $this->_data['properties']['end-coordinates'][0]);
  93. if($end) {
  94. $endAdr['properties']['locality'] = $end->locality;
  95. $endAdr['properties']['region'] = $end->region;
  96. $endAdr['properties']['country'] = $end->country;
  97. Log::info('Found end: '.$end->full_name.' '.$end->timezone);
  98. }
  99. } else {
  100. $end = false;
  101. }
  102. // Set the timezone of the dates based on the location
  103. $startDate = new DateTime($this->_data['properties']['start']);
  104. if($start && $start->timezone) {
  105. $startDate->setTimeZone(new DateTimeZone($start->timezone));
  106. }
  107. $endDate = new DateTime($this->_data['properties']['end']);
  108. if($end && $end->timezone) {
  109. $endDate->setTimeZone(new DateTimeZone($end->timezone));
  110. }
  111. $params = [
  112. 'h' => 'entry',
  113. 'created' => $endDate->format('c'),
  114. 'trip' => [
  115. 'type' => 'h-trip',
  116. 'properties' => [
  117. 'mode-of-transport' => $this->_data['properties']['mode'],
  118. 'start' => $startDate->format('c'),
  119. 'end' => $endDate->format('c'),
  120. 'route' => 'route.json'
  121. // TODO: avgpace for runs
  122. // TODO: avgspeed for bike rides
  123. // TODO: avg heart rate if available
  124. ]
  125. ]
  126. ];
  127. if($startAdr) {
  128. $params['trip']['properties']['start-location'] = $startAdr;
  129. }
  130. if($endAdr) {
  131. $params['trip']['properties']['end-location'] = $endAdr;
  132. }
  133. if(array_key_exists('distance', $this->_data['properties'])) {
  134. $params['trip']['properties']['distance'] = [
  135. 'type' => 'h-measure',
  136. 'properties' => [
  137. 'num' => round($this->_data['properties']['distance']),
  138. 'unit' => 'meter'
  139. ]
  140. ];
  141. }
  142. if(array_key_exists('duration', $this->_data['properties'])) {
  143. $params['trip']['properties']['duration'] = [
  144. 'type' => 'h-measure',
  145. 'properties' => [
  146. 'num' => round($this->_data['properties']['duration']),
  147. 'unit' => 'second'
  148. ]
  149. ];
  150. }
  151. if(array_key_exists('cost', $this->_data['properties'])) {
  152. $params['trip']['properties']['cost'] = [
  153. 'type' => 'h-measure',
  154. 'properties' => [
  155. 'num' => round($this->_data['properties']['cost'], 2),
  156. 'unit' => 'USD'
  157. ]
  158. ];
  159. }
  160. // echo "Micropub Params\n";
  161. // print_r($params);
  162. $multipart = new Multipart();
  163. $multipart->addArray($params);
  164. $multipart->addFile('route.json', $file_path, 'application/json');
  165. $httpheaders = [
  166. 'Authorization: Bearer ' . $db->micropub_token,
  167. 'Content-type: ' . $multipart->contentType()
  168. ];
  169. Log::info('Sending to the Micropub endpoint: '.$db->micropub_endpoint);
  170. // Post to the Micropub endpoint
  171. $ch = curl_init($db->micropub_endpoint);
  172. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  173. curl_setopt($ch, CURLOPT_POST, true);
  174. curl_setopt($ch, CURLOPT_HTTPHEADER, $httpheaders);
  175. curl_setopt($ch, CURLOPT_POSTFIELDS, $multipart->data());
  176. curl_setopt($ch, CURLOPT_HEADER, true);
  177. $response = curl_exec($ch);
  178. Log::info("Done!");
  179. Log::info($response);
  180. // echo "========\n";
  181. // echo $response."\n========\n";
  182. //
  183. // echo "\n";
  184. }
  185. public static function geocode($lat, $lng) {
  186. $ch = curl_init(env('ATLAS_BASE').'api/geocode?latitude='.$lat.'&longitude='.$lng);
  187. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  188. curl_setopt($ch, CURLOPT_TIMEOUT, 8);
  189. $response = curl_exec($ch);
  190. if($response) {
  191. return json_decode($response);
  192. }
  193. }
  194. }