Stat.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. <?php
  2. /**
  3. * Niushop商城系统 - 团队十年电商经验汇集巨献!
  4. * =========================================================
  5. * Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
  6. * ----------------------------------------------
  7. * 官方网址: https://www.niushop.com
  8. * =========================================================
  9. */
  10. namespace app\model\system;
  11. use app\model\BaseModel;
  12. use app\model\order\OrderCommon;
  13. use app\model\order\OrderRefund;
  14. use app\model\stat\GoodsCartStat;
  15. use app\model\stat\GoodsStat;
  16. use app\model\stat\MemberStat;
  17. use app\model\stat\MemberWithdrawStat;
  18. use app\model\stat\OrderStat;
  19. use app\model\stat\RechargeStat;
  20. use app\model\stat\VisitStat;
  21. use Carbon\Carbon;
  22. use think\facade\Cache;
  23. use think\facade\Db;
  24. use think\facade\Log;
  25. /**
  26. * 统计
  27. * @author Administrator
  28. *
  29. */
  30. class Stat extends BaseModel
  31. {
  32. /**
  33. * 添加店铺统计(按照天统计)
  34. * @param array $data
  35. */
  36. public function addShopStat($data)
  37. {
  38. Log::write('addShopStat' . '-' . date('y-m-d H:i:s', time()) . '-' . json_encode($data));
  39. $site_id = $data[ 'site_id' ];
  40. $carbon = Carbon::now();
  41. $condition = [
  42. 'site_id' => $site_id,
  43. 'year' => $carbon->year,
  44. 'month' => $carbon->month,
  45. 'day' => $carbon->day
  46. ];
  47. $info = model('stat_shop')->getInfo($condition, 'id');
  48. //在这里会整体处理总支出 总收入 总预计收入
  49. $stat_data = $this->getStatData($data);
  50. if (empty($info)) {
  51. $insert_data = [
  52. 'site_id' => $site_id,
  53. 'year' => $carbon->year,
  54. 'month' => $carbon->month,
  55. 'day' => $carbon->day,
  56. 'day_time' => time(),
  57. 'create_time' => time()
  58. ];
  59. $insert_data = array_merge($insert_data, $stat_data);
  60. $res = model('stat_shop')->add(
  61. $insert_data
  62. );
  63. } else {
  64. $update_data = array ();
  65. if (!empty($stat_data)) {
  66. foreach ($stat_data as $k => $v) {
  67. if ($v > 0) {
  68. $update_data[ $k ] = Db::raw($k . '+' . $v);
  69. } else if ($v < 0) {
  70. $update_data[ $k ] = Db::raw($k . '-' . abs($v));
  71. }
  72. }
  73. }
  74. if (!empty($update_data)) {
  75. $res = Db::name('stat_shop')->where($condition)
  76. ->update($update_data);
  77. Log::write('addShopStat' . Db::name('stat_shop')->getLastSql());
  78. Cache::tag("cache_table" . "stat_shop")->clear();
  79. }
  80. }
  81. //增加当天时统计
  82. $this->addShopHourStat($data, $carbon);
  83. return $this->success($res ?? 0);
  84. }
  85. /**
  86. * 增加当日的时统计记录
  87. * @param $data
  88. */
  89. public function addShopHourStat($data, $carbon)
  90. {
  91. $site_id = $data[ 'site_id' ];
  92. // $carbon = Carbon::now();
  93. $condition = [
  94. 'site_id' => $site_id,
  95. 'year' => $carbon->year,
  96. 'month' => $carbon->month,
  97. 'day' => $carbon->day,
  98. 'hour' => $carbon->hour
  99. ];
  100. $info = model('stat_shop_hour')->getInfo($condition, 'id');
  101. //在这里会整体处理总支出 总收入 总预计收入
  102. $stat_data = $this->getStatData($data);
  103. if (empty($info)) {
  104. $insert_data = [
  105. 'site_id' => $site_id,
  106. 'year' => $carbon->year,
  107. 'month' => $carbon->month,
  108. 'day' => $carbon->day,
  109. 'day_time' => time(),
  110. 'create_time' => time(),
  111. 'hour' => $carbon->hour
  112. ];
  113. $insert_data = array_merge($insert_data, $stat_data);
  114. $res = model('stat_shop_hour')->add(
  115. $insert_data
  116. );
  117. } else {
  118. $update_data = array ();
  119. if (!empty($stat_data)) {
  120. foreach ($stat_data as $k => $v) {
  121. if ($v > 0) {
  122. $update_data[ $k ] = Db::raw($k . '+' . $v);
  123. } else if ($v < 0) {
  124. $update_data[ $k ] = Db::raw($k . '-' . abs($v));
  125. }
  126. }
  127. }
  128. if (!empty($update_data)) {
  129. $res = Db::name('stat_shop_hour')->where($condition)
  130. ->update($update_data);
  131. Cache::tag("cache_table" . "stat_shop_hour")->clear();
  132. }
  133. }
  134. return $this->success($res ?? 0);
  135. }
  136. /**
  137. * 整理数据
  138. * @param $data
  139. * @return mixed
  140. */
  141. public function getStatData($data)
  142. {
  143. unset($data[ 'site_id' ]);
  144. $order_pay_money = $data[ 'order_pay_money' ] ?? 0;//订单总额
  145. $member_recharge_total_money = $data[ 'member_recharge_total_money' ] ?? 0;//会员充值总额
  146. $member_level_total_money = $data[ 'member_level_total_money' ] ?? 0;//超级会员卡销售额
  147. $member_giftcard_total_money = $data[ 'member_giftcard_total_money' ] ?? 0;//礼品卡订单总额
  148. $earnings_total_money = $order_pay_money + $member_recharge_total_money + $member_level_total_money + $member_giftcard_total_money;//预计总收入
  149. $order_refund_total_money = $data[ 'refund_total' ] ?? 0;//订单退款总额
  150. $member_withdraw_total_money = $data[ 'member_withdraw_total_money' ] ?? 0;//会员提现总额
  151. $expenditure_total_money = $order_refund_total_money + $member_withdraw_total_money;//总支出
  152. $expected_earnings_total_money = $earnings_total_money - $expenditure_total_money;
  153. Log::write('getStatData' . json_encode([ $earnings_total_money, $expenditure_total_money, $expected_earnings_total_money ]));
  154. $data[ 'earnings_total_money' ] = $earnings_total_money;
  155. $data[ 'expenditure_total_money' ] = $expenditure_total_money;
  156. $data[ 'expected_earnings_total_money' ] = $expected_earnings_total_money;
  157. $data = array_filter($data);
  158. return $data;
  159. }
  160. /**
  161. * 获取店铺统计(按照天查询)
  162. * @param unknown $site_id 0表示平台
  163. * @param unknown $year
  164. * @param unknown $month
  165. * @param unknown $day
  166. */
  167. public function getStatShop($site_id, $year, $month, $day)
  168. {
  169. $condition = [
  170. 'site_id' => $site_id,
  171. 'year' => $year,
  172. 'month' => $month,
  173. 'day' => $day
  174. ];
  175. $info = model('stat_shop')->setIsCache(0)->getInfo($condition, '*');
  176. if (empty($info)) {
  177. $condition[ 'day_time' ] = strtotime(date("{$year}-{$month}-{$day}"));
  178. model('stat_shop')->add($condition);
  179. $info = model('stat_shop')->getInfo($condition, '*');
  180. }
  181. $info[ 'goods_order_count' ] = numberFormat($info[ 'goods_order_count' ]);
  182. return $this->success($info);
  183. }
  184. /**
  185. * 获取店铺统计信息
  186. * @param unknown $site_id
  187. * @param unknown $start_time
  188. */
  189. public function getShopStatSum($site_id, $start_time = 0, $end_time = 0)
  190. {
  191. $condition = [
  192. [ 'site_id', '=', $site_id ]
  193. ];
  194. if (!empty($start_time)) {
  195. $condition[] = [ 'day_time', '>=', $start_time ];
  196. }
  197. if (!empty($end_time)) {
  198. $condition[] = [ 'day_time', '<=', $end_time ];
  199. }
  200. $field = array_map(function($field) {
  201. switch ( $field ) {
  202. case 'earnings_total_money':
  203. return "sum(earnings_total_money) + sum(cashier_billing_money) + sum(cashier_buycard_money) as earnings_total_money";
  204. break;
  205. case 'expenditure_total_money':
  206. // return "sum(expenditure_total_money) + sum(cashier_refund_money) as expenditure_total_money";
  207. return "sum(expenditure_total_money) as expenditure_total_money";
  208. break;
  209. case 'cashier_billing_money':
  210. return "sum(cashier_billing_money) + sum(cashier_buycard_money) as cashier_order_pay_money";
  211. break;
  212. case 'refund_total':
  213. return "sum(refund_total) + sum(cashier_refund_money) as refund_total";
  214. break;
  215. case 'expected_earnings_total_money':
  216. // return "sum(expected_earnings_total_money) + sum(cashier_billing_money) + sum(cashier_buycard_money) - sum(cashier_refund_money) as expected_earnings_total_money";
  217. return "sum(earnings_total_money) + sum(cashier_billing_money) + sum(cashier_buycard_money) - sum(expenditure_total_money) as expected_earnings_total_money";
  218. break;
  219. case 'order_pay_count':
  220. return "sum(order_pay_count) + sum(cashier_billing_count) + sum(cashier_buycard_count) as order_pay_count";
  221. break;
  222. default:
  223. return "sum($field) as $field";
  224. }
  225. }, $this->getStatField());
  226. // outFileLog($condition,'stat','$condition');
  227. // outFileLog($field,'stat','$field');
  228. $info = model('stat_shop')->getInfo($condition, $field);
  229. if (isset($info[ 'goods_order_count' ])) {
  230. $info[ 'goods_order_count' ] = numberFormat($info[ 'goods_order_count' ]);
  231. }
  232. return $this->success($info);
  233. }
  234. /**
  235. * 获取店铺统计列表
  236. * @param unknown $site_id
  237. * @param unknown $start_time
  238. */
  239. public function getShopStatList($site_id, $start_time, $end_time)
  240. {
  241. $condition = [
  242. [ 'site_id', '=', $site_id ],
  243. [ 'day_time', '>=', $start_time ],
  244. [ 'day_time', '<=', $end_time ],
  245. ];
  246. $list = model('stat_shop')->getList($condition, $this->handleStatField());
  247. foreach ($list as $k => $v) {
  248. $list[ $k ][ 'goods_order_count' ] = numberFormat($list[ $k ][ 'goods_order_count' ]);
  249. }
  250. return $this->success($list);
  251. }
  252. /**
  253. * 处理查询字段
  254. */
  255. private function handleStatField()
  256. {
  257. $fields = Db::name('stat_shop')->getTableFields('');
  258. foreach ($fields as $k => $field) {
  259. switch ( $field ) {
  260. case 'earnings_total_money':
  261. $fields[ $k ] = "earnings_total_money + cashier_billing_money + cashier_buycard_money as earnings_total_money";
  262. break;
  263. case 'expenditure_total_money':
  264. $fields[ $k ] = "expenditure_total_money + cashier_refund_money as expenditure_total_money";
  265. break;
  266. case 'cashier_billing_money':
  267. $fields[ $k ] = "cashier_billing_money + cashier_buycard_money as cashier_order_pay_money";
  268. break;
  269. case 'refund_total':
  270. $fields[ $k ] = "refund_total + cashier_refund_money as refund_total";
  271. break;
  272. case 'expected_earnings_total_money':
  273. $fields[ $k ] = "expected_earnings_total_money + cashier_billing_money + cashier_buycard_money - cashier_refund_money as expected_earnings_total_money";
  274. break;
  275. case 'order_pay_count':
  276. $fields[ $k ] = "order_pay_count + cashier_billing_count + cashier_buycard_count as order_pay_count";
  277. break;
  278. }
  279. }
  280. return implode(',', $fields);
  281. }
  282. /**
  283. * 获取小时统计数据
  284. * @param $site_id
  285. * @param $year
  286. * @param $month
  287. * @param $day
  288. * @return array
  289. */
  290. public function getShopStatHourList($site_id, $year, $month, $day)
  291. {
  292. $condition = [
  293. [ 'site_id', '=', $site_id ],
  294. [ 'year', '=', $year ],
  295. [ 'month', '=', $month ],
  296. [ 'day', '=', $day ],
  297. ];
  298. $list = model('stat_shop_hour')->getList($condition, $this->handleStatHourField(), 'id desc');
  299. foreach ($list as $k => $v) {
  300. $list[ $k ][ 'goods_order_count' ] = numberFormat($list[ $k ][ 'goods_order_count' ]);
  301. }
  302. return $this->success($list);
  303. }
  304. /**
  305. * 处理查询字段
  306. */
  307. private function handleStatHourField()
  308. {
  309. $fields = Db::name('stat_shop_hour')->getTableFields('');
  310. foreach ($fields as $k => $field) {
  311. switch ( $field ) {
  312. case 'earnings_total_money':
  313. $fields[ $k ] = "earnings_total_money + cashier_billing_money + cashier_buycard_money as earnings_total_money";
  314. break;
  315. case 'expenditure_total_money':
  316. $fields[ $k ] = "expenditure_total_money + cashier_refund_money as expenditure_total_money";
  317. break;
  318. case 'refund_total':
  319. $fields[ $k ] = "refund_total + cashier_refund_money as refund_total";
  320. break;
  321. case 'expected_earnings_total_money':
  322. $fields[ $k ] = "expected_earnings_total_money + cashier_billing_money + cashier_buycard_money - cashier_refund_money as expected_earnings_total_money";
  323. break;
  324. case 'order_pay_count':
  325. $fields[ $k ] = "order_pay_count + cashier_billing_count + cashier_buycard_count as order_pay_count";
  326. break;
  327. }
  328. }
  329. $fields[] = 'cashier_billing_money + cashier_buycard_money as cashier_order_pay_money';
  330. return implode(',', $fields);
  331. }
  332. /**
  333. * 获取天统计表统计字段
  334. * @return array
  335. */
  336. public function getStatField()
  337. {
  338. $fields = Db::name('stat_shop')->getTableFields('');
  339. $fields = array_values(array_diff($fields, [ 'id', 'site_id', 'year', 'month', 'day', 'day_time' ]));
  340. return $fields;
  341. }
  342. /**
  343. * 获取时统计表统计字段
  344. * @return array
  345. */
  346. public function getStatHourField()
  347. {
  348. $fields = Db::name('stat_shop_hour')->getTableFields('');
  349. $fields = array_values(array_diff($fields, [ 'id', 'site_id', 'year', 'month', 'day', 'hour', 'day_time' ]));
  350. return $fields;
  351. }
  352. /**
  353. * 获取商品销量排行榜
  354. * @param $site_id
  355. * @param string $start_time
  356. * @param string $end_time
  357. */
  358. public function getGoodsSaleNumRankingList($site_id, $start_time, $end_time, $page_index, $page_size)
  359. {
  360. $condition = [
  361. [ 'o.site_id', '=', $site_id ],
  362. [ 'o.pay_status', '=', 1 ],
  363. [ 'g.is_delete', '=', 0 ],
  364. [ 'o.order_status', '<>', OrderCommon::ORDER_CLOSE ],
  365. [ 'og.refund_status', '<>', OrderRefund::REFUND_COMPLETE ]
  366. ];
  367. if (!empty($start_time) && empty($end_time)) {
  368. // $condition[] = [ 'o.create_time', '>=', date_to_time($start_time) ];
  369. $condition[] = [ 'o.create_time', '>=', $start_time ];
  370. } elseif (empty($start_time) && !empty($end_time)) {
  371. $condition[] = [ 'o.create_time', '<=', $end_time ];
  372. } elseif (!empty($start_time) && !empty($end_time)) {
  373. $condition[] = [ 'o.create_time', 'between', [ $start_time, $end_time ] ];
  374. }
  375. $join = [
  376. [ 'order o', 'og.order_id = o.order_id', 'left' ],
  377. [ 'goods g', 'og.goods_id = g.goods_id', 'right' ]
  378. ];
  379. $list = model('order_goods')->pageList($condition, 'og.goods_id,g.goods_name,g.goods_state,SUM(FLOOR(og.num)) AS sale_num', 'sale_num desc', $page_index, $page_size, 'og', $join, 'og.goods_id');
  380. // foreach($list as &$v){
  381. // $v['sale_num'] = round($v['sale_num'],0);
  382. // }
  383. return $this->success($list);
  384. }
  385. /**
  386. * 获取商品销量排行榜
  387. * @param $site_id
  388. * @param string $start_time
  389. * @param string $end_time
  390. * @param $page_index
  391. * @param $page_size
  392. * @return array
  393. */
  394. public function getGoodsSaleMoneyRankingList($site_id, $start_time, $end_time, $page_index, $page_size)
  395. {
  396. $condition = [
  397. [ 'o.site_id', '=', $site_id ],
  398. [ 'o.pay_status', '=', 1 ],
  399. [ 'g.is_delete', '=', 0 ],
  400. [ 'o.order_status', '<>', OrderCommon::ORDER_CLOSE ],
  401. [ 'og.refund_status', '<>', OrderRefund::REFUND_COMPLETE ]
  402. ];
  403. if (!empty($start_time) && empty($end_time)) {
  404. $condition[] = [ 'o.create_time', '>=', $start_time ];
  405. } elseif (empty($start_time) && !empty($end_time)) {
  406. $condition[] = [ 'o.create_time', '<=', $end_time ];
  407. } elseif (!empty($start_time) && !empty($end_time)) {
  408. $condition[] = [ 'o.create_time', 'between', [ $start_time, $end_time ] ];
  409. }
  410. $join = [
  411. [ 'order o', 'og.order_id = o.order_id', 'left' ],
  412. [ 'goods g', 'og.goods_id = g.goods_id', 'right' ]
  413. ];
  414. $list = model('order_goods')->pageList($condition, 'og.goods_id,g.goods_name,g.goods_state,SUM(o.order_money) AS order_money', 'order_money desc', $page_index, $page_size, 'og', $join, 'og.goods_id');
  415. return $this->success($list);
  416. }
  417. public function switchStat($params)
  418. {
  419. $type = $params[ 'type' ];
  420. $temp_params = $params[ 'data' ];
  421. $result = event('AddStat', $params, true);
  422. if (empty($result)) {
  423. switch ( $type ) {
  424. case 'order_create'://订单支付
  425. $order_stat_model = new OrderStat();
  426. $result = $order_stat_model->addOrderCreateStat($temp_params);
  427. break;
  428. case 'order_pay'://订单支付
  429. $order_stat_model = new OrderStat();
  430. $result = $order_stat_model->addOrderPayStat($temp_params);
  431. break;
  432. case 'order_refund'://退款
  433. $order_stat_model = new OrderStat();
  434. $result = $order_stat_model->addOrderRefundStat($temp_params);
  435. break;
  436. case 'add_goods'://添加商品
  437. $goods_stat_model = new GoodsStat();
  438. $result = $goods_stat_model->addGoodsStat($temp_params);
  439. break;
  440. case 'collect_goods':
  441. $goods_stat_model = new GoodsStat();
  442. $result = $goods_stat_model->addGoodsCollectStat($temp_params);
  443. break;
  444. case 'recharge':
  445. $recharge_model = new RechargeStat();
  446. $result = $recharge_model->addRechargeStat($temp_params);
  447. break;
  448. case 'visit':
  449. $visit_model = new VisitStat();
  450. $result = $visit_model->addVisitStat($temp_params);
  451. break;
  452. case 'member_withdraw':
  453. $withdraw_model = new MemberWithdrawStat();
  454. $result = $withdraw_model->addMemberWithdrawStat($temp_params);
  455. break;
  456. case 'add_member':
  457. $member_model = new MemberStat();
  458. $result = $member_model->addMemberStat($temp_params);
  459. break;
  460. case 'goods_cart'://购物车加购
  461. $goods_cart_stat_model = new GoodsCartStat();
  462. $result = $goods_cart_stat_model->addGoodsCartStat($temp_params);
  463. break;
  464. case 'goods_visit':
  465. $goods_stat_model = new GoodsStat();
  466. $result = $goods_stat_model->addGoodsVisit($temp_params);
  467. break;
  468. case 'goods_on'://上下架
  469. $goods_stat_model = new GoodsStat();
  470. $result = $goods_stat_model->addGoodsOnStat($temp_params);
  471. break;
  472. }
  473. }
  474. return $result;
  475. }
  476. }