using System; using System.Collections.Generic; using System.Linq; using System.Text; using CP.Model; using HtmlAgilityPack; using Newtonsoft.Json; using Quartz; using SCC.Common; using SCC.Crawler.Tools; using SCC.Interface; using SCC.Models; namespace SCC.Crawler.QG { /// /// 七星彩 /// public class QXCJob : IJob { public QXCJob() { log = new LogHelper(); email = IOC.Resolve(); } #region QXC执行入口 /// /// 作业执行入口 /// /// 作业执行上下文 public void Execute(IJobExecutionContext context) { Config = CommonHelper.GetConfigFromDataMap(context.JobDetail.JobDataMap); //预设节假日不开奖 if (Config.SkipDate.Contains(CommonHelper.SCCSysDateTime.ToString("yyyyMMdd"))) return; LatestItem = context.JobDetail.JobDataMap["LatestItem"] as TcqxcLongInfo; try { //服务启动时配置初始数据 if (LatestItem == null) { if (LatestItem == null) LatestItem = new TcqxcLongInfo { qi = CommonHelper.GenerateQiHaoYYYYQQQ(0), date = new DateTime(CommonHelper.SCCSysDateTime.Year, 1, 1) }; } //程序时间第二天,程序根据配置检查是否昨天有开奖 isGetData = false; if (CommonHelper.CheckDTIsNeedGetData(Config)) DoMainUrl(); //当今日开奖并且当前时间是晚上8点过后开始抓取 if (CommonHelper.CheckTodayIsOpenDay(Config) && CommonHelper.SCCSysDateTime.Hour > 12) DoMainUrl(); } catch (Exception ex) { log.Error(GetType(), string.Format("【{0}】抓取时发生错误,错误信息【{1}】", Config.Area + currentLottery, ex.Message)); } //保存最新期号 context.JobDetail.JobDataMap["LatestItem"] = LatestItem; } #endregion #region /// /// 把获取到的数据list存入数据库 /// private void DoMainUrl() { if (!string.IsNullOrEmpty(Config.MainUrl)) { var openList = GetOpenListFromMainUrl(Config.MainUrl); if (openList == null || openList.Count == 0) return; //无抓取数据 //抓取到的最新期数 var newestQiHao = Convert.ToInt32(openList.OrderByDescending(m => m.qi).First().qi.ToString()); //数据库里面最新期数 LatestItem = TcqxcData.GetLastOne(); var startQiNum = Convert.ToInt32(LatestItem.qi.ToString()); //startQiNum = 2018125; if (startQiNum > newestQiHao) return; //无最新数据 //处理最新开奖数据 TcqxcLongInfo matchItem = null; for (var i = startQiNum; i <= newestQiHao; i++) { matchItem = openList.FirstOrDefault(r => r.qi.ToString() == i.ToString()); if (matchItem != null) { //add db matchItem.addtime = DateTime.Now; TcqxcData.Add(matchItem); //Do Success Log log.Info(GetType(), CommonHelper.GetJobMainLogInfo(Config, i.ToString())); LatestItem = matchItem; isGetData = true; } } } } /// /// 通过主站点爬取开奖数据 /// private List GetOpenListFromMainUrl(string mainUrl) { var result = new List(); try { var url = new Uri(mainUrl); var htmlResource = NetHelper.GetUrlResponse(mainUrl, Encoding.GetEncoding("utf-8")); if (htmlResource == null) return result; var doc = new HtmlDocument(); doc.LoadHtml(htmlResource); var table = doc.DocumentNode.SelectSingleNode("//table"); if (table == null) return result; var trs = table.ChildNodes.Where(node => node.Name == "tr").ToList(); TcqxcLongInfo model = null; var optimizeUrl = string.Empty; for (var i = 0; i < trs.Count; i++) //第一二行为表头 { var trstyle = trs[i].Attributes["style"]; if (trstyle != null && trstyle.Value == "display:none") continue; var tds = trs[i].ChildNodes.Where(node => node.Name == "td").ToList(); if (tds.Count < 12) continue; model = new TcqxcLongInfo(); var Term = tds[0].InnerText.Trim(); if (Term.Length < 7) { model.qi = Convert.ToInt32("20" + Term); } else { model.qi = Convert.ToInt32(Term); } optimizeUrl = model.qi.ToString(); model.date = Convert.ToDateTime(tds[1].InnerText.Substring(0, 5)); if (tds[2].ChildNodes.Count == 0) continue; var b = tds[2].ChildNodes.Where(n => n.Name.ToLower() == "b").ToList(); var span = b[0].ChildNodes.Where(n => n.Name.ToLower() == "span").ToList(); model.n1 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(0, 1).Trim()); model.n2 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(1, 1).Trim()); model.n3 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(2, 1).Trim()); model.n4 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(3, 1).Trim()); model.n5 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(4, 1).Trim()); model.n6 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(5, 1).Trim()); model.n7 = Convert.ToInt32(span[0].InnerText.Replace(" ", "").Substring(6, 1).Trim()); var details = GetKaijiangDetails(optimizeUrl); var list = new List(); model.zj1 = details.zj1; model.jo1 = details.jo1; list.Add(new Winbonus() { item = "一等奖", wincount = model.zj1, winmoney = model.jo1 }); model.zj2 = details.zj2; model.jo2 = details.jo2; list.Add(new Winbonus() { item = "二等奖", wincount = model.zj2, winmoney = model.jo2 }); model.zj3 = details.zj3; model.jo3 = details.jo3; list.Add(new Winbonus() { item = "三等奖", wincount = model.zj3, winmoney = model.jo3 }); model.zj4 = details.zj4; list.Add(new Winbonus() { item = "四等奖", wincount = model.zj4, winmoney = "300", }); model.zj5 = details.zj5; list.Add(new Winbonus() { item = "五等奖", wincount = model.zj5, winmoney = "20", }); model.zj6 = details.zj6; list.Add(new Winbonus() { item = "六等奖", wincount = model.zj6, winmoney = "5", }); model.winbonus = JsonConvert.SerializeObject(list); model.tzmoney = details.tzmoney; model.nextmoney = details.nextmoney; if (model.tzmoney != "0") model.isupdetails = true; result.Add(model); } //var checkDataHelper = new CheckDataHelper(); //var dbdata = services.GetListS(currentLottery) // .ToDictionary(w => w.Term.ToString(), w => w.GetCodeStr()); //checkDataHelper.CheckData(dbdata, result.ToDictionary(w => w.Term.ToString(), w => w.GetCodeStr()), // Config.Area, currentLottery); //result = result.OrderByDescending(S => S.Term).ToList(); } catch (Exception ex) { log.Error(GetType(), string.Format("【{0}】通过主站抓取开奖列表时发生错误,错误信息【{1}】", Config.Area + currentLottery, ex.Message)); } return result; } /// /// 开奖详情 /// /// 地址 /// private TcqxcLongInfo GetKaijiangDetails(string optimizeUrl) { var url = "https://www.8200.cn/kjh/qxc/" + optimizeUrl + ".htm"; var htmlResource = NetHelper.GetUrlResponse(url, Encoding.GetEncoding("utf-8")); var doc = new HtmlDocument(); doc.LoadHtml(htmlResource); var div = doc.DocumentNode.SelectSingleNode("//div[@class='text-16']"); if (div == null) return null; //爬去奖金 var jiangjin = div.ChildNodes.Where(node => node.Name == "p").ToList(); //爬去奖项 //var tbody = div.ChildNodes.Where(node => node.Name == "tbody").ToList(); var table = doc.DocumentNode.SelectSingleNode("//table"); var trs = table.ChildNodes.Where(node => node.Name == "tr").ToList(); var gdje = jiangjin[2].InnerText.Replace(" 万元", "").Replace("奖池滚存:", "").Replace("--", "0").Replace(",", "") .Trim(); var trje = jiangjin[1].InnerText.Replace(" 万元", "").Replace("本期销量:", "").Replace("--", "0").Replace(",", "") .Trim(); var entity = new TcqxcLongInfo { nextmoney = gdje == "0" ? "0" : (double.Parse(gdje) * 10000).ToString(), tzmoney = trje == "0" ? "0" : (double.Parse(trje) * 10000).ToString() }; //TODO //组装详情 for (var i = 0; i < trs.Count; i++) { var tds = trs[i].ChildNodes.Where(node => node.Name == "td").ToList(); var TotalMoney = tds[1].InnerText.Replace("元", "").Replace("--", "0").Replace(",", "").Trim(); string name = tds[0].InnerText.Trim(); string money = TotalMoney == "0" ? "0" : double.Parse(TotalMoney).ToString(); string quantity = tds[2].InnerText.Trim().Replace(" 注", "").Replace("--", "0").Trim(); if (name == "一等奖") { entity.zj1 = quantity; entity.jo1 = money; } else if (name == "二等奖") { entity.zj2 = quantity; entity.jo2 = money; } else if (name == "三等奖") { entity.zj3 = quantity; entity.jo3 = money; } else if (name == "四等奖") { entity.zj4 = quantity; } else if (name == "五等奖") { entity.zj5 = quantity; } else if (name == "六等奖") { entity.zj6 = quantity; } } return entity; } #endregion #region QXC初始化 /// /// 配置信息 /// private SCCConfig Config; /// /// 当天抓取的最新一期开奖记录 /// private TcqxcLongInfo LatestItem; #pragma warning disable CS0414 // 字段“QXCJob.FailedQiHaoList”已被赋值,但从未使用过它的值 /// /// 当天抓取失败列表 /// private List FailedQiHaoList = null; #pragma warning restore CS0414 // 字段“QXCJob.FailedQiHaoList”已被赋值,但从未使用过它的值 /// /// 日志对象 /// private readonly LogHelper log; /// /// 当前彩种 /// private SCCLottery currentLottery => SCCLottery.QXC; /// /// 邮件接口 /// private IEmail email; #pragma warning disable CS0414 // 字段“QXCJob.isGetData”已被赋值,但从未使用过它的值 /// /// 是否本次运行抓取到开奖数据 /// private bool isGetData; #pragma warning restore CS0414 // 字段“QXCJob.isGetData”已被赋值,但从未使用过它的值 #endregion } }