Recruitment.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. <?php namespace App\Http\Controllers\Api\Lottery;
  2. use App\Http\Controllers\Api\Api;
  3. use App\Models\Lottery\Recruitment as Model;
  4. use App\Models\Custom;
  5. use App\Models\CustomCoupon;
  6. use App\Models\CustomScore;
  7. use App\Models\CustomAmount;
  8. use App\Models\Lottery\RecruitmentRecord;
  9. use App\Models\Lottery\RecruitmentReward as RecruitmentReward;
  10. use Illuminate\Support\Facades\DB;
  11. use App\Models\WeiBan\Tags as WeiBanTags;
  12. /**
  13. * 拉新抽奖
  14. *
  15. * @author 刘相欣
  16. *
  17. * */
  18. class Recruitment extends Api{
  19. /**
  20. * 获取抽奖配置 /api/lottery_recruitment/get_detail
  21. *
  22. *
  23. * */
  24. public function get_detail(Model $Model,Custom $Custom,RecruitmentReward $RecruitmentReward,RecruitmentRecord $RecruitmentRecord,WeiBanTags $WeiBanTags){
  25. // 接口验签
  26. // $this->verify_sign();
  27. // 检查登录
  28. $uid = $this->checkLogin();
  29. // 接收参数
  30. $id = request('id',0);
  31. // 获取活动
  32. $data = $Model->getOne($id);
  33. // 如果存在的话
  34. if( !$data ) return json_send(['code'=>'error','msg'=>'暂无活动','data'=>$data]);
  35. // 获取客户信息
  36. $custom = $Custom->getOne($uid);
  37. // 如果存在的话
  38. if( !$custom ) return json_send(['code'=>'no_login','msg'=>'请登录','data'=>['error'=>'无对应客户']]);
  39. // 如果客户注册时间不在活动期间
  40. if( $data['start_time'] > $custom['insert_time'] || $data['end_time'] < $custom['insert_time'] ) return json_send(['code'=>'error','msg'=>'账号不在活动范围','data'=>['error'=>'不在活动时间内注册']]);
  41. // 如果城市限制并且不在在城市范围内,不允许参加
  42. if ( $data['city_ids'] && !in_array($custom['city_id'],explode(',',$data['city_ids'])) ) return json_send(['code'=>'error','msg'=>'账号不在城市范围,请核对您的城市','data'=>['error'=>'不在标签范围内']]);
  43. // 是否需要获取用户标签
  44. $tags = $data['tag_scope'] || $data['tag_except'] ? $WeiBanTags->getListByWeibanExtid($custom['weiban_extid']) : [];
  45. // 获取用户的标签
  46. $tags = $tags ? array_column($tags,'name') : [];
  47. // 判断是不是可以参与
  48. if( $data['tag_scope'] ) {
  49. // 解析数组
  50. $data['tag_scope'] = explode(',',$data['tag_scope']);
  51. // 计算交集
  52. $intersect = count(array_intersect($data['tag_scope'],$tags));
  53. // 判断标签是否必须同时满足
  54. if( $data['tag_scope_type'] == 1 ) {
  55. // 判断用户的标签是否全部满足于标签限制范围内
  56. if( $intersect != count($data['tag_scope']) ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  57. }else{
  58. // 判断标签限制并且不在标签限制范围内,不允许参加
  59. if( !$intersect ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  60. }
  61. }
  62. // 判断是不是可以参与
  63. if( $data['tag_except'] ) {
  64. // 解析数组
  65. $data['tag_except'] = explode(',',$data['tag_except']);
  66. // 计算交集
  67. $intersect = count(array_intersect($data['tag_except'],$tags));
  68. // 如果存在交集,在排除范围,即不可参与
  69. if( $intersect ) return json_send(['code'=>'error','msg'=>'账号不在范围','data'=>['error'=>'在标签排除范围内']]);
  70. }
  71. // 奖品
  72. $reward = $RecruitmentReward->getListByLottery($data['id']);
  73. // 活动暂无奖品
  74. if( !$reward ) return json_send(['code'=>'error','msg'=>'活动暂未配置奖品','data'=>$data]);
  75. // 查询条件
  76. $map = [];
  77. // 判断周期
  78. if( !empty($data['freq']) ) {
  79. if( $data['freq'] == 1 ) $map = [['insert_time','>=',now()->startOfDay()->getTimestamp()],['insert_time','<=',now()->endOfDay()->getTimestamp()]];
  80. if( $data['freq'] == 2 ) $map = [['insert_time','>=',now()->startOfWeek()->getTimestamp()],['insert_time','<=',now()->endOfWeek()->getTimestamp()]];
  81. if( $data['freq'] == 3 ) $map = [['insert_time','>=',now()->startOfMonth()->getTimestamp()],['insert_time','<=',now()->endOfMonth()->getTimestamp()]];
  82. }
  83. // logo
  84. $data['logo'] = $data['logo'] ? path_compat($data['logo']) : '';
  85. // 查询用户已参与次数
  86. $data['join_num'] = (int)$RecruitmentRecord->query()->where([['custom_uid','=',$uid],['lottery_id','=',$id]])->where($map)->count();
  87. // 计算剩余次数
  88. $data['number'] = $data['lucky_num'] - $data['join_num'];
  89. // 最少为0,避免显示异常
  90. $data['number'] = $data['number'] < 0 ? 0 : $data['number'];
  91. // 时间处理
  92. $data['start_date'] = date('Y/m/d H:i',$data['start_time']);
  93. // 时间处理
  94. $data['end_date'] = date('Y/m/d H:i',$data['end_time']);
  95. // 通过活动ID,查询奖品
  96. $data['reward_list'] = [];
  97. // 奖品数据
  98. foreach ($reward as $value) {
  99. // 奖项
  100. $data['reward_list'][] = ['id'=>$value['id'],'name'=>$value['reward_name'],'img'=>$value['reward_thumb'],'reward_type'=>$value['reward_type']];
  101. }
  102. // 默认可以参加活动
  103. $data['allow_join'] = 1;
  104. // 返回结果
  105. return json_send(['code'=>'success','msg'=>'获取成功','data'=>$data]);
  106. }
  107. /**
  108. * 抽奖 /api/lottery_recruitment/get_reward
  109. *
  110. * */
  111. public function get_reward(Model $Model,Custom $Custom,RecruitmentRecord $RecruitmentRecord,RecruitmentReward $RecruitmentReward,CustomCoupon $CustomCoupon,CustomScore $CustomScore,CustomAmount $CustomAmount,WeiBanTags $WeiBanTags){
  112. // 接口验签
  113. // $this->verify_sign();
  114. // 检查登录
  115. $uid = $this->checkLogin();
  116. // 获取活动
  117. $lotteryId = request('lottery_id',0);
  118. // 如果存在的话
  119. if( !$lotteryId ) return json_send(['code'=>'error','msg'=>'请选择参与的活动','data'=>['error'=>"抽奖活动ID有误"]]);
  120. // 获取客户信息
  121. $custom = $Custom->getOne($uid);
  122. // 如果存在的话
  123. if( !$custom ) return json_send(['code'=>'no_login','msg'=>'请登录','data'=>['error'=>'无对应客户']]);
  124. // 通过活动ID,查询奖品
  125. $data = $Model->getOne($lotteryId);
  126. // 如果存在的话
  127. if( !$data ) return json_send(['code'=>'error','msg'=>'活动不存在或未开始','data'=>$data]);
  128. // 活动时间判断
  129. if( $data['start_time'] > time() ) return json_send(['code'=>'error','msg'=>'活动暂未开始','data'=>$data]);
  130. // 活动时间判断
  131. if( $data['end_time'] < time() ) return json_send(['code'=>'error','msg'=>'活动已结束','data'=>$data]);
  132. // 如果客户注册时间不在活动期间
  133. if( $data['start_time'] > $custom['insert_time'] || $data['end_time'] < $custom['insert_time'] ) return json_send(['code'=>'error','msg'=>'账号不在活动范围','data'=>['error'=>'不在活动时间内注册']]);
  134. // 如果城市限制并且不在在城市范围内,不允许参加
  135. if ( $data['city_ids'] && !in_array($custom['city_id'],explode(',',$data['city_ids'])) ) return json_send(['code'=>'error','msg'=>'账号不在城市范围,请核对您的城市','data'=>['error'=>'不在标签范围内']]);
  136. // 是否需要获取用户标签
  137. $tags = $data['tag_scope'] || $data['tag_except'] ? $WeiBanTags->getListByWeibanExtid($custom['weiban_extid']) : [];
  138. // 获取用户的标签
  139. $tags = $tags ? array_column($tags,'name') : [];
  140. // 判断是不是可以参与
  141. if( $data['tag_scope'] ) {
  142. // 解析数组
  143. $data['tag_scope'] = explode(',',$data['tag_scope']);
  144. // 计算交集
  145. $intersect = count(array_intersect($data['tag_scope'],$tags));
  146. // 判断标签是否必须同时满足
  147. if( $data['tag_scope_type'] == 1 ) {
  148. // 判断用户的标签是否全部满足于标签限制范围内
  149. if( $intersect != count($data['tag_scope']) ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  150. }else{
  151. // 判断标签限制并且不在标签限制范围内,不允许参加
  152. if( !$intersect ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  153. }
  154. }
  155. // 判断是不是可以参与
  156. if( $data['tag_except'] ) {
  157. // 解析数组
  158. $data['tag_except'] = explode(',',$data['tag_except']);
  159. // 计算交集
  160. $intersect = count(array_intersect($data['tag_except'],$tags));
  161. // 如果存在交集,在排除范围,即不可参与
  162. if( $intersect ) return json_send(['code'=>'error','msg'=>'账号在不可参与范围','data'=>['error'=>'账号在标签排除范围内']]);
  163. }
  164. // 奖品
  165. $reward = $RecruitmentReward->getListByLottery($data['id']);
  166. // 活动暂无奖品
  167. if( !$reward ) return json_send(['code'=>'error','msg'=>'活动暂未配置奖品','data'=>$data]);
  168. // 查询条件
  169. $map = [['insert_time','>=',$data['start_time']],['insert_time','<=',$data['end_time']]];
  170. // 判断周期
  171. if( !empty($data['freq']) ) {
  172. if( $data['freq'] == 1 ) $map = [['insert_time','>=',now()->startOfDay()->getTimestamp()],['insert_time','<=',now()->endOfDay()->getTimestamp()]];
  173. if( $data['freq'] == 2 ) $map = [['insert_time','>=',now()->startOfWeek()->getTimestamp()],['insert_time','<=',now()->endOfWeek()->getTimestamp()]];
  174. if( $data['freq'] == 3 ) $map = [['insert_time','>=',now()->startOfMonth()->getTimestamp()],['insert_time','<=',now()->endOfMonth()->getTimestamp()]];
  175. }
  176. // 查询用户已参与次数
  177. $data['join_num'] = (int)$RecruitmentRecord->query()->where([['custom_uid','=',$uid],['lottery_id','=',$lotteryId]])->where($map)->count();
  178. // 计算剩余次数
  179. $data['join_num'] = $data['lucky_num'] - $data['join_num'];
  180. // 如果次数不够
  181. if( $data['join_num'] <= 0 ) return json_send(['code'=>'error','msg'=>'抽奖次数已用完','data'=>['error'=>'抽奖次数已用完']]);
  182. // 限制中奖则获取中奖次数
  183. $rewarTotal = $data['max_reward'] ? $RecruitmentRecord->query()->where([['lottery_id','=',$data['id']],['custom_uid','=',$uid],['reward_id','>',0]])->where($map)->count() : 0;
  184. // 中奖上限以后不再中奖, 默认获取未中奖
  185. $rewardResult = ($data['max_reward'] && $rewarTotal >= $data['max_reward']) ? ['reward_list'=>$reward,'index'=>0] : $RecruitmentReward->getRewardResult($lotteryId);
  186. // 未抽中
  187. if( !$rewardResult ) return json_send(['code'=>'error','msg'=>'谢谢参与','data'=>['error'=>'谢谢参与']]);
  188. // 奖品索引
  189. $rewardIndex = $rewardResult['index'];
  190. // 奖品列表
  191. $reward = $rewardResult['reward_list'];
  192. // 获取奖品信息
  193. $rewardResult = $reward[$rewardIndex];
  194. // 是否记录结果
  195. $record = ['custom_uid'=>$uid,'lottery_id'=>$lotteryId,'reward_id'=>$rewardResult['id'],'reward_name'=>$rewardResult['reward_name'],'status'=>(in_array($rewardResult['reward_type'],[4,5,6]) ? 1 : 8)];
  196. // 实物状态需要填写地址
  197. if( $rewardResult['reward_type'] == 5 ) $record['status'] = 0;
  198. // 开启事务
  199. DB::beginTransaction();
  200. try{
  201. // 如果存在奖品
  202. if( $rewardResult['id'] ){
  203. // 奖品数量减少
  204. $result = $RecruitmentReward->edit($rewardResult['id'],['reward_total'=>DB::raw('reward_total+-1'),'lottery_id'=>$rewardResult['lottery_id']]);
  205. // 如果奖品数量减少失败
  206. if( !$result ) {
  207. // 回退数据
  208. DB::rollBack();
  209. // 提醒重试
  210. return json_send(['code'=>'error','msg'=>'抽奖失败,请重试','data'=>['error'=>'奖品数量扣减失败']]);
  211. }
  212. }
  213. // 如果是积分
  214. if( $rewardResult['reward_type'] == 1 ){
  215. // 积分大于0
  216. if( $rewardResult['reward_info'] > 0 ){
  217. // 积分发放
  218. $result = $CustomScore->trade($uid,$lotteryId,$rewardResult['reward_info'],7,3);
  219. // 发放失败,改为未中奖
  220. if( isset($result['error']) ) {
  221. // 回退数据
  222. DB::rollBack();
  223. // 通知重试
  224. return json_send(['code'=>'error','msg'=>'出了点小差,请重新抽奖','data'=>$result]);
  225. }
  226. }
  227. }
  228. // 优惠券,先进行发放
  229. if( $rewardResult['reward_type'] == 2 ){
  230. // 优惠券存在ID
  231. if( $rewardResult['reward_info'] > 0 ){
  232. // 积分给与
  233. $result = $CustomCoupon->giveCoupon($rewardResult['reward_info'],$uid);
  234. // 发放失败,改为未中奖
  235. if( isset($result['error']) ) {
  236. // 回退数据
  237. DB::rollBack();
  238. // 通知重试
  239. return json_send(['code'=>'error','msg'=>'出了点小差,请重新抽奖','data'=>$result]);
  240. }
  241. }
  242. }
  243. // 红包
  244. if( $rewardResult['reward_type'] == 3 ){
  245. // 积分大于0
  246. if( $rewardResult['reward_info'] > 0 ){
  247. // 积分发放
  248. $result = $CustomAmount->trade($uid,$lotteryId,$rewardResult['reward_info'],6,1,'下单抽奖');
  249. // 发放失败,改为未中奖
  250. if( isset($result['error']) ) {
  251. // 回退数据
  252. DB::rollBack();
  253. // 通知重试
  254. return json_send(['code'=>'error','msg'=>'出了点小差,请重新抽奖','data'=>$result]);
  255. }
  256. }
  257. }
  258. // 扣减数量
  259. $result = $RecruitmentRecord->add($record);
  260. // 记录失败
  261. if( !$result ) {
  262. // 回退数据
  263. DB::rollBack();
  264. // 下单失败提示
  265. return json_send(['code'=>'error','msg'=>'抽奖失败,请重试','data'=>['error'=>'奖品记录失败']]);
  266. }
  267. // 提交事务
  268. DB::commit();
  269. // 通过活动ID,查询奖品
  270. $rewardList = [];
  271. // 奖品数据
  272. foreach ($reward as $value) {
  273. $rewardList[] = ['id'=>$value['id'],'name'=>$value['reward_name'],'img'=>$value['reward_thumb'],'reward_type'=>$value['reward_type']];
  274. }
  275. // 默认可以参加活动
  276. $data['allow_join'] = 1;
  277. // 返回结果
  278. return json_send(['code'=>'success','msg'=>'抽奖成功','data'=>['reward_list'=>$rewardList,'reward_index'=>$rewardIndex,'join_num'=>$data['join_num']-1]]);
  279. // 异常处理
  280. } catch (\Throwable $th) {
  281. // 回退数据
  282. DB::rollBack();
  283. // 下单失败提示
  284. return json_send(['code'=>'error','msg'=>'抽奖失败,请重试','data'=>['error'=>$th->getMessage().$th->getLine()]]);
  285. }
  286. }
  287. }