You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

406 lines
18 KiB

6 years ago
6 years ago
  1. <?php
  2. namespace App\Service;
  3. use App\Commons\Log;
  4. use App\Constants\LogLabel;
  5. use App\Model\FinancialRecord;
  6. use App\Model\Market;
  7. use App\Model\MmInfo;
  8. use App\Model\MpInfo;
  9. use App\Model\Order;
  10. use App\Model\OrderMain;
  11. use App\Model\ServiceReward;
  12. use App\Model\Store;
  13. use App\Model\StoreAccount;
  14. use App\Model\SystemConfig;
  15. use App\Model\UserBalance;
  16. use App\Model\UserRelationBind;
  17. use App\Model\Users;
  18. use App\TaskWorker\SSDBTask;
  19. use Hyperf\DbConnection\Db;
  20. use Hyperf\Di\Annotation\Inject;
  21. use Hyperf\Utils\ApplicationContext;
  22. class SeparateAccountsService implements SeparateAccountsServiceInterface
  23. {
  24. /**
  25. * @Inject
  26. * @var Log
  27. */
  28. protected $log;
  29. /**
  30. * @Inject
  31. * @var UserServiceInterface
  32. */
  33. protected $userService;
  34. /**
  35. * @Inject
  36. * @var FinancialRecordServiceInterface
  37. */
  38. protected $financialRecordService;
  39. /**
  40. * @Inject
  41. * @var MiniprogramServiceInterface
  42. */
  43. protected $miniprogramService;
  44. /**
  45. * @inheritDoc
  46. */
  47. public function orderOnlinePaid($global_order_id)
  48. {
  49. // 线上订单支付完成
  50. // 订单
  51. $orderMain = OrderMain::query()
  52. ->where(['global_order_id' => $global_order_id])
  53. ->first();
  54. if (empty($orderMain)) {
  55. return false;
  56. }
  57. // =======用户支付流水 / Start=======
  58. $this->financialRecordService->userByOLOrderPaid($orderMain->user_id, $global_order_id, $orderMain->money);
  59. // =======用户支付流水 / End=======
  60. }
  61. /**
  62. * @inheritDoc
  63. */
  64. public function orderOnlineCompleted($global_order_id)
  65. {
  66. $ssdb = ApplicationContext::getContainer()->get(SSDBTask::class);
  67. // 线上订单完成(用户点击确认收货完成/管理后台点击完成/配送员点击完成/自动收货等),进行相关分账
  68. // 订单
  69. $orderMain = OrderMain::query()
  70. ->where(['global_order_id' => $global_order_id])
  71. ->whereIn('state', [OrderMain::ORDER_STATE_COMPLETE,OrderMain::ORDER_STATE_EVALUATED,OrderMain::ORDER_STATE_UNREFUND])
  72. ->first();
  73. if (empty($orderMain)) {
  74. return false;
  75. }
  76. $currentTime = time();
  77. Db::beginTransaction();
  78. try {
  79. // =======商户订单收入流水 / Start=======
  80. // 查询子订单
  81. $orders = Order::query()->select(['id', 'money', 'user_id', 'store_id', 'pay_time'])
  82. ->where(['order_main_id' => $orderMain->id])
  83. ->get()->toArray();
  84. foreach ($orders as $key => &$order) {
  85. // 商户
  86. $store = Store::find($order['store_id']);
  87. // 旧商户流水基础数据 TODO 直接移除或后续考虑移除
  88. $storeAccountBase = [
  89. 'user_id' => $order['user_id'],
  90. 'order_id' => $order['id'],
  91. 'store_id' => $order['store_id'],
  92. 'type' => 1,
  93. 'time' => date('Y-m-d H:i:s', $currentTime),
  94. 'add_time' => $currentTime,
  95. ];
  96. // 旧商户流水 TODO 直接移除或后续考虑移除
  97. $storeAccount = [
  98. 'money' => $order['money'],
  99. 'note' => '线上订单',
  100. 'category' => 1,
  101. ];
  102. StoreAccount::query()->insert(array_merge($storeAccountBase, $storeAccount));
  103. // 新商户流水
  104. $this->financialRecordService->storeByOLOrderComp($store->user_id, $global_order_id ,$order['money']);
  105. }
  106. // =======商户订单收入流水 / End=======
  107. // =======社区服务点分账 / Start=======
  108. // 前提:用户线上下单并且订单完成
  109. // 奖励规则A:用户是平台新用户,奖励社区服务点平台新用户奖励x元+平台新用户首单奖励y元+订单商品金额z%的分成
  110. // 奖励规则B:用户是非新用户,奖励社区服务点订单实际支付金额z%的分成
  111. // =======社区服务点分账 / Start=======
  112. // 当前用户的社区服务点绑定关系
  113. $communityBind = UserRelationBind::query()
  114. ->where(['bind_type' => UserRelationBind::BIND_TYPE_COMMUNITY, 'user_id' => $orderMain->user_id])
  115. ->first();
  116. if ($communityBind) {
  117. // 奖励/分账金额
  118. $award = ServiceReward::query()->where(['type' => ServiceReward::TYPE_COMMUNITY])->first();
  119. if (empty($award)) {
  120. Db::rollBack();
  121. return false;
  122. }
  123. $award = $award->set_reward;
  124. // 平台新用户
  125. if ($this->userService->isPlatformNewUser($orderMain->user_id, $orderMain->id)) {
  126. $this->financialRecordService->communityAwardByPlatNewUser($communityBind->source_id, $global_order_id, $award['new_user_reward']);
  127. $this->financialRecordService->communityAwardByPlatNewUserFirstOLOrder($communityBind->source_id, $global_order_id, $award['first_reward']);
  128. }
  129. // 账单分成
  130. if ($orderMain->type == OrderMain::ORDER_TYPE_ONLINE) {
  131. $money = bcmul($orderMain->money, bcdiv($award['flow_reward'], 100, 6), 2);
  132. $this->financialRecordService->communitySeparateAccountsByOrderComp($communityBind->source_id, $global_order_id, $money);
  133. }
  134. }
  135. // =======社区服务点分账 / End=======
  136. // =======服务商、市场经理奖励分账 / Start=======
  137. // 前提A:新用户下单并且订单完成(线上、线下都行)
  138. // 奖励规则A:用户是平台新用户,奖励市场经理 1 元,服务商 0.5 元,如果是线上订单,服务商有6%的订单分成
  139. // 前提B:新商户旗下产生 10 个新用户
  140. // 奖励规则B:奖励市场经理 25 元,服务商 10 元(仅仅奖励一次)
  141. // 前提C:用户线上下单并且订单完成
  142. // 奖励规则C:奖励服务商账单 6% 分成
  143. // 判断是新商户:入驻绑定的时候把关系存在SSDB
  144. // =======服务商、市场经理奖励分账 / Start=======
  145. $MmMpAwardConfig = [
  146. 'mm_new_user' => 1,
  147. 'mm_new_store' => 25,
  148. 'mp_new_user' => 0.5,
  149. 'mp_new_store' => 10,
  150. 'separate_rate' => 6,
  151. 'limit_new_user_number' => 10,
  152. ];
  153. foreach ($orders as $key => &$order) {
  154. // 当前订单(子)对应商户是否有市场经理绑定关系
  155. $store = Store::query()->find($order['store_id']); // 商户
  156. $mmInfo = MmInfo::query()->where(['user_id' => $store->mm_user_id])->first(); // 市场经理
  157. $market = Market::query()->find($mmInfo->market_id); // 市场
  158. $mpInfo = MpInfo::query()->find($market->mp_id); // 服务商
  159. $ssdbName = 'mm_'.$mmInfo->id.'_award_'.$store->id;
  160. $record = $ssdb->exec('hgetall', $ssdbName);
  161. // 平台新用户
  162. if ($this->userService->isPlatformNewUser($orderMain->user_id, $orderMain->id)) {
  163. $ssdb->exec('hincr', $ssdbName, 'new_user_number', 1);
  164. $this->financialRecordService->mmAwardByPlatNewUser($mmInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mm_new_user'], '发展新用户'); // 市场经理新用户奖励
  165. $this->financialRecordService->mpAwardByPlatNewUser($mpInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mp_new_user'], '市场经理「'.$mmInfo->name.'」发展新用户'); // 服务商新用户奖励
  166. }
  167. // 判断是否已经奖励过新拓展商户的奖励,没有的话判断新用户个数是否达到要求,进行奖励
  168. if (
  169. !empty($record)
  170. &&$record['is_awarded']==0
  171. &&$record['new_user_number']>=$MmMpAwardConfig['limit_new_user_number']
  172. ) { // 存在记录且未发放奖励,同时新用户数已经超过10
  173. $this->financialRecordService->mmAwardByNewStore($mmInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mm_new_store'], '发展新商户【'.$store->name.'】'); // 市场经理新商户奖励
  174. $this->financialRecordService->mpAwardByNewStore($mpInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mp_new_store'], '市场经理「'.$mmInfo->name.'」发展新新商户【'.$store->name.'】'); // 服务商新商户奖励
  175. }
  176. }
  177. // 线上订单服务商分账
  178. if ($orderMain->type == OrderMain::ORDER_TYPE_ONLINE) {
  179. $money = bcmul($orderMain->money, bcdiv($MmMpAwardConfig['separate_rate'], 100, 6), 2);
  180. $this->financialRecordService->mpSeparateAccountByOLOrderComp($mpInfo->admin_user_id, $global_order_id, $money);
  181. }
  182. // =======服务商、市场经理奖励分账 / End=======
  183. Db::commit();
  184. return true;
  185. } catch (\Exception $e) {
  186. $this->log->event(LogLabel::SEPARATE_ACCOUNTS_LOG, ['exception' => $e->getMessage(), 'order_main' => json_encode($orderMain)]);
  187. Db::rollBack();
  188. return false;
  189. }
  190. }
  191. /**
  192. * @inheritDoc
  193. */
  194. public function orderOfflinePaid($global_order_id)
  195. {
  196. // 线下订单支付完成
  197. // 订单
  198. $orderMain = OrderMain::query()
  199. ->where(['global_order_id' => $global_order_id])
  200. ->first();
  201. if (empty($orderMain)) {
  202. return false;
  203. }
  204. // 查询子订单,当面付目前实际上只有一个子订单
  205. $order = Order::query()->select(['id', 'money', 'user_id', 'store_id', 'pay_time'])
  206. ->where(['order_main_id' => $orderMain->id])
  207. ->first();
  208. if (empty($order)) {
  209. return false;
  210. }
  211. $currentTime = time();
  212. Db::beginTransaction();
  213. try {
  214. // =======用户支付流水 / Start=======
  215. $this->financialRecordService->userByOFLOrderPaid($orderMain->user_id, $global_order_id, $orderMain->money);
  216. // =======用户支付流水 / End=======
  217. // =======线下订单支付完成商户分账 / Start=======
  218. // 前提:用户线上下单并且支付完成
  219. // 奖励规则A:用户是平台新用户,奖励商户2元
  220. // 奖励规则B:用户是非新用户,但是是商户当日首单,奖励商户0.05元
  221. // =======线下订单支付完成商户分账 / Start=======
  222. // 旧商户订单流水基础数据 TODO 直接移除或后续考虑移除
  223. $storeAccountBase = [
  224. 'user_id' => $order->user_id,
  225. 'order_id' => $order->id,
  226. 'store_id' => $order->store_id,
  227. 'type' => 1,
  228. 'time' => date('Y-m-d H:i:s', $currentTime),
  229. 'add_time' => $currentTime,
  230. ];
  231. // 旧商户订单流水 TODO 直接移除或后续考虑移除
  232. $storeAccount = [
  233. 'money' => $order->money,
  234. 'note' => '当面付订单收入',
  235. 'category' => 2,
  236. ];
  237. StoreAccount::query()->insert(array_merge($storeAccountBase, $storeAccount));
  238. // 商户
  239. $store = Store::find($order->store_id);
  240. // 新商户订单流水
  241. $this->financialRecordService->storeByOFLOrderComp($store->user_id, $global_order_id, $order->money);
  242. $needAward = false;
  243. $awardAmount = 0;
  244. // 新用户商户奖励
  245. if ($this->userService->isPlatformNewUser($orderMain->user_id, $orderMain->id)) {
  246. $awardAmount = SystemConfig::query()->where(['type' => 1, 'menu_name' => 'award_new_user'])->value('value');
  247. // 旧商户流水 TODO 直接移除或后续考虑移除
  248. $storeAccount = [
  249. 'money' => $awardAmount,
  250. 'note' => '新用户下单成功,平台奖励',
  251. 'category' => 3,
  252. ];
  253. // 新商户流水
  254. $this->financialRecordService->storeAwardByPlatNewUserOFLOrder($store->user_id, $global_order_id, $awardAmount);
  255. $needAward = true;
  256. }
  257. else {
  258. // 商户当日首单奖励
  259. if (
  260. $this->userService->isStoreFirstOrderToday(
  261. $order->user_id,
  262. $order->store_id,
  263. $order->id,
  264. FinancialRecord::OFL_FIRST_AWARD_LIMIT_AMOUNT
  265. )
  266. && $order->money >= FinancialRecord::OFL_FIRST_AWARD_LIMIT_AMOUNT
  267. ) {
  268. $awardAmount = SystemConfig::query()->where(['type' => 1, 'menu_name' => 'award_each_order'])->value('value');
  269. // 旧商户流水 TODO 直接移除或后续考虑移除
  270. $storeAccount = [
  271. 'money' => $awardAmount,
  272. 'note' => '用户下单成功,平台奖励',
  273. 'category' => 4,
  274. ];
  275. // 新商户流水
  276. $this->financialRecordService->storeAwardByTodayFirstOFLOrder($store->user_id, $global_order_id, $awardAmount);
  277. $needAward = true;
  278. }
  279. }
  280. if ($needAward && $awardAmount) {
  281. // 旧商户流水 TODO 直接移除或后续考虑移除
  282. StoreAccount::query()->insert(array_merge($storeAccountBase, $storeAccount));
  283. // 发模板消息
  284. $openid = Users::query()->where(['id' => $store['user_id']])->value('openid');
  285. $res = $this->miniprogramService->sendTemMsgForAward($storeAccount['money'], $storeAccount['note'], $openid, $storeAccountBase['time']);
  286. }
  287. // =======线下订单支付完成商户分账 / End=======
  288. // =======服务商、市场经理奖励分账 / Start=======
  289. // 前提A:新用户下单并且订单完成(线上、线下都行)
  290. // 奖励规则A:用户是平台新用户,奖励市场经理 1 元,服务商 0.5 元,如果是线上订单,服务商有6%的订单分成
  291. // 前提B:新商户旗下产生 10 个新用户
  292. // 奖励规则B:奖励市场经理 25 元,服务商 10 元(仅仅奖励一次)
  293. // 前提C:用户线上下单并且订单完成
  294. // 奖励规则C:奖励服务商账单 6% 分成
  295. // 判断是新商户:入驻绑定的时候把关系存在SSDB
  296. // =======服务商、市场经理奖励分账 / Start=======
  297. $MmMpAwardConfig = [
  298. 'mm_new_user' => 1,
  299. 'mm_new_store' => 25,
  300. 'mp_new_user' => 0.5,
  301. 'mp_new_store' => 10,
  302. 'separate_rate' => 6,
  303. 'limit_new_user_number' => 10,
  304. ];
  305. // 当前订单(子)对应商户是否有市场经理绑定关系
  306. $store = Store::query()->find($order['store_id']); // 商户
  307. $mmInfo = MmInfo::query()->where(['user_id' => $store->mm_user_id])->first(); // 市场经理
  308. $market = Market::query()->find($mmInfo->market_id); // 市场
  309. $mpInfo = MpInfo::query()->find($market->mp_id); // 服务商
  310. $ssdb = ApplicationContext::getContainer()->get(SSDBTask::class);
  311. $ssdbName = 'mm_'.$mmInfo->id.'_award_'.$store->id;
  312. $record = $ssdb->exec('hgetall', $ssdbName);
  313. // 平台新用户
  314. if ($this->userService->isPlatformNewUser($orderMain->user_id, $orderMain->id)) {
  315. $ssdb->exec('hincr', $ssdbName, 'new_user_number', 1);
  316. $this->financialRecordService->mmAwardByPlatNewUser($mmInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mm_new_user'], '发展新用户'); // 市场经理新用户奖励
  317. $this->financialRecordService->mpAwardByPlatNewUser($mpInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mp_new_user'], '市场经理「'.$mmInfo->name.'」发展新用户'); // 服务商新用户奖励
  318. }
  319. // 判断是否已经奖励过新拓展商户的奖励,没有的话判断新用户个数是否达到要求,进行奖励
  320. if (
  321. !empty($record)
  322. &&$record['is_awarded']==0
  323. &&$record['new_user_number']>=$MmMpAwardConfig['limit_new_user_number']
  324. ) { // 存在记录且未发放奖励,同时新用户数已经超过10
  325. $this->financialRecordService->mmAwardByNewStore($mmInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mm_new_store'], '发展新商户【'.$store->name.'】'); // 市场经理新商户奖励
  326. $this->financialRecordService->mpAwardByNewStore($mpInfo->admin_user_id, $global_order_id, $MmMpAwardConfig['mp_new_store'], '市场经理「'.$mmInfo->name.'」发展新新商户【'.$store->name.'】'); // 服务商新商户奖励
  327. }
  328. // =======服务商、市场经理奖励分账 / End=======
  329. Db::commit();
  330. return true;
  331. } catch (\Exception $e) {
  332. $this->log->event(LogLabel::SEPARATE_ACCOUNTS_LOG, ['exception' => $e->getMessage(), 'order_main' => json_encode($orderMain)]);
  333. Db::rollBack();
  334. return false;
  335. }
  336. }
  337. }