CloudWallet.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\library\LianLianPay;
  4. use llianpay\accp\security\LLianPayAccpSignature;
  5. use think\Db;
  6. use llianpay\accp\client\LLianPayClient;
  7. date_default_timezone_set('Asia/Shanghai');
  8. header('Content-type:application/json;charset=utf-8');
  9. require_once env('root_path').'application/common/library/llp/src/cfg.php';
  10. require_once env('root_path').'application/common/library/llp/src/client/LLianPayClient.php';
  11. /**
  12. * @title 云钱包【提现、充值和支付还是用之前接接口】
  13. * @desc 用的是测试环境的用户钱包,测试环境忽略user_id不同接口结果不一致的场景
  14. * @controller CloudWallet
  15. * @package app\api\controller
  16. * @group worker
  17. */
  18. class CloudWallet extends Base
  19. {
  20. protected $env = 0; // 0 测试环境 1正式环境
  21. protected $prefix = 'xl_';
  22. protected static $domain_name = 'test';
  23. protected $cloud_url = [
  24. //用户申请接入[云钱包申请开户]
  25. 'apply'=>[
  26. 'https://accpgw-ste.lianlianpay-inc.com/v1/acctmgr/openacct-apply',//测试环境
  27. 'https://accpgw.lianlianpay.com/v1/acctmgr/openacct-apply',//生产环境
  28. ],
  29. // 激活云钱包
  30. 'activate'=>[
  31. 'https://accpapi-ste.lianlianpay-inc.com/v1/acctmgr/openacct-activate-apply',//测试环境
  32. 'https://accpapi.lianlianpay.com/v1/acctmgr/openacct-activate-apply',//生产环境
  33. ],
  34. // 随机因子
  35. 'random'=>[
  36. 'https://accpapi-ste.lianlianpay-inc.com/v1/acctmgr/get-random',//测试环境
  37. 'https://accpapi.lianlianpay.com/v1/acctmgr/get-random',//生产环境
  38. ],
  39. // 获取云钱包用户信息
  40. 'userinfo'=>[
  41. 'https://accpapi-ste.lianlianpay-inc.com/v1/acctmgr/query-userinfo',//测试环境
  42. 'https://accpapi.lianlianpay.com/v1/acctmgr/query-userinfo',//生产环境
  43. ],
  44. // 获取云钱包账号信息
  45. 'acctinfo'=>[
  46. 'https://accpapi-ste.lianlianpay-inc.com/v1/acctmgr/query-acctinfo',//测试环境
  47. 'https://accpapi.lianlianpay.com/v1/acctmgr/query-acctinfo',//生产环境
  48. ],
  49. // 云钱包提现
  50. 'withdrawal'=>[
  51. 'https://accpapi-ste.lianlianpay-inc.com/v1/txn/withdrawal',//测试环境
  52. 'https://accpapi.lianlianpay.com/v1/txn/withdrawal',//生产环境
  53. ],
  54. //交易二次短信验证
  55. 'validation'=>[
  56. 'https://accpapi-ste.lianlianpay-inc.com/v1/txn/validation-sms',//测试环境
  57. 'https://accpapi.lianlianpay.com/v1/txn/validation-sms',//生产环境
  58. ],
  59. // 支付
  60. 'cashier'=>[
  61. 'https://accpgw-ste.lianlianpay-inc.com/v1/cashier/paycreate',//测试环境
  62. 'https://accpgw.lianlianpay.com/v1/cashier/paycreate',//生产环境
  63. ],
  64. ];
  65. public function initialize(){
  66. parent::initialize();
  67. parent::check_login();
  68. if(!checkAuth($this->uid))$this->error('暂未开放.....');
  69. }
  70. /**
  71. * @title 申请接入[开户]
  72. * @desc 申请接入[开户]
  73. * @author qc
  74. * @method POST
  75. * @tag 编辑信息
  76. * @url /api/Cloud_wallet/openAcctApply
  77. * @header name:Authorization require:1 desc:Token
  78. * @param name:user_id type:int require:0 default:0 desc:用户id
  79. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  80. * @return name:ret_msg type:string default:0 desc:请求结果描述
  81. * @return name:user_status type:string default:0 desc:用户状态【NORMAL:正常,其他异常】
  82. * @return name:bank_open_flag type:int default:0 desc:银行开户标识【0未开户,1已开户,2开户中】
  83. * @return name:bank_account type:string default:0 desc:ACCP合作银行账户
  84. */
  85. public function openAcctApply()
  86. {
  87. $user_id = $this->uid;
  88. $user_info = Db::name('store_member')->find($user_id);
  89. //if(empty($user_info)) $user_info = Db::name('store_member')->find(6);
  90. if(empty($user_info)) $this->error('用户信息有误');
  91. if(!$user_info['true_name'] || !$user_info['id_card']) $this->error('请先实名认证');
  92. //if(!$user_info['bank_num']) $this->error('请绑定银行卡');
  93. $current = date("YmdHis");
  94. $params = [
  95. 'timestamp' => $current,
  96. 'oid_partner' => OID_PARTNER,
  97. 'user_id' => $this->prefix.$user_id,
  98. 'txn_seqno' => get_order_sn(),
  99. 'txn_time' =>$current,
  100. 'flag_chnl' => "H5",
  101. 'notify_url' => 'https://'.$_SERVER['HTTP_HOST'] . '/api/Notify/openAcctApplyNotify',
  102. 'user_type' => 'INNERUSER',
  103. 'basicInfo'=>[
  104. 'reg_phone' => $user_info['phone'] ,
  105. 'user_name' => $user_info['true_name'] ,
  106. 'id_type' => 'ID_CARD',
  107. 'id_no' => $user_info['id_card'] ,
  108. 'occupation' =>18,
  109. ],
  110. 'accountInfo' => [
  111. 'account_type' => 'PERSONAL_PAYMENT_ACCOUNT'
  112. ],
  113. 'linkedAcctInfo'=>[
  114. //'linked_accttype'=> 'LEGALREPT_BANK_CARD',
  115. //'linked_acctno'=> $user_info['bank_num'] ,
  116. ],
  117. ];
  118. $url = $this->cloud_url['apply'][$this->env];
  119. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  120. $this->success('ok',json_decode($result,true));
  121. }
  122. /**
  123. * @title 激活申请
  124. * @desc 激活申请
  125. * @author qc
  126. * @method POST
  127. * @tag 编辑信息
  128. * @url /api/Cloud_wallet/activateApply
  129. * @header name:Authorization require:1 desc:Token
  130. * @param name:user_id type:int require:0 default:0 desc:用户id
  131. * @param name:password type:string require:0 default:0 desc:支付密码。6-12位的字母、数字,不可以是连续或者重复的数字和字母,正则:^[a-zA-Z0-9]{6,12}$。通过密码控件加密成密文传输
  132. * @param name:pkg_name type:string require:0 default:test desc:APP包名[h5或是测试不用传]
  133. * @param name:app_name type:string require:0 default:test desc:APP应用名[h5或是测试不用传]
  134. * @param name:flag_chnl type:string require:0 default:H5 desc:交易发起渠道传H5或APP
  135. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  136. * @return name:ret_msg type:string default:0 desc:请求结果描述
  137. * @return name:user_status type:string default:0 desc:用户状态【NORMAL:正常,其他异常】
  138. * @return name:bank_open_flag type:int default:0 desc:银行开户标识【0未开户,1已开户,2开户中】
  139. * @return name:bank_account type:string default:0 desc:ACCP合作银行账户
  140. */
  141. public function activateApply()
  142. {
  143. $user_id = $this->uid;
  144. $password = input('post.password');
  145. $flag_chnl = input('get.flag_chnl','H5');
  146. $pkg_name = input('get.pkg_name');
  147. $app_name = input('get.app_name');
  148. $current = date("YmdHis");
  149. $rand_arr = self::getRandomKey($this->prefix.$user_id, $this->cloud_url['random'][$this->env],$flag_chnl,$pkg_name,$app_name);
  150. if($rand_arr['ret_code'] != '0000') $this->error($rand_arr['ret_msg']);
  151. $params = [
  152. 'timestamp' => $current,
  153. 'oid_partner' => OID_PARTNER,
  154. 'user_id' => $this->prefix.$user_id,
  155. 'txn_seqno' => get_order_sn(),
  156. 'txn_time' =>$current,
  157. 'notify_url' => 'https://'.$_SERVER['HTTP_HOST'] . '/api/Notify/openAcctApplyNotify',
  158. 'password' => $password,
  159. 'random_key' => $rand_arr['random_key'],
  160. 'accountInfo'=>[
  161. 'account_type' => 'PERSONAL_PAYMENT_ACCOUNT'
  162. ]
  163. ];
  164. $url = $this->cloud_url['activate'][$this->env];
  165. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  166. $this->success('ok',json_decode($result,true));
  167. }
  168. /**
  169. * @title 获取随机因子
  170. * @desc 获取随机因子
  171. * @author qc
  172. * @method GET
  173. * @tag 编辑信息
  174. * @url /api/Cloud_wallet/getRandomKeys
  175. * @header name:Authorization require:1 desc:Token
  176. * @param name:user_id type:int require:0 default:0 desc:用户id
  177. * @param name:pkg_name type:string require:0 default:test desc:APP包名[h5或是测试不用传]
  178. * @param name:app_name type:string require:0 default:test desc:APP应用名[h5或是测试不用传]
  179. * @param name:flag_chnl type:string require:0 default:H5 desc:交易发起渠道传H5或APP
  180. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  181. * @return name:ret_msg type:string default:0 desc:请求结果描述
  182. * @return name:random_key type:string default:0 desc:随机因子key,有效期30分钟
  183. * @return name:random_value type:string default:0 desc:随机因子值,有效期30分钟
  184. */
  185. public function getRandomKeys()
  186. {
  187. $user_id = $this->uid;
  188. $flag_chnl = input('get.flag_chnl','H5');
  189. $pkg_name = input('get.pkg_name');
  190. $app_name = input('get.app_name');
  191. $res = self::getRandomKey($this->prefix.$user_id, $this->cloud_url['random'][$this->env],$flag_chnl,$pkg_name,$app_name);
  192. $this->success('ok',$res);
  193. }
  194. // 获取随机因子
  195. public static function getRandomKey($user_id,$url,$flag_chnl,$pkg_name,$app_name)
  196. {
  197. $current = date("YmdHis");
  198. $params = [
  199. 'timestamp' => $current,
  200. 'oid_partner' => OID_PARTNER,
  201. 'user_id' => $user_id,
  202. 'flag_chnl' => $flag_chnl,
  203. 'pkg_name' => $flag_chnl == 'H5' ? static::$domain_name :($pkg_name ? $pkg_name : 'test') ,
  204. 'app_name' => $flag_chnl == 'H5' ? static::$domain_name :($app_name ? $app_name : 'test') ,
  205. ];
  206. //var_dump($params);
  207. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  208. return json_decode($result,JSON_UNESCAPED_UNICODE);
  209. }
  210. /**
  211. * @title 获取云钱包用户信息
  212. * @desc 获取云钱包用户信息
  213. * @author qc
  214. * @method GET
  215. * @tag 编辑信息
  216. * @url /api/Cloud_wallet/getUserInfo
  217. * @header name:Authorization require:1 desc:Token
  218. * @param name:user_id type:int require:0 default:0 desc:用户id
  219. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  220. * @return name:ret_msg type:string default:0 desc:请求结果描述
  221. * @return name:user_status type:string default:0 desc:用户状态【NORMAL:正常,其他异常】
  222. * @return name:bank_open_flag type:int default:0 desc:银行开户标识【0未开户,1已开户,2开户中】
  223. * @return name:bank_account type:string default:0 desc:ACCP合作银行账户
  224. */
  225. public function getUserInfo()
  226. {
  227. $user_id = $this->uid;
  228. $current = date("YmdHis");
  229. $params = [
  230. 'timestamp' => $current,
  231. 'oid_partner' => OID_PARTNER,
  232. 'user_id' => $this->prefix.$user_id,
  233. ];
  234. $url = $url = $this->cloud_url['userinfo'][$this->env];
  235. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  236. $res = LianLianPay::userLevel($this->uid);
  237. // var_dump($res);
  238. $this->success('ok',json_decode($result,true));
  239. }
  240. /**
  241. * @title 获取云钱包账户信息
  242. * @desc 获取云钱包账户信息
  243. * @author qc
  244. * @method GET
  245. * @tag 编辑信息
  246. * @url /api/Cloud_wallet/getAcctInfo
  247. * @header name:Authorization require:1 desc:Token
  248. * @param name:user_id type:int require:0 default:0 desc:用户id
  249. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  250. * @return name:ret_msg type:string default:0 desc:请求结果描述
  251. * @return name:ret_msg type:string default:0 desc:请求结果描述
  252. * @return name:acctinfo_list type:array default:0 desc:账户列表
  253. * @return name:acctinfo_list.oid_acctno type:string default:0 desc:账户号
  254. * @return name:acctinfo_list.acct_state type:string default:0 desc:账户状态【NORMAL:正常,其他异常】
  255. * @return name:acctinfo_list.amt_balcur type:float default:0 desc:资金余额[单位:元]
  256. * @return name:acctinfo_list.amt_balaval type:float default:0 desc:可用余额[单位:元]
  257. * @return name:acctinfo_list.amt_balfrz type:float default:0 desc:冻结金额[单位:元]
  258. * @return name:acctinfo_list.amt_lastbal type:float default:0 desc:昨日资金余额[单位:元]
  259. * @return name:acctinfo_list.amt_lastaval type:float default:0 desc:昨日可用余额[单位:元]
  260. * @return name:acctinfo_list.amt_lastfrz type:float default:0 desc:昨日冻结金额[单位:元]
  261. * @return name:total_money type:float default:0 desc:总资产
  262. * @return name:total_balaval type:float default:0 desc:总可用
  263. * @return name:total_balfrz type:float default:0 desc:总冻结
  264. *
  265. */
  266. public function getAcctInfo()
  267. {
  268. $user_id = $this->uid;
  269. $current = date("YmdHis");
  270. $params = [
  271. 'timestamp' => $current,
  272. 'oid_partner' => OID_PARTNER,
  273. 'user_id' => $this->prefix.$user_id,
  274. 'user_type' => 'INNERUSER',
  275. ];
  276. $url = $url = $this->cloud_url['acctinfo'][$this->env];
  277. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  278. $res = json_decode($result,true);
  279. if(isset($res['acctinfo_list'])) {
  280. $total_money = bcadd($res['acctinfo_list'][0]['amt_balcur'],$res['acctinfo_list'][1]['amt_balcur'],2);
  281. $total_balaval = bcadd($res['acctinfo_list'][0]['amt_balaval'],$res['acctinfo_list'][1]['amt_balaval'],2);
  282. $total_balfrz = bcadd($res['acctinfo_list'][0]['amt_balfrz'],$res['acctinfo_list'][1]['amt_balfrz'],2);
  283. }
  284. $res['total_money'] = !empty($total_money) ? $total_money : 0;
  285. $res['total_balaval'] = !empty($total_balaval) ? $total_balaval : 0;
  286. $res['total_balfrz'] = !empty($total_balfrz) ? $total_balfrz : 0;
  287. $this->success('ok',$res);
  288. }
  289. /**
  290. * @title 交易二次短信验证
  291. * @desc 交易二次短信验证
  292. * @author qc
  293. * @method GET
  294. * @tag 编辑信息
  295. * @url /api/Cloud_wallet/validationSms
  296. * @header name:Authorization require:1 desc:Token
  297. * @param name:user_id type:int require:0 default:0 desc:用户id
  298. * @param name:txn_seqno type:int require:0 default:0 desc:订单号
  299. * @param name:total_amount type:int require:0 default:0 desc:订单金额【提现金额】
  300. * @param name:token type:int require:0 default:0 desc:token
  301. * @param name:verify_code type:int require:0 default:0 desc:验证码
  302. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  303. * @return name:ret_msg type:string default:0 desc:请求结果描述
  304. */
  305. public function validationSms()
  306. {
  307. $user_id = $this->uid;
  308. $token = input('get.token');
  309. $txn_seqno = input('get.txn_seqno');
  310. $total_amount = input('get.total_amount');
  311. $verify_code = input('get.verify_code');
  312. $current = date("YmdHis");
  313. $params = [
  314. 'timestamp' => $current,
  315. 'oid_partner' => OID_PARTNER,
  316. 'payer_type'=>"USER",
  317. 'payer_id' => $this->prefix.$user_id,
  318. 'txn_seqno'=>$txn_seqno,
  319. 'total_amount'=>$total_amount,
  320. 'token'=>$token,
  321. 'verify_code'=>$verify_code
  322. ];
  323. $url = $this->cloud_url['validation'][$this->env];
  324. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  325. $this->success('ok',json_decode($result,true));
  326. }
  327. /**
  328. * @title 收银台支付测试【测试用不对接】
  329. * @desc 收银台支付测试
  330. * @author qc
  331. * @method GET
  332. * @tag 编辑信息
  333. * @url /api/Cloud_wallet/cashierPay
  334. */
  335. public function cashierPay()
  336. {
  337. $voucher_order = [];
  338. $voucher_order['order_no'] = get_order_sn();
  339. $money = 10;
  340. $from = input('post.from',1);
  341. $user = Db::name('store_member')->where('id',$this->uid)->find();
  342. $notify_url = 'https://'.$_SERVER['HTTP_HOST'] . '/api/Pay/lianLianNotify';
  343. $return_url ='https://'.$_SERVER['HTTP_HOST']."/h5/pages/mine/mine";//回调地址
  344. $risk_item= risk_item($user,'余额充值',$from == 1 ? 16 : 10);
  345. $res= LianLianPay::cashierPay($voucher_order['order_no'],$money,$this->uid,$notify_url,$return_url,'余额充值',[[
  346. 'payee_id'=>OID_PARTNER,
  347. 'payee_type' => 'MERCHANT',
  348. 'payee_amount' => $money,
  349. ]],$risk_item);
  350. $this->success('ok',$res);
  351. }
  352. /**
  353. * @title 云钱包提现【测试用不对接】
  354. * @desc 云钱包提现
  355. * @author qc
  356. * @method POST
  357. * @tag 编辑信息
  358. * @url /api/Cloud_wallet/withdrawal
  359. * @header name:Authorization require:1 desc:Token
  360. * @param name:user_id type:int require:0 default:0 desc:用户id
  361. * @param name:source type:int require:0 default:10 desc:app传10,h5传16
  362. * @param name:pkg_name type:string require:0 default:test desc:APP包名[h5或是测试不用传]
  363. * @param name:app_name type:string require:0 default:test desc:APP应用名[h5或是测试不用传]
  364. * @param name:flag_chnl type:string require:0 default:H5 desc:交易发起渠道传H5或ANDROID或IOS
  365. * @return name:ret_code type:string default:0 desc:0000:成功,其他异常
  366. * @return name:ret_msg type:string default:0 desc:请求结果描述
  367. */
  368. public function withdrawal()
  369. {
  370. $user_id = $this->uid;
  371. $money = input('post.money');
  372. $password = input('post.password');
  373. $flag_chnl = input('get.flag_chnl','H5');
  374. $pkg_name = input('get.pkg_name');
  375. $app_name = input('get.app_name');
  376. $random_key = self::getRandomKey($this->prefix.$user_id, $this->cloud_url['random'][$this->env],$flag_chnl,$pkg_name,$app_name);
  377. $current = date("YmdHis");
  378. $user_info = Db::name('store_member')->where('id',$user_id)->find();
  379. $risk_item = risk_item($user_info,'余额提现',input('post,source',10));
  380. $notify_url = 'https://'.$_SERVER['HTTP_HOST'] . '/api/Pay/withdrawalNotify';
  381. $params = [
  382. 'timestamp' => $current,
  383. 'oid_partner' => OID_PARTNER,
  384. 'notify_url'=> $notify_url,
  385. 'risk_item'=>json_encode($risk_item),
  386. 'linked_acctno'=>$user_info['bank_num'],
  387. 'check_flag'=> 'N',
  388. 'pay_time_type'=> 'TRANS_NORMAL',
  389. 'orderInfo'=>[
  390. 'txn_seqno'=>get_order_sn(),
  391. 'txn_time' => $current,
  392. 'total_amount' => $money,
  393. ],
  394. 'payerInfo'=>[
  395. 'payer_type'=> 'USER',
  396. 'payer_id'=> $this->prefix.$user_id,
  397. 'payer_accttype'=> 'USEROWN',
  398. 'password'=> $password,
  399. 'random_key'=> $random_key['random_key'],
  400. ]
  401. ];
  402. $url = $url = $this->cloud_url['withdrawal'][$this->env];
  403. $result = LLianPayClient::sendRequest($url, json_encode($params,JSON_UNESCAPED_UNICODE));
  404. $this->success('ok',json_decode($result,true));
  405. }
  406. public function payNotifyTest()
  407. {
  408. $result = '{"oid_partner":"2020042200284052","payerInfo":[{"amount":"1.00","method":"BALANCE","payer_type":"USER","payer_id":"1"}],"txn_type":"GENERAL_CONSUME","payeeInfo":[{"amount":"1.00","payee_id":"2020042200284052","payee_type":"MERCHANT"}],"orderInfo":{"total_amount":"1.00","txn_seqno":"202210091020109197255438","txn_time":"20221009102010"},"txn_status":"TRADE_SUCCESS","accounting_date":"20221009","finish_time":"20221009102029","accp_txno":"2022100916760526"}';
  409. $header = '{"accept-encoding":"gzip,deflate","host":"xianglian-api.hdlkeji.com","content-length":"458","signature-type":"RSA","signature-data":"a5BMrkODcWl6FJ2vOEMG\/dnDbHfsBp34yY9ls9KmaWv5zYYO2EoK+DwL9PAFWTFzb0sKKYaOgrO8mFjl4CclTEd5UWvDFTAhv+EijImHTm4rGwbv1k2fpiw5MCSci1JTAE3ROsocgDWbLSsuFbTQ\/d97oR9KJbrjIInlVPEjO04=","correlationid":"2b9369df-ca95-49df-8f25-10fca9854c02","user-agent":"httpcomponents","connection":"Close","content-type":"text\/json;charset=UTF-8"}';
  410. $result = json_decode($result,true);
  411. $header = json_decode($header,true);
  412. // var_dump($result,$header);
  413. $check_sign = LLianPayAccpSignature::checkSign(json_encode($result,JSON_UNESCAPED_UNICODE),$header['signature-data']);
  414. var_dump($check_sign);
  415. $orderNo = $result['orderInfo']['txn_seqno'];
  416. var_dump(['out_trade_no' => $orderNo,'user_id'=> trim($result['payerInfo'][0]['payer_id'],$this->prefix)]);
  417. }
  418. }