GoodsService.php 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861
  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\AgentModel;
  13. use App\Models\CartsModel;
  14. use App\Models\GoodsCategoryModel;
  15. use App\Models\GoodsModel;
  16. use App\Models\GoodsSkuModel;
  17. use App\Models\MemberModel;
  18. use App\Models\MerchantModel;
  19. use App\Models\OrderModel;
  20. use App\Models\ShopModel;
  21. use App\Models\TradeModel;
  22. use App\Services\BaseService;
  23. use App\Services\ConfigService;
  24. use App\Services\RedisService;
  25. use App\Services\SupplyService;
  26. use App\Services\WalletService;
  27. use BN\Red;
  28. use Illuminate\Support\Facades\DB;
  29. /**
  30. * 商品管理-服务类
  31. * @author laravel开发员
  32. * @since 2020/11/11
  33. * Class GoodsService
  34. * @package App\Services\Api
  35. */
  36. class GoodsService extends BaseService
  37. {
  38. // 静态对象
  39. protected static $instance = null;
  40. /**
  41. * 构造函数
  42. * @author laravel开发员
  43. * @since 2020/11/11
  44. * GoodsService constructor.
  45. */
  46. public function __construct()
  47. {
  48. $this->model = new GoodsModel();
  49. }
  50. /**
  51. * 静态入口
  52. * @return static|null
  53. */
  54. public static function make()
  55. {
  56. if (!self::$instance) {
  57. self::$instance = (new static());
  58. }
  59. return self::$instance;
  60. }
  61. /**
  62. * 商品列表
  63. * @param $params
  64. * @param int $pageSize
  65. * @param int $userId
  66. * @return array
  67. */
  68. public function getDataList($params, $pageSize = 12, $userId = 0)
  69. {
  70. $model = $this->model->with(['skuList'])->from('goods as a')
  71. ->where(['a.status' => 1, 'a.mark' => 1])
  72. ->where('a.retail_price', '>', 0)
  73. ->where(function ($query) use ($params) {
  74. $supplyType = isset($params['supply_type']) ? intval($params['supply_type']) : 0;
  75. if ($supplyType > 0) {
  76. $query->where('a.supply_type', $supplyType);
  77. }
  78. $cateId = isset($params['cate_id']) ? intval($params['cate_id']) : 0;
  79. if ($cateId > 0) {
  80. $subIds = GoodsCategoryModel::where(['pid'=> $cateId,'mark'=>1,'status'=>1])->pluck('cate_id');
  81. $query->where(function($query) use($cateId,$subIds){
  82. if($subIds){
  83. $query->whereIn('a.cate_id', $subIds)
  84. ->orWhere('a.cate_id', $cateId);
  85. }else{
  86. $query->where('a.cate_id', $cateId);
  87. }
  88. });
  89. }
  90. })
  91. ->where(function ($query) use ($params) {
  92. $keyword = isset($params['kw']) ? $params['kw'] : '';
  93. if ($keyword) {
  94. $query->where('a.goods_name', 'like', "%{$keyword}%")
  95. ->orWhere('a.spu_name', 'like', "%{$keyword}%")
  96. ->orWhere('a.tag', 'like', "%{$keyword}%");
  97. }
  98. })
  99. ->select(['a.*']);
  100. // 排序
  101. $sortType = isset($params['sort_type'])? $params['sort_type'] : 1;
  102. if ($sortType == 2){
  103. $model = $model->orderBy('a.is_recommend','asc')->orderBy('a.sales', 'desc');
  104. }
  105. $list = $model->orderBy('a.create_time', 'desc')
  106. ->paginate($pageSize > 0 ? $pageSize : 9999999);
  107. $list = $list ? $list->toArray() : [];
  108. if ($list) {
  109. $locale = RedisService::get("caches:locale:lang_{$userId}");
  110. $locale = $locale ? $locale : session('locale_lang');
  111. $locale = $locale ? $locale : 'zh-cn';
  112. $supplyList = config('goods.supplyList');
  113. $usdtPrice = RedisService::get("caches:wallets:usdt_rate");
  114. if($usdtPrice<=0){
  115. $usdtCnyPrice = ConfigService::make()->getConfigByCode('usdt_cny_price', 7.2);
  116. $usdtPrice = $usdtCnyPrice>0 && $usdtCnyPrice< 100? $usdtCnyPrice : 0;
  117. }
  118. $xdPrice = ConfigService::make()->getConfigByCode('xd_price', 100);
  119. $xdPrice = $xdPrice > 0 && $xdPrice <= 10000 ? $xdPrice : 100;
  120. foreach ($list['data'] as &$item) {
  121. $item['detail_img'] = isset($item['detail_img']) && $item['detail_img'] ? json_decode($item['detail_img'], true) : [];
  122. $item['supply_name'] = isset($supplyList[$item['supply_type']]) ? $supplyList[$item['supply_type']] : '';
  123. $item['usdt_price'] = $usdtPrice;
  124. $item['xd_price_rate'] = $xdPrice;
  125. $item['original_price'] = $item['retail_price'];
  126. $item['retail_price'] = $usdtPrice > 0 ? moneyFormat($item['retail_price'] / $usdtPrice * $xdPrice, 2) : $item['retail_price'];
  127. }
  128. unset($item);
  129. } else {
  130. $this->updateGoods();
  131. }
  132. return [
  133. 'total' => isset($list['total']) ? $list['total'] : 9,
  134. 'pageSize' => $pageSize,
  135. 'list' => isset($list['data']) ? $list['data'] : [],
  136. ];
  137. }
  138. /**
  139. * 详情
  140. * @param $id
  141. * @return array
  142. */
  143. public function getInfo($goodsId, $userId=0, $updateView=true)
  144. {
  145. $field = ['a.*'];
  146. $info = $this->model->from('goods as a')->with(['category','skuList'])
  147. ->where(['a.goods_id'=> $goodsId,'a.status'=>1,'a.mark'=>1])
  148. ->select($field)
  149. ->first();
  150. $info = $info? $info->toArray() : [];
  151. if($info){
  152. if(isset($info['main_img'])){
  153. $info['main_img'] = $info['main_img']? get_image_url($info['main_img']) : '';
  154. }
  155. if(isset($info['detail_img'])){
  156. $info['detail_img'] = $info['detail_img']? json_decode($info['detail_img'], true) : [];
  157. }
  158. $supplyList = config('goods.supplyList');
  159. $info['supply_name'] = isset($supplyList[$info['supply_type']]) ? $supplyList[$info['supply_type']] : '';
  160. $usdtPrice = RedisService::get("caches:wallets:usdt_rate");
  161. if($usdtPrice<=0){
  162. $usdtCnyPrice = ConfigService::make()->getConfigByCode('usdt_cny_price', 7.2);
  163. $usdtPrice = $usdtCnyPrice>0 && $usdtCnyPrice< 100? $usdtCnyPrice : 0;
  164. }
  165. $xdPrice = ConfigService::make()->getConfigByCode('xd_price', 100);
  166. $xdPrice = $xdPrice > 0 && $xdPrice <= 10000 ? $xdPrice : 100;
  167. $info['usdt_price_rate'] = $usdtPrice;
  168. $info['xd_price'] = $xdPrice;
  169. $info['custom_uid'] = ConfigService::make()->getConfigByCode('xl_custom_id',100001);
  170. if(isset($info['retail_price']) && $info['retail_price']){
  171. $info['retail_price'] = moneyFormat($info['retail_price']/$usdtPrice * $xdPrice,2);
  172. }
  173. if(isset($info['sku_list']) && $info['sku_list']){
  174. foreach ($info['sku_list'] as &$v){
  175. $v['detail_img'] = $v['detail_img']? json_decode($v['detail_img'], true) : [];
  176. $v['attr'] = $v['attr']? json_decode($v['attr'], true) : [];
  177. $v['main_img'] = $v['main_img']? get_image_url($v['main_img']) : '';
  178. $v['retail_price'] = $v['retail_price']? moneyFormat($v['retail_price']/$usdtPrice * $xdPrice,2) : $info['retail_price'];
  179. }
  180. unset($v);
  181. }
  182. // 更新访问量
  183. if($updateView){
  184. $this->updateView($userId, $goodsId);
  185. }
  186. }
  187. return $info;
  188. }
  189. /**
  190. * 获取要购买或结算的商品列表
  191. * @param $userId
  192. * @param array $params
  193. * @return array|false|\Illuminate\Database\Eloquent\Builder[]|\Illuminate\Database\Eloquent\Collection
  194. */
  195. public function getBuyList($userId, $params=[])
  196. {
  197. $goodsId = isset($params['goods_id'])? $params['goods_id'] : 0;
  198. $skuId = isset($params['sku_id'])? $params['sku_id'] : 0;
  199. $num = isset($params['num'])? $params['num'] : 0;
  200. $cartIds = isset($params['cart_ids'])? $params['cart_ids'] : '';
  201. $cartIds = $cartIds? explode('|', $cartIds) : [];
  202. if(empty($goodsId) && empty($cartIds)){
  203. $this->error = 2901;
  204. return false;
  205. }
  206. if($goodsId && (empty($skuId) || $num<=0)){
  207. $this->error = 2901;
  208. return false;
  209. }
  210. $cacheKey = "caches:goods:buyList:{$userId}_".md5(json_encode($params,256));
  211. $datas = RedisService::get($cacheKey);
  212. if($datas){
  213. return $datas;
  214. }
  215. $goods = [];
  216. $skuList = [];
  217. $usdtPrice = RedisService::get("caches:wallets:usdt_rate");
  218. if($usdtPrice<=0){
  219. $usdtCnyPrice = ConfigService::make()->getConfigByCode('usdt_cny_price', 7.2);
  220. $usdtPrice = $usdtCnyPrice>0 && $usdtCnyPrice< 100? $usdtCnyPrice : 0;
  221. }
  222. $xdPrice = ConfigService::make()->getConfigByCode('xd_price', 100);
  223. $xdPrice = $xdPrice > 0 && $xdPrice <= 10000 ? $xdPrice : 100;
  224. if($goodsId){
  225. $info = $this->getInfo($goodsId, $userId, false);
  226. if($info){
  227. $info['num'] = $num;
  228. $info['sku_id'] = $skuId;
  229. $skuInfo = GoodsSkuModel::where(['goods_id'=> $goodsId,'sku_id'=> $skuId,'mark'=>1])->first();
  230. $retailPrice = isset($skuInfo['retail_price'])? $skuInfo['retail_price'] : 0;
  231. if($retailPrice){
  232. $info['retail_price'] = $usdtPrice > 0 ? moneyFormat($retailPrice / $usdtPrice * $xdPrice, 2) : $retailPrice;
  233. }
  234. if(isset($skuInfo['attr']) && $skuInfo['attr']){
  235. $skuInfo['attr'] = json_decode($skuInfo['attr'], true);
  236. }
  237. $skuList[$goodsId] = [
  238. 'sku_id'=> $skuId,
  239. 'num'=> $num
  240. ];
  241. $info['sku'] = $skuInfo;
  242. $goods[] = $info;
  243. }
  244. }else {
  245. $goods = CartsModel::with(['sku'])->from('carts as a')
  246. ->leftJoin('goods as b','b.goods_id','=','a.goods_id')
  247. ->leftJoin('goods_sku as c','c.sku_id','=','a.sku_id')
  248. ->whereIn('a.id', $cartIds)
  249. ->where(['a.status' => 1, 'a.mark' => 1,'b.status'=>1,'b.mark'=>1])
  250. ->where('b.retail_price', '>', 0)
  251. ->where('a.num', '>', 0)
  252. ->select(['b.goods_id','b.merch_id','b.goods_name','b.supply_type','b.main_img','b.cost_price','b.retail_price','b.limit_num','b.lowest_num','b.brand_name','a.num','a.sku_id'])
  253. ->get();
  254. if($goods){
  255. // 价格等参数格式化
  256. $locale = RedisService::get("caches:locale:lang_{$userId}");
  257. $locale = $locale ? $locale : session('locale_lang');
  258. $locale = $locale ? $locale : 'zh-cn';
  259. $supplyList = config('goods.supplyList');
  260. foreach ($goods as &$item) {
  261. $item['detail_img'] = isset($item['detail_img']) && $item['detail_img'] ? json_decode($item['detail_img'], true) : [];
  262. $item['supply_name'] = isset($supplyList[$item['supply_type']]) ? $supplyList[$item['supply_type']] : '';
  263. $item['usdt_price'] = $usdtPrice;
  264. $item['xd_price_rate'] = $xdPrice;
  265. $item['original_price'] = $item['retail_price'];
  266. $skuInfo = isset($item['sku'])? $item['sku'] : [];
  267. if(isset($skuInfo['attr']) && $skuInfo['attr']){
  268. $skuInfo['attr'] = json_decode($skuInfo['attr'], true);
  269. }
  270. $item['sku'] = $skuInfo;
  271. $retailPrice = isset($skuInfo['retail_price'])? $skuInfo['retail_price'] : 0;
  272. if($retailPrice){
  273. $item['retail_price'] = $usdtPrice > 0 ? moneyFormat($retailPrice / $usdtPrice * $xdPrice, 2) : $retailPrice;
  274. }else{
  275. $item['retail_price'] = $usdtPrice > 0 ? moneyFormat($item['retail_price'] / $usdtPrice * $xdPrice, 2) : $item['retail_price'];
  276. }
  277. $skuList[$item['goods_id']] = [
  278. 'sku_id'=> $item['sku_id'],
  279. 'num'=> $item['num']
  280. ];
  281. }
  282. unset($item);
  283. }
  284. }
  285. if(empty($goods)){
  286. $this->error = 2901;
  287. return false;
  288. }
  289. RedisService::set($cacheKey, ['sku_list'=> array_values($skuList), 'goods'=> $goods], rand(2,3));
  290. return ['sku_list'=> array_values($skuList), 'goods'=> $goods];
  291. }
  292. /**
  293. * 订单商品
  294. * @param $userId
  295. * @param $ids
  296. * @return array
  297. */
  298. public function getOrderGoods($userId, $ids)
  299. {
  300. $cacheKey = "caches:goods:order_{$userId}_".md5(json_encode($ids,256));
  301. $datas = RedisService::get($cacheKey);
  302. if($datas){
  303. return $datas;
  304. }
  305. $goods = $this->model->from('goods as a')
  306. ->whereIn('a.goods_id',$ids)
  307. ->where(['a.status' => 1, 'a.mark' => 1])
  308. ->where('a.retail_price', '>', 0)
  309. ->select(['a.goods_id','a.merch_id','a.goods_name','a.cate_id','a.supply_type','a.main_img','a.retail_price'])
  310. ->get();
  311. $goods = $goods? $goods->toArray() : [];
  312. if($goods){
  313. RedisService::set($cacheKey, $goods, rand(3,5));
  314. }
  315. return $goods;
  316. }
  317. /**
  318. * 添加/更新购物车
  319. * @param $userId
  320. * @param $goodsId
  321. * @param $params
  322. * @return array|false
  323. */
  324. public function updateCart($userId, $goodsId, $params)
  325. {
  326. $skuId = isset($params['sku_id'])? $params['sku_id'] : 0;
  327. $status = isset($params['status'])? $params['status'] : 0;
  328. $merchId = isset($params['merch_id'])? $params['merch_id'] : 0;
  329. $num = isset($params['num'])? $params['num'] : 1;
  330. if($skuId<=0 || $goodsId<=0 || $userId<=0 || $num<=0){
  331. $this->error = 2014;
  332. return false;
  333. }
  334. if(CartsModel::where(['user_id'=> $userId,'status'=>1,'mark'=>1])->count('id') > 20){
  335. $this->error = 1050;
  336. return false;
  337. }
  338. $cartId = CartsModel::where(['user_id'=> $userId,'goods_id'=> $goodsId,'sku_id'=>$skuId])->value('id');
  339. if($cartId){
  340. CartsModel::where(['id'=> $cartId])->update(['num'=> $num,'merch_id'=>$merchId,'status'=> $status,'mark'=>1,'update_time'=>time()]);
  341. RedisService::clear("caches:members:cartList:{$userId}");
  342. $count = $this->getCartCount($userId,true);
  343. return ['id'=> $cartId,'count'=> $count];
  344. }else{
  345. $cartId = CartsModel::insertGetId(['user_id'=> $userId,'goods_id'=>$goodsId,'merch_id'=>$merchId,'sku_id'=>$skuId,'num'=> $num,'status'=> $status,'mark'=>1,'create_time'=>time()]);
  346. RedisService::clear("caches:members:cartList:{$userId}");
  347. $count = $this->getCartCount($userId, true);
  348. return ['id'=> $cartId,'count'=>$count];
  349. }
  350. }
  351. /**
  352. * 购物车列表
  353. * @param $userId
  354. * @param int $pageSize
  355. * @return array|\Illuminate\Database\Eloquent\Builder[]|\Illuminate\Database\Eloquent\Collection|mixed
  356. */
  357. public function getCartList($userId, $pageSize = 30)
  358. {
  359. $cacheKey = "caches:members:cartList:{$userId}";
  360. $cacheCountKey = "caches:members:cartCount:{$userId}";
  361. $datas = RedisService::get($cacheKey);
  362. if($datas){
  363. return $datas;
  364. }
  365. $datas = CartsModel::with(['sku'])->from('carts as a')
  366. ->leftJoin('goods as b','b.goods_id','=','a.goods_id')
  367. ->leftJoin('goods_sku as c','c.sku_id','=','a.sku_id')
  368. ->where(['a.status' => 1, 'a.mark' => 1,'b.status'=>1,'b.mark'=>1])
  369. ->where('b.cost_price', '>', 0)
  370. ->where('a.num', '>', 0)
  371. ->select(['b.goods_id','b.merch_id','b.goods_name','b.supply_type','b.main_img','b.cost_price','b.retail_price','b.limit_num','b.lowest_num','b.brand_name','a.num','a.sku_id'])
  372. ->orderBy('a.create_time','desc')
  373. ->limit($pageSize)
  374. ->get();
  375. if($datas){
  376. // 价格等参数格式化
  377. $locale = RedisService::get("caches:locale:lang_{$userId}");
  378. $locale = $locale ? $locale : session('locale_lang');
  379. $locale = $locale ? $locale : 'zh-cn';
  380. $supplyList = config('goods.supplyList');
  381. $usdtPrice = RedisService::get("caches:wallets:usdt_rate");
  382. if($usdtPrice<=0){
  383. $usdtCnyPrice = ConfigService::make()->getConfigByCode('usdt_cny_price', 7.2);
  384. $usdtPrice = $usdtCnyPrice>0 && $usdtCnyPrice< 100? $usdtCnyPrice : 0;
  385. }
  386. $xdPrice = ConfigService::make()->getConfigByCode('xd_price', 100);
  387. $xdPrice = $xdPrice > 0 && $xdPrice <= 10000 ? $xdPrice : 100;
  388. foreach ($datas as &$item) {
  389. $item['detail_img'] = isset($item['detail_img']) && $item['detail_img'] ? json_decode($item['detail_img'], true) : [];
  390. $item['supply_name'] = isset($supplyList[$item['supply_type']]) ? $supplyList[$item['supply_type']] : '';
  391. $item['usdt_price'] = $usdtPrice;
  392. $item['xd_price_rate'] = $xdPrice;
  393. $item['original_price'] = $item['retail_price'];
  394. $skuInfo = isset($item['sku'])? $item['sku'] : [];
  395. $retailPrice = isset($skuInfo['retail_price'])? $skuInfo['retail_price'] : 0;
  396. if($retailPrice){
  397. $item['retail_price'] = $usdtPrice > 0 ? moneyFormat($retailPrice / $usdtPrice * $xdPrice, 2) : $retailPrice;
  398. }else{
  399. $item['retail_price'] = $usdtPrice > 0 ? moneyFormat($item['retail_price'] / $usdtPrice * $xdPrice, 2) : $item['retail_price'];
  400. }
  401. }
  402. unset($item);
  403. RedisService::set($cacheCountKey, count($datas), rand(300, 600));
  404. RedisService::set($cacheKey, $datas, rand(300, 600));
  405. }
  406. return $datas;
  407. }
  408. /**
  409. * 购物车中数量
  410. * @param $userId
  411. * @return array|mixed
  412. */
  413. public function getCartCount($userId, $refresh = false)
  414. {
  415. $cacheKey = "caches:member:cartCount:{$userId}";
  416. $data = RedisService::get($cacheKey);
  417. if($data>0 && !$refresh){
  418. return $data;
  419. }
  420. $data = CartsModel::from('carts as a')
  421. ->leftJoin('goods as b','b.goods_id','=','a.goods_id')
  422. ->where(['a.status' => 1, 'a.mark' => 1,'b.status'=>1,'b.mark'=>1])
  423. ->where('a.num', '>', 0)
  424. ->where('b.retail_price', '>', 0)
  425. ->count('a.id');
  426. if($data){
  427. RedisService::set($cacheKey, $data, rand(300, 600));
  428. }
  429. return $data;
  430. }
  431. /**
  432. * 运费
  433. */
  434. public function getFreight($userId, $addressId,$skuList)
  435. {
  436. $cacheKey = "caches:goods:freight:{$userId}_{$addressId}_".md5(json_encode($skuList,256));
  437. $data = RedisService::get($cacheKey);
  438. if($data){
  439. return $data;
  440. }
  441. if(empty($addressId)){
  442. $address = MemberAddressService::make()->getBindInfo($userId);
  443. $streetCode = isset($address['street_code'])? $address['street_code'] : '';
  444. $addressId = $streetCode? $streetCode : (isset($address['district_code'])? $address['district_code'] : '');
  445. }
  446. $result = SupplyService::make()->getApiData('getFreight',['address_id'=> intval($addressId),'sku_list'=> $skuList]);
  447. $freight = isset($result['freight'])? floatval($result['freight']) : -1;
  448. if($freight>=0){
  449. // 价格参数
  450. $usdtPrice = RedisService::get("caches:wallets:usdt_rate");
  451. if($usdtPrice<=0){
  452. $usdtCnyPrice = ConfigService::make()->getConfigByCode('usdt_cny_price', 7.2);
  453. $usdtPrice = $usdtCnyPrice>0 && $usdtCnyPrice< 100? $usdtCnyPrice : 0;
  454. }
  455. $xdPrice = ConfigService::make()->getConfigByCode('xd_price', 100);
  456. $xdPrice = $xdPrice > 0 && $xdPrice <= 10000 ? $xdPrice : 100;
  457. $xdFreight = $freight? moneyFormat($freight/$usdtPrice * $xdPrice, 2) : 0;
  458. RedisService::set($cacheKey, ['freight'=>$xdFreight,'fee'=> $freight], rand(5,10));
  459. return ['freight'=>$xdFreight,'fee'=> $freight];
  460. }else{
  461. $errorCode = SupplyService::make()->getError();
  462. $this->error = $errorCode? $errorCode : 1052;
  463. return false;
  464. }
  465. }
  466. /**
  467. * 更新浏览量
  468. * @param $userId
  469. * @param $dynamicId
  470. * @return array|mixed
  471. */
  472. public function updateView($userId, $id)
  473. {
  474. $cacheKey = "caches:goods:views:u{$userId}_d{$id}";
  475. $data = RedisService::get($cacheKey);
  476. if($data){
  477. return false;
  478. }
  479. $data = $this->model->where(['goods_id'=> $id])->update(['views'=>DB::raw('views + 1'),'update_time'=>time()]);
  480. RedisService::set($cacheKey, $id, rand(1,3)*7200);
  481. return $data;
  482. }
  483. /**
  484. * 更新商品SKU数据到本地
  485. * @param int $pageSize
  486. * @param $params 参数
  487. * @return array|false
  488. */
  489. public function updateGoodsSku($pageSize = 100, $params = [])
  490. {
  491. $cacheKey = "caches:supply:goods_sku_update_{$pageSize}";
  492. if (RedisService::get($cacheKey)) {
  493. $this->error = 1047;
  494. return false;
  495. }
  496. $page = RedisService::get($cacheKey . '_page');
  497. $page = $page ? $page + 1 : 1;
  498. $lastDate = GoodsSkuModel::where(['mark' => 1])->orderBy('last_update_at', 'desc')->value('last_update_at');
  499. $params = [
  500. 'limit' => $pageSize > 0 ? $pageSize : 50,
  501. 'page' => $page,
  502. 'date' => $lastDate ? $lastDate : '', // 开始时间
  503. ];
  504. $goods = [];
  505. $updated = 0;
  506. $error = 0;
  507. $datas = SupplyService::make()->getApiData('getSkuUpdate', $params);
  508. if ($datas && $datas['list']) {
  509. foreach ($datas['list'] as &$item) {
  510. $goodsId = isset($item['goods_id']) ? $item['goods_id'] : 0;
  511. $goodsSkuSn = isset($item['sku_sn']) ? $item['sku_sn'] : '';
  512. $changeType = isset($item['change_type']) ? $item['change_type'] : '';
  513. if ($goodsId && $goodsSkuSn && $changeType) {
  514. $skuInfo = SupplyService::make()->getApiData('getSkuDetail', ['sku_sn' => $goodsSkuSn]);
  515. if ($skuInfo) {
  516. $updateData = ['goods_id'=>$goodsId,'sku_sn'=> $goodsSkuSn,'remark'=>'SKU更新','update_time' => time(),'mark'=>1, 'last_update_at' => $item['update_time']];
  517. if (isset($skuInfo['sku_name']) && $skuInfo['sku_name']) {
  518. $updateData['sku_name'] = $skuInfo['sku_name'];
  519. }
  520. if (isset($skuInfo['main_img']) && $skuInfo['main_img']) {
  521. $updateData['main_img'] = $skuInfo['main_img'];
  522. }
  523. if (isset($skuInfo['spu_sn']) && $skuInfo['spu_sn']) {
  524. $updateData['spu_sn'] = $skuInfo['spu_sn'];
  525. }
  526. if (isset($skuInfo['sku_id']) && $skuInfo['sku_id']) {
  527. $updateData['sku_id'] = intval($skuInfo['sku_id']);
  528. }
  529. if (isset($skuInfo['status']) && $skuInfo['status']) {
  530. $updateData['status'] = intval($skuInfo['status']);
  531. }
  532. if (isset($skuInfo['retail_price']) && $skuInfo['retail_price']) {
  533. $updateData['retail_price'] = floatval($skuInfo['retail_price']);
  534. }
  535. if (isset($skuInfo['plat_price']) && $skuInfo['plat_price']) {
  536. $updateData['plat_price'] = floatval($skuInfo['plat_price']);
  537. }
  538. if (isset($skuInfo['detail_img']) && $skuInfo['detail_img']) {
  539. $updateData['detail_img'] = json_encode($skuInfo['detail_img'], 256);
  540. }
  541. if (isset($skuInfo['attr']) && $skuInfo['attr']) {
  542. $updateData['attr'] = json_encode($skuInfo['attr'], 256);
  543. }
  544. if(GoodsSkuModel::where(['goods_id' => $goodsId])->value('id')){
  545. GoodsSkuModel::where(['goods_id' => $goodsId, 'mark' => 1])->update($updateData);
  546. $updated++;
  547. }else{
  548. $error++;
  549. }
  550. } else {
  551. $error++;
  552. }
  553. } else {
  554. $error++;
  555. }
  556. }
  557. unset($item);
  558. RedisService::set($cacheKey . '_page', $page, rand(300, 600));
  559. }else{
  560. RedisService::set($cacheKey . '_page', 0, rand(300, 600));
  561. }
  562. return ['count' => count($goods), 'updated' => $updated, 'errorCount' => $error,'page'=>$page];
  563. }
  564. /**
  565. * 更新商品
  566. * @param int $pageSize
  567. * @param $params 参数
  568. * @return array|false
  569. */
  570. public function updateGoods($pageSize = 100, $params = [])
  571. {
  572. set_time_limit(0);
  573. $cacheKey = "caches:supply:goods_list_update_{$pageSize}";
  574. if (RedisService::get($cacheKey)) {
  575. $this->error = 1047;
  576. return false;
  577. }
  578. $page = RedisService::get($cacheKey . '_page');
  579. $page = $page ? $page + 1 : 1;
  580. $lastTime = $this->model->where(['mark' => 1])->orderBy('create_time', 'desc')->value('create_time');
  581. $params = [
  582. 'limit' => $pageSize > 0 ? $pageSize : 50,
  583. 'page' => $page,
  584. 'status' => isset($params['status']) ? $params['status'] : 1, // 状态:0-全部,1-上架的,2-下架的
  585. 'supply_type' => isset($params['supply_type']) ? $params['supply_type'] : 0, // 渠道商
  586. 'title' => isset($params['title']) ? $params['title'] : '', // 标题关键词
  587. 'cate_id' => isset($params['cate_id']) ? $params['cate_id'] : '', // 分类ID
  588. 'begin_time' => $lastTime ? $lastTime : '', // 开始时间
  589. ];
  590. $goods = [];
  591. $skus = [];
  592. $updated = 0;
  593. $error = 0;
  594. $datas = SupplyService::make()->getApiData('getGoodsList', $params);
  595. if ($datas && $datas['list']) {
  596. foreach ($datas['list'] as &$item) {
  597. $goodsId = isset($item['goods_id']) ? $item['goods_id'] : 0;
  598. if ($goodsId && !$this->checkGoods($goodsId)) {
  599. $info = $this->getApiInfo($goodsId);
  600. if ($info) {
  601. $skuList = isset($info['sku_list']) ? $info['sku_list'] : [];
  602. $goods[] = [
  603. 'goods_id' => $goodsId,
  604. 'supply_type' => isset($item['supply_type']) ? $item['supply_type'] : 0,
  605. 'spu_sn' => isset($item['spu_sn']) ? $item['spu_sn'] : '',
  606. 'spu_name' => isset($info['spu_name']) ? $info['spu_name'] : '',
  607. 'main_img' => isset($info['main_img']) ? $info['main_img'] : '',
  608. 'detail_img' => isset($info['detail_img']) ? json_encode($info['detail_img'], 256) : '',
  609. 'goods_name' => isset($item['goods_name']) ? $item['goods_name'] : '',
  610. 'brand_name' => isset($info['brand_name']) ? $info['brand_name'] : '',
  611. 'limit_num' => isset($info['limit_num']) ? intval($info['limit_num']) : 0,
  612. 'lowest_num' => isset($info['lowest_num']) ? intval($info['lowest_num']) : 1,
  613. 'cost_price' => isset($info['cost_price']) ? floatval($info['cost_price']) : 0,
  614. 'retail_price' => isset($info['retail_price']) ? floatval($info['retail_price']) : 0,
  615. 'profit' => isset($info['profit']) ? floatval($info['profit']) : 0,
  616. 'sku_list' => $skuList? json_encode($skuList,256):'',
  617. 'sku_total' => isset($info['sku_total']) ? intval($info['sku_total']) : 0,
  618. 'tag' => isset($item['tag']) ? json_encode($item['tag'], 256) : '',
  619. 'status' => isset($info['status']) ? intval($info['status']) : 1,
  620. 'cate_id' => isset($item['cate_id']) ? intval($item['cate_id']) : 0,
  621. 'last_update_at' => isset($info['update_time']) ? $info['update_time'] : (isset($item['time']) && $item['time'] ? $item['time'] : date('Y-m-d H:i:s')),
  622. 'create_time' => time(),
  623. ];
  624. foreach($skuList as $v){
  625. $skus[] = [
  626. 'sku_id'=> isset($v['sku_id'])? $v['sku_id'] : 0,
  627. 'goods_id'=> $goodsId,
  628. 'spu_sn'=> isset($v['spu_sn'])? $v['spu_sn'] : '',
  629. 'sku_sn'=> isset($v['sku_sn'])? $v['sku_sn'] : '',
  630. 'sku_name'=> isset($v['sku_name'])? $v['sku_name'] : '',
  631. 'main_img'=> isset($v['main_img'])? $v['main_img'] : '',
  632. 'status'=> isset($v['status'])? $v['status'] : 1,
  633. 'source_type'=> isset($v['source_type'])? $v['source_type'] : 0,
  634. 'retail_price'=> isset($v['retail_price'])? floatval($v['retail_price']) : 0,
  635. 'plat_price'=> isset($v['plat_price'])? floatval($v['plat_price']) : 0,
  636. 'profit'=> isset($v['profit'])? floatval($v['profit']) : 0,
  637. 'last_update_at'=> isset($v['update_time'])? $v['update_time'] : date('Y-m-d H:i:s'),
  638. 'detail_img'=> isset($v['detail_img'])? json_encode($v['detail_img'],256) : '',
  639. 'attr'=> isset($v['attr'])? json_encode($v['attr'],256) : '',
  640. 'remark'=>'SKU同步创建',
  641. ];
  642. }
  643. $updated++;
  644. } else {
  645. $error++;
  646. }
  647. } else {
  648. $error++;
  649. }
  650. }
  651. unset($item);
  652. } else {
  653. RedisService::set($cacheKey . '_page', 0, rand(300, 600));
  654. }
  655. if ($goods) {
  656. RedisService::set($cacheKey . '_page', $page, rand(300, 600));
  657. RedisService::set($cacheKey, $goods, rand(5, 10));
  658. DB::beginTransaction();
  659. try {
  660. $this->model->insertAll($goods);
  661. if($skus){
  662. GoodsSkuModel::insert($skus);
  663. }
  664. DB::commit();
  665. }catch (\Exception $exception){
  666. DB::rollBack();
  667. }
  668. }
  669. return ['count' => count($goods), 'updated' => $updated, 'errorCount' => $error,'page'=>$page];
  670. }
  671. /**
  672. * 更新商品分类
  673. * @param int $pid 上级ID
  674. * @param int $pageSize
  675. * @param $params 参数
  676. * @return array|false
  677. */
  678. public function updateGoodsCategory($pid=0, $pageSize = 200, $params = [])
  679. {
  680. set_time_limit(0);
  681. $cacheKey = "caches:supply:goods_category_update_{$pid}_{$pageSize}";
  682. if (RedisService::get($cacheKey)) {
  683. $this->error = 1047;
  684. return false;
  685. }
  686. $params = [
  687. 'limit' => $pageSize > 0 ? $pageSize : 50,
  688. 'page' => 1,
  689. 'pid' => $pid, // 上级ID
  690. ];
  691. $categorys = [];
  692. $updated = 0;
  693. $error = 0;
  694. $datas = SupplyService::make()->getApiData('getGoodsCategory', $params);
  695. if ($datas && $datas['data']) {
  696. foreach ($datas['data'] as &$item) {
  697. $cateId = isset($item['id']) ? $item['id'] : 0;
  698. if ($cateId && !$this->checkCategory($cateId)) {
  699. $categorys[] = [
  700. 'cate_id' => $cateId,
  701. 'name' => isset($item['name']) ? $item['name'] : '',
  702. 'pid' => isset($item['pid']) ? intval($item['pid']) : 0,
  703. 'create_time' => time(),
  704. ];
  705. $updated++;
  706. } else {
  707. $error++;
  708. }
  709. }
  710. unset($item);
  711. }
  712. if ($categorys) {
  713. RedisService::set($cacheKey, $categorys, rand(5, 10));
  714. GoodsCategoryModel::insert($categorys);
  715. }
  716. return ['count' => count($categorys), 'updated' => $updated,'pid'=>$pid, 'errorCount' => $error];
  717. }
  718. /**
  719. * 验证
  720. * @param $goodsId
  721. * @return bool
  722. */
  723. public function checkGoods($goodsId)
  724. {
  725. $cacheKey = "caches:goods:check_{$goodsId}";
  726. if (RedisService::get($cacheKey) || RedisService::exists($cacheKey)) {
  727. return true;
  728. }
  729. $data = $this->model->where(['goods_id' => $goodsId, 'mark' => 1])->value('id');
  730. RedisService::set($cacheKey, $data, rand(30, 60));
  731. return $data;
  732. }
  733. /**
  734. * 验证分类
  735. * @param $cateId
  736. * @return bool
  737. */
  738. public function checkCategory($cateId)
  739. {
  740. $cacheKey = "caches:goods:category_check_{$cateId}";
  741. if (RedisService::get($cacheKey) || RedisService::exists($cacheKey)) {
  742. return true;
  743. }
  744. $data = GoodsCategoryModel::where(['cate_id' => $cateId, 'mark' => 1])->value('id');
  745. RedisService::set($cacheKey, $data, rand(30, 60));
  746. return $data;
  747. }
  748. /**
  749. * 接口商品详情
  750. * @param $goodsId 商品ID
  751. * @param int $isReal 是否实时数据,0-是,1-否
  752. * @param int $type 数据类型:0-详情,1-仅SKU数据
  753. * @return array|false|mixed|string
  754. */
  755. public function getApiInfo($goodsId, $isReal = 0, $type = 0)
  756. {
  757. $cacheKey = "caches:goods:detail_{$goodsId}_{$isReal}_{$type}";
  758. $info = RedisService::get($cacheKey);
  759. if (empty($info)) {
  760. $params = [
  761. 'goods_id' => $goodsId,
  762. 'is_real' => $isReal,
  763. 'type' => $type
  764. ];
  765. $info = SupplyService::make()->getApiData('getGoodsDetail', $params);
  766. if ($info) {
  767. RedisService::set($cacheKey, $info, rand(5, 10));
  768. }
  769. }
  770. return $info;
  771. }
  772. public function apiCategory($num)
  773. {
  774. }
  775. }