BalanceLogService.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | LARAVEL8.0 框架 [ LARAVEL ][ RXThinkCMF ]
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2017~2021 LARAVEL研发中心
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://www.laravel.cn
  8. // +----------------------------------------------------------------------
  9. // | Author: laravel开发员 <laravel.qq.com>
  10. // +----------------------------------------------------------------------
  11. namespace App\Services\Api;
  12. use App\Models\AccountLogModel;
  13. use App\Models\ActionLogModel;
  14. use App\Models\BalanceLogModel;
  15. use App\Models\MemberBankModel;
  16. use App\Models\MemberModel;
  17. use App\Services\BaseService;
  18. use App\Services\ConfigService;
  19. use App\Services\RedisService;
  20. use Illuminate\Support\Facades\DB;
  21. /**
  22. * 余额管理-服务类
  23. * @author laravel开发员
  24. * @since 2020/11/11
  25. */
  26. class BalanceLogService extends BaseService
  27. {
  28. public static $instance = null;
  29. /**
  30. * 构造函数
  31. * @author laravel开发员
  32. * @since 2020/11/11
  33. * AccountService constructor.
  34. */
  35. public function __construct()
  36. {
  37. $this->model = new BalanceLogModel();
  38. }
  39. /**
  40. * 静态入口
  41. * @return static|null
  42. */
  43. public static function make()
  44. {
  45. if (!self::$instance) {
  46. self::$instance = (new static());
  47. }
  48. return self::$instance;
  49. }
  50. /**
  51. * @param $params
  52. * @param int $pageSize
  53. * @return array
  54. */
  55. public function getDataList($params, $pageSize = 15)
  56. {
  57. $query = $this->getQuery($params);
  58. $list = $query->select(['a.*'])
  59. ->orderBy('a.status', 'asc')
  60. ->orderBy('a.create_time', 'desc')
  61. ->orderBy('a.id', 'desc')
  62. ->paginate($pageSize > 0 ? $pageSize : 9999999);
  63. $list = $list ? $list->toArray() : [];
  64. if ($list) {
  65. foreach ($list['data'] as &$item) {
  66. $item['create_time'] = $item['create_time'] ? datetime($item['create_time'], 'Y-m-d H:i:s') : '';
  67. $item['time_text'] = $item['create_time'] ? datetime($item['create_time'], 'Y年m月d日') : '';
  68. }
  69. }
  70. return [
  71. 'pageSize' => $pageSize,
  72. 'total' => isset($list['total']) ? $list['total'] : 0,
  73. 'list' => isset($list['data']) ? $list['data'] : []
  74. ];
  75. }
  76. public function getQuery($params)
  77. {
  78. $where = ['a.mark' => 1];
  79. $status = isset($params['status']) ? $params['status'] : 0;
  80. $type = isset($params['type']) ? $params['type'] : 0;
  81. if ($status > 0) {
  82. $where['a.status'] = $status;
  83. }
  84. if ($type > 0) {
  85. $where['a.type'] = $type;
  86. }
  87. return $this->model->with(['member'])->from("balance_logs as a")
  88. ->leftJoin('member as b', 'b.id', '=', 'a.user_id')
  89. ->where($where)
  90. ->where(function ($query) use ($params) {
  91. $keyword = isset($params['keyword']) ? $params['keyword'] : '';
  92. $userId = isset($params['user_id']) ? $params['user_id'] : 0;
  93. if ($userId) {
  94. $query->where('a.user_id', $userId);
  95. }
  96. if ($keyword) {
  97. $query->where(function ($query) use ($keyword) {
  98. $query->where('b.nickname', 'like', "%{$keyword}%")
  99. ->orWhere('b.mobile', 'like', "%{$keyword}%")
  100. ->orWhere('b.realname', 'like', "%{$keyword}%");
  101. });
  102. }
  103. $orderNo = isset($params['order_no']) ? trim($params['order_no']) : '';
  104. if ($orderNo) {
  105. $query->where(function ($query) use ($orderNo) {
  106. $query->where('a.order_no', 'like', "%{$orderNo}%");
  107. });
  108. }
  109. $account = isset($params['account']) ? trim($params['account']) : '';
  110. if ($account) {
  111. $query->where(function ($query) use ($account) {
  112. $query->where('a.account', 'like', "%{$account}%");
  113. });
  114. }
  115. })
  116. ->where(function ($query) use ($params) {
  117. // 日期
  118. $date = isset($params['date']) ? $params['date'] : [];
  119. $start = isset($date[0]) ? $date[0] : '';
  120. $end = isset($date[1]) ? $date[1] : '';
  121. $end = $start >= $end ? '' : $end;
  122. if ($start) {
  123. $query->where('a.create_time', '>=', strtotime($start));
  124. }
  125. if ($end) {
  126. $query->where('a.create_time', '<=', strtotime($end));
  127. }
  128. });
  129. }
  130. /**
  131. * 收入提现
  132. * @param $userId
  133. * @param $params
  134. * @return array|false
  135. */
  136. public function withdraw($userId, $params)
  137. {
  138. // 参数验证
  139. $payType = isset($params['pay_type']) && $params['pay_type'] ? intval($params['pay_type']) : 10;
  140. $accountType = isset($params['type']) && $params['type'] ? intval($params['type']) : 1;
  141. $money = isset($params['money']) ? floatval($params['money']) : 0;
  142. $accountId = isset($params['account_id']) ? intval($params['account_id']) : 0;
  143. if ($money <= 0) {
  144. $this->error = '请输入提现金额';
  145. return false;
  146. }
  147. if ($payType != 10 && $accountId <= 0) {
  148. $this->error = '请选择收款账户';
  149. return false;
  150. }
  151. $openWithdraw = ConfigService::make()->getConfigByCode("withdraw_{$accountType}_open", 1);
  152. if (!$openWithdraw) {
  153. $this->error = 2304;
  154. return false;
  155. }
  156. $withdrawMin = ConfigService::make()->getConfigByCode("withdraw_min", 0.1);
  157. if ($withdrawMin > 0 && $money < $withdrawMin) {
  158. $this->error = lang(2305, ['money' => $withdrawMin]);
  159. return false;
  160. }
  161. // 锁
  162. $cacheLockKey = "caches:members:withdraw:{$userId}";
  163. if (RedisService::get($cacheLockKey)) {
  164. $this->error = 1034;
  165. return false;
  166. }
  167. if($accountId){
  168. $accountInfo = MemberBankModel::where(['id' => $accountId, 'user_id' => $userId, 'mark' => 1])->first();
  169. $realname = isset($accountInfo['realname']) ? $accountInfo['realname'] : '';
  170. $account = isset($accountInfo['account']) ? $accountInfo['account'] : '';
  171. $accountName = isset($accountInfo['account_name']) ? $accountInfo['account_name'] : '';
  172. $accountRemark = isset($accountInfo['account_remark']) ? $accountInfo['account_remark'] : '';
  173. if (empty($accountInfo) || empty($realname) || empty($accountName) || empty($account)) {
  174. $this->error = '抱歉,当前收款账户错误请更换后重试';
  175. return false;
  176. }
  177. }else{
  178. $accountName = '微信支付';
  179. $account = '微信零钱';
  180. $accountRemark = '';
  181. }
  182. // 判断用户账号状态
  183. $fields = [1=>'balance',2=>'property',4=>'ls_score'];
  184. $field = isset($fields[$accountType])? $fields[$accountType]:'balance';
  185. RedisService::set($cacheLockKey, ['user_id' => $userId, 'params' => $params], rand(10, 20));
  186. $userInfo = MemberModel::where(['id' => $userId, 'mark' => 1])->select(['id', 'balance','property','bd_score','ls_score', 'status'])->first();
  187. $realname = isset($userInfo['realname']) ? $userInfo['realname'] : '';
  188. $status = isset($userInfo['status']) ? $userInfo['status'] : 0;
  189. $balance = isset($userInfo[$field]) ? $userInfo[$field] : 0;
  190. if (empty($userInfo) || $status != 1) {
  191. $this->error = 2016;
  192. RedisService::clear($cacheLockKey);
  193. return false;
  194. }
  195. if ($money > $balance) {
  196. $this->error = '该账户可提现余额不足';
  197. RedisService::clear($cacheLockKey);
  198. return false;
  199. }
  200. // 计算实际到账金额
  201. $poolMoney = 0;
  202. $ptMoney = 0;
  203. $ptRate = 0;
  204. $poolRate = 0;
  205. $total = $money;
  206. $actualMoney = $money;
  207. if($accountType==2){
  208. // 提现金额
  209. $price = PriceService::make()->getTodayPrice(1);
  210. if($price<=0){
  211. $this->error = '请等候今日资产价格刷新后重试~';
  212. RedisService::clear($cacheLockKey);
  213. return false;
  214. }
  215. $rate = ConfigService::make()->getConfigByCode('withdraw_2_rate',0);
  216. $rate = $rate>0 && $rate<=100? $rate : 100;
  217. $total = moneyFormat($total * $price,2);
  218. $actualMoney = moneyFormat($total * $rate/100, 2);
  219. // 结算比例
  220. $rate = ConfigService::make()->getConfigByCode('withdraw_2_rate',0);
  221. $rate = $rate>0 && $rate<=100? $rate : 100;
  222. $actualMoney = moneyFormat($actualMoney * $rate/100, 2);
  223. // 底池比例
  224. $poolRate = ConfigService::make()->getConfigByCode('withdraw_2_back_rate',0);
  225. $poolRate = $poolRate>0 && $poolRate<50? $poolRate : 0;
  226. $poolMoney = round($total * $poolRate/100, 2);
  227. // 运营账户比例
  228. $ptRate = ConfigService::make()->getConfigByCode('withdraw_2_pt_rate',0);
  229. $ptRate = $ptRate>0 && $ptRate<=50? $ptRate : 0;
  230. $ptMoney = round($total * $ptRate/100, 2);
  231. }
  232. // 手续费
  233. $feeRate = ConfigService::make()->getConfigByCode("withdraw_{$accountType}_fee",0);
  234. $feeRate = $feeRate>0 && $feeRate<=50? $feeRate : 0;
  235. $fee = round($actualMoney * $feeRate/100, 2);
  236. $actualMoney = moneyFormat($actualMoney - $fee, 2);
  237. $accountTypeName = ['账户','收益余额','数字资产','报单积分','绿色积分'][$accountType];
  238. $accountTypeName = $accountTypeName?$accountTypeName:'账户';
  239. // 提现处理
  240. DB::beginTransaction();
  241. $updateData = ["{$field}" => DB::raw("{$field} - {$money}"), 'update_time' => time()];
  242. // 会员账户
  243. $userType = 1;
  244. if (!MemberModel::where(['id' => $userId])->update($updateData)) {
  245. DB::rollBack();
  246. $this->error = '提现处理失败';
  247. RedisService::clear($cacheLockKey);
  248. return false;
  249. }
  250. $orderNo = get_order_num('JW');
  251. $order = [
  252. 'user_id' => $userId,
  253. 'order_no' => $orderNo,
  254. 'money' => $money,
  255. 'total' => $total,
  256. 'after_money' => moneyFormat(max(0, $balance - $money), 2),
  257. 'actual_money' => $actualMoney,
  258. 'fee' => $fee,
  259. 'pool_money' => $poolMoney,
  260. 'pool_rate' => $poolRate,
  261. 'pt_money' => $ptMoney,
  262. 'pt_rate' => $ptRate,
  263. 'user_type' => $userType,
  264. 'type' => 2,
  265. 'account_type' => $accountType,
  266. 'pay_type' => $payType,
  267. 'realname' => $realname,
  268. 'account_name' => $accountName,
  269. 'account' => $account,
  270. 'account_remark' => $accountRemark,
  271. 'date' => date('Y-m-d'),
  272. 'create_time' => time(),
  273. 'status' => 1,
  274. 'mark' => 1
  275. ];
  276. if (!$orderId = $this->model::insertGetId($order)) {
  277. DB::rollBack();
  278. $this->error = '提现处理失败';
  279. RedisService::clear($cacheLockKey);
  280. return false;
  281. }
  282. $log = [
  283. 'user_id' => $userId,
  284. 'source_order_no' => $orderNo,
  285. 'user_type' => $userType,
  286. 'account_type' => $accountType,
  287. 'type' => 4,
  288. 'money' => -$money,
  289. 'after_money' => moneyFormat(max(0, $balance - $money), 2),
  290. 'date' => date('Y-m-d'),
  291. 'create_time' => time(),
  292. 'remark' => "提现到{$account}",
  293. 'status' => 1,
  294. 'mark' => 1,
  295. ];
  296. if (!$accountId = AccountLogModel::insertGetId($log)) {
  297. DB::rollBack();
  298. $this->error = '提现处理失败';
  299. RedisService::clear($cacheLockKey);
  300. return false;
  301. }
  302. DB::commit();
  303. // 操作日志
  304. ActionLogModel::setRecord($userId, ['type' => 2, 'title' => "{$accountTypeName}提现", 'content' => "姓名:{$realname},账号:{$accountName}/{$account}/{$accountRemark},提现{$money}元,单号:{$orderNo}", 'module' => 'balanceLog']);
  305. ActionLogModel::record();
  306. RedisService::clear($cacheLockKey);
  307. $this->error = '提现申请成功,请耐心等候审核~';
  308. return ['id' => $orderId, 'aid' => $accountId, 'money' => $money];
  309. }
  310. }