date.php 62 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353
  1. <?php
  2. /**
  3. * 任务计划 - 天执行的任务
  4. *
  5. *
  6. *
  7. *
  8. */
  9. use bonus\account;
  10. defined('InShopNC') or exit('Access Invalid!');
  11. require_once(BASE_ROOT_PATH . '/helper/notify_helper.php');
  12. require_once(BASE_ROOT_PATH . '/helper/stat_helper.php');
  13. require_once(BASE_ROOT_PATH . '/helper/bonus_helper.php');
  14. require_once(BASE_DATA_PATH . '/mobile/omsid.php');
  15. class dateControl extends BaseCronControl
  16. {
  17. /**
  18. * 该文件中所有任务执行频率,默认1天,单位:秒
  19. * @var int
  20. */
  21. const EXE_TIMES = 86400;
  22. /**
  23. * 优惠券即将到期提醒时间,单位:天
  24. * @var int
  25. */
  26. const VOUCHER_INTERVAL = 5;
  27. /**
  28. * 兑换码即将到期提醒时间,单位:天
  29. * @var int
  30. */
  31. const VR_CODE_INTERVAL = 5;
  32. /**
  33. * 订单结束后可评论时间,15天,60*60*24*15
  34. * @var int
  35. */
  36. const ORDER_EVALUATE_TIME = 1296000;
  37. /**
  38. * 每次到货通知消息数量
  39. * @var int
  40. */
  41. const ARRIVAL_NOTICE_NUM = 100;
  42. /**
  43. * 默认方法
  44. */
  45. public function indexOp()
  46. {
  47. //更新订单商品佣金值
  48. $this->_order_commis_rate_update();
  49. //订单超期后不允许评价
  50. $this->_order_eval_expire_update();
  51. //增加会员积分和经验值
  52. $this->_add_points();
  53. //自提点中,已经关闭的订单删除
  54. $this->_order_delivery_cancel_del();
  55. //更新订单扩展表收货人所在省份ID
  56. $this->_order_reciver_provinceid_update();
  57. //更新退款申请超时处理
  58. Model('trade')->editRefundConfirm();
  59. //代金券即将过期提醒
  60. $this->_voucher_will_expire();
  61. //虚拟兑换码即将过期提醒
  62. $this->_vr_code_will_expire();
  63. //更新商品访问量
  64. $this->_goods_click_update();
  65. //更新商品促销到期状态
  66. $this->_goods_promotion_state_update();
  67. //更新浏览量
  68. $this->_goods_browse_update();
  69. //缓存订单及订单商品相关数据
  70. $this->_order_goods_cache();
  71. //会员相关数据统计
  72. $this->_member_stat();
  73. bonus\activity_bonus::gen_daily_bonus();
  74. $this->_stat_daily();
  75. }
  76. //三点开始统计日志服务,这个会在APPServer 上执行,切记不可放后台任务
  77. public function third_clockOp()
  78. {
  79. stat_helper::onDaiyBehavior(time() - 86400);
  80. }
  81. public function nine_clockOp()
  82. {
  83. //订单自动完成
  84. $this->_order_auto_complete();
  85. }
  86. public function ten_clockOp()
  87. {
  88. //已领取但未使用的红包过期通知
  89. $this->bonus_expire_notify();
  90. //商品到货通知提醒
  91. $this->_arrival_notice();
  92. //F码过期提醒
  93. $this->fcode_expire_notify();
  94. }
  95. public function testOp()
  96. {
  97. $this->fcode_expire_notify();
  98. }
  99. public function omsidOp()
  100. {
  101. $serials = omsid();
  102. $mod_goods = Model('goods');
  103. foreach ($serials as $goods_serial => $omsid)
  104. {
  105. $ret = $mod_goods->where(['goods_serial' => $goods_serial])->update(['oms_id' => $omsid]);
  106. if($ret == false) {
  107. Log::record("cannot update goods_serial = {$goods_serial}",Log::ERR);
  108. }
  109. }
  110. }
  111. public function init_relationOp()
  112. {
  113. $i = 0;
  114. while (true)
  115. {
  116. $start = $i * 1000;
  117. $items = Model()->table('member_relation')->field('member_id,invited_user')->order('member_mobile asc')->limit("{$start},1000")->select();
  118. if(empty($items)) {
  119. return;
  120. }
  121. $i++;
  122. foreach ($items as $item)
  123. {
  124. $member_id = intval($item['member_id']);
  125. if($member_id <= 0) continue;
  126. $invitees = unserialize($item['invited_user']);
  127. if(empty($invitees)) continue;
  128. foreach ($invitees as $userid) {
  129. $this->update_invitee($member_id,$userid);
  130. }
  131. }
  132. }
  133. }
  134. public function skupriceOp()
  135. {
  136. $this->add_sku();
  137. }
  138. private function update_invitee($invitorid,$userid)
  139. {
  140. $mod = Model('member');
  141. $invitee = $mod->where(['member_id' => $userid])->find();
  142. if(empty($invitee)) return false;
  143. $inviter_id = intval($invitee['inviter_id']);
  144. if($inviter_id > 0) return false;
  145. $ret = $mod->editMember(['member_id' => $userid],['inviter_id' => $invitorid]);
  146. if($ret == false) {
  147. Log::record(__METHOD__ . " update {$userid} invitor id err",Log::ERR);
  148. }
  149. else {
  150. Log::record(__METHOD__ . " update {$userid} invitor id success",Log::DEBUG);
  151. }
  152. }
  153. public function init_inviterOp()
  154. {
  155. $path = BASE_DATA_PATH . "/mobile/invitors.txt";
  156. $file = fopen($path,'r');
  157. if(!is_resource($file)) {
  158. Log::record(__METHOD__ . " file is not exist",Log::ERR);
  159. return false;
  160. }
  161. while(!feof($file)) {
  162. $line = fgets($file);
  163. $kv = preg_split('/,/',$line);
  164. if(count($kv) != 2) continue;
  165. $key = trim($kv[0]);
  166. $val = trim($kv[1]);
  167. $this->update_invitor($key,$val);
  168. }
  169. fclose($file);
  170. }
  171. private function update_invitor($invitor,$user)
  172. {
  173. if(empty($invitor)) return false;
  174. $mod = Model('member');
  175. $invitor_info = $mod->where(['member_mobile' => $invitor])->find();
  176. if(empty($invitor_info)) return false;
  177. $member_id = intval($invitor_info['member_id']);
  178. $user_info = $mod->where(['member_mobile' => $user])->find();
  179. if(empty($user_info)) return false;
  180. $inviter_id = intval($user_info['inviter_id']);
  181. if($inviter_id > 0) return false;
  182. $ret = $mod->editMember(['member_mobile' => $user],['inviter_id' => $member_id]);
  183. if($ret == false) {
  184. Log::record(__METHOD__ . " update {$user} invitor id err",Log::ERR);
  185. }
  186. else {
  187. Log::record(__METHOD__ . " update {$user} invitor id success",Log::DEBUG);
  188. }
  189. }
  190. public function available_bonusOp()
  191. {
  192. $mod_member = Model('member');
  193. $i = 0;
  194. while (true)
  195. {
  196. $start = $i * 1000;
  197. $items = Model()->table('member')->field('*')->order('member_id asc')->limit("{$start},1000")->select();
  198. if(empty($items)) {
  199. return;
  200. }
  201. $i++;
  202. foreach ($items as $item)
  203. {
  204. $member_id = $item['member_id'];
  205. $pred = new account($member_id);
  206. $amount = $pred->total_bonus();
  207. $mod_member->editMember(['member_id' => $member_id],['available_bonus' => $amount]);
  208. }
  209. }
  210. }
  211. public function _stat_daily()
  212. {
  213. $cur_date = new DateTime();
  214. $inter = new DateInterval('P1D');
  215. $cur_date->sub($inter);
  216. stat_helper::onDaiyMember($cur_date->getTimestamp());
  217. stat_helper::onDaiyBonus($cur_date->getTimestamp());
  218. stat_helper::onDaiyOrder($cur_date->getTimestamp());
  219. stat_helper::onDailySelfDiscount($cur_date->getTimestamp());
  220. }
  221. public function init_member_bonusOp()
  222. {
  223. stat_helper::onInitSelfDiscount();
  224. }
  225. /**
  226. * 订单自动完成
  227. */
  228. private function _order_auto_complete()
  229. {
  230. //虚拟订单过使用期自动完成
  231. $_break = false;
  232. $model_order = Model('vr_order');
  233. $logic_order = Logic('vr_order');
  234. $condition = array();
  235. $condition['order_state'] = ORDER_STATE_PAY;
  236. $condition['vr_indate'] = array('lt',time());
  237. //分批,每批处理100个订单,最多处理5W个订单
  238. for ($i = 0; $i < 500; $i++)
  239. {
  240. if ($_break) {
  241. break;
  242. }
  243. $order_list = $model_order->getOrderList($condition, '', 'order_id,order_sn', 'vr_indate asc', 100);
  244. if (empty($order_list)) break;
  245. foreach ($order_list as $order_info) {
  246. $result = $logic_order->changeOrderStateSuccess($order_info['order_id']);
  247. if (!$result['state']) {
  248. $this->log('虚拟订单过使用期自动完成失败SN:'.$order_info['order_sn']); $_break = true; break;
  249. }
  250. }
  251. }
  252. //实物订单发货后,超期自动收货完成
  253. $_break = false;
  254. $model_order = Model('order');
  255. $logic_order = Logic('order');
  256. $condition = array();
  257. $condition['order_state'] = ORDER_STATE_SEND;
  258. $condition['lock_state'] = 0;
  259. $condition['delay_time'] = array('lt',time() - ORDER_AUTO_RECEIVE_DAY * 86400);
  260. //分批,每批处理100个订单,最多处理5W个订单
  261. for ($i = 0; $i < 500; $i++){
  262. if ($_break) {
  263. break;
  264. }
  265. $order_list = $model_order->getOrderList($condition, '', '*', 'delay_time asc', 100);
  266. if (empty($order_list)) break;
  267. foreach ($order_list as $order_info) {
  268. $result = $logic_order->changeOrderStateReceive($order_info,'system','系统','超期未收货系统自动完成订单');
  269. if (!$result['state']) {
  270. $this->log('实物订单超期未收货自动完成订单失败SN:'.$order_info['order_sn']); $_break = true; break;
  271. }
  272. }
  273. }
  274. }
  275. /**
  276. * 自提订单中,已经关闭订单的,删除
  277. */
  278. private function _order_delivery_cancel_del() {
  279. $model_delivery = Model('delivery_order');
  280. $model_order = Model('order');
  281. for($i = 0; $i < 10; $i++) {
  282. $delivery_list = $model_delivery->getDeliveryOrderDefaultList(array(), '*', 0, 'order_id asc', 100);
  283. if (!empty($delivery_list)) {
  284. $order_ids = array();
  285. foreach ($delivery_list as $k => $v) {
  286. $order_ids[] = $v['order_id'];
  287. }
  288. $condition = array();
  289. $condition['order_state'] = ORDER_STATE_CANCEL;
  290. $condition['order_id'] = array('in',$order_ids);
  291. $order_list = $model_order->getOrderList($condition,'','order_id');
  292. if (!empty($order_list)) {
  293. $order_ids = array();
  294. foreach ($order_list as $k => $v) {
  295. $order_ids[] = $v['order_id'];
  296. }
  297. $del = $model_delivery->delDeliveryOrder(array('order_id'=>array('in',$order_ids)));
  298. if (!del) {
  299. $this->log('删除自提点订单失败');
  300. }
  301. } else {
  302. break;
  303. }
  304. } else {
  305. break;
  306. }
  307. }
  308. }
  309. /**
  310. * 更新订单扩展表中收货人所在省份ID
  311. */
  312. private function _order_reciver_provinceid_update()
  313. {
  314. $model_order = Model('order');
  315. $model_area = Model('area');
  316. //每次最多处理5W个订单
  317. $condition = array();
  318. $condition['reciver_province_id'] = 0;
  319. $condition['reciver_city_id'] = array('neq',0);
  320. for($i = 0; $i < 500; $i++)
  321. {
  322. $order_list = $model_order->getOrderCommonList($condition, 'reciver_city_id','order_id desc', 100);
  323. if (!empty($order_list))
  324. {
  325. $city_ids = array();
  326. foreach ($order_list as $v) {
  327. if (!in_array($v['reciver_city_id'],$city_ids)) {
  328. $city_ids[] = $v['reciver_city_id'];
  329. }
  330. }
  331. $area_list = $model_area->getAreaList(array('area_id'=>array('in',$city_ids)),'area_parent_id,area_id');
  332. if (!empty($area_list))
  333. {
  334. foreach ($area_list as $v) {
  335. $update = $model_order->editOrderCommon(array('reciver_province_id'=>$v['area_parent_id']),array('reciver_city_id'=>$v['area_id']));
  336. if (!$update) {
  337. $this->log('更新订单扩展表中收货人所在省份ID失败');break;
  338. }
  339. }
  340. }
  341. }
  342. else {
  343. break;
  344. }
  345. }
  346. }
  347. /**
  348. * 增加会员积分和经验值
  349. */
  350. private function _add_points()
  351. {
  352. return;
  353. $model_points = Model('points');
  354. $model_exppoints = Model('exppoints');
  355. //24小时之内登录的会员送积分和经验值,每次最多处理5W个会员
  356. $model_member = Model('member');
  357. $condition = array();
  358. $condition['member_login_time'] = array('gt',time() - self::EXE_TIMES);
  359. for($i = 0; $i < 50000; $i=$i+100)
  360. {
  361. $member_list = $model_member->getMemberList($condition, 'member_name,member_id',0,'', "{$i},100");
  362. if (!empty($member_list))
  363. {
  364. foreach ($member_list as $member_info)
  365. {
  366. if (C('points_isuse')) {
  367. $model_points->savePointsLog('login',array('pl_memberid'=>$member_info['member_id'],'pl_membername'=>$member_info['member_name']),true);
  368. }
  369. $model_exppoints->saveExppointsLog('login',array('exp_memberid'=>$member_info['member_id'],'exp_membername'=>$member_info['member_name']),true);
  370. }
  371. }
  372. else {
  373. break;
  374. }
  375. }
  376. //24小时之内注册的会员送积分,每次最多处理5W个会员
  377. if (C('points_isuse'))
  378. {
  379. $condition = array();
  380. $condition['member_time'] = array('gt',time() - self::EXE_TIMES);
  381. for($i = 0; $i < 50000; $i=$i+100)
  382. {
  383. $member_list = $model_member->getMemberList($condition, 'member_name,member_id',0,'member_id desc', "{$i},100");
  384. if (!empty($member_list)) {
  385. foreach ($member_list as $member_info) {
  386. $model_points->savePointsLog('regist',array('pl_memberid'=>$member_info['member_id'],'pl_membername'=>$member_info['member_name']),true);
  387. }
  388. } else {
  389. break;
  390. }
  391. }
  392. }
  393. //24小时之内完成了实物订单送积分和经验值,每次最多处理5W个订单
  394. $model_order = Model('order');
  395. $condition = array();
  396. $condition['finnshed_time'] = array('gt',time() - self::EXE_TIMES);
  397. for($i = 0; $i < 50000; $i=$i+100)
  398. {
  399. $order_list = $model_order->getOrderList($condition,'','buyer_name,buyer_id,order_amount,order_sn,order_id','', "{$i},100");
  400. if (!empty($order_list))
  401. {
  402. foreach ($order_list as $order_info) {
  403. if (C('points_isuse')) {
  404. $model_points->savePointsLog('order',array('pl_memberid'=>$order_info['buyer_id'],'pl_membername'=>$order_info['buyer_name'],'orderprice'=>$order_info['order_amount'],'order_sn'=>$order_info['order_sn'],'order_id'=>$order_info['order_id']),true);
  405. }
  406. $model_exppoints->saveExppointsLog('order',array('exp_memberid'=>$order_info['buyer_id'],'exp_membername'=>$order_info['buyer_name'],'orderprice'=>$order_info['order_amount'],'order_sn'=>$order_info['order_sn'],'order_id'=>$order_info['order_id']),true);
  407. }
  408. } else {
  409. break;
  410. }
  411. }
  412. //24小时之内完成了实物订单送积分和经验值,每次最多处理5W个订单
  413. $model_order = Model('vr_order');
  414. $condition = array();
  415. $condition['finnshed_time'] = array('gt',time() - self::EXE_TIMES);
  416. for($i = 0; $i < 50000; $i=$i+100)
  417. {
  418. $order_list = $model_order->getOrderList($condition,'','buyer_name,buyer_id,order_amount,order_sn,order_id','', "{$i},100");
  419. if (!empty($order_list))
  420. {
  421. foreach ($order_list as $order_info) {
  422. if (C('points_isuse')) {
  423. $model_points->savePointsLog('order',array('pl_memberid'=>$order_info['buyer_id'],'pl_membername'=>$order_info['buyer_name'],'orderprice'=>$order_info['order_amount'],'order_sn'=>$order_info['order_sn'],'order_id'=>$order_info['order_id']),true);
  424. }
  425. $model_exppoints->saveExppointsLog('order',array('exp_memberid'=>$order_info['buyer_id'],'exp_membername'=>$order_info['buyer_name'],'orderprice'=>$order_info['order_amount'],'order_sn'=>$order_info['order_sn'],'order_id'=>$order_info['order_id']),true);
  426. }
  427. } else {
  428. break;
  429. }
  430. }
  431. }
  432. /**
  433. * 代金券即将过期提醒
  434. */
  435. private function _voucher_will_expire()
  436. {
  437. $time_start = mktime(0, 0, 0, date("m") , date("d")+self::VOUCHER_INTERVAL, date("Y"));
  438. $time_stop = $time_start + self::EXE_TIMES - 1;
  439. $where = array();
  440. $where['voucher_end_date'] = array(array('egt', $time_start), array('elt', $time_stop), 'and');
  441. $list = Model('voucher')->getVoucherUnusedList($where);
  442. if (!empty($list))
  443. {
  444. foreach ($list as $val) {
  445. $param = array();
  446. $param['code'] = 'voucher_will_expire';
  447. $param['member_id'] = $val['voucher_owner_id'];
  448. $param['param'] = array(
  449. 'indate' => date('Y-m-d H:i:s', $val['voucher_end_date']),
  450. 'voucher_url' => urlShop('member_voucher', 'index')
  451. );
  452. QueueClient::push('sendMemberMsg', $param);
  453. }
  454. }
  455. }
  456. /**
  457. * 虚拟兑换码即将过期提醒
  458. */
  459. private function _vr_code_will_expire() {
  460. $time_start = mktime(0, 0, 0, date("m") , date("d")+self::VR_CODE_INTERVAL, date("Y"));
  461. $time_stop = $time_start + self::EXE_TIMES - 1;
  462. $where = array();
  463. $where['vr_indate'] = array(array('egt', $time_start), array('elt', $time_stop), 'and');
  464. $list = Model('vr_order')->getCodeUnusedList($where);
  465. if (!empty($list)) {
  466. foreach ($list as $val) {
  467. $param = array();
  468. $param['code'] = 'vr_code_will_expire';
  469. $param['member_id'] = $val['buyer_id'];
  470. $param['param'] = array(
  471. 'indate' => date('Y-m-d H:i:s', $val['vr_indate']),
  472. 'vr_order_url' => urlShop('member_vr_order', 'index')
  473. );
  474. QueueClient::push('sendMemberMsg', $param);
  475. }
  476. }
  477. }
  478. /**
  479. * 订单超期后不允许评价
  480. */
  481. private function _order_eval_expire_update() {
  482. //实物订单超期未评价自动更新状态,每次最多更新1000个订单
  483. $model_order = Model('order');
  484. $condition = array();
  485. $condition['order_state'] = ORDER_STATE_SUCCESS;
  486. $condition['evaluation_state'] = 0;
  487. $condition['finnshed_time'] = array('lt',time() - self::ORDER_EVALUATE_TIME);
  488. $update = array();
  489. $update['evaluation_state'] = 2;
  490. $update = $model_order->editOrder($update,$condition,1000);
  491. if (!$update) {
  492. $this->log('更新实物订单超期不能评价失败'); return;
  493. }
  494. //虚拟订单超期未评价自动更新状态,每次最多更新1000个订单
  495. $model_order = Model('vr_order');
  496. $condition = array();
  497. $condition['order_state'] = ORDER_STATE_SUCCESS;
  498. $condition['evaluation_state'] = 0;
  499. $condition['use_state'] = 1;
  500. $condition['finnshed_time'] = array('lt',time() - self::ORDER_EVALUATE_TIME);
  501. $update = array();
  502. $update['evaluation_state'] = 2;
  503. $update = $model_order->editOrder($update,$condition,1000);
  504. if (!$update) {
  505. $this->log('更新虚拟订单超期不能评价失败'); return;
  506. }
  507. }
  508. /**
  509. * 更新商品访问量(redis)
  510. */
  511. private function _goods_click_update() {
  512. $data = rcache('updateRedisDate', 'goodsClick');
  513. foreach ($data as $key => $val) {
  514. Model('goods')->editGoodsById(array('goods_click' => array('exp', 'goods_click +'.$val)), $key);
  515. }
  516. dcache('updateRedisDate', 'goodsClick');
  517. }
  518. /**
  519. * 更新商品促销到期状态(目前只有满即送)
  520. */
  521. private function _goods_promotion_state_update() {
  522. //满即送过期
  523. Model('p_mansong')->editExpireMansong();
  524. }
  525. /**
  526. * 商品到货通知提醒
  527. */
  528. private function _arrival_notice()
  529. {
  530. $strat_time = strtotime("-60 day"); // 只通知最近60天的记录
  531. $mod_anotice = Model('arrival_notice');
  532. // 删除60天之前的记录
  533. $mod_anotice->delArrivalNotice(array('an_addtime' => array('lt', $strat_time), 'an_type' => 1));
  534. $count = $mod_anotice->getArrivalNoticeCount(array());
  535. $times = ceil($count/self::ARRIVAL_NOTICE_NUM);
  536. if ($times == 0) return false;
  537. for ($i = 0; $i <= $times; $i++)
  538. {
  539. $notice_list = $mod_anotice->getArrivalNoticeList(array(), '*', $i.','.self::ARRIVAL_NOTICE_NUM);
  540. if (empty($notice_list)) continue;
  541. // 查询商品是否已经上架
  542. $goodsid_array = array();
  543. foreach ($notice_list as $val) {
  544. $goodsid_array[] = $val['goods_id'];
  545. }
  546. $goodsid_array = array_unique($goodsid_array);
  547. $goods_list = Model('goods')->getGoodsOnlineList(array('goods_id' => array('in', $goodsid_array), 'goods_storage' => array('gt', 0)), 'goods_id');
  548. if (empty($goods_list)) continue;
  549. // 需要通知到货的商品
  550. $goodsid_array = array();
  551. foreach ($goods_list as $val) {
  552. $goodsid_array[] = $val['goods_id'];
  553. }
  554. // 根据商品id重新查询需要通知的列表
  555. $notice_list = $mod_anotice->getArrivalNoticeList(array('goods_id' => array('in', $goodsid_array)), '*');
  556. if (empty($notice_list)) continue;
  557. foreach ($notice_list as $val)
  558. {
  559. $param = array();
  560. $param['code'] = 'arrival_notice';
  561. $param['member_id'] = $val['member_id'];
  562. $param['param'] = array(
  563. 'goods_name' => $val['goods_name'],
  564. 'goods_url' => urlShop('goods', 'index', array('goods_id' => $val['goods_id']))
  565. );
  566. $param['number'] = array('mobile' => $val['an_mobile'], 'email' => $val['an_email']);
  567. $param['sms_param'] = array('goods_name' => $val['goods_name']);
  568. QueueClient::push('sendMemberMsg', $param);
  569. }
  570. // 清楚发送成功的数据
  571. $mod_anotice->delArrivalNotice(array('goods_id' => array('in', $goodsid_array)));
  572. }
  573. }
  574. /**
  575. * 将缓存中的浏览记录存入数据库中,并删除30天前的浏览历史
  576. */
  577. private function _goods_browse_update(){
  578. $model = Model('goods_browse');
  579. //将cache中的记录存入数据库
  580. if (C('cache_open')){//如果浏览记录已经存入了缓存中,则将其整理到数据库中
  581. //上次更新缓存的时间
  582. $latest_record = $model->getGoodsbrowseOne(array(),'','browsetime desc');
  583. $starttime = ($t = intval($latest_record['browsetime']))?$t:0;
  584. $monthago = strtotime(date('Y-m-d',time())) - 86400*30;
  585. $model_member = Model('member');
  586. //查询会员信息总条数
  587. $countnum = $model_member->getMemberCount(array());
  588. $eachnum = 100;
  589. for ($i=0; $i<$countnum; $i+=$eachnum){//每次查询100条
  590. $member_list = $model_member->getMemberList(array(), '*', 0, 'member_id asc', "$i,$eachnum");
  591. foreach ((array)$member_list as $k=>$v){
  592. $insert_arr = array();
  593. $goodsid_arr = array();
  594. //生成缓存的键值
  595. $hash_key = $v['member_id'];
  596. $browse_goodsid = rcache($hash_key,'goodsbrowse','goodsid');
  597. if ($browse_goodsid) {
  598. //删除缓存中多余的浏览历史记录,仅保留最近的30条浏览历史,先取出最近30条浏览历史的商品ID
  599. $cachegoodsid_arr = $browse_goodsid['goodsid']?unserialize($browse_goodsid['goodsid']):array();
  600. unset($browse_goodsid['goodsid']);
  601. if ($cachegoodsid_arr){
  602. $cachegoodsid_arr = array_slice($cachegoodsid_arr,-30,30,true);
  603. }
  604. //处理存入数据库的浏览历史缓存信息
  605. $_cache = rcache($hash_key, 'goodsbrowse');
  606. foreach((array)$_cache as $c_k=>$c_v){
  607. $c_v = unserialize($c_v);
  608. if ($c_v['browsetime'] >= $starttime){//如果 缓存中的数据未更新到数据库中(即添加时间大于上次更新到数据库中的数据时间)则将数据更新到数据库中
  609. $tmp_arr = array();
  610. $tmp_arr['goods_id'] = $c_v['goods_id'];
  611. $tmp_arr['member_id'] = $v['member_id'];
  612. $tmp_arr['browsetime'] = $c_v['browsetime'];
  613. $tmp_arr['gc_id'] = $c_v['gc_id'];
  614. $tmp_arr['gc_id_1'] = $c_v['gc_id_1'];
  615. $tmp_arr['gc_id_2'] = $c_v['gc_id_2'];
  616. $tmp_arr['gc_id_3'] = $c_v['gc_id_3'];
  617. $insert_arr[] = $tmp_arr;
  618. $goodsid_arr[] = $c_v['goods_id'];
  619. }
  620. //除了最近的30条浏览历史之外多余的浏览历史记录或者30天之前的浏览历史从缓存中删除
  621. if (!in_array($c_v['goods_id'], $cachegoodsid_arr) || $c_v['browsetime'] < $monthago){
  622. unset($_cache[$c_k]);
  623. }
  624. }
  625. //删除已经存在的该商品浏览记录
  626. if ($goodsid_arr){
  627. $model->delGoodsbrowse(array('member_id'=>$v['member_id'],'goods_id'=>array('in',$goodsid_arr)));
  628. }
  629. //将缓存中的浏览历史存入数据库
  630. if ($insert_arr){
  631. $model->addGoodsbrowseAll($insert_arr);
  632. }
  633. //重新赋值浏览历史缓存
  634. dcache($hash_key, 'goodsbrowse');
  635. $_cache['goodsid'] = serialize($cachegoodsid_arr);
  636. wcache($hash_key,$_cache,'goodsbrowse');
  637. }
  638. }
  639. }
  640. }
  641. //删除30天前的浏览历史
  642. $model->delGoodsbrowse(array('browsetime'=>array('lt',$monthago)));
  643. }
  644. /**
  645. * 缓存订单及订单商品相关数据
  646. */
  647. private function _order_goods_cache(){
  648. $model = Model('stat');
  649. //查询最后统计的记录
  650. $latest_record = $model->table('stat_ordergoods')->order('stat_updatetime desc,rec_id desc')->find();
  651. $stime = 0;
  652. if ($latest_record){
  653. $start_time = strtotime(date('Y-m-d',$latest_record['stat_updatetime']));
  654. } else {
  655. $start_time = strtotime(date('Y-m-d',strtotime(C('setup_date'))));//从系统的安装时间开始统计
  656. }
  657. for ($stime = $start_time; $stime < time(); $stime = $stime+86400){
  658. $etime = $stime + 86400 - 1;
  659. //避免重复统计,开始时间必须大于最后一条记录的记录时间
  660. $search_stime = $latest_record['stat_updatetime'] > $stime?$latest_record['stat_updatetime']:$stime;
  661. //统计一天的数据,如果结束时间大于当前时间,则结束时间为当前时间,避免因为查询时间的延迟造成数据遗落
  662. $search_etime = ($t = ($stime + 86400 - 1)) > time() ? time() : ($stime + 86400 - 1);
  663. //查询时间段内新订单或者更新过的订单,在缓存表中需要将新订单和更新过的订单进行重新缓存
  664. $where = array();
  665. $where['log_time'] = array('between',array($search_stime,$search_etime));
  666. //查询记录总条数
  667. $countnum_arr = $model->table('order_log')->field('COUNT(DISTINCT order_id) as countnum')->where($where)->find();
  668. $countnum = intval($countnum_arr['countnum']);
  669. for ($i=0; $i<$countnum; $i+=100){//每次查询100条
  670. $orderlog_list = array();
  671. $orderlog_list = $model->table('order_log')->field('DISTINCT order_id')->where($where)->limit($i.',100')->select();
  672. if ($orderlog_list){
  673. //店铺ID数组
  674. $storeid_arr = array();
  675. //商品ID数组
  676. $goodsid_arr = array();
  677. //商品公共表ID数组
  678. $goods_commonid_arr = array();
  679. //订单ID数组
  680. $orderid_arr = array();
  681. //整理需要缓存的订单ID
  682. foreach ((array)$orderlog_list as $k=>$v){
  683. $orderid_arr[] = $v['order_id'];
  684. }
  685. unset($orderlog_list);
  686. //查询订单数据
  687. $field = 'order_id,order_sn,store_id,buyer_id,buyer_name,add_time,payment_code,order_amount,shipping_fee,evaluation_state,order_state,refund_state,refund_amount,order_from';
  688. $order_list_tmp = $model->table('order')->field($field)->where(array('order_id'=>array('in',$orderid_arr)))->select();
  689. $order_list = array();
  690. foreach ((array)$order_list_tmp as $k=>$v){
  691. //判读订单是否计入统计(在线支付订单已支付或者经过退款的取消订单或者货到付款订单订单已成功)
  692. $v['order_isvalid'] = 0;
  693. if ($v['payment_code'] != 'offline' && $v['order_state'] != ORDER_STATE_NEW && $v['order_state'] != ORDER_STATE_CANCEL){//在线支付并且已支付并且未取消
  694. $v['order_isvalid'] = 1;
  695. } elseif ($v['order_state'] == ORDER_STATE_CANCEL && $v['refund_state'] != 0) {//经过退款的取消订单
  696. $v['order_isvalid'] = 1;
  697. } elseif ($v['payment_code'] == 'offline' && $v['order_state'] == ORDER_STATE_SUCCESS) {//货到付款订单,订单成功之后才计入统计
  698. $v['order_isvalid'] = 1;
  699. }
  700. $order_list[$v['order_id']] = $v;
  701. $storeid_arr[] = $v['store_id'];
  702. }
  703. unset($order_list_tmp);
  704. //查询订单扩展数据
  705. $field = 'order_id,reciver_province_id';
  706. $order_common_list_tmp = $model->table('order_common')->field($field)->where(array('order_id'=>array('in',$orderid_arr)))->select();
  707. $order_common_list = array();
  708. foreach ((array)$order_common_list_tmp as $k=>$v){
  709. $order_common_list[$v['order_id']] = $v;
  710. }
  711. unset($order_common_list_tmp);
  712. //查询店铺信息
  713. $field = 'store_id,store_name,grade_id,sc_id';
  714. $store_list_tmp = $model->table('store')->field($field)->where(array('store_id'=>array('in',$storeid_arr)))->select();
  715. $store_list = array();
  716. foreach ((array)$store_list_tmp as $k=>$v){
  717. $store_list[$v['store_id']] = $v;
  718. }
  719. unset($store_list_tmp);
  720. //查询订单商品
  721. $field = 'rec_id,order_id,goods_id,goods_name,goods_price,goods_num,goods_image,goods_pay_price,store_id,buyer_id,goods_type,promotions_id,commis_rate,gc_id';
  722. $ordergoods_list = $model->table('order_goods')->field($field)->where(array('order_id'=>array('in',$orderid_arr)))->select();
  723. foreach ((array)$ordergoods_list as $k=>$v){
  724. $goodsid_arr[] = $v['goods_id'];
  725. }
  726. //查询商品信息
  727. $field = 'goods_id,goods_commonid,goods_price,goods_serial,gc_id,gc_id_1,gc_id_2,gc_id_3,goods_image';
  728. $goods_list_tmp = $model->table('goods')->field($field)->where(array('goods_id'=>array('in',$goodsid_arr)))->select();
  729. foreach ((array)$goods_list_tmp as $k=>$v){
  730. $goods_commonid_arr[] = $v['goods_commonid'];
  731. }
  732. //查询商品公共信息
  733. $field = 'goods_commonid,goods_name,brand_id,brand_name';
  734. $goods_common_list_tmp = $model->table('goods_common')->field($field)->where(array('goods_commonid'=>array('in',$goods_commonid_arr)))->select();
  735. $goods_common_list = array();
  736. foreach ((array)$goods_common_list_tmp as $k=>$v){
  737. $goods_common_list[$v['goods_commonid']] = $v;
  738. }
  739. unset($goods_common_list_tmp);
  740. //处理商品数组
  741. $goods_list = array();
  742. foreach ((array)$goods_list_tmp as $k=>$v){
  743. $v['goods_commonname'] = $goods_common_list[$v['goods_commonid']]['goods_name'];
  744. $v['brand_id'] = $goods_common_list[$v['goods_commonid']]['brand_id'];
  745. $v['brand_name'] = $goods_common_list[$v['goods_commonid']]['brand_name'];
  746. $goods_list[$v['goods_id']] = $v;
  747. }
  748. unset($goods_list_tmp);
  749. //查询订单缓存是否存在,存在则删除
  750. $model->table('stat_ordergoods')->where(array('order_id'=>array('in',$orderid_arr)))->delete();
  751. //查询订单缓存是否存在,存在则删除
  752. $model->table('stat_order')->where(array('order_id'=>array('in',$orderid_arr)))->delete();
  753. //整理新增数据
  754. $ordergoods_insert_arr = array();
  755. foreach ((array)$ordergoods_list as $k=>$v){
  756. $tmp = array();
  757. $tmp['rec_id'] = $v['rec_id'];
  758. $tmp['stat_updatetime'] = $search_etime;
  759. $tmp['order_id'] = $v['order_id'];
  760. $tmp['order_sn'] = $order_list[$v['order_id']]['order_sn'];
  761. $tmp['order_add_time'] = $order_list[$v['order_id']]['add_time'];
  762. $tmp['payment_code'] = $order_list[$v['order_id']]['payment_code'];
  763. $tmp['order_amount'] = $order_list[$v['order_id']]['order_amount'];
  764. $tmp['shipping_fee'] = $order_list[$v['order_id']]['shipping_fee'];
  765. $tmp['evaluation_state'] = $order_list[$v['order_id']]['evaluation_state'];
  766. $tmp['order_state'] = $order_list[$v['order_id']]['order_state'];
  767. $tmp['refund_state'] = $order_list[$v['order_id']]['refund_state'];
  768. $tmp['refund_amount'] = $order_list[$v['order_id']]['refund_amount'];
  769. $tmp['order_from'] = $order_list[$v['order_id']]['order_from'];
  770. $tmp['order_isvalid'] = $order_list[$v['order_id']]['order_isvalid'];
  771. $tmp['reciver_province_id'] = $order_common_list[$v['order_id']]['reciver_province_id'];
  772. $tmp['store_id'] = $v['store_id'];
  773. $tmp['store_name'] = $store_list[$v['store_id']]['store_name'];
  774. $tmp['grade_id'] = $store_list[$v['store_id']]['grade_id'];
  775. $tmp['sc_id'] = $store_list[$v['store_id']]['sc_id'];
  776. $tmp['buyer_id'] = $order_list[$v['order_id']]['buyer_id'];
  777. $tmp['buyer_name'] = $order_list[$v['order_id']]['buyer_name'];
  778. $tmp['goods_id'] = $v['goods_id'];
  779. $tmp['goods_name'] = $v['goods_name'];
  780. $tmp['goods_commonid'] = intval($goods_list[$v['goods_id']]['goods_commonid']);
  781. $tmp['goods_commonname'] = ($t = $goods_list[$v['goods_id']]['goods_commonname'])?$t:'';
  782. $tmp['gc_id'] = intval($goods_list[$v['goods_id']]['gc_id']);
  783. $tmp['gc_parentid_1'] = intval($goods_list[$v['goods_id']]['gc_id_1']);
  784. $tmp['gc_parentid_2'] = intval($goods_list[$v['goods_id']]['gc_id_2']);
  785. $tmp['gc_parentid_3'] = intval($goods_list[$v['goods_id']]['gc_id_3']);
  786. $tmp['brand_id'] = intval($goods_list[$v['goods_id']]['brand_id']);
  787. $tmp['brand_name'] = ($t = $goods_list[$v['goods_id']]['brand_name'])?$t:'';
  788. $tmp['goods_serial'] = ($t = $goods_list[$v['goods_id']]['goods_serial'])?$t:'';
  789. $tmp['goods_price'] = $v['goods_price'];
  790. $tmp['goods_num'] = $v['goods_num'];
  791. $tmp['goods_image'] = $goods_list[$v['goods_id']]['goods_image'];
  792. $tmp['goods_pay_price'] = $v['goods_pay_price'];
  793. $tmp['goods_type'] = $v['goods_type'];
  794. $tmp['promotions_id'] = $v['promotions_id'];
  795. $tmp['commis_rate'] = $v['commis_rate'];
  796. $ordergoods_insert_arr[] = $tmp;
  797. }
  798. $model->table('stat_ordergoods')->insertAll($ordergoods_insert_arr);
  799. $order_insert_arr = array();
  800. foreach ((array)$order_list as $k=>$v){
  801. $tmp = array();
  802. $tmp['order_id'] = $v['order_id'];
  803. $tmp['order_sn'] = $v['order_sn'];
  804. $tmp['order_add_time'] = $v['add_time'];
  805. $tmp['payment_code'] = $v['payment_code'];
  806. $tmp['order_amount'] = $v['order_amount'];
  807. $tmp['shipping_fee'] = $v['shipping_fee'];
  808. $tmp['evaluation_state'] = $v['evaluation_state'];
  809. $tmp['order_state'] = $v['order_state'];
  810. $tmp['refund_state'] = $v['refund_state'];
  811. $tmp['refund_amount'] = $v['refund_amount'];
  812. $tmp['order_from'] = $v['order_from'];
  813. $tmp['order_isvalid'] = $v['order_isvalid'];
  814. $tmp['reciver_province_id'] = $order_common_list[$v['order_id']]['reciver_province_id'];
  815. $tmp['store_id'] = $v['store_id'];
  816. $tmp['store_name'] = $store_list[$v['store_id']]['store_name'];
  817. $tmp['grade_id'] = $store_list[$v['store_id']]['grade_id'];
  818. $tmp['sc_id'] = $store_list[$v['store_id']]['sc_id'];
  819. $tmp['buyer_id'] = $v['buyer_id'];
  820. $tmp['buyer_name'] = $v['buyer_name'];
  821. $order_insert_arr[] = $tmp;
  822. }
  823. $model->table('stat_order')->insertAll($order_insert_arr);
  824. }
  825. }
  826. }
  827. }
  828. /**
  829. * 会员相关数据统计
  830. */
  831. private function _member_stat()
  832. {
  833. $model = Model('stat');
  834. //查询最后统计的记录
  835. $latest_record = $model->getOneStatmember(array(), '', 'statm_id desc');
  836. $stime = 0;
  837. if ($latest_record){
  838. $start_time = strtotime(date('Y-m-d',$latest_record['statm_updatetime']));
  839. } else {
  840. $start_time = strtotime(date('Y-m-d',strtotime(C('setup_date'))));//从系统的安装时间开始统计
  841. }
  842. $j = 1;
  843. for ($stime = $start_time; $stime < time(); $stime = $stime+86400){
  844. //数据库更新数据数组
  845. $insert_arr = array();
  846. $update_arr = array();
  847. $etime = $stime + 86400 - 1;
  848. //避免重复统计,开始时间必须大于最后一条记录的记录时间
  849. $search_stime = $latest_record['statm_updatetime'] > $stime ? $latest_record['statm_updatetime'] : $stime;
  850. //统计一天的数据,如果结束时间大于当前时间,则结束时间为当前时间,避免因为查询时间的延迟造成数据遗落
  851. $search_etime = ($t = ($stime + 86400 - 1)) > time() ? time() : ($stime + 86400 - 1);
  852. //统计订单下单量和下单金额
  853. $field = ' order.order_id,add_time,buyer_id,buyer_name,order_amount';
  854. $where = array();
  855. $where['order.order_state'] = array('neq',ORDER_STATE_NEW);//去除未支付订单
  856. $where['order.refund_state'] = array('exp',"!(order.order_state = '".ORDER_STATE_CANCEL."' and order.refund_state = 0)");//没有参与退款的取消订单,不记录到统计中
  857. $where['order_log.log_time'] = array('between',array($search_stime,$search_etime));//按照订单付款的操作时间统计
  858. //货到付款当交易成功进入统计,非货到付款当付款后进入统计
  859. $where['payment_code'] = array('exp',"(order.payment_code='offline' and order_log.log_orderstate = '".ORDER_STATE_SUCCESS."') or (order.payment_code<>'offline' and order_log.log_orderstate = '".ORDER_STATE_PAY."' )");
  860. $orderlist_tmp = $model->statByOrderLog($where, $field, 0, 0, 'order_id');//此处由于底层的限制,仅能查询1000条,如果日下单量大于1000,则需要limit的支持
  861. $order_list = array();
  862. $orderid_list = array();
  863. foreach ((array)$orderlist_tmp as $k=>$v){
  864. $addtime = strtotime(date('Y-m-d',$v['add_time']));
  865. if ($addtime != $stime){//订单如果隔天支付的话,需要进行统计数据更新
  866. $update_arr[$addtime][$v['buyer_id']]['statm_membername'] = $v['buyer_name'];
  867. $update_arr[$addtime][$v['buyer_id']]['statm_ordernum'] = intval($update_arr[$addtime][$v['buyer_id']]['statm_ordernum'])+1;
  868. $update_arr[$addtime][$v['buyer_id']]['statm_orderamount'] = floatval($update_arr[$addtime][$v['buyer_id']]['statm_orderamount']) + (($t = floatval($v['order_amount'])) > 0?$t:0);
  869. } else {
  870. $order_list[$v['buyer_id']]['buyer_name'] = $v['buyer_name'];
  871. $order_list[$v['buyer_id']]['ordernum'] = intval($order_list[$v['buyer_id']]['ordernum']) + 1;
  872. $order_list[$v['buyer_id']]['orderamount'] = floatval($order_list[$v['buyer_id']]['orderamount']) + (($t = floatval($v['order_amount'])) > 0?$t:0);
  873. }
  874. //记录订单ID数组
  875. $orderid_list[] = $v['order_id'];
  876. }
  877. //统计下单商品件数
  878. if ($orderid_list){
  879. $field = ' add_time,order.buyer_id,order.buyer_name,goods_num ';
  880. $where = array();
  881. $where['order.order_id'] = array('in',$orderid_list);
  882. $ordergoods_tmp = $model->statByOrderGoods($where, $field, 0, 0, 'order.order_id');
  883. $ordergoods_list = array();
  884. foreach ((array)$ordergoods_tmp as $k=>$v){
  885. $addtime = strtotime(date('Y-m-d',$v['add_time']));
  886. if ($addtime != $stime){//订单如果隔天支付的话,需要进行统计数据更新
  887. $update_arr[$addtime][$v['buyer_id']]['statm_goodsnum'] = intval($update_arr[$addtime][$v['buyer_id']]['statm_goodsnum']) + (($t = floatval($v['goods_num'])) > 0?$t:0);
  888. } else {
  889. $ordergoods_list[$v['buyer_id']]['goodsnum'] = $ordergoods_list[$v['buyer_id']]['goodsnum'] + (($t = floatval($v['goods_num'])) > 0?$t:0);
  890. }
  891. }
  892. }
  893. //统计的预存款记录
  894. $field = ' lg_member_id,lg_member_name,SUM(IF(lg_av_amount>=0,lg_av_amount,0)) as predincrease, SUM(IF(lg_av_amount<=0,lg_av_amount,0)) as predreduce ';
  895. $where = array();
  896. $where['lg_add_time'] = array('between',array($stime,$etime));
  897. $predeposit_tmp = $model->getPredepositInfo($where, $field, 0, 'lg_member_id', 0, 'lg_member_id');
  898. $predeposit_list = array();
  899. foreach ((array)$predeposit_tmp as $k=>$v){
  900. $predeposit_list[$v['lg_member_id']] = $v;
  901. }
  902. //统计的积分记录
  903. $field = ' pl_memberid,pl_membername,SUM(IF(pl_points>=0,pl_points,0)) as pointsincrease, SUM(IF(pl_points<=0,pl_points,0)) as pointsreduce ';
  904. $where = array();
  905. $where['pl_addtime'] = array('between',array($stime,$etime));
  906. $points_tmp = $model->statByPointslog($where, $field, 0, 0, '', 'pl_memberid');
  907. $points_list = array();
  908. foreach ((array)$points_tmp as $k=>$v){
  909. $points_list[$v['pl_memberid']] = $v;
  910. }
  911. //处理需要更新的数据
  912. foreach ((array)$update_arr as $k=>$v){
  913. foreach ($v as $m_k=>$m_v){
  914. //查询记录是否存在
  915. $statmember_info = $model->getOneStatmember(array('statm_time'=>$k,'statm_memberid'=>$m_k));
  916. if ($statmember_info){
  917. $m_v['statm_ordernum'] = intval($statmember_info['statm_ordernum']) + $m_v['statm_ordernum'];
  918. $m_v['statm_orderamount'] = floatval($statmember_info['statm_ordernum']) + $m_v['statm_orderamount'];
  919. $m_v['statm_updatetime'] = $search_etime;
  920. $model->updateStatmember(array('statm_time'=>$k,'statm_memberid'=>$m_k),$m_v);
  921. } else {
  922. $tmp = array();
  923. $tmp['statm_memberid'] = $m_k;
  924. $tmp['statm_membername'] = $m_v['statm_membername'];
  925. $tmp['statm_time'] = $k;
  926. $tmp['statm_updatetime'] = $search_etime;
  927. $tmp['statm_ordernum'] = ($t = intval($m_v['statm_ordernum'])) > 0?$t:0;
  928. $tmp['statm_orderamount'] = ($t = floatval($m_v['statm_orderamount']))>0?$t:0;
  929. $tmp['statm_goodsnum'] = ($t = intval($m_v['statm_goodsnum']))?$t:0;
  930. $tmp['statm_predincrease'] = 0;
  931. $tmp['statm_predreduce'] = 0;
  932. $tmp['statm_pointsincrease'] = 0;
  933. $tmp['statm_pointsreduce'] = 0;
  934. $insert_arr[] = $tmp;
  935. }
  936. unset($statmember_info);
  937. }
  938. }
  939. //处理获得所有会员ID数组
  940. $memberidarr_order = $order_list?array_keys($order_list):array();
  941. $memberidarr_ordergoods = $ordergoods_list?array_keys($ordergoods_list):array();
  942. $memberidarr_predeposit = $predeposit_list?array_keys($predeposit_list):array();
  943. $memberidarr_points = $points_list?array_keys($points_list):array();
  944. $memberid_arr = array_merge($memberidarr_order,$memberidarr_ordergoods,$memberidarr_predeposit,$memberidarr_points);
  945. //查询会员信息
  946. $memberid_list = Model('member')->getMemberList(array('member_id'=>array('in',$memberid_arr)), '*', 0);
  947. //查询记录是否存在
  948. $statmemberlist_tmp = $model->statByStatmember(array('statm_time'=>$stime));
  949. $statmemberlist = array();
  950. foreach ((array)$statmemberlist_tmp as $k=>$v){
  951. $statmemberlist[$v['statm_memberid']] = $v;
  952. }
  953. foreach ((array)$memberid_list as $k=>$v){
  954. $tmp = array();
  955. $tmp['statm_memberid'] = $v['member_id'];
  956. $tmp['statm_membername'] = $v['member_name'];
  957. $tmp['statm_time'] = $stime;
  958. $tmp['statm_updatetime'] = $search_etime;
  959. //因为记录可能已经存在,所以加上之前的统计记录
  960. $tmp['statm_ordernum'] = intval($statmemberlist[$tmp['statm_memberid']]['statm_ordernum']) + (($t = intval($order_list[$tmp['statm_memberid']]['ordernum'])) > 0?$t:0);
  961. $tmp['statm_orderamount'] = floatval($statmemberlist[$tmp['statm_memberid']]['statm_orderamount']) + (($t = floatval($order_list[$tmp['statm_memberid']]['orderamount']))>0?$t:0);
  962. $tmp['statm_goodsnum'] = intval($statmemberlist[$tmp['statm_memberid']]['statm_goodsnum']) + (($t = intval($ordergoods_list[$tmp['statm_memberid']]['goodsnum']))?$t:0);
  963. $tmp['statm_predincrease'] = (($t = floatval($predeposit_list[$tmp['statm_memberid']]['predincrease']))?$t:0);
  964. $tmp['statm_predreduce'] = (($t = floatval($predeposit_list[$tmp['statm_memberid']]['predreduce']))?$t:0);
  965. $tmp['statm_pointsincrease'] = (($t = intval($points_list[$tmp['statm_memberid']]['pointsincrease']))?$t:0);
  966. $tmp['statm_pointsreduce'] = (($t = intval($points_list[$tmp['statm_memberid']]['pointsreduce']))?$t:0);
  967. $insert_arr[] = $tmp;
  968. }
  969. //删除旧的统计数据
  970. $model->delByStatmember(array('statm_time'=>$stime));
  971. $model->table('stat_member')->insertAll($insert_arr);
  972. }
  973. }
  974. private function bonus_expire_notify()
  975. {
  976. notify_helper::onBonusExpire(10,5);
  977. }
  978. private function fcode_expire_notify()
  979. {
  980. notify_helper::onFcodeWarning();
  981. }
  982. public function update_voteOp()
  983. {
  984. $mod_special = Model('mb_special');
  985. $items = $mod_special->field('*')->where(['from_user' => 1])->limit(false)->select();
  986. foreach ($items as $item)
  987. {
  988. $spid = intval($item['special_id']);
  989. $params = $item['params'];
  990. $vote_result = $item['vote_result'];
  991. if(!empty($params) && !empty($vote_result))
  992. {
  993. $params = unserialize($params);
  994. $vresult = unserialize($vote_result);
  995. $vote = $params['vote'];
  996. if(!empty($vote))
  997. {
  998. $params['vote'] = [];
  999. $params['vote'][] = $vote;
  1000. $vote_result = [];
  1001. $vote_result[] = $vresult;
  1002. $mod_special->editUserSpecial($spid,['special_id' =>$spid],['params' => serialize($params),'vote_result' => serialize($vote_result)]);
  1003. }
  1004. }
  1005. }
  1006. }
  1007. private function add_sku()
  1008. {
  1009. $path = BASE_DATA_PATH . "/sales/sku.csv";
  1010. $lines = explode("\r",$this->read_file($path));
  1011. $mod_price = Model('');
  1012. foreach($lines as $line)
  1013. {
  1014. $row = explode(';',$line);
  1015. $omsid = intval($row[0]);
  1016. $market_price = intval($row[1] * 100 + 0.5) / 100;
  1017. $purchase_price = intval($row[2] * 100 + 0.5) / 100;
  1018. $sku_price = intval($row[3] * 100 + 0.5) / 100;
  1019. if($omsid <= 0) continue;
  1020. $val = ['omsid' => $omsid,'purchase_price' => $purchase_price,'market_price' => $market_price,'sku_price' => $sku_price];
  1021. $ret = $mod_price->table('goods_orgprice')->insert($val);
  1022. if($ret == false) {
  1023. Log::record("cannot insert row",Log::DEBUG);
  1024. }
  1025. }
  1026. }
  1027. private function read_file($path)
  1028. {
  1029. $file = fopen($path, "r");
  1030. $datas = '';
  1031. while (!feof($file)) {
  1032. $datas .= fgets($file);
  1033. }
  1034. return $datas;
  1035. }
  1036. //订单系统,通道,商户订单统计.
  1037. public function _order_stats()
  1038. {
  1039. $this->system_order_stats();
  1040. $this->provider_order_stats();
  1041. $this->merchant_order_stats();
  1042. }
  1043. private function day_order_stats($type , $condition , $start_time)
  1044. {
  1045. if($start_time == 0) {
  1046. $start_time = strtotime(date("Y-m-d" , strtotime("-1 day")));
  1047. }
  1048. $stats = Model('')->table('refill_stats')->where($condition)->find();
  1049. $end_time = $start_time + 3600*24;
  1050. $condition['refill_order.order_time'] = [ ['egt', $start_time] , ['lt', $end_time] , 'and'];
  1051. $condition['inner_status'] = 0;
  1052. $counts = Model('')->table('refill_order,vr_order')->join('inner')
  1053. ->on('refill_order.order_id=vr_order.order_id')
  1054. ->field('count(*) as order_count, sum(refill_amount) as refill_amounts, sum(channel_amount) as channel_amounts, sum(mch_amount) as mch_amounts, order_state')
  1055. ->where($condition)
  1056. ->group('order_state')
  1057. ->select();
  1058. $cancel_count = $success_count = $order_count = 0;
  1059. $success_refill_amounts = $success_channel_amounts = $success_mch_amounts = $profit_amounts = 0;
  1060. foreach ($counts as $count)
  1061. {
  1062. if ($count['order_state'] == ORDER_STATE_SUCCESS) {
  1063. $success_refill_amounts = $count['refill_amounts'];
  1064. $success_channel_amounts = $count['channel_amounts'];
  1065. $success_mch_amounts = $count['mch_amounts'];
  1066. $success_count = $count['order_count'];
  1067. $profit_amounts = ncPriceFormat($success_mch_amounts - $success_channel_amounts);
  1068. } elseif ($count['order_state'] == ORDER_STATE_CANCEL) {
  1069. $cancel_count = $count['order_count'];
  1070. }
  1071. $order_count += $count['order_count'];
  1072. }
  1073. $params['time_text'] = date("Y-m-d" , $start_time);
  1074. $params['time_stamp'] = $start_time;
  1075. $params['success_count'] = $success_count;
  1076. $params['success_refill_amounts'] = $success_refill_amounts;
  1077. $params['success_channel_amounts'] = $success_channel_amounts;
  1078. $params['success_mch_amounts'] = $success_mch_amounts;
  1079. $params['profit_amounts'] = $profit_amounts;
  1080. $params['order_count'] = $order_count;
  1081. $params['cancel_count'] = $cancel_count;
  1082. if($order_count == 0) {
  1083. $params['success_ratio'] = '0%';
  1084. }else{
  1085. $params['success_ratio'] = ncPriceFormat($success_count / $order_count) * 100 . '%';
  1086. }
  1087. return $params;
  1088. }
  1089. private function system_order_stats()
  1090. {
  1091. $type = 'system';
  1092. $start_time = strtotime(date("Y-m-d" , strtotime("-1 day")));
  1093. $system_stats = function($params , $type){
  1094. $params['type'] = $type;
  1095. $params['cid'] = 0;
  1096. $params['cname'] = $type;
  1097. $success_refill_amounts = $params['success_refill_amounts'];
  1098. if($success_refill_amounts > 0 && $success_refill_amounts <= 15000000) {
  1099. $params['service_amounts'] = ncPriceFormat($params['success_refill_amounts'] * 0.001);
  1100. }elseif ($success_refill_amounts > 15000000 && $success_refill_amounts <= 30000000) {
  1101. $params['service_amounts'] = 15000;
  1102. }else{
  1103. $params['service_amounts'] = ncPriceFormat($params['success_refill_amounts'] * 0.0005);
  1104. }
  1105. Model('')->table('refill_stats')->insert($params);
  1106. };
  1107. $need_init = $this->check_stats($type);
  1108. if($need_init == false){
  1109. $stats = Model('')->table('refill_stats')->where(['type' => $type , 'time_stamp' => $start_time])->find();
  1110. if(empty($stats)) {
  1111. $params = $this->day_order_stats($type , [] , $start_time);
  1112. $system_stats($params,$type);
  1113. }
  1114. }else{
  1115. Model('')->table('refill_stats')->where(['type' => $type])->delete();
  1116. foreach ($need_init as $date) {
  1117. $params = $this->day_order_stats($type , [] , strtotime($date));
  1118. $system_stats($params,$type);
  1119. }
  1120. }
  1121. }
  1122. private function provider_order_stats()
  1123. {
  1124. $type = 'provider';
  1125. $provider_list = Model('')->table('refill_provider,store')->field('refill_provider.store_id,store.store_name')->join('inner')
  1126. ->on('store.store_id=refill_provider.store_id')->limit(100)->select();
  1127. if(empty($provider_list)) {
  1128. return ;
  1129. }
  1130. $start_time = strtotime(date("Y-m-d" , strtotime("-1 day")));
  1131. $provider_stats = function($params , $type , $provider){
  1132. $params['type'] = $type;
  1133. $params['cid'] = $provider['store_id'];
  1134. $params['cname'] = $provider['store_name'];
  1135. $params['service_amounts'] = 0;
  1136. return $params;
  1137. };
  1138. $insert = [];
  1139. $need_init = $this->check_stats($type);
  1140. foreach ($provider_list as $provider) {
  1141. $store_id = $provider['store_id'];
  1142. $condition['store_id'] = $store_id;
  1143. if($need_init == false){
  1144. $stats = Model('')->table('refill_stats')->where(['type' => $type , 'time_stamp' => $start_time , 'cid' => $store_id])->find();
  1145. if(empty($stats)) {
  1146. $params = $this->day_order_stats($type , $condition , $start_time);
  1147. $insert[] = $provider_stats($params , $type , $provider);
  1148. }
  1149. }else {
  1150. Model('')->table('refill_stats')->where(['type' => $type])->delete();
  1151. foreach ($need_init as $date) {
  1152. $params = $this->day_order_stats($type , $condition , strtotime($date));
  1153. $insert[] = $provider_stats($params, $type , $provider);
  1154. }
  1155. }
  1156. }
  1157. Model('')->table('refill_stats')->insertAll($insert);
  1158. }
  1159. private function merchant_order_stats()
  1160. {
  1161. $type = 'merchant';
  1162. $merchant_list = Model('')->table('merchant')->limit(100)->select();
  1163. if(empty($merchant_list)) {
  1164. return ;
  1165. }
  1166. $start_time = strtotime(date("Y-m-d" , strtotime("-1 day")));
  1167. $merchant_stats = function($params , $type , $merchant){
  1168. $params['type'] = $type;
  1169. $params['cid'] = $merchant['mchid'];
  1170. $params['cname'] = $merchant['company_name'] == '' ? $merchant['name'] : $merchant['company_name'];
  1171. $params['service_amounts'] = 0;
  1172. return $params;
  1173. };
  1174. $insert = [];
  1175. $need_init = $this->check_stats($type);
  1176. foreach ($merchant_list as $merchant) {
  1177. $mchid = $merchant['mchid'];
  1178. $condition['mchid'] = $mchid;
  1179. if($need_init == false){
  1180. $stats = Model('')->table('refill_stats')->where(['type' => $type , 'time_stamp' => $start_time , 'cid' => $mchid])->find();
  1181. if(empty($stats)) {
  1182. $params = $this->day_order_stats($type , $condition , $start_time);
  1183. $insert[] = $merchant_stats($params , $type , $merchant);
  1184. }
  1185. }else{
  1186. Model('')->table('refill_stats')->where(['type' => $type])->delete();
  1187. foreach ($need_init as $date) {
  1188. $params = $this->day_order_stats($type , $condition , strtotime($date));
  1189. $insert[] = $merchant_stats($params , $type , $merchant);
  1190. }
  1191. }
  1192. }
  1193. Model('')->table('refill_stats')->insertAll($insert);
  1194. }
  1195. //检测前天是否有统计
  1196. public function check_stats($type)
  1197. {
  1198. $date = strtotime(date("Y-m-d" , strtotime("-2 day")));
  1199. $condition['type'] = $type;
  1200. $condition['time_stamp'] = $date;
  1201. $stats = Model('')->table('refill_stats')->where($condition)->find();
  1202. if(empty($stats)) {
  1203. $first_order = Model('')->table('refill_order')->order('order_time asc')->find();
  1204. $days = (strtotime(date("Y-m-d")) - strtotime(date("Y-m-d" , $first_order['order_time']))) / (3600 * 24);
  1205. for ($i = $days; $i > 0; $i--) {
  1206. $dates[] = date("Y-m-d" , strtotime("-{$i}day"));
  1207. }
  1208. return $dates;
  1209. }
  1210. return false;
  1211. }
  1212. }