Taxi.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. <?php
  2. namespace app\api\controller\v1;
  3. use app\api\controller\ApiController;
  4. use app\api\service\SmsCode;
  5. use app\common\validate\IDMustBePositiveInt;
  6. use EasyWeChat\Factory;
  7. use think\facade\Cache;
  8. class Taxi extends ApiController
  9. {
  10. /**
  11. * 获取车辆
  12. *
  13. * @url GET /taxi/nearby?page=1
  14. * @author 许祖兴 < zuxing.xu@lettered.cn>
  15. * @date 2020/7/3 16:29
  16. *
  17. * @return \think\response\Json
  18. * @throws \think\db\exception\DataNotFoundException
  19. * @throws \think\db\exception\ModelNotFoundException
  20. * @throws \think\exception\DbException
  21. */
  22. public function getNearby()
  23. {
  24. $param = $this->request->param();
  25. $param['category_id'] = $this->request->param('category_id', 0);
  26. $limit = 10;
  27. $map['status'] = 1;
  28. if ($param['category_id']) {
  29. $map['category_id'] = $param['category_id'];
  30. }
  31. //
  32. $taxi = model('common/Taxi')->with(['user'])
  33. ->where($map)
  34. ->select();
  35. return $this->ApiJson(0,'获取成功', $taxi);
  36. }
  37. /**
  38. * 用车信息
  39. *
  40. * @url GET /taxi/:id
  41. * @author 许祖兴 < zuxing.xu@lettered.cn>
  42. * @date 2020/7/1 10:37
  43. *
  44. * @param int $id 用车ID
  45. * @return \think\response\Json
  46. * @throws \Lettered\Support\Exceptions\EvidentException
  47. * @throws \think\db\exception\DataNotFoundException
  48. * @throws \think\db\exception\ModelNotFoundException
  49. * @throws \think\exception\DbException
  50. */
  51. public function getTaxiByID($id)
  52. {
  53. (new IDMustBePositiveInt())->valid();
  54. $taxi = model('common/Taxi')->with(['user'])->hidden(['user.id_card'])
  55. ->find($id);
  56. if (!$taxi){
  57. return $this->ApiJson(-1, '不存在用车编号');
  58. }
  59. return $this->ApiJson(0, '获取用车信息成功', $taxi);
  60. }
  61. /**
  62. * 用车
  63. */
  64. public function callTaxi()
  65. {
  66. $params = $this->request->param();
  67. // 参数校验
  68. $valid = $this->validate($params, [
  69. 'taxi_id|关联车辆' => 'require',
  70. 'mobile|联系方式' => 'require',
  71. 'count|乘车人数' => 'require',
  72. 'depart|起始位置' => 'require',
  73. 'arrive|送达位置' => 'require',
  74. 'km|公里数' => 'require',
  75. 'price|价格' => 'require',
  76. ]);
  77. // 错误返回
  78. if(true !== $valid){
  79. return $this->ApiJson(-1, $valid);
  80. };
  81. $user = $this->auth->user();
  82. // 统计订单
  83. $count = model('common/TaxiOrder')->where('status','>=','2')->where(['is_free' => '1','user_id' => $user['id']])->count('id');
  84. // 创建订单
  85. $params['order_no'] = get_order_no();
  86. $params['user_id'] = $user['id'];
  87. $order = model('common/TaxiOrder')::create($params,true);
  88. //
  89. if ($order){
  90. // 创建对应支付记录
  91. $trade_no = get_order_no();
  92. $logID = model('common/OrderPaylog')->storeBy([
  93. 'out_trade_no' => $trade_no,
  94. 'total_price' => $params['price'],
  95. 'order_idx' => $order['id'],
  96. 'ascription' => 'motor' // 归属订单
  97. ]);
  98. // 免单优惠 20201201 添加条件 仅限xx 公里以下 不包含 xx 公里
  99. if ($count < sys_config('taxi_order_free_num','store') && $params['km'] < sys_config('taxi_order_free_km','store')){
  100. model('common/OrderPaylog')->updateBy($logID,[
  101. 'pay_price' => $params['price'],
  102. 'is_pay' => 1
  103. ]);
  104. model('common/TaxiOrder')->updateBy($order['id'], [
  105. 'is_free' => 1, // 免单
  106. 'status' => 2
  107. ]);
  108. // 后台推送
  109. // push_socket_data('motor',[
  110. // 'id' => $order['id'],
  111. // 'msg' => '有新的(免费)摩的订单等待处理哟,点击前往!'
  112. // ]);
  113. return $this->ApiJson(0,"本单免费,等待师傅接驾", [
  114. 'type' => 'free',
  115. 'success' => "ok!"
  116. ]);
  117. }
  118. // 返回支付单号
  119. return $this->ApiJson(0,'订单提交成功', [
  120. 'type' => 'wx',
  121. 'success' => "ok!",
  122. 'trade_no' => $trade_no
  123. ]);
  124. }
  125. return $this->ApiJson(-1,'发生异常,请骚后重试...');
  126. }
  127. /**
  128. * 用车,新的地图下单方式
  129. */
  130. public function newCallTaxi()
  131. {
  132. $params = $this->request->param();
  133. // 参数校验
  134. $valid = $this->validate($params, [
  135. 'mobile|联系方式' => 'require',
  136. 'count|乘车人数' => 'require',
  137. 'depart|起始位置' => 'require',
  138. 'arrive|送达位置' => 'require',
  139. 'km|公里数' => 'require',
  140. 'price|价格' => 'require',
  141. ]);
  142. // 错误返回
  143. if(true !== $valid){
  144. return $this->ApiJson(-1, $valid);
  145. };
  146. $user = $this->auth->user();
  147. // 统计订单
  148. $count = model('common/TaxiOrder')->where('status','>=','2')->where(['is_free' => '1','user_id' => $user['id']])->count('id');
  149. // 创建订单
  150. $params['order_no'] = get_order_no();
  151. $params['user_id'] = $user['id'];
  152. // 验证是否有未支付订单,有则直接更新订单为新订单
  153. $hasOrderId = model('common/TaxiOrder')->where(['user_id'=> $user['id'],'status'=> 1])->order('created_at','desc')->value('id');
  154. if($hasOrderId){
  155. model('common/TaxiOrder')->updateBy($hasOrderId, $params);
  156. }else{
  157. $order = model('common/TaxiOrder')::create($params,true);
  158. }
  159. // 订单信息验证处理
  160. if ($order){
  161. // 创建对应支付记录
  162. $trade_no = get_order_no();
  163. $logID = model('common/OrderPaylog')->storeBy([
  164. 'out_trade_no' => $trade_no,
  165. 'total_price' => $params['price'],
  166. 'order_idx' => $order['id'],
  167. 'ascription' => 'motor' // 归属订单
  168. ]);
  169. // 免单优惠 20201201 添加条件 仅限xx 公里以下 不包含 xx 公里
  170. if ($count < sys_config('taxi_order_free_num','store') && $params['km'] < sys_config('taxi_order_free_km','store')){
  171. model('common/OrderPaylog')->updateBy($logID,[
  172. 'pay_price' => $params['price'],
  173. 'is_pay' => 1
  174. ]);
  175. model('common/TaxiOrder')->updateBy($order['id'], [
  176. 'is_free' => 1, // 免单
  177. 'status' => 2
  178. ]);
  179. // 后台推送
  180. // push_socket_data('motor',[
  181. // 'id' => $order['id'],
  182. // 'msg' => '有新的(免费)摩的订单等待处理哟,点击前往!'
  183. // ]);
  184. return $this->ApiJson(0,"本单免费,等待师傅接驾", [
  185. 'type' => 'free',
  186. 'success' => "ok!"
  187. ]);
  188. }
  189. // 返回支付单号
  190. return $this->ApiJson(0,'订单提交成功', [
  191. 'type' => 'wx',
  192. 'success' => "ok!",
  193. 'trade_no' => $trade_no
  194. ]);
  195. }
  196. return $this->ApiJson(-1,'发生异常,请骚后重试...');
  197. }
  198. /**
  199. * 订单详情
  200. *
  201. * @author 许祖兴 < zuxing.xu@lettered.cn>
  202. * @date 2020/7/6 11:09
  203. *
  204. * @return \think\response\Json
  205. * @throws \Lettered\Support\Exceptions\FailedException
  206. * @throws \think\db\exception\DataNotFoundException
  207. * @throws \think\db\exception\ModelNotFoundException
  208. * @throws \think\exception\DbException
  209. */
  210. public function orderInfo()
  211. {
  212. $param = $this->request->param();
  213. // 数据校验
  214. $valid = $this->validate($param, [
  215. 'order_no' => 'require',
  216. ]);
  217. // 错误
  218. if (true !== $valid) {
  219. return $this->ApiJson(-1, $valid);
  220. }
  221. $info = model('common/TaxiOrder')
  222. ->with(['paylog','taxi','user'])
  223. ->where(['user_id' => $this->auth->user()['id'], 'order_no'=> $param['order_no']])
  224. ->findOrFail();
  225. return $this->ApiJson(0,'获取成功', $info);
  226. }
  227. /**
  228. * 订单
  229. *
  230. * @author 许祖兴 < zuxing.xu@lettered.cn>
  231. * @date 2020/7/6 11:09
  232. *
  233. * @return \think\response\Json
  234. * @throws \Lettered\Support\Exceptions\FailedException
  235. * @throws \think\db\exception\DataNotFoundException
  236. * @throws \think\db\exception\ModelNotFoundException
  237. * @throws \think\exception\DbException
  238. */
  239. public function order()
  240. {
  241. $param = $this->request->param();
  242. $limit = 10;
  243. // 数据校验
  244. $valid = $this->validate($param, [
  245. 'page' => 'require',
  246. ]);
  247. // 错误
  248. if (true !== $valid) {
  249. return $this->ApiJson(-1, $valid);
  250. }
  251. $orders = model('common/TaxiOrder')
  252. ->with(['paylog'])
  253. ->where(['user_id' => $this->auth->user()['id']])
  254. ->limit((($param['page'] - 1) * $limit) . "," . $limit)
  255. ->order(['id' => 'desc'])
  256. ->select();
  257. return $this->ApiJson(0,'获取成功', $orders);
  258. }
  259. /**
  260. * @desc 取消订单
  261. * @return \think\response\Json
  262. * @throws \think\db\exception\DataNotFoundException
  263. * @throws \think\db\exception\ModelNotFoundException
  264. * @throws \think\exception\DbException
  265. * @author weichuanbao<654745815@qq.com>
  266. * @date 2021/12/2 0002
  267. */
  268. public function cancelOrder()
  269. {
  270. $param = $this->request->param();
  271. // 数据校验
  272. $valid = $this->validate($param, [
  273. 'id' => 'require|number',
  274. ]);
  275. // 错误
  276. if (true !== $valid) {
  277. return $this->ApiJson(-1, $valid);
  278. }
  279. $row = model('common/TaxiOrder')->with(['paylog'])
  280. ->field('id,status,created_at')
  281. ->whereNotIn('status', [0,5])
  282. ->find($param['id']);
  283. $user = $this->auth->user();
  284. if ($row) {
  285. if (time() - $row['created_at'] < (60 * 10)) {
  286. // return $this->ApiJson(-1, '10分钟内无法取消订单');
  287. }
  288. if ($row['status'] == 2 || $row['status'] == 3) {
  289. $config = Cache::get('system_config');
  290. $total_price = $row->paylog['total_price'];
  291. // 如果平台已指派司机,则扣除部分金额
  292. if ($row['status'] == 3) {
  293. $total_price = $total_price * (1 - $config['store']['store_cancel_order']/100);
  294. }
  295. if ($row->paylog['pay_type'] == 'balance') {
  296. model('common/Users')->changeBalance(
  297. $user['id'],
  298. $total_price,
  299. "取消成功,取消金额【" . $total_price . "】",
  300. true
  301. );
  302. } else if ($row->paylog['pay_type'] == 'property') {
  303. model('common/Users')->changeProperty(
  304. $user['id'],
  305. $total_price,
  306. "取消成功,取消资产【{$total_price}】",
  307. true
  308. );
  309. } else if ($row->paylog['pay_type'] == 'wechat') {
  310. // 加载配置
  311. $wechat = sys_config('', 'wechat');
  312. $config = [
  313. // 前面的appid什么的也得保留哦
  314. 'app_id' => $wechat['mini_appid'],
  315. 'mch_id' => $wechat['pay_mch_id'],
  316. 'key' => $wechat['pay_secret_key'],
  317. // 如需使用敏感接口(如退款、发送红包等)需要配置 API 证书路径(登录商户平台下载 API 证书)
  318. 'cert_path' => $wechat['cert_path'], // XXX: 绝对路径!!!!
  319. 'key_path' => $wechat['key_path'], // XXX: 绝对路径!!!!
  320. // 'notify_url' => 'https://api.gxrrj.cn/api/v1/wechat/notify',
  321. // 'notify_url' => 'http://rrj.gxnwsoft.com/api/v1/wechat/refundNotify',
  322. // 'sandbox' => true
  323. ];
  324. // 创建应用实例
  325. $app = Factory::payment($config);
  326. // Example:
  327. $result = $app->refund->byOutTradeNumber($row->paylog['out_trade_no'], get_order_no(), $row->paylog['total_price'] * 100, $total_price * 100, [
  328. // 可在此处传入其他参数,详细参数见微信支付文档
  329. 'refund_desc' => '用户申请退款',
  330. ]);
  331. app()->log(json_encode($result));
  332. if ($result['return_code'] == 'SUCCESS') {
  333. if ($result['result_code'] != 'SUCCESS') {
  334. $this->ApiJson(-1, '失败');
  335. }
  336. }
  337. else {
  338. $this->ApiJson(-1, '成功');
  339. }
  340. }
  341. }
  342. // 后台推送
  343. push_socket_data('motor',[
  344. 'id' => $row['id'],
  345. 'msg' => '有用户取消订单,点击前往!'
  346. ]);
  347. $row->status = 5;
  348. $row->save();
  349. return $this->ApiJson(0, '取消成功', $row);
  350. }
  351. return $this->ApiJson(-1, '取消失败');
  352. }
  353. }