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.
 
 
 
 
 
 

519 line
22 KiB

  1. <?php
  2. /**
  3. * Created by PhpStorm.
  4. * User: Steven
  5. * Date: 2017/2/17
  6. * Time: 14:06
  7. */
  8. namespace backend\modules\motorcade\models;
  9. use phpDocumentor\Reflection\Types\Null_;
  10. use Yii;
  11. class BusGps
  12. {
  13. private $AK = "qnG9Q2oSgfGlwLUBeYBp7gTvFUCk31me"; //百度访问应用AK
  14. private $GeocodingUrl = "http://api.map.baidu.com/geocoder/v2/";
  15. /**
  16. * 获取车辆的最新GPS数据(运营主体下有GPS的车辆)
  17. * User: Steven
  18. */
  19. public function getNewGpsData($bus_list = '', $first = '')
  20. {
  21. $main_cooperation_id = Yii::$app->user->identity->MAIN_CORP_ID2;
  22. // $main_cooperation_id = 7;
  23. if ($bus_list == '') {
  24. $sql_str = "";
  25. } else {
  26. $sql_str = " and bus_no in($bus_list)";
  27. }
  28. $conn = Yii::$app->db;
  29. $date = date('Y-m-d', time());
  30. /*if ($first == '') {
  31. $sql = "select bus_no from gps_new as a where create_time = (select max(b.create_time) from gps_new as b where a.bus_no = b.bus_no ) and a.main_corp_id={$main_cooperation_id} $sql_str;";
  32. } else {
  33. $sql = "select * from gps_new as a where create_time = (select b.create_time from gps_new as b where a.bus_no = b.bus_no order by b.create_time desc limit 1,1 ) and a.main_corp_id={$main_cooperation_id} $sql_str;";
  34. }*/
  35. $sql = "select a.bus_no,a.run_date,a.latitude,a.longitude,a.gpsSpeed,a.direction from gps_new a ,(
  36. select max(ID) as id from gps_new where main_corp_id=$main_cooperation_id and run_date='{$date}' and create_time>'{$date} 00:00:00' GROUP BY bus_no) b where a.id=b.id $sql_str";
  37. $res = $conn->createCommand($sql)->queryAll();
  38. return $res;
  39. }
  40. /**
  41. * 获取聚合状态下的数据
  42. * User: Steven
  43. * @return array
  44. */
  45. public function getPolymerData()
  46. {
  47. $main_cooperation_id = Yii::$app->user->identity->MAIN_CORP_ID2;
  48. $conn = Yii::$app->db;
  49. $today = date('Y-m-d', time());
  50. /*$sql = "select city as title,group_concat('''',bus_no,'''') as car_list, count(1) as num from gps_new as a where create_time =
  51. (select max(b.create_time) from gps_new as b where a.bus_no = b.bus_no and b.run_date='{$today}') and a.run_date='{$today}' and a.main_corp_id={$main_cooperation_id} GROUP BY city;";*/
  52. $sql = "select city as title,group_concat('''',bus_no,'''') as car_list, count(1) as num from(
  53. select a.bus_no,a.run_date,a.latitude,a.longitude,a.gpsSpeed,a.direction,a.city from gps_new a ,(
  54. select max(ID) as id from gps_new where main_corp_id=$main_cooperation_id and run_date='{$today}' and create_time>'{$today} 00:00:00' GROUP BY bus_no) b where a.id=b.id) as c GROUP BY c.city";
  55. $res = $conn->createCommand($sql)->queryAll();
  56. return $res;
  57. }
  58. /**
  59. * 查询车辆详细信息
  60. * User: Steven
  61. * @param $gpsData
  62. */
  63. public function getCarDataService($gpsData, $arr)
  64. {
  65. $conn = Yii::$app->db;
  66. $date = date('Y-m-d', time());
  67. /*$res = $conn->createCommand("select b.bus_no,b.seat_desc,a.itinerary_name,a.start_time,a.run_status,c.driver_name,d.start_time as end_time
  68. from bus_order a,base_bus b,base_driver c,bus_itinerary d
  69. where a.send_bus_res_id=b.BUS_ID and b.BUS_NO='{$gpsData['bus_no']}' and a.send_bus_driver_res_id=c.DRIVER_ID
  70. and a.itinerary_id=d.itinerary_id and a.run_date='{$date}' and a.cancel_flag=0 and run_status=437
  71. ORDER BY d.station_seq_id DESC limit 1;
  72. ")->queryAll();*/
  73. $res = $conn->createCommand("select a.bus_no,a.seat_desc,b.itinerary_name,b.start_time,b.run_status,c.driver_name,d.start_time as end_time from base_bus a
  74. LEFT JOIN bus_order b on a.BUS_ID=b.send_bus_res_id and b.run_date='{$date}' and b.cancel_flag=0 and b.run_status=437
  75. LEFT JOIN base_driver c on b.send_bus_driver_res_id=c.DRIVER_ID
  76. LEFT JOIN bus_itinerary d on b.itinerary_id=d.itinerary_id
  77. where a.BUS_NO='{$gpsData['bus_no']}' and a.cancel_flag=0 ORDER BY d.station_seq_id DESC limit 1;
  78. ")->queryAll();
  79. if ($res[0]['run_status'] == 437) { //出行中的车次
  80. $arr[$gpsData['bus_no']] = array(
  81. 'plateCode' => $gpsData['bus_no'],
  82. 'seat_desc' => isset($res[0]['seat_desc']) ? $res[0]['seat_desc'] : '',
  83. 'bus_status' => 1, //车辆状态
  84. 'bus_desc' => '出车中',
  85. 'lat' => $gpsData['latitude'],
  86. 'lng' => $gpsData['longitude'],
  87. 'dspeed' => $gpsData['gpsSpeed'], //gps速
  88. 'direction' => $gpsData['direction'],
  89. 'driver' => isset($res[0]['driver_name']) ? $res[0]['driver_name'] : '',
  90. 'itinerary_name' => isset($res[0]['itinerary_name']) ? $res[0]['itinerary_name'] : '',
  91. 'start_time' => isset($res[0]['start_time']) ? $res[0]['start_time'] : '',
  92. 'end_time' => isset($res[0]['end_time']) ? $res[0]['end_time'] : ''
  93. );
  94. } else { //当前车次当前没有出行
  95. $arr[$gpsData['bus_no']] = array(
  96. 'plateCode' => $gpsData['bus_no'],
  97. 'seat_desc' => $res[0]['seat_desc'],
  98. 'bus_status' => 0, //车辆状态
  99. 'bus_desc' => '空闲',
  100. 'lat' => $gpsData['latitude'],
  101. 'lng' => $gpsData['longitude'],
  102. 'dspeed' => $gpsData['gpsSpeed'], //gps速
  103. 'direction' => $gpsData['direction'],
  104. 'driver' => '',
  105. 'itinerary_name' => '',
  106. 'start_time' => '',
  107. 'end_time' => ''
  108. );
  109. }
  110. return $arr;
  111. }
  112. /**
  113. * 获取车辆运营概况统计数据
  114. * User: Steven
  115. */
  116. public function getStatistics($bus_no)
  117. {
  118. $main_cooperation_id = Yii::$app->user->identity->MAIN_CORP_ID2;
  119. $conn = Yii::$app->db;
  120. $date = date('Y-m-d', time());
  121. /* $sql = "select bus_id,bus_no from base_bus where GPS_ID <>0 and CANCEL_FLAG=0 and MAIN_CORP_ID={$main_cooperation_id}";
  122. $bus_list = $conn->createCommand($sql)->queryAll();*/
  123. $bus_count = count($bus_no); //所有车辆个数
  124. /*$bus_no_list = '';
  125. foreach ($bus_list as $value) {
  126. $bus_no_list .= $bus_no_list == '' ? $value['bus_id'] : ',' . $value['bus_id'];
  127. }*/
  128. //TODO:查询当前出车的车辆个数
  129. $sql_status = "select COUNT(1) as count from bus_order a,base_user b
  130. where a.create_user_id=b.id and b.MAIN_CORP_ID2=$main_cooperation_id and a.run_date='{$date}' and a.run_status=437 and a.cancel_flag=0";
  131. $bus_status = $conn->createCommand($sql_status)->queryAll();
  132. $online_count = $bus_status[0]['count']; //当前出车车辆
  133. //TODO:今日出车线路分布
  134. $res = $conn->createCommand("select a.bus_number,a.itinerary_name,a.line_type,a.run_status,b.bus_no
  135. from bus_order a
  136. inner join base_bus b on a.send_bus_res_id =b.bus_id and b.main_corp_id=$main_cooperation_id
  137. where a.run_date='{$date}' and a.run_status in (435,436,437,438);")->queryAll();
  138. $all_count = 0;
  139. $provice_count = 0;
  140. $country_count = 0;
  141. $shdsn_count = 0;
  142. $qdh_count = 0;
  143. $xt_count = 0;
  144. $wz_count = 0;
  145. $zz_count = 0;
  146. $ts_count = 0;
  147. $poi_arr = array();
  148. if ($res) {
  149. foreach ($res as $value) {
  150. $all_count = $all_count + 1;
  151. /*if (in_array($value['bus_no'], $bus_no)) {
  152. //统计车辆运营概况
  153. if ($value['run_status'] == 437) { //出车
  154. $on_run_count = $on_run_count + 1;
  155. $on_bus_list .= $on_bus_list == '' ? $value["bus_no"] : "," . $value["bus_no"];
  156. } else {
  157. $off_run_count = $off_run_count + 1;
  158. $off_bus_list .= $off_bus_list == '' ? $value["bus_no"] : "," . $value["bus_no"];
  159. }
  160. }*/
  161. //统计出车线路分布
  162. if ($value['line_type'] == 1) { //1:省级 2:市内
  163. $provice_count = $provice_count + 1;
  164. } else {
  165. $country_count = $country_count + 1;
  166. }
  167. //上海迪士尼,千岛湖,西塘,乌镇,周庄,汤山
  168. if (strstr($value['itinerary_name'], '上海迪士尼') != false) { // 包含
  169. $shdsn_count = $shdsn_count + 1;
  170. }
  171. if (strstr($value['itinerary_name'], '千岛湖') != false) {
  172. $qdh_count = $qdh_count + 1;
  173. }
  174. if (strstr($value['itinerary_name'], '西塘') != false) {
  175. $xt_count = $xt_count + 1;
  176. }
  177. if (strstr($value['itinerary_name'], '乌镇') != false) {
  178. $wz_count = $wz_count + 1;
  179. }
  180. if (strstr($value['itinerary_name'], '周庄') != false) {
  181. $zz_count = $zz_count + 1;
  182. }
  183. if (strstr($value['itinerary_name'], '汤山') != false) {
  184. $ts_count = $ts_count + 1;
  185. }
  186. }
  187. $array = array('shdsn_count' => $shdsn_count, 'qdh_count' => $qdh_count, 'xt_count' => $xt_count, 'wz_count' => $wz_count, 'zz_count' => $zz_count, 'ts_count' => $ts_count);
  188. arsort($array);
  189. $poi_name = '';
  190. foreach ($array as $key => $value) {
  191. if ($value != 0) {
  192. switch ($key) {
  193. case 'shdsn_count':
  194. $poi_name = '上海迪士尼';
  195. break;
  196. case 'qdh_count':
  197. $poi_name = '千岛湖';
  198. break;
  199. case 'xt_count':
  200. $poi_name = '西塘';
  201. break;
  202. case 'wz_count':
  203. $poi_name = '乌镇';
  204. break;
  205. case 'zz_count':
  206. $poi_name = '周庄';
  207. break;
  208. case 'ts_count':
  209. $poi_name = '汤山';
  210. break;
  211. }
  212. $poi_arr[] = array(
  213. 'poi_name' => $poi_name,
  214. 'num' => $value,
  215. 'percent' => round(($value / $all_count) * 100, 2),
  216. );
  217. }
  218. }
  219. }
  220. $result = array( //当前车辆运营状况
  221. 'run_data' => array(
  222. 0 => array(
  223. 'sum_run' => $bus_count,
  224. 'run' => $online_count,
  225. 'bus_list' => ''
  226. ),
  227. 1 => array(
  228. 'sum_run' => $bus_count,
  229. 'run' => $bus_count - $online_count,
  230. 'bus_list' => ''
  231. ),
  232. 2 => array(
  233. 'sum_run' => $bus_count,
  234. 'run' => 0,
  235. 'bus_list' => ''
  236. )
  237. ),
  238. 'line_data' => array( //今日出车线路分布
  239. 'provice_count' => $provice_count,
  240. 'city_count' => 0,
  241. 'city_inside_count' => $country_count
  242. ),
  243. 'poi_data' => $poi_arr
  244. );
  245. /*$res_task = $conn->createCommand("select count(*) as count,line_type from (select distinct a.send_bus_res_id,a.line_type from bus_order a,base_bus b
  246. where a.send_bus_res_id=b.BUS_ID and b.BUS_NO in({$bus_no}) and a.run_date='{$date}' and a.cancel_flag=0) c GROUP BY line_type;")->queryAll();*/
  247. return $result;
  248. }
  249. /**
  250. *
  251. * User: Steven
  252. * @param $bus_num_list
  253. * @return array
  254. */
  255. /*public function getBusOrderByNum($bus_num_list)
  256. {
  257. $conn = Yii::$app->db;
  258. $res = $conn->createCommand("select a.run_date,a.itinerary_name,b.bus_no,c.driver_name from bus_order a
  259. left join base_bus b on a.send_bus_res_id=b.bus_id
  260. left join base_driver c on a.send_bus_driver_res_id=c.driver_id
  261. where a.bus_number in ($bus_num_list)")->queryAll();
  262. return $res;
  263. }*/
  264. /**
  265. * 获取历史轨迹页面左侧车辆列表
  266. * User: Steven
  267. * @param $date
  268. * @return array
  269. */
  270. public function getCarList($date, $key_word)
  271. {
  272. $main_cooperation_id = Yii::$app->user->identity->MAIN_CORP_ID2;
  273. $conn = Yii::$app->db;
  274. $res = $conn->createCommand("select a.bus_no,count(b.send_bus_res_id) as count from base_bus a
  275. LEFT JOIN bus_order b on a.BUS_ID=b.send_bus_res_id and b.run_date='{$date}' and b.cancel_flag=0
  276. where a.main_corp_id=$main_cooperation_id and a.gps_id<>0 and a.CANCEL_FLAG=0 and a.bus_no like '%{$key_word}%'
  277. GROUP BY a.bus_no ORDER BY count DESC")->queryAll();
  278. return $res;
  279. }
  280. function objectToArray ($object) {
  281. if(!is_object($object) && !is_array($object)) {
  282. return $object;
  283. }
  284. return array_map('objectToArray', (array) $object);
  285. }
  286. /**
  287. * 存储GPS数据
  288. * User: Steven
  289. * @param $result
  290. * @return int
  291. */
  292. public function uploadGpsData($result)
  293. {
  294. $conn = Yii::$app->db;
  295. $data_arr = array();
  296. foreach ($result as $item) {
  297. if ($item['IsOnLine'] == '1') {
  298. $item['WeiDu'] = $item['WeiDu'] / 1000000;
  299. $item['JingDu'] = $item['JingDu'] / 1000000;
  300. $location = $item['WeiDu'] . ',' . $item['JingDu'];
  301. $data = $this->GeocodingAPI($location, 2);
  302. $data_arr[] = array(
  303. date('Y-m-d H:i:s', strtotime($item['DateTime'])), 7, 628,
  304. $item['CarLPN'], //车牌号
  305. date('Y-m-d', time()),
  306. $item['WeiDu'], //维度
  307. $item['JingDu'], //经度
  308. $item['Speed'], //gps速度
  309. $item['Speed'], //速度
  310. 0, //高度
  311. $item['Direction'], //方向
  312. $data['addressComponent']['province'], //所属市区
  313. $data['addressComponent']['city'], //所属市区
  314. 0, //里程数
  315. 0, //油量
  316. );/**/
  317. }
  318. }
  319. $res = $conn->createCommand()->batchInsert('gps_new', ['create_time', 'main_corp_id', 'org_id', 'bus_no', 'run_date', 'latitude', 'longitude', 'gpsSpeed', 'speed', 'height', 'direction', 'province', 'city', 'mileage', 'oil'], $data_arr)->execute();
  320. return $res;
  321. }
  322. /**
  323. * 上传我们采购的GPS的数据
  324. * User: Steven
  325. */
  326. public
  327. function uploadGpsZz($data)
  328. {
  329. $conn = Yii::$app->db;
  330. $data_arr = array();
  331. foreach ($data['data'] as $item) {
  332. if ($item['device_info'] == 0) {
  333. $location = $item['lat'] . ',' . $item['lng'];
  334. $Geocoding = $this->GeocodingAPI($location, 2);
  335. //IMEI:352477070153196,352477070154186,352477070154426,352477070156637,352477070156678
  336. $bus_no = '';
  337. switch ($item['imei']) {
  338. case '352477070153196':
  339. $bus_no = '沪ZZ8888';
  340. break;
  341. case '352477070154186':
  342. $bus_no = '沪ZZ9999';
  343. break;
  344. case '352477070154426':
  345. $bus_no = '沪ZZ0000';
  346. break;
  347. case '352477070156637':
  348. $bus_no = '沪ZZ6666';
  349. break;
  350. case '352477070156678':
  351. $bus_no = '沪ZZ5555';
  352. break;
  353. }
  354. $data_arr[] = array(
  355. date('Y-m-d H:i:s', time()), 10, 1283,
  356. $bus_no, //车牌号
  357. date('Y-m-d', time()),
  358. $item['lat'], //维度
  359. $item['lng'], //经度
  360. $item['speed'], //gps速度
  361. $item['speed'], //速度
  362. 0, //高度
  363. $item['course'], //方向
  364. $Geocoding['addressComponent']['province'], //所属市区
  365. $Geocoding['addressComponent']['city'] //所属市区
  366. );
  367. }
  368. }
  369. $res = $conn->createCommand()->batchInsert('gps_new', ['create_time', 'main_corp_id', 'org_id', 'bus_no', 'run_date', 'latitude', 'longitude', 'gpsSpeed', 'speed', 'height', 'direction', 'province', 'city'], $data_arr)->execute();
  370. return $res;
  371. }
  372. /**
  373. * 获取历史轨迹所需要的数据
  374. * User: Steven
  375. * @param $bus_no
  376. * @param $run_date
  377. * @param $begin_time
  378. * @param $end_time
  379. * @return array
  380. */
  381. public
  382. function getHistoryData($bus_no, $run_date, $begin_time, $end_time)
  383. {
  384. $main_cooperation_id = Yii::$app->user->identity->MAIN_CORP_ID2;
  385. $conn = Yii::$app->db;
  386. /*$res = $conn->createCommand("select latitude,longitude,bus_no,count(longitude)*30 as delay_time
  387. from gps where bus_no='{$bus_no}' and create_time>'{$run_date} {$begin_time}' and create_time<'{$run_date} {$end_time}'
  388. GROUP BY longitude ORDER BY create_time")->queryAll();*/
  389. $sql = "select latitude,longitude,bus_no,count(longitude)*25 as delay_time , create_time
  390. from gps_new where bus_no='{$bus_no}' and run_date='{$run_date}' and main_corp_id=$main_cooperation_id and cancel_flag=0
  391. GROUP BY longitude ORDER BY create_time "; // limit 37,2
  392. $res = $conn->createCommand($sql)->queryAll();
  393. return $res;
  394. }
  395. /**
  396. * 提供从地址到经纬度坐标或者从经纬度坐标到地址的转换服务
  397. * User: Steven
  398. * @param $location (纬度,精度)
  399. * @param $type 1:从地址到经纬度坐标 2:从经纬度坐标到地址
  400. * @return array|bool
  401. */
  402. public
  403. function GeocodingAPI($location, $type)
  404. {
  405. if ($type == 1) { //从地址到经纬度坐标
  406. $data = array('address' => $location, 'output' => 'json', 'city' => $location, 'ak' => $this->AK);
  407. } else { //从经纬度坐标到地址
  408. $data = array('location' => $location, 'output' => 'json', 'tactics' => 11, 'ak' => $this->AK);
  409. }
  410. $res = $this->send_get($this->GeocodingUrl, $data);
  411. $addressList = json_decode($res, true);
  412. if (isset($addressList['status']) && $addressList['status'] == 0) {
  413. if ($type == 1) {
  414. $data['lng'] = $addressList['result']['location']['lng'];
  415. $data['lat'] = $addressList['result']['location']['lat'];
  416. } else {
  417. $data = $addressList['result'];
  418. }
  419. return $data;
  420. } else {
  421. return false;
  422. }
  423. }
  424. /**
  425. * 发送get参数(链接,参数数组)
  426. * User: Steven
  427. * @param $url
  428. * @param $get_data
  429. * @return string
  430. */
  431. public
  432. function send_get($url, $get_data)
  433. {
  434. $url_param = http_build_query($get_data);
  435. $result = file_get_contents($url . "?{$url_param}");
  436. return $result;
  437. }
  438. //发送HTTP请求
  439. public
  440. function httpsPost($url, $param = array())
  441. {
  442. $ch = curl_init(); // 初始化一个 cURL 对象
  443. curl_setopt($ch, CURLOPT_URL, $url); // 设置需要抓取的URL
  444. curl_setopt($ch, CURLOPT_HEADER, 0); // // 设置header
  445. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 设置cURL 参数,要求结果保存到字符串中还是输出到屏幕上。
  446. // 如果你想PHP去做一个正规的HTTP POST,设置这个选项为一个非零值。这个POST是普通的 application/x-www-from-urlencoded 类型,多数被HTML表单使用。
  447. curl_setopt($ch, CURLOPT_POST, 1);
  448. curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($param)); // 传递一个作为HTTP “POST”操作的所有数据的字符串。//http_build_query:生成 URL-encode 之后的请求字符串
  449. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
  450. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
  451. curl_setopt($ch, CURLOPT_HTTPHEADER, array(
  452. 'Content-type:application/x-www-form-urlencoded;charset=utf-8'
  453. ));
  454. $rtn = curl_exec($ch); // 运行cURL,请求网页
  455. if ($errno = curl_errno($ch)) {
  456. throw new \Exception ('Curl Error(' . $errno . '):' . curl_error($ch));
  457. }
  458. curl_close($ch); // 关闭URL请求
  459. return $rtn; // 返回获取的数据
  460. }
  461. /**
  462. * 临时:更新历史数据的市级地理位置
  463. * User: Steven
  464. * @return int
  465. */
  466. public
  467. function test()
  468. {
  469. $conn = Yii::$app->db;
  470. $sql = "SELECT * FROM gps_new_copy WHERE run_date='2017-03-07' AND province IS NULL ORDER BY create_time LIMIT 0,5000";
  471. $res = $conn->createCommand($sql)->queryAll();
  472. foreach ($res as $item) {
  473. $lo = $item['latitude'] . ',' . $item['longitude'];
  474. $res = $this->GeocodingAPI($lo, 2);
  475. $sql = "update gps_new_copy set province='{$res['addressComponent']['province']}',city='{$res['addressComponent']['city']}' where id={$item['ID']}";
  476. $RES = $conn->createCommand($sql)->execute();
  477. }
  478. return $RES;
  479. }
  480. public
  481. function test1($plateCode, $vehicleId)
  482. {
  483. $conn = Yii::$app->db;
  484. $sql = "update base_bus set GPS_ID={$vehicleId} where MAIN_CORP_ID=7 and org_id=628 and bus_no='{$plateCode}'";
  485. $RES = $conn->createCommand($sql)->execute();
  486. return $RES;
  487. }
  488. }