RejectedController.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. <?php
  2. namespace App\Http\Controllers\api\thirdPart\jianshang;
  3. use App\Components\ErrorPush;
  4. use App\Http\Controllers\Controller;
  5. use App\Logistic;
  6. use App\QualityLabel;
  7. use App\RejectedBill;
  8. use App\RejectedBillItem;
  9. use Illuminate\Http\Request;
  10. use Illuminate\Support\Collection;
  11. use Illuminate\Support\Facades\Auth;
  12. use Zttp\Zttp;
  13. use Zttp\ZttpResponse;
  14. class RejectedController extends Controller
  15. {
  16. use ErrorPush;
  17. function sendRejected($rejected){
  18. $amount = $rejected['amount'] ?? '';
  19. $created_at = $rejected['created_at'] ?? '';
  20. $fee_collected = $rejected['fee_collected'] ?? '';
  21. $goods_barcode = $rejected['barcode_goods'] ?? '';
  22. $goods_name = $rejected['name_goods'] ?? '';
  23. $logistic_name_return = Logistic::nameById($rejected['id_logistic_return']);
  24. $logistic_number = $rejected['logistic_number'] ?? '';
  25. $logistic_number_return = $rejected['logistic_number_return'] ?? '';
  26. $order_number = $rejected['order_number'] ?? '';
  27. $quality_label = QualityLabel::where('id',$rejected['id_quality_label'])->first();
  28. $quality_label = $quality_label?$quality_label['name']:'';
  29. $sender_mobile = $rejected['mobile_sender'] ?? '';
  30. $sender_name = $rejected['sender'] ?? '';
  31. $json = $this->packSendToRemote($amount,$created_at,$fee_collected,$goods_barcode,
  32. $goods_name,$logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  33. $quality_label,$sender_mobile,$sender_name);
  34. if(!$json){
  35. app('LogService')->log(__METHOD__,'error'.'_'.$logistic_number_return,"没有返回");
  36. return false;
  37. }
  38. if(!$json['success']){
  39. app('LogService')->log(__METHOD__,'error'.'_'.$logistic_number_return,"返回错误:".json_encode($json));
  40. return false;
  41. }else{
  42. app('LogService')->log(__METHOD__,'JianshangResponse'.'_'.$logistic_number_return,json_encode($json));
  43. }
  44. return $json;
  45. }
  46. function sendRejectedByApi(Request $request){
  47. $amount = $request->input('amount') ?? '';
  48. $created_at = $request->input('created_at') ?? '';
  49. $fee_collected = $request->input('fee_collected') ?? '';
  50. $goods_barcode = $request->input('goods_barcode') ?? '';
  51. $goods_name = $request->input('goods_name') ?? '';
  52. $logistic_name_return = $request->input('logistic_name_return') ?? '';
  53. $logistic_number = $request->input('logistic_number') ?? '';
  54. $logistic_number_return = $request->input('logistic_number_return') ?? '';
  55. $order_number = $request->input('order_number') ?? '';
  56. $quality_label = $request->input('quality_label') ?? '';
  57. $sender_mobile = $request->input('sender_mobile') ?? '';
  58. $sender_name = $request->input('sender_name') ?? '';
  59. $json = $this->packSendToRemote($amount,$created_at,$fee_collected,$goods_barcode,
  60. $goods_name,$logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  61. $quality_label,$sender_mobile,$sender_name);
  62. if(!$json)return ['success'=>'false','fail_info'=>'没有JSON回复'];
  63. if(!$json['success']){return $this->apiError(__METHOD__,"返回错误:".json_encode($json));}
  64. return $json;
  65. }
  66. private function packSendToRemote($amount,$created_at,$fee_collected,$goods_barcode,
  67. $goods_name,$logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  68. $quality_label,$sender_mobile,$sender_name){
  69. if(env('api_faking')){
  70. app('LogService')->log(__METHOD__,'error_'.__FUNCTION__,'API FAKING');
  71. return false;
  72. }
  73. $data="{\"amount\":\"$amount\", \"created_at\":\"$created_at\", \"fee_collected\":\"$fee_collected\",
  74. \"goods_barcode\":\"$goods_barcode\", \"goods_name\":\"$goods_name\", \"logistic_name_return\":\"$logistic_name_return\",
  75. \"logistic_number\":\"$logistic_number\", \"logistic_number_return\":\"$logistic_number_return\",
  76. \"order_number\":\"$order_number\", \"quality_label\":\"$quality_label\", \"sender_mobile\":\"$sender_mobile\", \"sender_name\":\"$sender_name\"}";
  77. $sortParamsAndMd5=function(Array $params,$timestamp,$nonce,$signKey){
  78. $params=new Collection($params);
  79. $params=$params->sortKeys();
  80. $str='';
  81. $params->each(function($param,$key)use(&$str){
  82. $val=trim($param);
  83. if($val){
  84. $str .= $key.$val;
  85. }
  86. });
  87. return strtolower(md5($str.$timestamp.$nonce.$signKey));
  88. };
  89. $timestamp=intval(microtime(true)*1000);
  90. $nonce=md5(microtime(true));
  91. $sign=$sortParamsAndMd5(json_decode($data,true),$timestamp,$nonce,config('api.sign_key_rejected_send_jianshang'));
  92. $url=config('api.url_rejected_send_jianshang');
  93. $response=Zttp::withHeaders([
  94. 'sign'=>$sign,
  95. 'nonce'=>$nonce,
  96. 'timestamp'=>$timestamp,
  97. 'Content-Type'=>'application/json'
  98. ])->post($url,json_decode($data,true));
  99. app('LogService')->log(__METHOD__,__FUNCTION__,$data);
  100. if(!$response){
  101. app('LogService')->log(__METHOD__,'error_'.__FUNCTION__,'回复异常:'.$response->body());
  102. return false;
  103. }
  104. if(!$response->json()){
  105. app('LogService')->log(__METHOD__,'error_'.__FUNCTION__,'json为空:'.$response->body());
  106. return false;
  107. }
  108. return $response->json();
  109. }
  110. function sendRejected2(array $rejected){
  111. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,__FUNCTION__,"发送笕尚接口,进入入口。退单号:{$rejected['logistic_number_return']}",Auth::user()['id']);
  112. $created_at = $rejected['created_at'] ?? '';
  113. $fee_collected = $rejected['fee_collected'] ?? '';
  114. $goodses=[];
  115. foreach ($rejected['goodses'] as $goods){
  116. $temGoods['amount']=$goods['amount'] ?? '';
  117. $temGoods['goods_barcode']=$goods['barcode_goods'] ?? '';
  118. $temGoods['goods_name']=$goods['name_goods'] ?? '';
  119. $temGoods['quality_label']=$goods['quality_label'] ?? '';
  120. $goodses[]=$temGoods;
  121. }
  122. $logistic_name_return = Logistic::nameById($rejected['id_logistic_return']);
  123. $logistic_number = $rejected['logistic_number'] ?? '';
  124. $logistic_number_return = $rejected['logistic_number_return'] ?? '';
  125. $order_number = $rejected['order_number'] ?? '';
  126. $sender_mobile = $rejected['mobile_sender'] ?? '';
  127. $sender_name = $rejected['sender'] ?? '';
  128. $json = $this->packSendToRemote2($created_at,$fee_collected,$goodses,
  129. $logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  130. $sender_mobile,$sender_name);
  131. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,__FUNCTION__,"发送笕尚接口调度完成。退单号:{$rejected['logistic_number_return']}",Auth::user()['id']);
  132. if(!$json){
  133. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,'error_'.__FUNCTION__,"发送笕尚接口异常,没有json返回。退单号:{$rejected['logistic_number_return']}",Auth::user()['id']);
  134. return 'fail';
  135. }else if(!isset($json['success'])||!$json['success']){
  136. if(strpos(json_encode($json),'已经接收过')!==false){
  137. return 'received';
  138. }
  139. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,'error_'.__FUNCTION__,"发送笕尚接口错误。退单号:{$rejected['logistic_number_return']}。".json_encode($json),Auth::user()['id']);
  140. return 'fail';
  141. }
  142. if(!$json['goodses_feedback']){
  143. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,'error_'.__FUNCTION__,"发送笕尚接口返回不全,没有商品信息返回(goodses)。退单号:{$rejected['logistic_number_return']}。".json_encode($json),Auth::user()['id']);
  144. }
  145. if(!$json['logistic_number_return']){
  146. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,'error_'.__FUNCTION__,"发送笕尚接口返回不全,没有退回快递单号。退单号:{$rejected['logistic_number_return']}。".json_encode($json),Auth::user()['id']);
  147. }
  148. $storable = (function ()use($json) {
  149. if(count($json['goodses_feedback'])==0)return false;
  150. foreach ($json['goodses_feedback'] as $goods){
  151. if(!$goods['storable']){
  152. return false;
  153. }
  154. }
  155. return true;
  156. })();
  157. if(!$storable){
  158. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,'error_'.__FUNCTION__,"发送笕尚接口提示,商品不可入库。退单号:{$rejected['logistic_number_return']}".json_encode($json),Auth::user()['id']);
  159. return 'none';
  160. }
  161. if(config('api.jianshang_rejecteds_log_switch'))Controller::logS(__METHOD__,__FUNCTION__,"发送笕尚接口完成,可入库,待其未来推单。退单号:{$rejected['logistic_number_return']}".json_encode($json),Auth::user()['id']);
  162. // $this->markItemsIsLoaded($json);
  163. return 'storable';
  164. }
  165. private function packSendToRemote2($created_at,$fee_collected,$goodses,
  166. $logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  167. $sender_mobile,$sender_name){
  168. foreach ($goodses as $goods){
  169. asort($goods);
  170. }
  171. $goodses = json_encode($goodses);
  172. $data="{\"created_at\":\"$created_at\", \"fee_collected\":\"$fee_collected\",
  173. \"goodses\":$goodses,\"logistic_name_return\":\"$logistic_name_return\",
  174. \"logistic_number\":\"$logistic_number\", \"logistic_number_return\":\"$logistic_number_return\",
  175. \"order_number\":\"$order_number\", \"sender_mobile\":\"$sender_mobile\", \"sender_name\":\"$sender_name\"}";
  176. $sortParamsAndMd5=function(Array $params,$timestamp,$nonce,$signKey){
  177. $params=new Collection($params);
  178. $params=$params->sortKeys();
  179. $str='';
  180. $params->each(function($param,$key)use(&$str){
  181. if(is_array($param)){return;
  182. // foreach ($param as &$paramValArr){
  183. // foreach ($paramValArr as &$paramVal){
  184. // $paramVal = trim($paramVal);
  185. // }
  186. // }
  187. // $param=json_encode($param);
  188. }
  189. $val=trim($param);
  190. if($val){
  191. $str .= $key.$val;
  192. }
  193. });
  194. $forSign = $str . $timestamp . $nonce . $signKey;
  195. return strtolower(md5($forSign));
  196. };
  197. if (!$data){
  198. app('LogService')->log(__METHOD__,'笕尚接口参数异常',json_encode(array($created_at,$fee_collected,$goodses,
  199. $logistic_name_return,$logistic_number,$logistic_number_return,$order_number,
  200. $sender_mobile,$sender_name)));
  201. return false;
  202. }
  203. $timestamp=intval(microtime(true)*1000);
  204. $nonce=md5(microtime(true));
  205. $sign=$sortParamsAndMd5(json_decode($data,true),$timestamp,$nonce,config('api.sign_key_rejected_send_jianshang'));
  206. app('LogService')->log(__METHOD__,'笕尚接口发送请求:',$data);
  207. $url=config('api.url_rejected_send_jianshang2');
  208. if(env('api_faking'))$url=url(config('api.fakingUrl_rejected_send_jianshang'));
  209. try{
  210. /** @var ZttpResponse $response */
  211. $response=Zttp::withHeaders([
  212. 'sign'=>$sign,
  213. 'nonce'=>$nonce,
  214. 'timestamp'=>$timestamp,
  215. 'Content-Type'=>'application/json'
  216. ])->post($url,json_decode($data,true));
  217. }catch (\Exception $e){
  218. app('LogService')->log(__METHOD__,'笕尚接口返回异常!!:',$e->getMessage().'||'.json_encode($data));
  219. $response=false;
  220. }
  221. if(!$response){
  222. app('LogService')->log(__METHOD__,'笕尚接口返回异常!!:',"沒有返回");
  223. return false;
  224. }
  225. app('LogService')->log(__METHOD__,'笕尚接口返回:',$response->body());
  226. return $response->json();
  227. }
  228. private function markItemsIsLoaded($json){
  229. $barcodes=[];
  230. foreach ($json['goodses_feedback'] as $goods){
  231. $barcodes[]=trim($goods['goods_barcode']);
  232. }
  233. $bill=RejectedBill::where('logistic_number_return',$json['logistic_number_return'])->first();
  234. $items=RejectedBillItem::where('id_rejected_bill',$bill['id'])->whereIn('barcode_goods',$barcodes)->get();
  235. $items->each(function(RejectedBillItem $item)use(&$json){
  236. foreach ($json['goodses_feedback'] as &$goods1){ //标记出有重复条码的商品,并且标记出其中是否同时有不能入库可和可以入库的
  237. $goods1['sameBarcode']=0;
  238. $goods1['hasUnStorable']=false;
  239. $goods1['hasStorable']=false;
  240. foreach ($json['goodses_feedback'] as &$goods2){
  241. if($goods1['goods_barcode']==$goods2['goods_barcode']){
  242. $goods1['sameBarcode']+=1;
  243. if(!$goods2['storable']){
  244. $goods1['hasUnStorable']=true;
  245. }else{
  246. $goods1['hasStorable']=true;
  247. }
  248. }
  249. }
  250. }
  251. foreach ($json['goodses_feedback'] as $goods){
  252. if($item['barcode_goods']==$goods['goods_barcode']){
  253. $item['is_loaded'] = $goods['storable']==='true'|| $goods['storable']===true ? '已入库':'未入库';
  254. if($goods['sameBarcode']>1){//如果该条码有不止一件,并且同时拥有入库和不入库的,就按品质标签来区分入库与否(因为回传不带是否正品消息,也没有ID)
  255. if($goods['hasUnStorable']&&$goods['hasStorable']){
  256. if($item['id_quality_label']==1){
  257. $item['is_loaded'] ='已入库';
  258. }else{
  259. $item['is_loaded'] ='未入库';
  260. }
  261. }
  262. }
  263. $item->update();
  264. }
  265. }
  266. });
  267. return $bill;
  268. }
  269. }