UserWalletApply.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  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 < $charge + $money) {
  74. throw \Exception('可提现余额不足');
  75. }
  76. // 检查每日最大提现次数
  77. if (isset($config['perday_num']) && $config['perday_num'] > 0) {
  78. $num = self::where(['user_id' => $user->id, 'createtime' => ['egt', strtotime(date("Y-m-d", time()))]])->count();
  79. if ($num >= $config['perday_num']) {
  80. throw \Exception('每日提现次数不能大于 ' . $config['perday_num'] . '次');
  81. }
  82. }
  83. // 检查每日最大提现金额
  84. if (isset($config['perday_amount']) && $config['perday_num'] > 0) {
  85. $amount = self::where(['user_id' => $user->id, 'createtime' => ['egt', strtotime(date("Y-m-d", time()))]])->sum('money');
  86. if ($amount >= $config['perday_amount']) {
  87. throw \Exception('每日提现金额不能大于 ' . $config['perday_amount'] . '元');
  88. }
  89. }
  90. // 检查提现账户信息
  91. $bank = \addons\shopro\model\UserBank::info($type, false);
  92. // 添加提现记录
  93. $platform = request()->header('platform');
  94. $apply = new self();
  95. $apply->apply_sn = self::getSn($user->id);
  96. $apply->user_id = $user->id;
  97. $apply->money = $money;
  98. $apply->charge_money = $charge;
  99. $apply->service_fee = $service_fee;
  100. $apply->apply_type = $type;
  101. $apply->platform = $platform;
  102. switch ($type) {
  103. case 'wechat':
  104. $applyInfo = [
  105. '微信用户' => $bank['real_name'],
  106. '微信ID' => $bank['card_no'],
  107. ];
  108. break;
  109. case 'alipay':
  110. $applyInfo = [
  111. '真实姓名' => $bank['real_name'],
  112. '支付宝账户' => $bank['card_no']
  113. ];
  114. break;
  115. case 'bank':
  116. $applyInfo = [
  117. '真实姓名' => $bank['real_name'],
  118. '开户行' => $bank['bank_name'],
  119. '银行卡号' => $bank['card_no']
  120. ];
  121. break;
  122. }
  123. if (!isset($applyInfo)) {
  124. throw \Exception('您的提现信息有误');
  125. }
  126. $apply->apply_info = $applyInfo;
  127. $apply->status = 0;
  128. $apply->save();
  129. self::handleLog($apply, '用户发起提现申请');
  130. // 扣除用户余额
  131. User::money(- ($money + $charge), $user->id, 'cash', $apply->id);
  132. // 检查是否执行自动打款
  133. $autoCheck = false;
  134. if ($type !== 'bank' && $config['wechat_alipay_auto']) {
  135. $autoCheck = true;
  136. }
  137. var_dump(66666);
  138. if ($autoCheck) {
  139. $apply = self::handleAgree($apply);
  140. $apply = self::handleWithdraw($apply);
  141. }
  142. return $apply;
  143. }
  144. public static function handleLog($apply, $oper_info)
  145. {
  146. $log = $apply->log;
  147. $oper = \addons\shopro\library\Oper::set();
  148. $log[] = [
  149. 'oper_type' => $oper['oper_type'],
  150. 'oper_id' => $oper['oper_id'],
  151. 'oper_info' => $oper_info,
  152. 'oper_time' => time()
  153. ];
  154. $apply->log = $log;
  155. $apply->save();
  156. return $apply;
  157. }
  158. // 同意
  159. public static function handleAgree($apply)
  160. {
  161. if ($apply->status != 0) {
  162. throw \Exception('请勿重复操作');
  163. }
  164. $apply->status = 1;
  165. $apply->save();
  166. return self::handleLog($apply, '同意提现申请');
  167. }
  168. // 处理打款
  169. public static function handleWithdraw($apply)
  170. {
  171. $withDrawStatus = false;
  172. if ($apply->status != 1) {
  173. throw \Exception('请勿重复操作');
  174. }
  175. if ($apply->apply_type !== 'bank') {
  176. $withDrawStatus = self::handleTransfer($apply);
  177. } else {
  178. $withDrawStatus = true;
  179. }
  180. if ($withDrawStatus) {
  181. $apply->status = 2;
  182. $apply->actual_money = $apply->money;
  183. $apply->save();
  184. return self::handleLog($apply, '已打款');
  185. }
  186. return $apply;
  187. }
  188. // 拒绝
  189. public static function handleReject($apply, $rejectInfo)
  190. {
  191. if ($apply->status != 0 && $apply->status != 1) {
  192. throw \Exception('请勿重复操作');
  193. }
  194. $apply->status = -1;
  195. $apply->save();
  196. User::money($apply->money + $apply->charge_money, $apply->user_id, 'cash_error', $apply->id);
  197. return self::handleLog($apply, '拒绝:' . $rejectInfo);
  198. }
  199. // 企业付款提现
  200. private static function handleTransfer($apply)
  201. {
  202. $type = $apply->apply_type;
  203. $platform = $apply->platform;
  204. // 1.企业自动付款
  205. $pay = new \addons\shopro\library\PayService($type, $platform, '', 'transfer');
  206. // 2.组装数据
  207. try {
  208. if ($type == 'wechat') {
  209. $payload = [
  210. 'partner_trade_no' => $apply->apply_sn,
  211. 'openid' => $apply->apply_info['微信ID'],
  212. 'check_name' => 'NO_CHECK',
  213. 'amount' => $apply->money,
  214. 'desc' => "用户[{$apply->apply_info['微信用户']}]提现"
  215. ];
  216. } elseif ($type == 'alipay') {
  217. $payload = [
  218. 'out_biz_no' => $apply->apply_sn,
  219. 'trans_amount' => $apply->money,
  220. 'product_code' => 'TRANS_ACCOUNT_NO_PWD',
  221. 'biz_scene' => 'DIRECT_TRANSFER',
  222. // 'order_title' => '余额提现到',
  223. 'remark' => '用户提现',
  224. 'payee_info' => [
  225. 'identity' => $apply->apply_info['支付宝账户'],
  226. 'identity_type' => 'ALIPAY_LOGON_ID',
  227. 'name' => $apply->apply_info['真实姓名'],
  228. ]
  229. ];
  230. }
  231. } catch (\Exception $e) {
  232. throw \Exception('提现信息不正确');
  233. }
  234. // 3.发起付款
  235. try {
  236. list($code, $response) = $pay->transfer($payload);
  237. if ($code === 1) {
  238. $apply->payment_json = json_encode($response, JSON_UNESCAPED_UNICODE);
  239. $apply->save();
  240. return true;
  241. }
  242. } catch (\Exception $e) {
  243. // \think\Log::error('提现失败:' . ' 行号:' . $e->getLine() . '文件:' . $e->getFile() . '错误信息:' . $e->getMessage());
  244. throw \Exception($e->getMessage());
  245. }
  246. return false;
  247. }
  248. /**
  249. * 提现类型列表
  250. */
  251. public function getApplyTypeList()
  252. {
  253. return ['bank' => '银行卡', 'wechat' => '微信零钱', 'alipay' => '支付宝账户'];
  254. }
  255. /**
  256. * 提现类型中文
  257. */
  258. public function getApplyTypeTextAttr($value, $data)
  259. {
  260. $value = isset($data['apply_type']) ? $data['apply_type'] : '';
  261. $list = $this->getApplyTypeList();
  262. return isset($list[$value]) ? $list[$value] : '';
  263. }
  264. /**
  265. * 提现信息
  266. */
  267. public function getApplyInfoAttr($value, $data)
  268. {
  269. $value = isset($data['apply_info']) ? $data['apply_info'] : $value;
  270. return json_decode($value, true);
  271. }
  272. /**
  273. * 提现信息 格式转换
  274. */
  275. public function setApplyInfoAttr($value, $data)
  276. {
  277. $value = isset($data['apply_info']) ? $data['apply_info'] : $value;
  278. $applyInfo = json_encode($value, JSON_UNESCAPED_UNICODE);
  279. return $applyInfo;
  280. }
  281. public function getStatusTextAttr($value, $data)
  282. {
  283. switch ($data['status']) {
  284. case 0:
  285. $status_name = '审核中';
  286. break;
  287. case 1:
  288. $status_name = '处理中';
  289. break;
  290. case 2:
  291. $status_name = '已处理';
  292. break;
  293. case -1:
  294. $status_name = '已拒绝';
  295. break;
  296. default:
  297. $status_name = '';
  298. }
  299. return $status_name;
  300. }
  301. public static function getWithdrawConfig()
  302. {
  303. $config = \addons\shopro\model\Config::where('name', 'withdraw')->find();
  304. return json_decode($config['value'], true);
  305. }
  306. /**
  307. * 获取日志字段数组
  308. */
  309. public function getLogAttr($value, $data)
  310. {
  311. $value = array_filter((array)json_decode($value, true));
  312. return (array)$value;
  313. }
  314. /**
  315. * 设置日志字段
  316. * @param mixed $value
  317. * @return string
  318. */
  319. public function setLogAttr($value)
  320. {
  321. $value = is_object($value) || is_array($value) ? json_encode($value, JSON_UNESCAPED_UNICODE) : $value;
  322. return $value;
  323. }
  324. }