UserWalletApply.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. <?php
  2. namespace addons\shopro\model;
  3. use think\Db;
  4. use think\Model;
  5. use traits\model\SoftDelete;
  6. use addons\shopro\exception\Exception;
  7. /**
  8. * 钱包
  9. */
  10. class UserWalletApply extends Model
  11. {
  12. use SoftDelete;
  13. // 表名,不含前缀
  14. protected $name = 'shopro_user_wallet_apply';
  15. // 自动写入时间戳字段
  16. protected $autoWriteTimestamp = 'int';
  17. // 定义时间戳字段名
  18. protected $createTime = 'createtime';
  19. protected $updateTime = 'updatetime';
  20. protected $deleteTime = false;
  21. protected $hidden = ['actual_money', 'log', 'payment_json', 'updatetime'];
  22. // 追加属性
  23. protected $append = [
  24. 'status_text',
  25. 'apply_type_text',
  26. ];
  27. /**
  28. * 获取提现单号
  29. *
  30. * @param int $user_id
  31. * @return string
  32. */
  33. public static function getSn($user_id)
  34. {
  35. $rand = $user_id < 9999 ? mt_rand(100000, 99999999) : mt_rand(100, 99999);
  36. $order_sn = date('Yhis') . $rand;
  37. $id = str_pad($user_id, (24 - strlen($order_sn)), '0', STR_PAD_BOTH);
  38. return 'W' . $order_sn . $id;
  39. }
  40. // 提现记录
  41. public static function getList()
  42. {
  43. $user = User::info();
  44. $walletApplys = self::where(['user_id' => $user->id])->order('id desc')->paginate(10);
  45. return $walletApplys;
  46. }
  47. /**
  48. * 申请提现
  49. *
  50. * @param int $type 提现方式 wechat|alipay|bank
  51. * @param int $money 提现金额
  52. */
  53. public static function apply($type, $money)
  54. {
  55. $user = User::info();
  56. $config = self::getWithdrawConfig();
  57. if (!in_array($type, $config['methods'])) {
  58. throw \Exception('暂不支持该提现方式');
  59. }
  60. $min = round(floatval($config['min']), 2);
  61. $max = round(floatval($config['max']), 2);
  62. $service_fee = round(floatval($config['service_fee']), 3); // 三位小数
  63. // 检查最小提现金额
  64. if ($money < $min || $money <= 0) {
  65. throw \Exception('提现金额不能少于 ' . $min . '元');
  66. }
  67. // 检查最大提现金额
  68. if ($max && $money > $max) {
  69. throw \Exception('提现金额不能大于 ' . $max . '元');
  70. }
  71. // 计算手续费
  72. $charge = $money * $service_fee;
  73. if ($user->money < $money) {
  74. throw \Exception('可提现余额不足');
  75. }
  76. // 检查提现账户信息
  77. $bank = \addons\shopro\model\UserBank::info($type, false);
  78. // 添加提现记录
  79. $platform = request()->header('platform');
  80. $apply = new self();
  81. $apply->apply_sn = self::getSn($user->id);
  82. $apply->user_id = $user->id;
  83. $apply->money = $money;
  84. $apply->charge_money = $charge;
  85. $apply->service_fee = $service_fee;
  86. $apply->apply_type = $type;
  87. $apply->platform = $platform;
  88. switch ($type) {
  89. case 'wechat':
  90. $applyInfo = [
  91. '微信用户' => $bank['real_name'],
  92. '微信ID' => $bank['card_no'],
  93. ];
  94. break;
  95. case 'alipay':
  96. $applyInfo = [
  97. '真实姓名' => $bank['real_name'],
  98. '支付宝账户' => $bank['card_no']
  99. ];
  100. break;
  101. case 'bank':
  102. $applyInfo = [
  103. '真实姓名' => $bank['real_name'],
  104. '开户行' => $bank['bank_name'],
  105. '银行卡号' => $bank['card_no']
  106. ];
  107. break;
  108. }
  109. if (!isset($applyInfo)) {
  110. throw \Exception('您的提现信息有误');
  111. }
  112. if($type == 'wechat' && $user->skm =='') throw \Exception('请在个人信息完善收款码!');
  113. $apply->apply_info = $applyInfo;
  114. $apply->wechat_code = $user->skm ;
  115. $apply->status = 0;
  116. $apply->save();
  117. self::handleLog($apply, '用户发起提现申请');
  118. // 扣除用户余额
  119. // User::money(- ($money + $charge), $user->id, 'cash', $apply->id);
  120. User::money(- $money , $user->id, 'cash', $apply->id);
  121. // 检查是否执行自动打款
  122. $autoCheck = false;
  123. if ($type !== 'bank' && $config['wechat_alipay_auto']) {
  124. $autoCheck = true;
  125. }
  126. if ($autoCheck) {
  127. $apply = self::handleAgree($apply);
  128. $apply = self::handleWithdraw($apply);
  129. }
  130. return $apply;
  131. }
  132. public static function handleLog($apply, $oper_info)
  133. {
  134. $log = $apply->log;
  135. $oper = \addons\shopro\library\Oper::set();
  136. $log[] = [
  137. 'oper_type' => $oper['oper_type'],
  138. 'oper_id' => $oper['oper_id'],
  139. 'oper_info' => $oper_info,
  140. 'oper_time' => time()
  141. ];
  142. $apply->log = $log;
  143. $apply->save();
  144. return $apply;
  145. }
  146. // 同意
  147. public static function handleAgree($apply)
  148. {
  149. if ($apply->status != 0) {
  150. throw \Exception('请勿重复操作');
  151. }
  152. $apply->status = 1;
  153. $apply->save();
  154. return self::handleLog($apply, '同意提现申请');
  155. }
  156. // 处理打款
  157. public static function handleWithdraw($apply)
  158. {
  159. $withDrawStatus = false;
  160. if ($apply->status != 1) {
  161. throw \Exception('请勿重复操作');
  162. }
  163. if ($apply->apply_type !== 'bank') {
  164. $withDrawStatus = true;//self::handleTransfer($apply);
  165. } else {
  166. $withDrawStatus = true;
  167. }
  168. if ($withDrawStatus) {
  169. $apply->status = 2;
  170. $apply->actual_money = $apply->money;
  171. $apply->save();
  172. return self::handleLog($apply, '已打款');
  173. }
  174. return $apply;
  175. }
  176. // 拒绝
  177. public static function handleReject($apply, $rejectInfo)
  178. {
  179. if ($apply->status != 0 && $apply->status != 1) {
  180. throw \Exception('请勿重复操作');
  181. }
  182. $apply->status = -1;
  183. $apply->save();
  184. User::money($apply->money + $apply->charge_money, $apply->user_id, 'cash_error', $apply->id);
  185. return self::handleLog($apply, '拒绝:' . $rejectInfo);
  186. }
  187. // 企业付款提现
  188. private static function handleTransfer($apply)
  189. {
  190. $type = $apply->apply_type;
  191. $platform = $apply->platform;
  192. // 1.企业自动付款
  193. $pay = new \addons\shopro\library\PayService($type, $platform, '', 'transfer');
  194. // 2.组装数据
  195. try {
  196. if ($type == 'wechat') {
  197. $payload = [
  198. 'partner_trade_no' => $apply->apply_sn,
  199. 'openid' => $apply->apply_info['微信ID'],
  200. 'check_name' => 'NO_CHECK',
  201. 'amount' => $apply->money,
  202. 'desc' => "用户[{$apply->apply_info['微信用户']}]提现"
  203. ];
  204. } elseif ($type == 'alipay') {
  205. $payload = [
  206. 'out_biz_no' => $apply->apply_sn,
  207. 'trans_amount' => $apply->money,
  208. 'product_code' => 'TRANS_ACCOUNT_NO_PWD',
  209. 'biz_scene' => 'DIRECT_TRANSFER',
  210. // 'order_title' => '余额提现到',
  211. 'remark' => '用户提现',
  212. 'payee_info' => [
  213. 'identity' => $apply->apply_info['支付宝账户'],
  214. 'identity_type' => 'ALIPAY_LOGON_ID',
  215. 'name' => $apply->apply_info['真实姓名'],
  216. ]
  217. ];
  218. }
  219. } catch (\Exception $e) {
  220. throw \Exception('提现信息不正确');
  221. }
  222. // 3.发起付款
  223. try {
  224. list($code, $response) = $pay->transfer($payload);
  225. if ($code === 1) {
  226. $apply->payment_json = json_encode($response, JSON_UNESCAPED_UNICODE);
  227. $apply->save();
  228. return true;
  229. }
  230. } catch (\Exception $e) {
  231. \think\Log::error('提现失败:' . ' 行号:' . $e->getLine() . '文件:' . $e->getFile() . '错误信息:' . $e->getMessage());
  232. throw \Exception($e->getMessage());
  233. }
  234. return false;
  235. }
  236. /**
  237. * 提现类型列表
  238. */
  239. public function getApplyTypeList()
  240. {
  241. return ['bank' => '银行卡', 'wechat' => '微信零钱', 'alipay' => '支付宝账户'];
  242. }
  243. /**
  244. * 提现类型中文
  245. */
  246. public function getApplyTypeTextAttr($value, $data)
  247. {
  248. $value = isset($data['apply_type']) ? $data['apply_type'] : '';
  249. $list = $this->getApplyTypeList();
  250. return isset($list[$value]) ? $list[$value] : '';
  251. }
  252. /**
  253. * 提现信息
  254. */
  255. public function getApplyInfoAttr($value, $data)
  256. {
  257. $value = isset($data['apply_info']) ? $data['apply_info'] : $value;
  258. return json_decode($value, true);
  259. }
  260. /**
  261. * 提现信息 格式转换
  262. */
  263. public function setApplyInfoAttr($value, $data)
  264. {
  265. $value = isset($data['apply_info']) ? $data['apply_info'] : $value;
  266. $applyInfo = json_encode($value, JSON_UNESCAPED_UNICODE);
  267. return $applyInfo;
  268. }
  269. public function getStatusTextAttr($value, $data)
  270. {
  271. switch ($data['status']) {
  272. case 0:
  273. $status_name = '审核中';
  274. break;
  275. case 1:
  276. $status_name = '处理中';
  277. break;
  278. case 2:
  279. $status_name = '已处理';
  280. break;
  281. case -1:
  282. $status_name = '已拒绝';
  283. break;
  284. default:
  285. $status_name = '';
  286. }
  287. return $status_name;
  288. }
  289. public static function getWithdrawConfig()
  290. {
  291. $config = \addons\shopro\model\Config::where('name', 'withdraw')->find();
  292. return json_decode($config['value'], true);
  293. }
  294. /**
  295. * 获取日志字段数组
  296. */
  297. public function getLogAttr($value, $data)
  298. {
  299. $value = array_filter((array)json_decode($value, true));
  300. return (array)$value;
  301. }
  302. /**
  303. * 设置日志字段
  304. * @param mixed $value
  305. * @return string
  306. */
  307. public function setLogAttr($value)
  308. {
  309. $value = is_object($value) || is_array($value) ? json_encode($value, JSON_UNESCAPED_UNICODE) : $value;
  310. return $value;
  311. }
  312. }