ActivityCache.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660
  1. <?php
  2. namespace addons\shopro\library\traits;
  3. use addons\shopro\exception\Exception;
  4. use addons\shopro\library\Redis;
  5. use addons\shopro\model\Activity;
  6. use addons\shopro\model\ActivityGoodsSkuPrice;
  7. use addons\shopro\model\Goods;
  8. use addons\shopro\model\GoodsSkuPrice;
  9. use addons\shopro\model\OrderItem;
  10. use addons\shopro\model\ScoreGoodsSkuPrice;
  11. /**
  12. * 活动 redis 缓存
  13. */
  14. trait ActivityCache
  15. {
  16. protected $zsetKey = 'zset-activity';
  17. protected $hashPrefix = 'hash-activity:';
  18. protected $hashGoodsPrefix = 'goods-';
  19. protected $hashGrouponPrefix = 'groupon-';
  20. public function hasRedis($is_interrupt = false) {
  21. $error_msg = '';
  22. try {
  23. $redis = $this->getRedis();
  24. // 检测连接是否正常
  25. $redis->ping();
  26. } catch (\BadFunctionCallException $e) {
  27. // 缺少扩展
  28. $error_msg = $e->getMessage() ? $e->getMessage() : "缺少 redis 扩展";
  29. } catch (\RedisException $e) {
  30. // 连接拒绝
  31. \think\Log::write('redis connection redisException fail: ' . $e->getMessage());
  32. $error_msg = $e->getMessage() ? $e->getMessage() : "redis 连接失败";
  33. } catch (\Exception $e) {
  34. // 异常
  35. \think\Log::write('redis connection fail: ' . $e->getMessage());
  36. $error_msg = $e->getMessage() ? $e->getMessage() : "redis 连接异常";
  37. }
  38. if ($error_msg) {
  39. if ($is_interrupt) {
  40. throw new \Exception($error_msg);
  41. } else {
  42. return false;
  43. }
  44. }
  45. return true;
  46. }
  47. public function getRedis() {
  48. if (!isset($GLOBALS['SPREDIS'])) {
  49. $GLOBALS['SPREDIS'] = (new Redis())->getRedis();
  50. }
  51. return $GLOBALS['SPREDIS'];
  52. }
  53. /**
  54. * 将活动设置到 redis 中
  55. *
  56. * @param [type] $activity
  57. * @param array $goodsList
  58. * @return void
  59. */
  60. public function setActivity($activity, $goodsList = []) {
  61. $redis = $this->getRedis();
  62. // hash 键值
  63. $hashKey = $this->getHashKey($activity['id'], $activity['type']);
  64. // 删除旧的可变数据,需要排除销量 key
  65. if ($redis->EXISTS($hashKey)) {
  66. // 如果 hashKey 存在,删除规格
  67. $hashs = $redis->HGETALL($hashKey);
  68. foreach ($hashs as $hashField => $hashValue) {
  69. // 是商品规格,并且不是销量
  70. if (strpos($hashField, $this->hashGoodsPrefix) !== false && strpos($hashField, '-sale') === false) {
  71. // 商品规格信息,删掉
  72. $redis->HDEL($hashKey, $hashField);
  73. }
  74. }
  75. }
  76. $redis->HMSET($hashKey, [
  77. 'id' => $activity['id'],
  78. 'title' => $activity['title'],
  79. 'type' => $activity['type'],
  80. 'richtext_id' => $activity['richtext_id'],
  81. 'richtext_title' => $activity['richtext_title'],
  82. 'starttime' => $activity['starttime'],
  83. 'endtime' => $activity['endtime'],
  84. 'rules' => is_array($activity['rules']) ? json_encode($activity['rules']) : $activity['rules'],
  85. 'goods_ids' => $activity['goods_ids']
  86. ]
  87. );
  88. if (in_array($activity['type'], ['groupon', 'seckill'])) {
  89. // 拼团或者秒杀,记录商品规格信息 (里面包含活动库存,价格等信息)
  90. foreach ($goodsList as $goods) {
  91. unset($goods['sales']); // 规格销量单独字段保存 goods-id-id-sale key
  92. $goods_sku_key = $this->getHashGoodsKey($goods['goods_id'], $goods['sku_price_id']);
  93. // 获取当前规格的销量,修改库存的时候,需要把 stock 加上这部分销量
  94. $cacheSale = $redis->HGET($hashKey, $goods_sku_key . '-sale');
  95. $goods['stock'] = $goods['stock'] + $cacheSale;
  96. $redis->HSET($hashKey, $this->getHashGoodsKey($goods['goods_id'], $goods['sku_price_id']), json_encode($goods));
  97. }
  98. }
  99. // 将 hash 键值存入 有序集合,score 为 id
  100. $redis->ZADD($this->zsetKey, $activity['starttime'], $hashKey);
  101. }
  102. /**
  103. * 获取所有活动(前端:秒杀商品列表,拼团商品列表)
  104. *
  105. * @param array $activityTypes 为空将查询所有类型的活动
  106. * @param string $status
  107. * @param string $format_type // 格式化类型,默认clear,清理多余的字段,比如拼团的 团信息
  108. * @return void
  109. */
  110. public function getActivityList($activityTypes = [], $status = 'all', $format_type = 'normal') {
  111. $redis = $this->getRedis();
  112. // 获取对应的活动类型的集合
  113. $activityHashList = $this->getActivityHashKeysByType($activityTypes);
  114. $activityList = [];
  115. if (!$activityHashList) { // 没有获取到,返回空数组
  116. return $activityList;
  117. }
  118. foreach ($activityHashList as $activityHashKey) {
  119. // 查询活动状态
  120. if ($status != 'all') {
  121. $activity_status = $this->getActivityStatusByHashKey($activityHashKey);
  122. if ($status != $activity_status) {
  123. continue;
  124. }
  125. }
  126. // 格式化活动
  127. $activity = $this->formatActivityByType($activityHashKey, $format_type);
  128. if ($activity) {
  129. $activityList[] = $activity;
  130. }
  131. }
  132. return $activityList;
  133. }
  134. /**
  135. * 查询商品列表,详情时,获取这个商品对应的秒杀拼团等活动
  136. *
  137. * @param [type] $goods_id
  138. * @param Array $activityTypes
  139. * @param integer $activity_id
  140. * @return void
  141. */
  142. public function getGoodsActivity($goods_id, $activityTypes = [], $activity_id = 0) {
  143. // 获取商品第一条活动的 hash key
  144. $activityHashKey = $this->getActivityHashKeyByGoods($goods_id, $activityTypes, 'first', $activity_id);
  145. // 如果存在活动
  146. if ($activityHashKey) {
  147. // 获取活动并且按照商品的要求格式化
  148. return $this->formatActivityByType($activityHashKey, 'goods', ['goods_id', $goods_id]);
  149. }
  150. return null;
  151. }
  152. /**
  153. * 查询商品列表,详情时,获取这个商品对应的满减,满折等活动
  154. *
  155. * @param int $goods_id 特定商品 id
  156. * @param Array $activityTypes 要查询的活动数组
  157. * @param string $type 查单条,还是全部 all|first
  158. * @param int $activity_id
  159. * @return void
  160. */
  161. public function getGoodsActivityDiscount($goods_id, $activityTypes = [], $type = 'all', $activity_id = 0) {
  162. // 获取活动的 hash key
  163. $activityHashKey = $this->getActivityHashKeyByGoods($goods_id, $activityTypes, $type, $activity_id);
  164. // 如果存在活动
  165. if ($activityHashKey) {
  166. if (is_array($activityHashKey)) {
  167. $activities = [];
  168. foreach ($activityHashKey as $key => $hashKey) {
  169. $activity = $this->formatActivityByType($hashKey, 'discount');
  170. if ($activity) {
  171. $activities[] = $activity;
  172. }
  173. }
  174. return $activities;
  175. } else {
  176. // 获取活动所有信息
  177. return $activity = $this->formatActivityByType($activityHashKey, 'discount');
  178. }
  179. }
  180. return $type == 'all' ? [] : null;
  181. }
  182. /**
  183. * 通过活动的键值,获取活动完整信息
  184. *
  185. * @param [type] $activityHashKey
  186. * @return array
  187. */
  188. public function getActivityByHashKey($activityHashKey)
  189. {
  190. $redis = $this->getRedis();
  191. // 取出整条 hash 记录
  192. $activityHash = $redis->HGETALL($activityHashKey);
  193. return $activityHash;
  194. }
  195. // 删除活动缓存
  196. public function delActivity($activity) {
  197. $redis = $this->getRedis();
  198. $hashKey = $this->getHashKey($activity['id'], $activity['type']);
  199. // 删除 hash
  200. $redis->DEL($hashKey);
  201. // 删除集合
  202. $redis->ZREM($this->zsetKey, $hashKey);
  203. }
  204. /**
  205. * 通过商品获取该商品参与的活动的hash key
  206. *
  207. * @param [type] $goods_id
  208. * @param Array $activityType
  209. * @param string $type 全部还是第一条
  210. * @param integer $activity_id
  211. * @return void
  212. */
  213. private function getActivityHashKeyByGoods($goods_id, $activityType = [], $type = 'first', $activity_id = 0) {
  214. $redis = $this->getRedis();
  215. // 获取对应类型的活动集合
  216. $activityHashList = $this->getActivityHashKeysByType($activityType, $activity_id);
  217. $activityHashKeys = [];
  218. if (!$activityHashList) { // 没有获取到,返回空数组
  219. return $activityHashKeys;
  220. }
  221. foreach ($activityHashList as $activityHashKey) {
  222. if (strpos($activityHashKey, 'seckill') === false && strpos($activityHashKey, 'groupon') === false) {
  223. // 不是拼团,秒杀,要校验活动时间,如果不在活动时间,跳过
  224. // 获取活动状态
  225. $activity_status = $this->getActivityStatusByHashKey($activityHashKey);
  226. if ($activity_status != 'ing') {
  227. continue;
  228. }
  229. }
  230. // 判断这条活动是否包含该商品
  231. $goods_ids = array_filter(explode(',', $redis->HGET($activityHashKey, 'goods_ids')));
  232. if (in_array($goods_id, $goods_ids) || empty($goods_ids)) {
  233. $activityHashKeys[] = $activityHashKey;
  234. if ($type == 'first') { // 只取第一条
  235. break;
  236. }
  237. }
  238. }
  239. if ($activity_id && !$activityHashKeys) {
  240. // 查询特定活动,没找到,抛出异常, 活动不存在
  241. new Exception('活动不存在');
  242. }
  243. return $type == 'first' ? ($activityHashKeys[0] ?? '') : $activityHashKeys;
  244. }
  245. /**
  246. * 获取活动的状态
  247. */
  248. private function getActivityStatusByHashKey($activityHashKey) {
  249. $redis = $this->getRedis();
  250. $starttime = $redis->HGET($activityHashKey, 'starttime');
  251. $endtime = $redis->HGET($activityHashKey, 'endtime');
  252. if ($starttime < time() && $endtime > time()) {
  253. $status = 'ing';
  254. }else if ($starttime > time()) {
  255. $status = 'nostart';
  256. }else if ($endtime < time()) {
  257. $status = 'ended';
  258. }
  259. return $status;
  260. }
  261. /**
  262. * 获取活动类型数组的所有活动hashkeys
  263. *
  264. * @param array|string $activityTypes
  265. * @return array
  266. */
  267. private function getActivityHashKeysByType($activityTypes = [], $activity_id = 0) {
  268. $redis = $this->getRedis();
  269. $activityTypes = is_array($activityTypes) ? $activityTypes : [$activityTypes];
  270. $activityTypes = array_values(array_filter($activityTypes)); // 过滤空值
  271. // 获取活动集合
  272. $hashList = $redis->ZRANGE($this->zsetKey, 0, 999999999);
  273. // 优先判断 activity_id,可以唯一确定 活动key, 不需要判断 activityTypes
  274. if ($activity_id) {
  275. $activityHashKeys = [];
  276. foreach ($hashList as $hashKey) {
  277. $suffix = ':' . $activity_id;
  278. // 判断是否是要找的活动id, 截取 hashKey 后面几位,是否为当前要查找的活动 id
  279. if (substr($hashKey, (strlen($hashKey) - strlen($suffix))) == $suffix) {
  280. $activityHashKeys[] = $hashKey;
  281. break;
  282. }
  283. }
  284. return $activityHashKeys;
  285. }
  286. // 判断是否传入了 需要的活动类型,默认取全部活动
  287. if ($activityTypes) {
  288. // 获取对应的活动类型的集合
  289. $activityHashKeys = [];
  290. foreach ($hashList as $hashKey) {
  291. // 循环要查找的活动类型数组
  292. foreach ($activityTypes as $type) {
  293. if (strpos($hashKey, $type) !== false) { // 是要查找的类型
  294. $activityHashKeys[] = $hashKey;
  295. break;
  296. }
  297. }
  298. }
  299. } else {
  300. // 全部活动
  301. $activityHashKeys = $hashList;
  302. }
  303. return $activityHashKeys;
  304. }
  305. // ------------------------格式化活动---------------------
  306. /**
  307. * 格式化活动
  308. *
  309. * @param string array $activityHash 活动 key 或者活动完整信息
  310. * @param string $type 格式化方式
  311. * @param array $data 额外参数
  312. * @return void
  313. */
  314. public function formatActivityByType($activityHash, $type = 'normal', $data = []) {
  315. switch($type) {
  316. case 'normal' :
  317. // 正常模式,只移除销量,团信息,保留全部商品规格数据
  318. $activity = $this->getActivityFormatNormal($activityHash, $data);
  319. break;
  320. case 'clear' :
  321. // 简洁模式,只保留活动表基本信息
  322. $activity = $this->getActivityFormatClear($activityHash, $data);
  323. break;
  324. case 'goods' :
  325. // 按照前端商品方式格式化
  326. $activity = $this->getActivityFormatGoods($activityHash, $data);
  327. break;
  328. case 'discount' :
  329. $activity = $this->getActivityFormatDiscount($activityHash, $data);
  330. break;
  331. default :
  332. $activity = $this->getActivityFormatNormal($activityHash, $data);
  333. break;
  334. }
  335. return $activity;
  336. }
  337. /**
  338. * 正常模式,只移除销量, 团信息,保留全部商品规格数据
  339. *
  340. * @param string $activityHashKey
  341. * @param array $data 额外数据,商品 id
  342. * @return void
  343. */
  344. private function getActivityFormatNormal($activityHashKey, $data = []) {
  345. // 传入的是活动的key
  346. $activityHash = $this->getActivityByHashKey($activityHashKey);
  347. $activity = [];
  348. foreach ($activityHash as $key => $value) {
  349. // 包含 -sale 全部跳过
  350. if (strpos($key, '-sale') !== false) {
  351. continue;
  352. } else if (strpos($key, $this->hashGrouponPrefix) !== false) {
  353. // 拼团的参团人数,团用户,移除
  354. continue;
  355. } else if ($key == 'rules') {
  356. $activity[$key] = json_decode($value, true);
  357. } else {
  358. // 普通键值
  359. $activity[$key] = $value;
  360. }
  361. }
  362. if ($activity) {
  363. // 处理活动状态
  364. $activity['status_code'] = Activity::getStatusCode($activity);
  365. }
  366. return $activity ?: null;
  367. }
  368. /**
  369. * 简洁模式,只保留活动表基本信息
  370. *
  371. * @param string $activityHashKey
  372. * @param array $data 额外数据,商品 id
  373. * @return void
  374. */
  375. private function getActivityFormatClear($activityHashKey, $data = []) {
  376. $activityHash = $this->getActivityByHashKey($activityHashKey);
  377. $activity = [];
  378. foreach ($activityHash as $key => $value) {
  379. // 包含 -sale 全部跳过
  380. if (strpos($key, $this->hashGoodsPrefix) !== false) {
  381. continue;
  382. } else if (strpos($key, $this->hashGrouponPrefix) !== false) {
  383. // 拼团的参团人数,团用户,移除
  384. continue;
  385. } else if ($key == 'rules') {
  386. $activity[$key] = json_decode($value, true);
  387. } else {
  388. // 普通键值
  389. $activity[$key] = $value;
  390. }
  391. }
  392. if ($activity) {
  393. // 处理活动状态
  394. $activity['status_code'] = Activity::getStatusCode($activity);
  395. }
  396. return $activity ?: null;
  397. }
  398. /**
  399. * 获取并按照商品展示格式化活动数据
  400. *
  401. * @param string $activityHashKey hash key
  402. * @param array $data 额外数据,商品 id
  403. * @return array
  404. */
  405. private function getActivityFormatGoods($activityHashKey, $data = [])
  406. {
  407. $goods_id = $data['goods_id'] ?? 0;
  408. // 传入的是活动的key
  409. $activityHash = $this->getActivityByHashKey($activityHashKey);
  410. $activity = [];
  411. // 商品前缀
  412. $goodsPrefix = $this->hashGoodsPrefix . ($goods_id ? $goods_id . '-' : '');
  413. foreach ($activityHash as $key => $value) {
  414. // 包含 -sale 全部跳过
  415. if (strpos($key, '-sale') !== false) {
  416. continue;
  417. } else if (strpos($key, $goodsPrefix) !== false) {
  418. // 商品规格信息,或者特定商品规格信息
  419. $goods = json_decode($value, true);
  420. // 计算销量库存数据
  421. $goods = $this->calcGoods($goods, $activityHashKey);
  422. // 商品规格项
  423. $activity['activity_goods_sku_price'][] = $goods;
  424. } else if ($goods_id && strpos($key, $this->hashGoodsPrefix) !== false) {
  425. // 需要特定商品时,移除别的非当前商品的数据
  426. continue;
  427. } else if (strpos($key, $this->hashGrouponPrefix) !== false) {
  428. // 拼团的参团人数,团用户,移除
  429. continue;
  430. } else if ($key == 'rules') {
  431. $activity[$key] = json_decode($value, true);
  432. } else {
  433. // 普通键值
  434. $activity[$key] = $value;
  435. }
  436. }
  437. if ($activity) {
  438. // 处理活动状态
  439. $activity['status_code'] = Activity::getStatusCode($activity);
  440. }
  441. return $activity ?: null;
  442. }
  443. /**
  444. * 获取并按照折扣格式展示格式化活动数据
  445. *
  446. * @param string $activityHashKey hash key
  447. * @param array $data 额外数据
  448. * @return void
  449. */
  450. private function getActivityFormatDiscount($activityHashKey, $data = [])
  451. {
  452. $activityHash = $this->getActivityByHashKey($activityHashKey);
  453. $activity = [];
  454. foreach ($activityHash as $key => $value) {
  455. if ($key == 'rules') {
  456. $rules = json_decode($value, true);
  457. // 存在折扣
  458. if (isset($rules['discounts']) && $rules['discounts']) {
  459. // 处理展示优惠,full 从小到大
  460. $discounts = $rules['discounts'] ?? [];
  461. $discountsKeys = array_column($discounts, null, 'full');
  462. ksort($discountsKeys);
  463. $rules['discounts'] = array_values($discountsKeys); // 优惠按照 full 从小到大排序
  464. }
  465. $activity[$key] = $rules;
  466. } else {
  467. // 普通键值
  468. $activity[$key] = $value;
  469. }
  470. }
  471. if ($activity) {
  472. // 处理活动状态
  473. $activity['status_code'] = Activity::getStatusCode($activity);
  474. }
  475. return $activity ?: null;
  476. }
  477. /**
  478. * 计算每个规格的真实库存、销量
  479. *
  480. * @param [type] $goods
  481. * @param [type] $activityHashKey
  482. * @return void
  483. */
  484. private function calcGoods($goods, $activityHashKey)
  485. {
  486. $redis = $this->getRedis();
  487. // 销量 key
  488. $saleKey = $this->getHashGoodsKey($goods['goods_id'], $goods['sku_price_id'], true);
  489. // 缓存中的销量
  490. $cacheSale = $redis->HGET($activityHashKey, $saleKey);
  491. $stock = $goods['stock'] - $cacheSale;
  492. $goods['stock'] = $stock > 0 ? $stock : 0;
  493. $goods['sales'] = $cacheSale;
  494. return $goods;
  495. }
  496. // 拼接 hash key
  497. private function getHashKey($activity_id, $activity_type) {
  498. // 示例 hash-activity:groupon:25
  499. return $this->hashPrefix . $activity_type . ':' . $activity_id;
  500. }
  501. // 拼接 hash 表中 goods 的 key
  502. private function getHashGoodsKey($goods_id, $sku_price_id, $is_sale = false)
  503. {
  504. // 示例 商品规格:goods-25-30 or 商品规格销量:goods-25-30-sale
  505. return $this->hashGoodsPrefix . $goods_id . '-' . $sku_price_id . ($is_sale ? '-sale' : '');
  506. }
  507. // 拼接 hash 表中 groupon 的 key
  508. private function getHashGrouponKey($groupon_id, $goods_id, $type = '')
  509. {
  510. return $this->hashGrouponPrefix . $groupon_id . '-' . $goods_id . ($type ? '-' . $type : '');
  511. }
  512. // 获取 key 集合
  513. public function getKeys($detail, $activity)
  514. {
  515. // 获取 hash key
  516. $activityHashKey = $this->getHashKey($activity['activity_id'], $activity['activity_type']);
  517. $goodsSkuPriceKey = '';
  518. $saleKey = '';
  519. if (isset($detail['goods_sku_price_id']) && $detail['goods_sku_price_id']) {
  520. // 获取 hash 表中商品 sku 的 key
  521. $goodsSkuPriceKey = $this->getHashGoodsKey($detail['goods_id'], $detail['goods_sku_price_id']);
  522. // 获取 hash 表中商品 sku 的 销量的 key
  523. $saleKey = $this->getHashGoodsKey($detail['goods_id'], $detail['goods_sku_price_id'], true);
  524. }
  525. // 需要拼团的字段
  526. $grouponKey = '';
  527. $grouponNumKey = '';
  528. $grouponUserlistKey = '';
  529. if (isset($detail['groupon_id']) && $detail['groupon_id']) {
  530. // 获取 hash 表中团 key
  531. $grouponKey = $this->getHashGrouponKey($detail['groupon_id'], $detail['goods_id']);
  532. // 获取 hash 表中团当前人数 key
  533. $grouponNumKey = $this->getHashGrouponKey($detail['groupon_id'], $detail['goods_id'], 'num');
  534. // 获取 hash 表中团当前人员列表 key
  535. $grouponUserlistKey = $this->getHashGrouponKey($detail['groupon_id'], $detail['goods_id'], 'userlist');
  536. }
  537. return compact('activityHashKey', 'goodsSkuPriceKey', 'saleKey', 'grouponKey', 'grouponNumKey', 'grouponUserlistKey');
  538. }
  539. }