Date.php 84 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487
  1. <?php
  2. namespace app\crontab\controller;
  3. use think\facade\Db;
  4. /**
  5. * ============================================================================
  6. *
  7. * ============================================================================
  8. * 版权所有 2014-2028 浙江惠利玛产业互联网有限公司,并保留所有权利。
  9. * 网站地址: https://www.valimart.net/
  10. * ----------------------------------------------------------------------------
  11. *
  12. * ============================================================================
  13. * 定时器
  14. */
  15. class Date extends BaseCron {
  16. /**
  17. * 该文件中所有任务执行频率,默认1天,单位:秒
  18. * @var int
  19. */
  20. const EXE_TIMES = 86400;
  21. /**
  22. * 优惠券即将到期提醒时间,单位:天
  23. * @var int
  24. */
  25. const VOUCHER_INTERVAL = 5;
  26. /**
  27. * 兑换码即将到期提醒时间,单位:天
  28. * @var int
  29. */
  30. const VR_CODE_INTERVAL = 5;
  31. /**
  32. * 订单结束后可评论时间,15天,60*60*24*15
  33. * @var int
  34. */
  35. const ORDER_EVALUATE_TIME = 1296000;
  36. /**
  37. * 每次到货通知消息数量
  38. * @var int
  39. */
  40. const ARRIVAL_NOTICE_NUM = 100;
  41. /**
  42. * 默认方法
  43. */
  44. public function index() {
  45. //小程序直播商品状态查询
  46. $this->_minipro_live_goods_state();
  47. //小程序直播商品删除
  48. $this->_minipro_live_goods_delete();
  49. //小程序直播商品创建
  50. $this->_minipro_live_goods_add();
  51. //小程序直播商品更新
  52. $this->_minipro_live_goods_update();
  53. //订单超期后不允许评价
  54. $this->_order_eval_expire_update();
  55. //未付款订单超期自动关闭
  56. $this->_order_timeout_cancel();
  57. //增加会员积分和经验值
  58. $this->_add_points();
  59. //订单自动完成
  60. $this->_order_auto_complete();
  61. //更新订单扩展表收货人所在省份ID
  62. $this->_order_reciver_provinceid_update();
  63. //更新退款申请超时处理
  64. model('trade')->editRefundConfirm();
  65. //店铺到期关闭
  66. $this->_store_auto_close();
  67. //生成店铺结算单
  68. $this->_creat_bill();
  69. //代金券即将过期提醒
  70. $this->_voucher_will_expire();
  71. //虚拟兑换码即将过期提醒
  72. $this->_vr_code_will_expire();
  73. //更新商品访问量
  74. $this->_goods_click_update();
  75. //更新商品促销到期状态
  76. $this->_goods_promotion_state_update();
  77. //商品到货通知提醒
  78. $this->_arrival_notice();
  79. //更新浏览量
  80. $this->_goods_browse_update();
  81. //缓存订单及订单商品相关数据
  82. $this->_order_goods_cache();
  83. //会员相关数据统计
  84. $this->_member_stat();
  85. }
  86. /*
  87. * 小程序直播商品创建(调用额度:500次/一天)
  88. */
  89. function _minipro_live_goods_add() {
  90. if (intval(config('ds_config.live_type')) != 1) {
  91. return;
  92. }
  93. $goods_list = Db::name('goodscommon')->alias('goodscommon')->join('minipro_live_goods minipro_live_goods', 'minipro_live_goods.goods_commonid=goodscommon.goods_commonid', 'LEFT')->field('goodscommon.gc_id,goodscommon.gc_id_1,goodscommon.gc_id_2,goodscommon.gc_id_3,goodscommon.gc_name,goodscommon.goods_name,goodscommon.goods_image,goodscommon.goods_price,goodscommon.goods_commonid,goodscommon.store_id,goodscommon.store_name')->where(array(array('minipro_live_goods.goods_commonid', 'null')))->limit(100)->select()->toArray();
  94. $wechat_model = model('wechat');
  95. $wechat_model->getOneWxconfig();
  96. $accessToken = $wechat_model->getAccessToken('miniprogram', 0);
  97. if ($wechat_model->error_code) {
  98. $this->log('获取小程序AccessToken失败' . $wechat_model->error_message);
  99. return;
  100. }
  101. $minipro_live_goods_model = model('minipro_live_goods');
  102. foreach ($goods_list as $goods_info) {
  103. $goods_id = Db::name('goods')->where(array(array('goods_commonid', '=', $goods_info['goods_commonid'])))->order('goods_id asc')->value('goods_id');
  104. if (!$goods_id) {
  105. $this->log('商品不存在');
  106. continue;
  107. }
  108. //上传临时素材
  109. $goods_image = str_replace(UPLOAD_SITE_URL, BASE_UPLOAD_PATH, goods_thumb($goods_info));
  110. $image_path = pathinfo($goods_image, PATHINFO_DIRNAME);
  111. $image_name = pathinfo($goods_image, PATHINFO_BASENAME);
  112. if (!strstr($image_name, '_240')) {
  113. create_albumpic_thumb($image_path, $image_name);
  114. $goods_image = str_ireplace('.', '_240.', $image_name);
  115. }
  116. $res = $wechat_model->uploadMedia('image', $goods_image, "image/jpeg", $image_name);
  117. if (!$res['code']) {
  118. $this->log('上传临时素材失败' . $res['msg']);
  119. continue;
  120. }
  121. $coverImgUrl = $res['data'];
  122. $data = array(
  123. 'goodsInfo' => array(
  124. 'coverImgUrl' => $coverImgUrl,
  125. 'name' => mb_substr($goods_info['goods_name'], 0, 14), //商品名称,最长14个汉字
  126. 'priceType' => 1,
  127. 'price' => $goods_info['goods_price'],
  128. 'url' => 'pages/index/index?ret_url=' . urlencode(config('ds_config.h5_site_url') . '/pages/home/goodsdetail/Goodsdetail?goods_id=' . $goods_id)
  129. )
  130. );
  131. $res = http_request('https://api.weixin.qq.com/wxaapi/broadcast/goods/add?access_token=' . $accessToken, 'POST', $data);
  132. $res = json_decode($res, true);
  133. if (!$res || $res['errcode']) {
  134. $this->log(isset($res['errmsg']) ? $res['errmsg'] : '添加商品出错,错误码:' . $res['errcode']);
  135. continue;
  136. }
  137. $minipro_live_goods_model->addMiniproLiveGoods(array(
  138. 'store_id' => $goods_info['store_id'],
  139. 'store_name' => $goods_info['store_name'],
  140. 'goods_id' => $goods_id,
  141. 'goods_commonid' => $goods_info['goods_commonid'],
  142. 'goods_name' => $goods_info['goods_name'],
  143. 'goods_price' => $goods_info['goods_price'],
  144. 'goods_image' => $goods_info['goods_image'],
  145. 'minipro_live_goods_add_time' => TIMESTAMP,
  146. 'minipro_live_goods_state' => 0,
  147. 'minipro_live_goods_result_id' => $res['goodsId'],
  148. 'minipro_live_goods_verify_id' => $res['auditId'],
  149. ));
  150. }
  151. }
  152. /*
  153. * 小程序直播商品删除(调用额度:1000次/一天)
  154. */
  155. function _minipro_live_goods_delete() {
  156. if (intval(config('ds_config.live_type')) != 1) {
  157. return;
  158. }
  159. $wechat_model = model('wechat');
  160. $wechat_model->getOneWxconfig();
  161. $accessToken = $wechat_model->getAccessToken('miniprogram', 0);
  162. if ($wechat_model->error_code) {
  163. $this->log('获取小程序AccessToken失败' . $wechat_model->error_message);
  164. return;
  165. }
  166. $minipro_live_goods_model = model('minipro_live_goods');
  167. //商品不存在的删除,审核中的、已审核的商品名称、商品图片不一致的删除
  168. $minipro_live_goods_result_ids = Db::name('minipro_live_goods')->alias('minipro_live_goods')->join('goodscommon goodscommon', 'minipro_live_goods.goods_commonid=goodscommon.goods_commonid', 'LEFT')->where(array(array('minipro_live_goods_close', '=', 0), array('goodscommon.goods_commonid', 'null')))->whereOr(function($query) {
  169. $query->where(array(array('minipro_live_goods_state', '<>', 0)))->where(function ($query) {
  170. $query->whereOr(array(array('goodscommon.goods_name', '<>', Db::raw('minipro_live_goods.goods_name'))))->whereOr(array(array('goodscommon.goods_image', '<>', Db::raw('minipro_live_goods.goods_image'))));
  171. });
  172. })->limit(100)->column('minipro_live_goods_result_id');
  173. foreach ($minipro_live_goods_result_ids as $minipro_live_goods_result_id) {
  174. $data = array(
  175. 'goodsId' => $minipro_live_goods_result_id
  176. );
  177. $res = http_request('https://api.weixin.qq.com/wxaapi/broadcast/goods/delete?access_token=' . $accessToken, 'POST', $data);
  178. $res = json_decode($res, true);
  179. if (!$res || $res['errcode']) {
  180. $this->log(isset($res['errmsg']) ? $res['errmsg'] : '删除商品出错,错误码:' . $res['errcode']);
  181. continue;
  182. }
  183. $minipro_live_goods_model->delMiniproLiveGoods(array(array('minipro_live_goods_result_id', '=', $minipro_live_goods_result_id)));
  184. }
  185. //审核中的、已审核的商品ID不存在的删除
  186. $subQuery = Db::name('goods')->field('goods_id')->where('1=1')->buildSql();
  187. $minipro_live_goods_result_ids = Db::name('minipro_live_goods')->where(array(array('minipro_live_goods_close', '=', 0), array('minipro_live_goods_state', '<>', 0), array('goods_id', 'not in', Db::raw($subQuery))))->limit(100)->column('minipro_live_goods_result_id');
  188. foreach ($minipro_live_goods_result_ids as $minipro_live_goods_result_id) {
  189. $data = array(
  190. 'goodsId' => $minipro_live_goods_result_id
  191. );
  192. $res = http_request('https://api.weixin.qq.com/wxaapi/broadcast/goods/delete?access_token=' . $accessToken, 'POST', $data);
  193. $res = json_decode($res, true);
  194. if (!$res || $res['errcode']) {
  195. $this->log(isset($res['errmsg']) ? $res['errmsg'] : '删除商品出错,错误码:' . $res['errcode']);
  196. continue;
  197. }
  198. $minipro_live_goods_model->delMiniproLiveGoods(array(array('minipro_live_goods_result_id', '=', $minipro_live_goods_result_id)));
  199. }
  200. }
  201. /*
  202. * 小程序直播商品更新(调用额度:1000次/一天)
  203. */
  204. function _minipro_live_goods_update() {
  205. if (intval(config('ds_config.live_type')) != 1) {
  206. return;
  207. }
  208. $wechat_model = model('wechat');
  209. $wechat_model->getOneWxconfig();
  210. $accessToken = $wechat_model->getAccessToken('miniprogram', 0);
  211. if ($wechat_model->error_code) {
  212. $this->log('获取小程序AccessToken失败' . $wechat_model->error_message);
  213. return;
  214. }
  215. $minipro_live_goods_model = model('minipro_live_goods');
  216. //未审核的商品、已审核的商品价格不一致的更新
  217. $goods_list = Db::name('minipro_live_goods')->alias('minipro_live_goods')->field('minipro_live_goods_result_id,goodscommon.goods_name,goodscommon.goods_image,minipro_live_goods.goods_image as goods_image2,goodscommon.goods_price,goodscommon.goods_commonid,goodscommon.store_id,goodscommon.store_name')->join('goodscommon goodscommon', 'minipro_live_goods.goods_commonid=goodscommon.goods_commonid')->where(array(array('minipro_live_goods_close', '=', 0)))->where(function ($query) {
  218. $query->where(array(array('minipro_live_goods_state', '=', 2)))->where(array(array('goodscommon.goods_price', '<>', Db::raw('minipro_live_goods.goods_price'))));
  219. })->whereOr(function ($query) {
  220. $query->where(array(array('minipro_live_goods_state', '=', 0)))->where(function ($query) {
  221. $query->whereOr(array(array('goodscommon.goods_price', '<>', Db::raw('minipro_live_goods.goods_price'))))->whereOr(array(array('goodscommon.goods_name', '<>', Db::raw('minipro_live_goods.goods_name'))))->whereOr(array(array('goodscommon.goods_image', '<>', Db::raw('minipro_live_goods.goods_image'))));
  222. });
  223. })->limit(100)->select()->toArray();
  224. foreach ($goods_list as $goods_info) {
  225. $goods_data = array(
  226. 'goodsId' => $goods_info['minipro_live_goods_result_id'],
  227. 'priceType' => 1,
  228. );
  229. if ($goods_info['minipro_live_goods_state'] == 2) {
  230. $goods_data['price'] = $goods_info['goods_price'];
  231. }
  232. if ($goods_info['minipro_live_goods_state'] == 0) {
  233. if ($goods_info['goods_image'] != $goods_info['goods_image2']) {
  234. //上传临时素材
  235. $goods_image = str_replace(UPLOAD_SITE_URL, BASE_UPLOAD_PATH, goods_thumb($goods_info));
  236. $image_path = pathinfo($goods_image, PATHINFO_DIRNAME);
  237. $image_name = pathinfo($goods_image, PATHINFO_BASENAME);
  238. if (!strstr($image_name, '_240')) {
  239. create_albumpic_thumb($image_path, $image_name);
  240. $goods_image = str_ireplace('.', '_240.', $image_name);
  241. }
  242. $res = $wechat_model->uploadMedia('image', $goods_image, "image/jpeg", $image_name);
  243. if (!$res['code']) {
  244. $this->log('上传临时素材失败' . $res['msg']);
  245. continue;
  246. }
  247. $coverImgUrl = $res['data'];
  248. $goods_data['coverImgUrl'] = $coverImgUrl;
  249. }
  250. $goods_data['name'] = mb_substr($goods_info['goods_name'], 0, 14); //商品名称,最长14个汉字
  251. $goods_data['price'] = $goods_info['goods_price'];
  252. }
  253. $data = array(
  254. 'goodsInfo' => $goods_data
  255. );
  256. $res = http_request('https://api.weixin.qq.com/wxaapi/broadcast/goods/update?access_token=' . $accessToken, 'POST', $data);
  257. $res = json_decode($res, true);
  258. if (!$res || $res['errcode']) {
  259. $this->log(isset($res['errmsg']) ? $res['errmsg'] : '更新商品出错,错误码:' . $res['errcode']);
  260. continue;
  261. }
  262. $minipro_live_goods_model->editMiniproLiveGoods(array(
  263. 'goods_name' => $goods_info['goods_name'],
  264. 'goods_price' => $goods_info['goods_price'],
  265. 'goods_image' => $goods_info['goods_image'],
  266. ), array(array('minipro_live_goods_result_id', '=', $goods_info['minipro_live_goods_result_id'])));
  267. }
  268. //未审核的商品ID不存在的更新
  269. $subQuery = Db::name('goods')->field('goods_id')->where('1=1')->buildSql();
  270. $goods_list = Db::name('minipro_live_goods')->where(array(array('minipro_live_goods_close', '=', 0), array('minipro_live_goods_state', '=', 0), array('goods_id', 'not in', Db::raw($subQuery))))->limit(100)->select()->toArray();
  271. foreach ($goods_list as $goods_info) {
  272. $goods_id = Db::name('goods')->where(array(array('goods_commonid', '=', $goods_info['goods_commonid'])))->order('goods_id asc')->value('goods_id');
  273. if (!$goods_id) {
  274. $this->log('商品不存在');
  275. continue;
  276. }
  277. $goods_data = array(
  278. 'goodsId' => $goods_info['minipro_live_goods_result_id'],
  279. 'url' => 'pages/index/index?ret_url=' . urlencode(config('ds_config.h5_site_url') . '/pages/home/goodsdetail/Goodsdetail?goods_id=' . $goods_id)
  280. );
  281. $data = array(
  282. 'goodsInfo' => $goods_data
  283. );
  284. $res = http_request('https://api.weixin.qq.com/wxaapi/broadcast/goods/update?access_token=' . $accessToken, 'POST', $data);
  285. $res = json_decode($res, true);
  286. if (!$res || $res['errcode']) {
  287. $this->log(isset($res['errmsg']) ? $res['errmsg'] : '更新商品出错,错误码:' . $res['errcode']);
  288. continue;
  289. }
  290. $minipro_live_goods_model->editMiniproLiveGoods(array(
  291. 'goods_id' => $goods_id,
  292. ), array(array('minipro_live_goods_result_id', '=', $goods_info['minipro_live_goods_result_id'])));
  293. }
  294. }
  295. /*
  296. * 小程序直播商品状态查询(调用额度:1000次/一天)
  297. */
  298. function _minipro_live_goods_state() {
  299. if (intval(config('ds_config.live_type')) != 1) {
  300. return;
  301. }
  302. //审核时间为1~7天
  303. $wechat_model = model('wechat');
  304. $wechat_model->getOneWxconfig();
  305. $accessToken = $wechat_model->getAccessToken('miniprogram', 0);
  306. if ($wechat_model->error_code) {
  307. $this->log('获取小程序AccessToken失败' . $wechat_model->error_message);
  308. return;
  309. }
  310. $ids = Db::name('minipro_live_goods')->where(array(array('minipro_live_goods_close', '=', 0), array('minipro_live_goods_state', '=', 0)))->limit(200)->column('minipro_live_goods_result_id');
  311. if (!empty($ids)) {
  312. $data = array(
  313. 'goods_ids' => $ids
  314. );
  315. $res = http_request('https://api.weixin.qq.com/wxa/business/getgoodswarehouse?access_token=' . $accessToken, 'POST', json_encode($data));
  316. $res = json_decode($res, true);
  317. if (!$res || $res['errcode']) {
  318. $this->log('小程序直播商品状态查询失败' . (isset($res['errmsg']) ? $res['errmsg'] : ',错误码:' . $res['errcode']));
  319. return;
  320. }
  321. $minipro_live_goods_model = model('minipro_live_goods');
  322. foreach ($res['goods'] as $val) {
  323. if (in_array($val['audit_status'], array(2, 3))) {
  324. $minipro_live_goods_model->editMiniproLiveGoods(array('minipro_live_goods_state' => $val['audit_status']), array(array('minipro_live_goods_result_id', '=', $val['goods_id'])));
  325. }
  326. }
  327. }
  328. }
  329. /*
  330. * 店铺到期关闭
  331. */
  332. function _store_auto_close() {
  333. $store_model = model('store');
  334. $condition = array();
  335. $condition[] = array('store_state', '=', 1);
  336. $condition[] = array('store_endtime', '>', 0);
  337. $condition[] = array('store_endtime', '<', TIMESTAMP);
  338. $store_ids = Db::name('store')->where($condition)->limit(100)->column('store_id');
  339. $goods_model = model('goods');
  340. foreach ($store_ids as $store_id) {
  341. Db::startTrans();
  342. try {
  343. $store_model->editStore(array('store_state' => 0, 'store_recommend' => 0), array(array('store_id', '=', $store_id)));
  344. //根据店铺状态修改该店铺所有商品状态
  345. $goods_model->editProducesOffline(array(array('store_id', '=', $store_id)));
  346. } catch (\Exception $e) {
  347. Db::rollback();
  348. $this->log('到期店铺关闭失败[店铺ID:' . $store_id . ']' . $e->getMessage());
  349. }
  350. Db::commit();
  351. }
  352. }
  353. /*
  354. * 生成店铺结算单
  355. */
  356. public function _creat_bill() {
  357. //搜索商家结算日期小于当前时间减结算周期的所有商家,每次100个
  358. $store_ids = Db::name('store')->where(array(array('store_bill_time', '<', strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400)))->limit(100)->column('store_id');
  359. $storemoneylog_mod = model('storemoneylog');
  360. $orderinviter_model = model('orderinviter');
  361. $bill_model = model('bill');
  362. $trade_model = model('trade');
  363. $order_refund_day = intval($trade_model->getMaxDay('order_refund'));
  364. $order_refund_time = strtotime(date('Y-m-d 0:0:0')) - $order_refund_day * 86400;
  365. $vr_order_refund_day = intval(config('ds_config.code_invalid_refund'));
  366. $vr_order_refund_time = strtotime(date('Y-m-d 0:0:0')) - $vr_order_refund_day * 86400;
  367. foreach ($store_ids as $store_id) {
  368. Db::startTrans();
  369. $store_info = Db::name('store')->where('store_id', $store_id)->field('store_id,store_name,seller_name,is_platform_store,store_bill_time')->lock(true)->find();
  370. if ($store_info) {
  371. try {
  372. $data = array();
  373. //如果店铺没有结算过,则查询与结算单相关项目的最小时间作为结算单开始时间
  374. if (!$store_info['store_bill_time']) {
  375. $ob_startdate = 0;
  376. $storecost_time = Db::name('storecost')->where(array(array('storecost_store_id', '=', $store_id), array('storecost_state', '=', 0), array('storecost_time', '<', strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400)))->order('storecost_time asc')->lock(true)->value('storecost_time');
  377. //取结算时间和退款过期时间的较小者
  378. $temp_time = $order_refund_time;
  379. if ($temp_time > (strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400)) {
  380. $temp_time = strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400;
  381. }
  382. $order_time = Db::name('order')->where(array(array('store_id', '=', $store_id), array('order_state', '=', ORDER_STATE_SUCCESS), array('delay_time', '<', $temp_time), array('payment_code', '<>', 'offline'), array('ob_no', '=', 0)))->order('add_time asc')->lock(true)->value('add_time');
  383. //不允许退款且超过使用期限的、允许退款且超过可退款期限、允许退款并且全部完成退款的虚拟订单
  384. $vr_order_time = Db::name('vrorder')->where(array(array('store_id', '=', $store_id), array('order_state', '=', ORDER_STATE_SUCCESS), array('finnshed_time', '<', strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400), array('ob_no', '=', 0)))->where('((vr_invalid_refund=0) || (vr_invalid_refund=1 AND vr_indate<' . $vr_order_refund_time . ') || (vr_invalid_refund=1 AND refund_state=2))')->order('add_time asc')->lock(true)->value('add_time');
  385. $ob_startdate = (!$ob_startdate || ($ob_startdate && $storecost_time && $storecost_time < $ob_startdate)) ? $storecost_time : $ob_startdate;
  386. $ob_startdate = (!$ob_startdate || ($ob_startdate && $order_time && $order_time < $ob_startdate)) ? $order_time : $ob_startdate;
  387. $ob_startdate = (!$ob_startdate || ($ob_startdate && $vr_order_time && $vr_order_time < $ob_startdate)) ? $vr_order_time : $ob_startdate;
  388. } else {
  389. $ob_startdate = $store_info['store_bill_time'];
  390. }
  391. if ($ob_startdate) {
  392. //零点开始
  393. $ob_startdate = strtotime(date('Y-m-d 0:0:0', $ob_startdate));
  394. $ob_enddate = $ob_startdate + intval(config('ds_config.store_bill_cycle')) * 86400;
  395. //店铺促销活动费用
  396. $storecost_sum = Db::name('storecost')->where(array(array('storecost_store_id', '=', $store_id), array('storecost_state', '=', 0), array('storecost_time', '<', $ob_enddate)))->field('SUM(storecost_price) AS ob_store_cost_totals,COUNT(*) AS count')->lock(true)->find();
  397. //取结算时间和退款过期时间的较小者
  398. $temp_time = $order_refund_time;
  399. if ($temp_time > $ob_enddate) {
  400. $temp_time = $ob_enddate;
  401. }
  402. $order_ids = Db::name('order')->where(array(array('store_id', '=', $store_id), array('order_state', '=', ORDER_STATE_SUCCESS), array('delay_time', '<', $temp_time), array('payment_code', '<>', 'offline'), array('ob_no', '=', 0)))->lock(true)->column('order_id');
  403. $vr_order_ids = Db::name('vrorder')->where(array(array('store_id', '=', $store_id), array('order_state', '=', ORDER_STATE_SUCCESS), array('finnshed_time', '<', $temp_time), array('ob_no', '=', 0)))->where('((vr_invalid_refund=0) || (vr_invalid_refund=1 AND vr_indate<' . $vr_order_refund_time . ') || (vr_invalid_refund=1 AND refund_state=2))')->lock(true)->column('order_id');
  404. if ($storecost_sum && $storecost_sum['count']) {
  405. $data['ob_store_cost_totals'] = $storecost_sum['ob_store_cost_totals'];
  406. }
  407. if ($order_ids) {
  408. //查看相关订单是否有未完成的退款
  409. $r_order_ids = Db::name('refundreturn')->where('order_id', 'in', $order_ids)->where('refund_state', 'not in', [3, 4])->lock(true)->column('order_id');
  410. if ($r_order_ids) {
  411. $order_ids = array_diff($order_ids, $r_order_ids);
  412. }
  413. //如果全部为待处理退款的订单,则不生成结算单,等退款完成再处理
  414. if ($order_ids) {
  415. //结算订单金额,结算运费,结算退单金额
  416. $order_sum = Db::name('order')->where('order_id', 'in', $order_ids)->field('SUM(order_amount) AS ob_order_totals,SUM(shipping_fee) AS ob_shipping_totals,SUM(refund_amount) AS ob_order_return_totals')->lock(true)->find();
  417. //结算平台代金券金额
  418. $ob_mall_voucher_totals = Db::name('ordercommon')->where('order_id','in',$order_ids)->field('SUM(mallvoucher_price) AS ob_mall_voucher_totals')->lock(true)->find();
  419. //结算佣金金额
  420. $ordergoods_sum = Db::name('ordergoods')->where('order_id', 'in', $order_ids)->field('SUM(ROUND(goods_pay_price*commis_rate/100,2)) AS ob_commis_totals')->lock(true)->find();
  421. //结算退还佣金,部分退款的
  422. $refundreturn_sum_1 = Db::name('refundreturn')->alias('refundreturn')->join('ordergoods ordergoods', 'refundreturn.order_goods_id = ordergoods.rec_id')->where('refundreturn.order_id', 'in', $order_ids)->where('refundreturn.refund_state', 3)->where('refundreturn.seller_state', 2)->where('refundreturn.order_goods_id', '>', 0)->field('SUM(ROUND(refundreturn.refund_amount*ordergoods.commis_rate/100,2)) AS ob_commis_return_totals')->lock(true)->find();
  423. $refundreturn_sum_2 = array();
  424. $refundreturn_rec_ids_2 = Db::name('refundreturn')->where('order_id', 'in', $order_ids)->where('refund_state', 3)->where('seller_state', 2)->where('order_goods_id', 0)->lock(true)->column('order_id');
  425. if ($refundreturn_rec_ids_2) {
  426. $refundreturn_sum_2 = Db::name('ordergoods')->where('order_id', 'in', $refundreturn_rec_ids_2)->field('SUM(ROUND(goods_pay_price*commis_rate/100,2)) AS ob_commis_return_totals')->lock(true)->find();
  427. }
  428. foreach ($order_ids as $order_id) {
  429. $orderinviter_model->giveMoney($order_id, 0);
  430. }
  431. //分销的佣金
  432. $orderinviter_sum = Db::name('orderinviter')->where('orderinviter_order_id', 'in', $order_ids)->where('orderinviter_valid', 1)->where('orderinviter_order_type', 0)->field('SUM(orderinviter_money) AS ob_inviter_totals')->lock(true)->find();
  433. $data['ob_order_totals'] = $order_sum['ob_order_totals'];
  434. $data['ob_shipping_totals'] = $order_sum['ob_shipping_totals'];
  435. $data['ob_order_return_totals'] = $order_sum['ob_order_return_totals'];
  436. $data['ob_commis_totals'] = $ordergoods_sum['ob_commis_totals'];
  437. $data['ob_commis_return_totals'] = bcadd(((!empty($refundreturn_sum_1)) ? $refundreturn_sum_1['ob_commis_return_totals'] : 0), ((!empty($refundreturn_sum_2)) ? $refundreturn_sum_2['ob_commis_return_totals'] : 0), 2);
  438. $data['ob_inviter_totals'] = (!empty($orderinviter_sum)) ? $orderinviter_sum['ob_inviter_totals'] : 0;
  439. }
  440. }
  441. if ($vr_order_ids) {
  442. foreach ($vr_order_ids as $order_id) {
  443. $orderinviter_model->giveMoney($order_id, 1);
  444. }
  445. //分销的佣金
  446. $orderinviter_sum = Db::name('orderinviter')->where('orderinviter_order_id', 'in', $vr_order_ids)->where('orderinviter_valid', 1)->where('orderinviter_order_type', 1)->field('SUM(orderinviter_money) AS ob_vr_inviter_totals')->lock(true)->find();
  447. $vr_order_sum = Db::name('vrorder')->where('order_id', 'in', $vr_order_ids)->field('SUM(order_amount) AS ob_vr_order_totals,SUM(ROUND(order_amount*commis_rate/100,2)) AS ob_vr_commis_totals,SUM(refund_amount) AS ob_vr_order_return_totals,SUM(ROUND(refund_amount*commis_rate/100,2)) AS ob_vr_commis_return_totals')->lock(true)->find();
  448. $data['ob_vr_order_totals'] = $vr_order_sum['ob_vr_order_totals'];
  449. $data['ob_vr_commis_totals'] = $vr_order_sum['ob_vr_commis_totals'];
  450. $data['ob_vr_order_return_totals'] = $vr_order_sum['ob_vr_order_return_totals'];
  451. $data['ob_vr_commis_return_totals'] = $vr_order_sum['ob_vr_commis_return_totals'];
  452. $data['ob_vr_inviter_totals'] = (!empty($orderinviter_sum)) ? $orderinviter_sum['ob_vr_inviter_totals'] : 0;
  453. }
  454. if (!empty($data)) {
  455. $data['ob_createdate'] = TIMESTAMP;
  456. $data['ob_startdate'] = $ob_startdate;
  457. $data['ob_enddate'] = $ob_enddate;
  458. $data['ob_store_id'] = $store_id;
  459. $data['ob_store_name'] = $store_info['store_name']; //平台自营店铺不需要增加店铺资金
  460. $data['ob_state'] = ($store_info['is_platform_store'] == 1) ? 4 : 1;
  461. $data['ob_mall_voucher_totals'] = 0;
  462. if (!isset($data['ob_order_totals'])) {
  463. $data['ob_order_totals'] = 0;
  464. }
  465. if (!isset($data['ob_store_cost_totals'])) {
  466. $data['ob_store_cost_totals'] = 0;
  467. }
  468. if (!isset($data['ob_inviter_totals'])) {
  469. $data['ob_inviter_totals'] = 0;
  470. }
  471. if (!isset($data['ob_order_return_totals'])) {
  472. $data['ob_order_return_totals'] = 0;
  473. }
  474. if (!isset($data['ob_commis_totals'])) {
  475. $data['ob_commis_totals'] = 0;
  476. }
  477. if (!isset($data['ob_commis_return_totals'])) {
  478. $data['ob_commis_return_totals'] = 0;
  479. }
  480. if (!isset($data['ob_vr_order_totals'])) {
  481. $data['ob_vr_order_totals'] = 0;
  482. }
  483. if (!isset($data['ob_vr_order_return_totals'])) {
  484. $data['ob_vr_order_return_totals'] = 0;
  485. }
  486. if (!isset($data['ob_vr_commis_totals'])) {
  487. $data['ob_vr_commis_totals'] = 0;
  488. }
  489. if (!isset($data['ob_vr_commis_return_totals'])) {
  490. $data['ob_vr_commis_return_totals'] = 0;
  491. }
  492. if (!isset($data['ob_vr_inviter_totals'])) {
  493. $data['ob_vr_inviter_totals'] = 0;
  494. }
  495. if(!isset($ob_mall_voucher_totals) && !isset($vr_order_sum['ob_vr_mall_voucher_totals'])){
  496. $data['ob_mall_voucher_totals'] = 0;
  497. }else{
  498. $data['ob_mall_voucher_totals'] = bcadd($ob_mall_voucher_totals['ob_mall_voucher_totals'], $vr_order_sum['ob_vr_mall_voucher_totals'],2);
  499. }
  500. //计算实际可得金额
  501. $data['ob_result_totals'] = round($data['ob_order_totals'] - $data['ob_store_cost_totals'] - $data['ob_inviter_totals'] - $data['ob_order_return_totals'] - $data['ob_commis_totals'] + $data['ob_commis_return_totals'] + $data['ob_vr_order_totals'] - $data['ob_vr_order_return_totals'] - $data['ob_vr_commis_totals'] + $data['ob_vr_commis_return_totals'] - $data['ob_vr_inviter_totals'] + $data['ob_mall_voucher_totals'], 2);
  502. //插入到结算账单
  503. $ob_no = $bill_model->addOrderbill($data);
  504. if (!$ob_no) {
  505. throw new \think\Exception('新增结算单失败', 10006);
  506. }
  507. if ($order_ids) {
  508. //订单更新结算单号
  509. if (!Db::name('order')->where('order_id', 'in', $order_ids)->update(array('ob_no' => $ob_no))) {
  510. throw new \think\Exception('更新实物订单结算单号失败', 10006);
  511. }
  512. }
  513. if ($vr_order_ids) {
  514. //订单更新结算单号
  515. if (!Db::name('vrorder')->where('order_id', 'in', $vr_order_ids)->update(array('ob_no' => $ob_no))) {
  516. throw new \think\Exception('更新虚拟订单结算单号失败', 10006);
  517. }
  518. }
  519. }
  520. //店铺活动改成已结算
  521. if (!empty($storecost_sum) && $storecost_sum['count'] && !Db::name('storecost')->where(array(array('storecost_store_id', '=', $store_id), array('storecost_state', '=', 0), array('storecost_time', '<', $ob_enddate)))->update(array('storecost_state' => 1))) {
  522. throw new \think\Exception('更新店铺活动结算单号失败', 10006);
  523. }
  524. } else {
  525. $ob_enddate = strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400;
  526. }
  527. //更新店铺结算时间
  528. if (!Db::name('store')->where('store_id', $store_id)->update(array('store_bill_time' => $ob_enddate))) {
  529. throw new \think\Exception('更新店铺结算时间失败', 10006);
  530. }
  531. } catch (\Exception $e) {
  532. Db::rollback();
  533. $this->log('[店铺名称:' . $store_info['store_name'] . ']' . $e->getMessage());
  534. }
  535. }
  536. Db::commit();
  537. }
  538. //如果还有未更新结算日期的店铺,则重定向继续生产结算单
  539. if (Db::name('store')->where(array(array('store_bill_time', '<', strtotime(date('Y-m-d 0:0:0')) - intval(config('ds_config.store_bill_cycle')) * 86400)))->count() > 0) {
  540. $this->redirect('date/index');
  541. }
  542. }
  543. /**
  544. * 未付款订单超期自动关闭
  545. */
  546. private function _order_timeout_cancel() {
  547. //实物订单超期未支付系统自动关闭
  548. $_break = false;
  549. $chain_order_model = model('chain_order');
  550. $order_model = model('order');
  551. $logic_order = model('order', 'logic');
  552. $condition = array();
  553. $condition[] = array('order_state', '=', ORDER_STATE_NEW);
  554. $condition[] = array('add_time', '<', TIMESTAMP - config('ds_config.order_auto_cancel_day') * self::EXE_TIMES);
  555. //分批,每批处理100个订单,最多处理5W个订单
  556. for ($i = 0; $i < 500; $i++) {
  557. if ($_break) {
  558. break;
  559. }
  560. $order_list = $order_model->getOrderList($condition, '', '*', '', 100);
  561. if (empty($order_list))
  562. break;
  563. foreach ($order_list as $order_info) {
  564. try{
  565. Db::startTrans();
  566. $logic_order->changeOrderStateCancel($order_info, 'system', '系统', '超期未支付系统自动关闭订单', true, false);
  567. } catch (\Exception $e) {
  568. Db::rollback();
  569. $this->log('实物订单超期未支付关闭失败SN:' . $order_info['order_sn']);
  570. $_break = true;
  571. break;
  572. }
  573. Db::commit();
  574. }
  575. }
  576. //虚拟订单超期未支付系统自动关闭
  577. $_break = false;
  578. $vrorder_model = model('vrorder');
  579. $logic_vrorder = model('vrorder', 'logic');
  580. $condition = array();
  581. $condition[] = array('order_state', '=', ORDER_STATE_NEW);
  582. $condition[] = array('add_time', '<', TIMESTAMP - config('ds_config.order_auto_cancel_day') * self::EXE_TIMES);
  583. //分批,每批处理100个订单,最多处理5W个订单
  584. for ($i = 0; $i < 500; $i++) {
  585. if ($_break) {
  586. break;
  587. }
  588. $order_list = $vrorder_model->getVrorderList($condition, '', '*', '', 100);
  589. if (empty($order_list))
  590. break;
  591. foreach ($order_list as $order_info) {
  592. $result = $logic_vrorder->changeOrderStateCancel($order_info, 'system', '超期未支付系统自动关闭订单', false);
  593. }
  594. if (!$result['code']) {
  595. $this->log('虚拟订单超期未支付关闭失败SN:' . $order_info['order_sn']);
  596. $_break = true;
  597. break;
  598. }
  599. }
  600. }
  601. /**
  602. * 订单自动完成
  603. */
  604. private function _order_auto_complete() {
  605. //虚拟订单过使用期自动完成
  606. $_break = false;
  607. $vrorder_model = model('vrorder');
  608. $logic_vrorder = model('vrorder', 'logic');
  609. $condition = array();
  610. $condition[] = array('order_state', '=', ORDER_STATE_PAY);
  611. $condition[] = array('vr_indate', '<', TIMESTAMP);
  612. //分批,每批处理100个订单,最多处理5W个订单
  613. for ($i = 0; $i < 500; $i++) {
  614. if ($_break) {
  615. break;
  616. }
  617. $order_list = $vrorder_model->getVrorderList($condition, '', 'order_id,order_sn', 'vr_indate asc', 100);
  618. if (empty($order_list))
  619. break;
  620. foreach ($order_list as $order_info) {
  621. $result = $logic_vrorder->changeOrderStateSuccess($order_info['order_id']);
  622. if (!$result['code']) {
  623. $this->log('虚拟订单过使用期自动完成失败SN:' . $order_info['order_sn']);
  624. $_break = true;
  625. break;
  626. }
  627. }
  628. }
  629. //实物订单发货后,超期自动收货完成
  630. $_break = false;
  631. $order_model = model('order');
  632. $logic_order = model('order', 'logic');
  633. $condition = array();
  634. $condition[] = array('order_state', '=', ORDER_STATE_SEND);
  635. $condition[] = array('lock_state', '=', 0);
  636. $condition[] = array('delay_time', '<', TIMESTAMP - config('ds_config.order_auto_receive_day') * 86400);
  637. //分批,每批处理100个订单,最多处理5W个订单
  638. for ($i = 0; $i < 500; $i++) {
  639. if ($_break) {
  640. break;
  641. }
  642. $order_list = $order_model->getOrderList($condition, '', '*', 'delay_time asc', 100);
  643. if (empty($order_list))
  644. break;
  645. foreach ($order_list as $order_info) {
  646. $result = $logic_order->changeOrderStateReceive($order_info, 'system', '系统', '超期未收货系统自动完成订单');
  647. if (!$result['code']) {
  648. $this->log('实物订单超期未收货自动完成订单失败SN:' . $order_info['order_sn']);
  649. $_break = true;
  650. break;
  651. }
  652. }
  653. }
  654. }
  655. /**
  656. * 更新订单扩展表中收货人所在省份ID
  657. */
  658. private function _order_reciver_provinceid_update() {
  659. $order_model = model('order');
  660. $area_model = model('area');
  661. //每次最多处理5W个订单
  662. $condition = array();
  663. $condition[] = array('reciver_province_id', '=', 0);
  664. $condition[] = array('reciver_city_id', '<>', 0);
  665. for ($i = 0; $i < 500; $i++) {
  666. $order_list = $order_model->getOrdercommonList($condition, 'reciver_city_id', 'order_id desc', 100);
  667. if (!empty($order_list)) {
  668. $city_ids = array();
  669. foreach ($order_list as $v) {
  670. if (!in_array($v['reciver_city_id'], $city_ids)) {
  671. $city_ids[] = $v['reciver_city_id'];
  672. }
  673. }
  674. $area_list = $area_model->getAreaList(array(array('area_id', 'in', $city_ids)), 'area_parent_id,area_id');
  675. if (!empty($area_list)) {
  676. foreach ($area_list as $v) {
  677. $update = $order_model->editOrdercommon(array('reciver_province_id' => $v['area_parent_id']), array('reciver_city_id' => $v['area_id']));
  678. if (!$update) {
  679. $this->log('更新订单扩展表中收货人所在省份ID失败');
  680. break;
  681. }
  682. }
  683. }
  684. } else {
  685. break;
  686. }
  687. }
  688. }
  689. /**
  690. * 增加会员积分和经验值
  691. */
  692. private function _add_points() {
  693. return;
  694. $points_model = model('points');
  695. $exppoints_model = model('exppoints');
  696. //24小时之内登录的会员送积分和经验值,每次最多处理5W个会员
  697. $member_model = model('member');
  698. $condition = array();
  699. $condition[] = array('member_logintime', '>', TIMESTAMP - self::EXE_TIMES);
  700. for ($i = 0; $i < 50000; $i = $i + 100) {
  701. $member_list = $member_model->getMemberList($condition, 'member_name,member_id', 0, '', "{$i},100");
  702. if (!empty($member_list)) {
  703. foreach ($member_list as $member_info) {
  704. if (config('ds_config.points_isuse')) {
  705. $points_model->savePointslog('login', array('pl_memberid' => $member_info['member_id'], 'pl_membername' => $member_info['member_name']), true);
  706. }
  707. $exppoints_model->saveExppointslog('login', array('explog_memberid' => $member_info['member_id'], 'explog_membername' => $member_info['member_name']), true);
  708. }
  709. } else {
  710. break;
  711. }
  712. }
  713. //24小时之内注册的会员送积分,每次最多处理5W个会员
  714. if (config('ds_config.points_isuse')) {
  715. $condition = array();
  716. $condition[] = array('member_addtime', '>', TIMESTAMP - self::EXE_TIMES);
  717. for ($i = 0; $i < 50000; $i = $i + 100) {
  718. $member_list = $member_model->getMemberList($condition, 'member_name,member_id', 0, 'member_id desc', "{$i},100");
  719. if (!empty($member_list)) {
  720. foreach ($member_list as $member_info) {
  721. $points_model->savePointslog('regist', array('pl_memberid' => $member_info['member_id'], 'pl_membername' => $member_info['member_name']), true);
  722. }
  723. } else {
  724. break;
  725. }
  726. }
  727. }
  728. //24小时之内完成了实物订单送积分和经验值,每次最多处理5W个订单
  729. $order_model = model('order');
  730. $condition = array();
  731. $condition[] = array('finnshed_time', '>', TIMESTAMP - self::EXE_TIMES);
  732. for ($i = 0; $i < 50000; $i = $i + 100) {
  733. $order_list = $order_model->getOrderList($condition, '', 'buyer_name,buyer_id,order_amount,order_sn,order_id', '', "{$i},100");
  734. if (!empty($order_list)) {
  735. foreach ($order_list as $order_info) {
  736. if (config('ds_config.points_isuse')) {
  737. $points_model->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);
  738. }
  739. $exppoints_model->saveExppointslog('order', array('explog_memberid' => $order_info['buyer_id'], 'explog_membername' => $order_info['buyer_name'], 'orderprice' => $order_info['order_amount'], 'order_sn' => $order_info['order_sn'], 'order_id' => $order_info['order_id']), true);
  740. }
  741. } else {
  742. break;
  743. }
  744. }
  745. //24小时之内完成了实物订单送积分和经验值,每次最多处理5W个订单
  746. $vrorder_model = model('vrorder');
  747. $condition = array();
  748. $condition[] = array('finnshed_time', '>', TIMESTAMP - self::EXE_TIMES);
  749. for ($i = 0; $i < 50000; $i = $i + 100) {
  750. $order_list = $vrorder_model->getVrorderList($condition, '', 'buyer_name,buyer_id,order_amount,order_sn,order_id', '', "{$i},100");
  751. if (!empty($order_list)) {
  752. foreach ($order_list as $order_info) {
  753. if (config('ds_config.points_isuse')) {
  754. $points_model->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);
  755. }
  756. $exppoints_model->saveExppointslog('order', array('explog_memberid' => $order_info['buyer_id'], 'explog_membername' => $order_info['buyer_name'], 'orderprice' => $order_info['order_amount'], 'order_sn' => $order_info['order_sn'], 'order_id' => $order_info['order_id']), true);
  757. }
  758. } else {
  759. break;
  760. }
  761. }
  762. }
  763. /**
  764. * 代金券即将过期提醒
  765. */
  766. private function _voucher_will_expire() {
  767. $time_start = mktime(0, 0, 0, date("m"), date("d") + self::VOUCHER_INTERVAL, date("Y"));
  768. $time_stop = $time_start + self::EXE_TIMES - 1;
  769. $where = array();
  770. $where[] = array('voucher_enddate', '>=', $time_start);
  771. $where[] = array('voucher_enddate', '<=', $time_stop);
  772. $list = model('voucher')->getVoucherUnusedList($where);
  773. if (!empty($list)) {
  774. foreach ($list as $val) {
  775. $param = array();
  776. $param['code'] = 'voucher_will_expire';
  777. $param['member_id'] = $val['voucher_owner_id'];
  778. $param['ali_param'] = array(
  779. 'indate' => date('Y-m-d H:i:s', $val['voucher_enddate']),
  780. );
  781. $param['ten_param'] = array(
  782. date('Y-m-d H:i:s', $val['voucher_enddate']),
  783. );
  784. $param['param'] = array_merge($param['ali_param'], array(
  785. 'voucher_url' => HOME_SITE_URL .'/Membervoucher/index'
  786. ));
  787. $param['weixin_param'] = array(
  788. 'url' => config('ds_config.h5_site_url') . '/pages/member/voucher/VoucherList',
  789. 'data' => array(
  790. "keyword1" => array(
  791. "value" => $val['voucher_code'],
  792. "color" => "#333"
  793. ),
  794. "keyword2" => array(
  795. "value" => date('Y-m-d', $val['voucher_startdate']) . '~' . date('Y-m-d', $val['voucher_enddate']),
  796. "color" => "#333"
  797. )
  798. ),
  799. );
  800. model('cron')->addCron(array('cron_exetime'=>TIMESTAMP,'cron_type'=>'sendMemberMsg','cron_value'=>serialize($param)));
  801. }
  802. }
  803. }
  804. /**
  805. * 虚拟兑换码即将过期提醒
  806. */
  807. private function _vr_code_will_expire() {
  808. $time_start = mktime(0, 0, 0, date("m"), date("d") + self::VR_CODE_INTERVAL, date("Y"));
  809. $time_stop = $time_start + self::EXE_TIMES - 1;
  810. $where = array();
  811. $where[] = array('vr_indate', '>=', $time_start);
  812. $where[] = array('vr_indate', '<=', $time_stop);
  813. $list = model('vrorder')->getCodeUnusedList($where);
  814. if (!empty($list)) {
  815. foreach ($list as $val) {
  816. $param = array();
  817. $param['code'] = 'vr_code_will_expire';
  818. $param['member_id'] = $val['buyer_id'];
  819. $param['ali_param'] = array(
  820. 'indate' => date('Y-m-d H:i:s', $val['vr_indate']),
  821. );
  822. $param['ten_param'] = array(
  823. date('Y-m-d H:i:s', $val['vr_indate']),
  824. );
  825. $param['param'] = array_merge($param['ali_param'], array(
  826. 'vr_order_url' => HOME_SITE_URL .'/Membervrorder/index'
  827. ));
  828. $vrorder = model('vrorder')->getVrorderInfo(array('order_id' => $val['order_id']));
  829. $param['weixin_param'] = array(
  830. 'url' => config('ds_config.h5_site_url') . '/pages/member/vrorder/OrderDetail?order_id=' . $val['order_id'],
  831. 'data' => array(
  832. "keyword1" => array(
  833. "value" => (!empty($vrorder)) ? $vrorder['goods_name'] : '',
  834. "color" => "#333"
  835. ),
  836. "keyword2" => array(
  837. "value" => date('Y-m-d', $val['vr_indate']),
  838. "color" => "#333"
  839. )
  840. ),
  841. );
  842. model('cron')->addCron(array('cron_exetime'=>TIMESTAMP,'cron_type'=>'sendMemberMsg','cron_value'=>serialize($param)));
  843. }
  844. }
  845. }
  846. /**
  847. * 订单超期后不允许评价
  848. */
  849. private function _order_eval_expire_update() {
  850. //实物订单超期未评价自动更新状态,每次最多更新1000个订单
  851. $order_model = model('order');
  852. $condition = array();
  853. $condition[] = array('order_state', '=', ORDER_STATE_SUCCESS);
  854. $condition[] = array('evaluation_state', '=', 0);
  855. $condition[] = array('finnshed_time', '<', TIMESTAMP - self::ORDER_EVALUATE_TIME);
  856. $update = array();
  857. $update['evaluation_state'] = 2;
  858. $update = $order_model->editOrder($update, $condition, 1000);
  859. if (!$update) {
  860. $this->log('更新实物订单超期不能评价失败');
  861. }
  862. //虚拟订单超期未评价自动更新状态,每次最多更新1000个订单
  863. $vrorder_model = model('vrorder');
  864. $condition = array();
  865. $condition[] = array('order_state', '=', ORDER_STATE_SUCCESS);
  866. $condition[] = array('evaluation_state', '=', 0);
  867. $condition[] = array('use_state', '=', 1);
  868. $condition[] = array('finnshed_time', '<', TIMESTAMP - self::ORDER_EVALUATE_TIME);
  869. $update = array();
  870. $update['evaluation_state'] = 2;
  871. $update = $vrorder_model->editVrorder($update, $condition, 1000);
  872. if (!$update) {
  873. $this->log('更新虚拟订单超期不能评价失败');
  874. }
  875. }
  876. /**
  877. * 更新商品访问量(redis)
  878. */
  879. private function _goods_click_update() {
  880. $data = rcache('updateRedisDate', 'goodsClick');
  881. if (is_array($data)) {
  882. foreach ($data as $key => $val) {
  883. model('goods')->editGoodsById(array('goods_click' => Db::raw('goods_click+' . $val)), $key);
  884. }
  885. }
  886. dcache('updateRedisDate', 'goodsClick');
  887. }
  888. /**
  889. * 更新商品促销到期状态(目前只有满即送)
  890. */
  891. private function _goods_promotion_state_update() {
  892. //满即送过期
  893. model('pmansong')->editExpireMansong();
  894. }
  895. /**
  896. * 商品到货通知提醒
  897. */
  898. private function _arrival_notice() {
  899. $arrivalnotice_model = model('arrivalnotice');
  900. $count = $arrivalnotice_model->getArrivalNoticeCount(array());
  901. $times = ceil($count / self::ARRIVAL_NOTICE_NUM);
  902. if ($times == 0)
  903. return false;
  904. for ($i = 0; $i <= $times; $i++) {
  905. $notice_list = $arrivalnotice_model->getArrivalNoticeList(array(), '*', $i . ',' . self::ARRIVAL_NOTICE_NUM);
  906. if (empty($notice_list))
  907. continue;
  908. // 查询商品是否已经上架
  909. $goodsid_array = array();
  910. foreach ($notice_list as $val) {
  911. $goodsid_array[] = $val['goods_id'];
  912. }
  913. $goodsid_array = array_unique($goodsid_array);
  914. $condition = array();
  915. $condition[] = array('goods_id', 'in', $goodsid_array);
  916. $condition[] = array('goods_storage', '>', 0);
  917. $goods_list = model('goods')->getGoodsOnlineList($condition, 'goods_id');
  918. if (empty($goods_list))
  919. continue;
  920. // 需要通知到货的商品
  921. $goodsid_array = array();
  922. $storage_array = array();
  923. foreach ($goods_list as $val) {
  924. $goodsid_array[] = $val['goods_id'];
  925. $storage_array[$val['goods_id']] = $val['goods_storage'];
  926. }
  927. // 根据商品id重新查询需要通知的列表
  928. $notice_list = $arrivalnotice_model->getArrivalNoticeList(array(array('goods_id', 'in', $goodsid_array)), '*');
  929. if (empty($notice_list))
  930. continue;
  931. foreach ($notice_list as $val) {
  932. $param = array();
  933. $param['code'] = 'arrival_notice';
  934. $param['member_id'] = $val['member_id'];
  935. $param['ali_param'] = array(
  936. 'goods_name' => $val['goods_name'],
  937. );
  938. $param['ten_param'] = array(
  939. $val['goods_name'],
  940. );
  941. $param['param'] = array_merge(array(
  942. 'goods_name' => $val['goods_name'],
  943. 'goods_url' => HOME_SITE_URL .'/Goods/index?goods_id='.$val['goods_id']
  944. ));
  945. $param['number'] = array('mobile' => $val['arrivalnotice_mobile'], 'email' => $val['arrivalnotice_email']);
  946. $param['weixin_param'] = array(
  947. 'url' => config('ds_config.h5_site_url') . '/pages/home/goodsdetail/Goodsdetail?goods_id=' . $val['goods_id'],
  948. 'data' => array(
  949. "keyword1" => array(
  950. "value" => $val['goods_name'],
  951. "color" => "#333"
  952. ),
  953. "keyword2" => array(
  954. "value" => isset($storage_array[$val['goods_id']]) ? $storage_array[$val['goods_id']] : '99',
  955. "color" => "#333"
  956. ),
  957. "keyword3" => array(
  958. "value" => date('Y-m-d'),
  959. "color" => "#333"
  960. )
  961. ),
  962. );
  963. model('cron')->addCron(array('cron_exetime'=>TIMESTAMP,'cron_type'=>'sendMemberMsg','cron_value'=>serialize($param)));
  964. }
  965. // 清楚发送成功的数据
  966. $arrivalnotice_model->editArrivalNotice(['arrivalnotice_state' => 2, 'arrivalnotice_time' => TIMESTAMP], array(array('goods_id', 'in', $goodsid_array)));
  967. }
  968. }
  969. /**
  970. * 将缓存中的浏览记录存入数据库中,并删除30天前的浏览历史
  971. */
  972. private function _goods_browse_update() {
  973. $goodsbrowse_model = model('goodsbrowse');
  974. //将cache中的记录存入数据库
  975. //如果浏览记录已经存入了缓存中,则将其整理到数据库中
  976. //上次更新缓存的时间
  977. $latest_record = $goodsbrowse_model->getOneGoodsbrowse(array(), '', 'goodsbrowse_time desc');
  978. $starttime = ($t = intval($latest_record['goodsbrowse_time'])) ? $t : 0;
  979. $monthago = strtotime(date('Y-m-d', TIMESTAMP)) - 86400 * 30;
  980. $member_model = model('member');
  981. //查询会员信息总条数
  982. $countnum = $member_model->getMemberCount(array());
  983. $eachnum = 100;
  984. for ($i = 0; $i < $countnum; $i += $eachnum) {//每次查询100条
  985. $member_list = $member_model->getMemberList(array(), '*', 0, 'member_id asc', "$i,$eachnum");
  986. foreach ((array) $member_list as $k => $v) {
  987. $insert_arr = array();
  988. $goodsid_arr = array();
  989. //生成缓存的键值
  990. $hash_key = $v['member_id'];
  991. $browse_goodsid = rcache($hash_key, 'goodsbrowse');
  992. if ($browse_goodsid) {
  993. //删除缓存中多余的浏览历史记录,仅保留最近的30条浏览历史,先取出最近30条浏览历史的商品ID
  994. $cachegoodsid_arr = $browse_goodsid['goodsid'] ? unserialize($browse_goodsid['goodsid']) : array();
  995. unset($browse_goodsid['goodsid']);
  996. if ($cachegoodsid_arr) {
  997. $cachegoodsid_arr = array_slice($cachegoodsid_arr, -30, 30, true);
  998. }
  999. //处理存入数据库的浏览历史缓存信息
  1000. $_cache = rcache($hash_key, 'goodsbrowse');
  1001. foreach ((array) $_cache as $c_k => $c_v) {
  1002. $c_v = unserialize($c_v);
  1003. if (isset($c_v['goodsbrowse_time']) && $c_v['goodsbrowse_time'] >= $starttime) {//如果 缓存中的数据未更新到数据库中(即添加时间大于上次更新到数据库中的数据时间)则将数据更新到数据库中
  1004. $tmp_arr = array();
  1005. $tmp_arr['goods_id'] = $c_v['goods_id'];
  1006. $tmp_arr['member_id'] = $v['member_id'];
  1007. $tmp_arr['goodsbrowse_time'] = $c_v['goodsbrowse_time'];
  1008. $tmp_arr['gc_id'] = $c_v['gc_id'];
  1009. $tmp_arr['gc_id_1'] = $c_v['gc_id_1'];
  1010. $tmp_arr['gc_id_2'] = $c_v['gc_id_2'];
  1011. $tmp_arr['gc_id_3'] = $c_v['gc_id_3'];
  1012. $insert_arr[] = $tmp_arr;
  1013. $goodsid_arr[] = $c_v['goods_id'];
  1014. }
  1015. //除了最近的30条浏览历史之外多余的浏览历史记录或者30天之前的浏览历史从缓存中删除
  1016. if (!in_array($c_v['goods_id'], $cachegoodsid_arr) || $c_v['goodsbrowse_time'] < $monthago) {
  1017. unset($_cache[$c_k]);
  1018. }
  1019. }
  1020. //删除已经存在的该商品浏览记录
  1021. if ($goodsid_arr) {
  1022. $condition = array();
  1023. $condition[] = array('member_id', '=', $v['member_id']);
  1024. $condition[] = array('goods_id', 'in', $goodsid_arr);
  1025. $goodsbrowse_model->delGoodsbrowse($condition);
  1026. }
  1027. //将缓存中的浏览历史存入数据库
  1028. if ($insert_arr) {
  1029. $goodsbrowse_model->addGoodsbrowseAll($insert_arr);
  1030. }
  1031. //重新赋值浏览历史缓存
  1032. dcache($hash_key, 'goodsbrowse');
  1033. $_cache['goodsid'] = serialize($cachegoodsid_arr);
  1034. wcache($hash_key, $_cache, 'goodsbrowse');
  1035. }
  1036. }
  1037. }
  1038. //删除30天前的浏览历史
  1039. $goodsbrowse_model->delGoodsbrowse(array(array('goodsbrowse_time', '<', $monthago)));
  1040. }
  1041. /**
  1042. * 缓存订单及订单商品相关数据
  1043. */
  1044. private function _order_goods_cache() {
  1045. //查询最后统计的记录
  1046. $latest_record = Db::name('statordergoods')->order('stat_updatetime desc,rec_id desc')->find();
  1047. $stime = 0;
  1048. if ($latest_record) {
  1049. $start_time = strtotime(date('Y-m-d', $latest_record['stat_updatetime']));
  1050. } else {
  1051. $start_time = strtotime(date('Y-m-d', strtotime(config('ds_config.setup_date')))); //从系统的安装时间开始统计
  1052. }
  1053. for ($stime = $start_time; $stime < TIMESTAMP; $stime = $stime + 86400) {
  1054. $etime = $stime + 86400 - 1;
  1055. //避免重复统计,开始时间必须大于最后一条记录的记录时间
  1056. $search_stime = $latest_record['stat_updatetime'] > $stime ? $latest_record['stat_updatetime'] : $stime;
  1057. //统计一天的数据,如果结束时间大于当前时间,则结束时间为当前时间,避免因为查询时间的延迟造成数据遗落
  1058. $search_etime = ($t = ($stime + 86400 - 1)) > TIMESTAMP ? TIMESTAMP : ($stime + 86400 - 1);
  1059. //查询时间段内新订单或者更新过的订单,在缓存表中需要将新订单和更新过的订单进行重新缓存
  1060. $where = array();
  1061. $where[] = array('log_time', 'between', array($search_stime, $search_etime));
  1062. //查询记录总条数
  1063. $countnum_arr = Db::name('orderlog')->field('COUNT(DISTINCT order_id) as countnum')->where($where)->find();
  1064. $countnum = intval($countnum_arr['countnum']);
  1065. for ($i = 0; $i < $countnum; $i += 100) {//每次查询100条
  1066. $orderlog_list = array();
  1067. $orderlog_list = Db::name('orderlog')->field('DISTINCT order_id')->where($where)->limit($i . ',100')->select()->toArray();
  1068. if ($orderlog_list) {
  1069. //店铺ID数组
  1070. $storeid_arr = array();
  1071. //商品ID数组
  1072. $goodsid_arr = array();
  1073. //商品公共表ID数组
  1074. $goods_commonid_arr = array();
  1075. //订单ID数组
  1076. $orderid_arr = array();
  1077. //整理需要缓存的订单ID
  1078. foreach ((array) $orderlog_list as $k => $v) {
  1079. $orderid_arr[] = $v['order_id'];
  1080. }
  1081. unset($orderlog_list);
  1082. //查询订单数据
  1083. $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';
  1084. $order_list_tmp = Db::name('order')->field($field)->where('order_id', 'in', $orderid_arr)->select()->toArray();
  1085. $order_list = array();
  1086. foreach ((array) $order_list_tmp as $k => $v) {
  1087. //判读订单是否计入统计(在线支付订单已支付或者经过退款的取消订单或者货到付款订单订单已成功)
  1088. $v['order_isvalid'] = 0;
  1089. if ($v['payment_code'] != 'offline' && $v['order_state'] != ORDER_STATE_NEW && $v['order_state'] != ORDER_STATE_CANCEL) {//在线支付并且已支付并且未取消
  1090. $v['order_isvalid'] = 1;
  1091. } elseif ($v['order_state'] == ORDER_STATE_CANCEL && $v['refund_state'] != 0) {//经过退款的取消订单
  1092. $v['order_isvalid'] = 1;
  1093. } elseif ($v['payment_code'] == 'offline' && $v['order_state'] == ORDER_STATE_SUCCESS) {//货到付款订单,订单成功之后才计入统计
  1094. $v['order_isvalid'] = 1;
  1095. }
  1096. $order_list[$v['order_id']] = $v;
  1097. $storeid_arr[] = $v['store_id'];
  1098. }
  1099. unset($order_list_tmp);
  1100. //查询订单扩展数据
  1101. $field = 'order_id,reciver_province_id';
  1102. $order_common_list_tmp = Db::name('ordercommon')->field($field)->where('order_id', 'in', $orderid_arr)->select()->toArray();
  1103. $order_common_list = array();
  1104. foreach ((array) $order_common_list_tmp as $k => $v) {
  1105. $order_common_list[$v['order_id']] = $v;
  1106. }
  1107. unset($order_common_list_tmp);
  1108. //查询店铺信息
  1109. $field = 'store_id,store_name,grade_id,storeclass_id';
  1110. $store_list_tmp = Db::name('store')->field($field)->where('store_id', 'in', $storeid_arr)->select()->toArray();
  1111. $store_list = array();
  1112. foreach ((array) $store_list_tmp as $k => $v) {
  1113. $store_list[$v['store_id']] = $v;
  1114. }
  1115. unset($store_list_tmp);
  1116. //查询订单商品
  1117. $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';
  1118. $ordergoods_list = Db::name('ordergoods')->field($field)->where('order_id', 'in', $orderid_arr)->select()->toArray();
  1119. foreach ((array) $ordergoods_list as $k => $v) {
  1120. $goodsid_arr[] = $v['goods_id'];
  1121. }
  1122. //查询商品信息
  1123. $field = 'goods_id,goods_commonid,goods_price,goods_serial,gc_id,gc_id_1,gc_id_2,gc_id_3,goods_image';
  1124. $goods_list_tmp = Db::name('goods')->field($field)->where('goods_id', 'in', $goodsid_arr)->select()->toArray();
  1125. foreach ((array) $goods_list_tmp as $k => $v) {
  1126. $goods_commonid_arr[] = $v['goods_commonid'];
  1127. }
  1128. //查询商品公共信息
  1129. $field = 'goods_commonid,goods_name,brand_id,brand_name';
  1130. $goods_common_list_tmp = Db::name('goodscommon')->field($field)->where('goods_commonid', 'in', $goods_commonid_arr)->select()->toArray();
  1131. $goods_common_list = array();
  1132. foreach ((array) $goods_common_list_tmp as $k => $v) {
  1133. $goods_common_list[$v['goods_commonid']] = $v;
  1134. }
  1135. unset($goods_common_list_tmp);
  1136. //处理商品数组
  1137. $goods_list = array();
  1138. foreach ((array) $goods_list_tmp as $k => $v) {
  1139. $v['goods_commonname'] = $goods_common_list[$v['goods_commonid']]['goods_name'];
  1140. $v['brand_id'] = $goods_common_list[$v['goods_commonid']]['brand_id'];
  1141. $v['brand_name'] = $goods_common_list[$v['goods_commonid']]['brand_name'];
  1142. $goods_list[$v['goods_id']] = $v;
  1143. }
  1144. unset($goods_list_tmp);
  1145. //查询订单缓存是否存在,存在则删除
  1146. Db::name('statordergoods')->where('order_id', 'in', $orderid_arr)->delete();
  1147. //查询订单缓存是否存在,存在则删除
  1148. Db::name('statorder')->where('order_id', 'in', $orderid_arr)->delete();
  1149. //整理新增数据
  1150. $ordergoods_insert_arr = array();
  1151. foreach ((array) $ordergoods_list as $k => $v) {
  1152. $tmp = array();
  1153. $tmp['rec_id'] = $v['rec_id'];
  1154. $tmp['stat_updatetime'] = $search_etime;
  1155. $tmp['order_id'] = $v['order_id'];
  1156. $tmp['order_sn'] = $order_list[$v['order_id']]['order_sn'];
  1157. $tmp['order_add_time'] = $order_list[$v['order_id']]['add_time'];
  1158. $tmp['payment_code'] = $order_list[$v['order_id']]['payment_code'];
  1159. $tmp['order_amount'] = $order_list[$v['order_id']]['order_amount'];
  1160. $tmp['shipping_fee'] = $order_list[$v['order_id']]['shipping_fee'];
  1161. $tmp['evaluation_state'] = (string) $order_list[$v['order_id']]['evaluation_state'];
  1162. $tmp['order_state'] = (string) $order_list[$v['order_id']]['order_state'];
  1163. $tmp['refund_state'] = $order_list[$v['order_id']]['refund_state'];
  1164. $tmp['refund_amount'] = $order_list[$v['order_id']]['refund_amount'];
  1165. $tmp['order_from'] = $order_list[$v['order_id']]['order_from'];
  1166. $tmp['order_isvalid'] = $order_list[$v['order_id']]['order_isvalid'];
  1167. $tmp['reciver_province_id'] = $order_common_list[$v['order_id']]['reciver_province_id'];
  1168. $tmp['store_id'] = $v['store_id'];
  1169. $tmp['store_name'] = $store_list[$v['store_id']]['store_name'];
  1170. $tmp['grade_id'] = $store_list[$v['store_id']]['grade_id'];
  1171. $tmp['storeclass_id'] = $store_list[$v['store_id']]['storeclass_id'];
  1172. $tmp['buyer_id'] = $order_list[$v['order_id']]['buyer_id'];
  1173. $tmp['buyer_name'] = $order_list[$v['order_id']]['buyer_name'];
  1174. $tmp['goods_id'] = $v['goods_id'];
  1175. $tmp['goods_name'] = $v['goods_name'];
  1176. $tmp['goods_commonid'] = intval($goods_list[$v['goods_id']]['goods_commonid']);
  1177. $tmp['goods_commonname'] = ($t = $goods_list[$v['goods_id']]['goods_commonname']) ? $t : '';
  1178. $tmp['gc_id'] = intval($goods_list[$v['goods_id']]['gc_id']);
  1179. $tmp['gc_parentid_1'] = intval($goods_list[$v['goods_id']]['gc_id_1']);
  1180. $tmp['gc_parentid_2'] = intval($goods_list[$v['goods_id']]['gc_id_2']);
  1181. $tmp['gc_parentid_3'] = intval($goods_list[$v['goods_id']]['gc_id_3']);
  1182. $tmp['brand_id'] = intval($goods_list[$v['goods_id']]['brand_id']);
  1183. $tmp['brand_name'] = ($t = $goods_list[$v['goods_id']]['brand_name']) ? $t : '';
  1184. $tmp['goods_serial'] = ($t = $goods_list[$v['goods_id']]['goods_serial']) ? $t : '';
  1185. $tmp['goods_price'] = $v['goods_price'];
  1186. $tmp['goods_num'] = $v['goods_num'];
  1187. $tmp['goods_image'] = $goods_list[$v['goods_id']]['goods_image'];
  1188. $tmp['goods_pay_price'] = $v['goods_pay_price'];
  1189. $tmp['goods_type'] = $v['goods_type'];
  1190. $tmp['promotions_id'] = $v['promotions_id'];
  1191. $tmp['commis_rate'] = $v['commis_rate'];
  1192. $ordergoods_insert_arr[] = $tmp;
  1193. }
  1194. Db::name('statordergoods')->insertAll($ordergoods_insert_arr);
  1195. $order_insert_arr = array();
  1196. foreach ((array) $order_list as $k => $v) {
  1197. $tmp = array();
  1198. $tmp['order_id'] = $v['order_id'];
  1199. $tmp['order_sn'] = $v['order_sn'];
  1200. $tmp['order_add_time'] = $v['add_time'];
  1201. $tmp['payment_code'] = $v['payment_code'];
  1202. $tmp['order_amount'] = $v['order_amount'];
  1203. $tmp['shipping_fee'] = $v['shipping_fee'];
  1204. $tmp['evaluation_state'] = (string) $v['evaluation_state'];
  1205. $tmp['order_state'] = (string) $v['order_state'];
  1206. $tmp['refund_state'] = $v['refund_state'];
  1207. $tmp['refund_amount'] = $v['refund_amount'];
  1208. $tmp['order_from'] = $v['order_from'];
  1209. $tmp['order_isvalid'] = $v['order_isvalid'];
  1210. $tmp['reciver_province_id'] = $order_common_list[$v['order_id']]['reciver_province_id'];
  1211. $tmp['store_id'] = $v['store_id'];
  1212. $tmp['store_name'] = $store_list[$v['store_id']]['store_name'];
  1213. $tmp['grade_id'] = $store_list[$v['store_id']]['grade_id'];
  1214. $tmp['storeclass_id'] = $store_list[$v['store_id']]['storeclass_id'];
  1215. $tmp['buyer_id'] = $v['buyer_id'];
  1216. $tmp['buyer_name'] = $v['buyer_name'];
  1217. $order_insert_arr[] = $tmp;
  1218. }
  1219. Db::name('statorder')->insertAll($order_insert_arr);
  1220. }
  1221. }
  1222. }
  1223. }
  1224. /**
  1225. * 会员相关数据统计
  1226. */
  1227. private function _member_stat() {
  1228. $stat_model = model('stat');
  1229. //查询最后统计的记录
  1230. $latest_record = $stat_model->getOneStatmember(array(), '', 'statm_id desc');
  1231. $stime = 0;
  1232. if ($latest_record) {
  1233. $start_time = strtotime(date('Y-m-d', $latest_record['statm_updatetime']));
  1234. } else {
  1235. $start_time = strtotime(date('Y-m-d', strtotime(config('ds_config.setup_date')))); //从系统的安装时间开始统计
  1236. }
  1237. $j = 1;
  1238. for ($stime = $start_time; $stime < TIMESTAMP; $stime = $stime + 86400) {
  1239. //数据库更新数据数组
  1240. $insert_arr = array();
  1241. $update_arr = array();
  1242. $etime = $stime + 86400 - 1;
  1243. //避免重复统计,开始时间必须大于最后一条记录的记录时间
  1244. $search_stime = $latest_record['statm_updatetime'] > $stime ? $latest_record['statm_updatetime'] : $stime;
  1245. //统计一天的数据,如果结束时间大于当前时间,则结束时间为当前时间,避免因为查询时间的延迟造成数据遗落
  1246. $search_etime = ($t = ($stime + 86400 - 1)) > TIMESTAMP ? TIMESTAMP : ($stime + 86400 - 1);
  1247. //统计订单下单量和下单金额
  1248. $field = ' order.order_id,add_time,buyer_id,buyer_name,order_amount';
  1249. $where = array();
  1250. $where[] = array('order.order_state', '<>', ORDER_STATE_NEW); //去除未支付订单
  1251. $where[] = array('order.refund_state', '<>', "0"); //没有参与退款的取消订单,不记录到统计中
  1252. $where[] = array('orderlog.log_time', 'between', array($search_stime, $search_etime)); //按照订单付款的操作时间统计
  1253. //货到付款当交易成功进入统计,非货到付款当付款后进入统计
  1254. $where = "(order.payment_code='offline' and orderlog.log_orderstate = '" . ORDER_STATE_SUCCESS . "') or (order.payment_code<>'offline' and orderlog.log_orderstate = '" . ORDER_STATE_PAY . "' )";
  1255. $orderlist_tmp = $stat_model->statByOrderLog($where, $field, 0, 0, 'order_id'); //此处由于底层的限制,仅能查询1000条,如果日下单量大于1000,则需要limit的支持
  1256. $order_list = array();
  1257. $orderid_list = array();
  1258. foreach ((array) $orderlist_tmp as $k => $v) {
  1259. $addtime = strtotime(date('Y-m-d', $v['add_time']));
  1260. if ($addtime != $stime) {//订单如果隔天支付的话,需要进行统计数据更新
  1261. $update_arr[$addtime][$v['buyer_id']]['statm_membername'] = $v['buyer_name'];
  1262. $update_arr[$addtime][$v['buyer_id']]['statm_ordernum'] = intval($update_arr[$addtime][$v['buyer_id']]['statm_ordernum']) + 1;
  1263. $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);
  1264. } else {
  1265. $order_list[$v['buyer_id']]['buyer_name'] = $v['buyer_name'];
  1266. $order_list[$v['buyer_id']]['ordernum'] = intval($order_list[$v['buyer_id']]['ordernum']) + 1;
  1267. $order_list[$v['buyer_id']]['orderamount'] = floatval($order_list[$v['buyer_id']]['orderamount']) + (($t = floatval($v['order_amount'])) > 0 ? $t : 0);
  1268. }
  1269. //记录订单ID数组
  1270. $orderid_list[] = $v['order_id'];
  1271. }
  1272. //统计下单商品件数
  1273. if ($orderid_list) {
  1274. $field = ' add_time,order.buyer_id,order.buyer_name,goods_num ';
  1275. $where = array();
  1276. $where[] = array('order.order_id', 'in', $orderid_list);
  1277. $ordergoods_tmp = $stat_model->statByOrderGoods($where, $field, 0, 0, 'order.order_id');
  1278. $ordergoods_list = array();
  1279. foreach ((array) $ordergoods_tmp as $k => $v) {
  1280. $addtime = strtotime(date('Y-m-d', $v['add_time']));
  1281. if ($addtime != $stime) {//订单如果隔天支付的话,需要进行统计数据更新
  1282. $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);
  1283. } else {
  1284. $ordergoods_list[$v['buyer_id']]['goodsnum'] = $ordergoods_list[$v['buyer_id']]['goodsnum'] + (($t = floatval($v['goods_num'])) > 0 ? $t : 0);
  1285. }
  1286. }
  1287. }
  1288. //统计的预存款记录
  1289. $field = ' lg_member_id,lg_member_name,lg_av_amount as predincrease, lg_av_amount as predreduce ';
  1290. $where = array();
  1291. $where[] = array('lg_addtime', 'between', array($stime, $etime));
  1292. $predeposit_tmp = $stat_model->getPredepositInfo($where, $field, 0, 'lg_member_id', 0, 'lg_member_id');
  1293. $predeposit_list = array();
  1294. foreach ((array) $predeposit_tmp as $k => $v) {
  1295. $predeposit_list[$v['lg_member_id']] = $v;
  1296. }
  1297. //统计的积分记录
  1298. $field = ' pl_memberid,pl_membername,pl_points as pointsincrease, pl_points as pointsreduce ';
  1299. $where = array();
  1300. $where[] = array('pl_addtime', 'between', array($stime, $etime));
  1301. $points_tmp = $stat_model->statByPointslog($where, $field, 0, 0, '', 'pl_memberid');
  1302. $points_list = array();
  1303. foreach ((array) $points_tmp as $k => $v) {
  1304. $points_list[$v['pl_memberid']] = $v;
  1305. }
  1306. //处理需要更新的数据
  1307. foreach ((array) $update_arr as $k => $v) {
  1308. foreach ($v as $m_k => $m_v) {
  1309. //查询记录是否存在
  1310. $statmember_info = $stat_model->getOneStatmember(array('statm_time' => $k, 'statm_memberid' => $m_k));
  1311. if ($statmember_info) {
  1312. $m_v['statm_ordernum'] = intval($statmember_info['statm_ordernum']) + $m_v['statm_ordernum'];
  1313. $m_v['statm_orderamount'] = floatval($statmember_info['statm_ordernum']) + $m_v['statm_orderamount'];
  1314. $m_v['statm_updatetime'] = $search_etime;
  1315. $stat_model->editStatmember(array('statm_time' => $k, 'statm_memberid' => $m_k), $m_v);
  1316. } else {
  1317. $tmp = array();
  1318. $tmp['statm_memberid'] = $m_k;
  1319. $tmp['statm_membername'] = $m_v['statm_membername'];
  1320. $tmp['statm_time'] = $k;
  1321. $tmp['statm_updatetime'] = $search_etime;
  1322. $tmp['statm_ordernum'] = ($t = intval($m_v['statm_ordernum'])) > 0 ? $t : 0;
  1323. $tmp['statm_orderamount'] = ($t = floatval($m_v['statm_orderamount'])) > 0 ? $t : 0;
  1324. $tmp['statm_goodsnum'] = ($t = intval($m_v['statm_goodsnum'])) ? $t : 0;
  1325. $tmp['statm_predincrease'] = 0;
  1326. $tmp['statm_predreduce'] = 0;
  1327. $tmp['statm_pointsincrease'] = 0;
  1328. $tmp['statm_pointsreduce'] = 0;
  1329. $insert_arr[] = $tmp;
  1330. }
  1331. unset($statmember_info);
  1332. }
  1333. }
  1334. //处理获得所有会员ID数组
  1335. $memberidarr_order = isset($order_list) ? array_keys($order_list) : array();
  1336. $memberidarr_ordergoods = isset($ordergoods_list) ? array_keys($ordergoods_list) : array();
  1337. $memberidarr_predeposit = isset($predeposit_list) ? array_keys($predeposit_list) : array();
  1338. $memberidarr_points = isset($points_list) ? array_keys($points_list) : array();
  1339. $memberid_arr = array_merge($memberidarr_order, $memberidarr_ordergoods, $memberidarr_predeposit, $memberidarr_points);
  1340. //查询会员信息
  1341. $memberid_list = model('member')->getMemberList(array(array('member_id', 'in', $memberid_arr)), '', 0);
  1342. //查询记录是否存在
  1343. $statmemberlist_tmp = $stat_model->statByStatmember(array('statm_time' => $stime));
  1344. $statmemberlist = array();
  1345. foreach ((array) $statmemberlist_tmp as $k => $v) {
  1346. $statmemberlist[$v['statm_memberid']] = $v;
  1347. }
  1348. foreach ((array) $memberid_list as $k => $v) {
  1349. $tmp = array();
  1350. $tmp['statm_memberid'] = $v['member_id'];
  1351. $tmp['statm_membername'] = $v['member_name'];
  1352. $tmp['statm_time'] = $stime;
  1353. $tmp['statm_updatetime'] = $search_etime;
  1354. //因为记录可能已经存在,所以加上之前的统计记录
  1355. $statmemberlist[$tmp['statm_memberid']] = isset($statmemberlist[$tmp['statm_memberid']]) ? $statmemberlist[$tmp['statm_memberid']] : 0;
  1356. $order_list[$tmp['statm_memberid']] = isset($order_list[$tmp['statm_memberid']]) ? $order_list[$tmp['statm_memberid']] : 0;
  1357. $ordergoods_list[$tmp['statm_memberid']] = isset($ordergoods_list[$tmp['statm_memberid']]) ? $ordergoods_list[$tmp['statm_memberid']] : 0;
  1358. $predeposit_list[$tmp['statm_memberid']] = isset($predeposit_list[$tmp['statm_memberid']]) ? $predeposit_list[$tmp['statm_memberid']] : 0;
  1359. $tmp['statm_ordernum'] = intval($statmemberlist[$tmp['statm_memberid']]['statm_ordernum']) + (($t = intval($order_list[$tmp['statm_memberid']]['ordernum'])) > 0 ? $t : 0);
  1360. $tmp['statm_orderamount'] = floatval($statmemberlist[$tmp['statm_memberid']]['statm_orderamount']) + (($t = floatval($order_list[$tmp['statm_memberid']]['orderamount'])) > 0 ? $t : 0);
  1361. $tmp['statm_goodsnum'] = intval($statmemberlist[$tmp['statm_memberid']]['statm_goodsnum']) + (($t = intval($ordergoods_list[$tmp['statm_memberid']]['goodsnum'])) ? $t : 0);
  1362. $tmp['statm_predincrease'] = (($t = floatval($predeposit_list[$tmp['statm_memberid']]['predincrease'])) ? $t : 0);
  1363. $tmp['statm_predreduce'] = (($t = floatval($predeposit_list[$tmp['statm_memberid']]['predreduce'])) ? $t : 0);
  1364. $tmp['statm_pointsincrease'] = (($t = intval($points_list[$tmp['statm_memberid']]['pointsincrease'])) ? $t : 0);
  1365. $tmp['statm_pointsreduce'] = (($t = intval($points_list[$tmp['statm_memberid']]['pointsreduce'])) ? $t : 0);
  1366. $insert_arr[] = $tmp;
  1367. }
  1368. //删除旧的统计数据
  1369. $stat_model->delByStatmember(array('statm_time' => $stime));
  1370. Db::name('statmember')->insertAll($insert_arr);
  1371. }
  1372. }
  1373. }
  1374. ?>