article.go 68 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994
  1. package services
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "github.com/PuerkitoBio/goquery"
  8. "hongze/hongze_cygx/models"
  9. "hongze/hongze_cygx/models/company"
  10. "hongze/hongze_cygx/utils"
  11. "html"
  12. "net/url"
  13. "regexp"
  14. "sort"
  15. "strconv"
  16. "strings"
  17. "time"
  18. )
  19. func GetReportContentTextSub(content string) (contentSub string, err error) {
  20. content = html.UnescapeString(content)
  21. doc, err := goquery.NewDocumentFromReader(strings.NewReader(content))
  22. docText := doc.Text()
  23. bodyRune := []rune(docText)
  24. bodyRuneLen := len(bodyRune)
  25. if bodyRuneLen > 200 {
  26. bodyRuneLen = 200
  27. }
  28. body := string(bodyRune[:bodyRuneLen])
  29. contentSub = body
  30. contentSub = strings.Replace(body, "Powered by Froala Editor", "", -1)
  31. return
  32. }
  33. func GetReportContentTextSubNew(content string) (contentSub string, err error) {
  34. content = html.UnescapeString(content)
  35. doc, errdoc := goquery.NewDocumentFromReader(strings.NewReader(content))
  36. if errdoc != nil {
  37. err = errdoc
  38. return
  39. }
  40. docText := doc.Text()
  41. bodyRune := []rune(docText)
  42. bodyRuneLen := len(bodyRune)
  43. body := string(bodyRune[:bodyRuneLen])
  44. contentSub = body
  45. contentSub = strings.Replace(contentSub, "Powered by Froala Editor", "", -1)
  46. contentSub = strings.Replace(contentSub, " ", "", -1)
  47. contentSub = strings.Replace(contentSub, "<p data-f-id=\"pbf\" style=\"text-align: center; font-size: 14px; margin-top: 30px; opacity: 0.65; font-family: sanered by <a href=\"https://www.froala.com/wysiwyg-editor?pb=1\" title=\"Froala Editor\">Froala Editor</a></p>", "", -1)
  48. return
  49. }
  50. func FixArticleImgUrl(body string) (contentSub string, err error) {
  51. r := strings.NewReader(string(body))
  52. doc, err := goquery.NewDocumentFromReader(r)
  53. if err != nil {
  54. fmt.Println(err)
  55. }
  56. doc.Find("img").Each(func(i int, s *goquery.Selection) {
  57. src, _ := s.Attr("src")
  58. if i == 0 && src != "" {
  59. contentSub = src
  60. }
  61. })
  62. return
  63. }
  64. func SynchronizationArtclehistory() {
  65. fmt.Println("同步开始")
  66. list, err := models.GetArticleHistoryList()
  67. if err != nil {
  68. fmt.Println("获取列表失败", err)
  69. }
  70. fmt.Println(len(list))
  71. for _, v := range list {
  72. //endDate := v.ModifyTime.Add(+time.Minute * 10).Format(utils.FormatDateTime)
  73. //detail, err := models.GetNewArticleHistoryRecordNewpv(v.UserId, v.ArticleId, endDate)
  74. //if err != nil && err.Error() != utils.ErrNoRow() {
  75. // fmt.Println("获取信息失败", err)
  76. //}
  77. v.OutType = 1
  78. //fmt.Println(v.Id)
  79. //if detail == nil {
  80. // _, err = models.AddCygxArticleViewRecordNewpv(v)
  81. // if err != nil {
  82. // fmt.Println("新增失败", err)
  83. // }
  84. //} else {
  85. // err = models.UpdateCygxArticleViewRecordNewpvList(v, v.StopTime)
  86. // if err != nil {
  87. // fmt.Println("修改失败", err)
  88. // }
  89. //}
  90. newId, err := models.AddCygxArticleViewRecordNewpv(v)
  91. fmt.Println("新增", newId)
  92. if err != nil {
  93. fmt.Println("新增失败", err)
  94. }
  95. }
  96. fmt.Println("同步结束")
  97. }
  98. // UserViewRedisData 阅读数据
  99. type UserViewRedisData struct {
  100. Mobile string `json:"mobile"`
  101. Email string `json:"email"`
  102. RealName string `json:"real_name"`
  103. CompanyName string `json:"company_name"`
  104. ViewTime string `json:"view_time" description:"阅读时间,格式:2022-02-17 13:06:13"`
  105. ProductId int `json:"product_id" description:"报告所属产品,ficc:1,权益:2"`
  106. CompanyId int `json:"company_id" description:"客户id"`
  107. UserId int `json:"user_id" description:"用户id"`
  108. ReportId int `json:"report_id" description:"报告id"`
  109. StopTime int `json:"stop_time" description:"停留时间"`
  110. ReportChapterId int `json:"report_chapter_id" description:"章节ID"`
  111. OutId int `json:"out_id" description:"记录ID"`
  112. }
  113. type ReportViewRecord struct {
  114. Id int `orm:"column(id);pk"`
  115. //UserId int `json:"user_id" description:"用户ID"`
  116. //ReportId int `description:"报告id"`
  117. //Mobile string `description:"手机号"`
  118. //Email string `description:"邮箱"`
  119. //RealName string `description:"用户实际姓名"`
  120. //CompanyName string `description:"公司名称"`
  121. //CreateTime time.Time `description:"创建时间"`
  122. //StopTime int `json:"stop_time" description:"停留时间"`
  123. //ReportChapterId int `json:"report_chapter_id" description:"章节ID"`
  124. //OutId int `json:"out_id" description:"记录ID"`
  125. Mobile string `json:"mobile"`
  126. Email string `json:"email"`
  127. RealName string `json:"real_name"`
  128. CompanyName string `json:"company_name"`
  129. ViewTime string `json:"view_time" description:"阅读时间,格式:2022-02-17 13:06:13"`
  130. ProductId int `json:"product_id" description:"报告所属产品,ficc:1,权益:2"`
  131. CompanyId int `json:"company_id" description:"客户id"`
  132. UserId int `json:"user_id" description:"用户id"`
  133. ReportId int `json:"report_id" description:"报告id"`
  134. StopTime int `json:"stop_time" description:"停留时间"`
  135. ReportChapterId int `json:"report_chapter_id" description:"章节ID"`
  136. OutId int `json:"out_id" description:"章节ID"`
  137. }
  138. // PushViewRecordNewRedisData 阅读数据加入到redis
  139. func PushViewRecordNewRedisData(reportViewRecord *ReportViewRecord, companyId int) bool {
  140. data := &UserViewRedisData{
  141. Mobile: reportViewRecord.Mobile,
  142. UserId: reportViewRecord.UserId,
  143. Email: reportViewRecord.Email,
  144. RealName: reportViewRecord.RealName,
  145. CompanyName: reportViewRecord.CompanyName,
  146. ViewTime: reportViewRecord.ViewTime,
  147. ProductId: 2,
  148. CompanyId: companyId,
  149. ReportId: reportViewRecord.ReportId,
  150. StopTime: reportViewRecord.StopTime,
  151. ReportChapterId: reportViewRecord.ReportChapterId,
  152. OutId: reportViewRecord.OutId,
  153. }
  154. if utils.Re == nil {
  155. err := utils.Rc.LPush(utils.CACHE_KEY_USER_VIEW, data)
  156. if err != nil {
  157. fmt.Println("PushViewRecordNewRedisData LPush Err:" + err.Error())
  158. }
  159. return true
  160. }
  161. return false
  162. }
  163. // 获取策略平台报告阅读数据
  164. func GetCeLueArticlePv(cont context.Context) (err error) {
  165. //func GetCeLueArticlePv() (err error) {
  166. defer func() {
  167. if err != nil {
  168. go utils.SendAlarmMsg("同步策略平台阅读数据失败"+err.Error(), 2)
  169. go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "GetCeLueArticlePv ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  170. }
  171. }()
  172. startTime := time.Now().Add(-time.Minute * 12).Format("2006-01-02 15:04:05")
  173. endTime := time.Now().Format("2006-01-02 15:04:05")
  174. requestUrl := utils.ApiUrl + "backend/statistics_access?take=1000&skip=0&sort=ASC&mode=all&"
  175. encodeData := url.Values{}
  176. encodeData.Add("start_dt", startTime)
  177. encodeData.Add("end_dt", endTime)
  178. encodeStr := encodeData.Encode()
  179. requestUrl += encodeStr
  180. authorization := utils.ApiAuthorization
  181. body, err := PublicGetDate(requestUrl, authorization)
  182. if err != nil {
  183. return
  184. }
  185. var chartResult models.CeLueArticleResultApi
  186. err = json.Unmarshal(body, &chartResult)
  187. if err != nil {
  188. fmt.Println(err)
  189. return err
  190. }
  191. mapMobileArticleId := make(map[string]int)
  192. //获取当天阅读记录
  193. listPv, err := models.GetArticleHistoryRecordAllList()
  194. if err != nil && err.Error() != utils.ErrNoRow() {
  195. fmt.Println("获取当天阅读记录失败", err)
  196. return err
  197. }
  198. if len(listPv) > 0 {
  199. for _, v := range listPv {
  200. mapMobileArticleId[fmt.Sprint(v.Mobile, "_", v.ArticleId)] = v.ArticleId
  201. }
  202. }
  203. var celueHistoryIds []int
  204. var mobiles []string
  205. for _, v := range chartResult.Data {
  206. celueHistoryIds = append(celueHistoryIds, v.CelueHistoryId)
  207. mobiles = append(mobiles, v.Mobile)
  208. }
  209. listcelueHistory, e := models.GetCeLueArticleListByIds(celueHistoryIds)
  210. if e != nil {
  211. err = errors.New("GetCeLueArticleListByIds, Err: " + e.Error())
  212. return
  213. }
  214. mapCelueHistoryIds := make(map[int]int)
  215. for _, v := range listcelueHistory {
  216. mapCelueHistoryIds[v.CelueHistoryId] = v.CelueHistoryId
  217. }
  218. listMobileCompany, e := company.GetCompanyProductListByMobiles(mobiles)
  219. if e != nil {
  220. err = errors.New("GetCompanyProductListByMobiles, Err: " + e.Error())
  221. return
  222. }
  223. mapUserCompany := make(map[string]*company.CompanyProductUser)
  224. for _, v := range listMobileCompany {
  225. mapUserCompany[v.Mobile] = v
  226. }
  227. var celueArticleHistoryItems []*models.CygxCelueArticleHistoryRecord
  228. var allRecordItems []*models.CygxArticleHistoryRecordAll
  229. for _, v := range chartResult.Data {
  230. articleId, _ := strconv.Atoi(v.ArticleId)
  231. if articleId == 0 {
  232. continue
  233. }
  234. //fmt.Println(v.ArticleId)
  235. item := new(models.CygxCelueArticleHistoryRecord)
  236. item.CelueHistoryId = v.CelueHistoryId
  237. item.Mobile = v.Mobile
  238. item.ArticleId = v.ArticleId
  239. if v.CompanyName != nil {
  240. item.CompanyName = v.CompanyName.RealName
  241. }
  242. if v.CrmUser != nil {
  243. item.RealName = v.CrmUser.RealName
  244. }
  245. if mapUserCompany[v.Mobile] != nil {
  246. mapUser := mapUserCompany[v.Mobile]
  247. item.CompanyName = mapUser.CompanyName
  248. item.CompanyStatus = mapUser.Status
  249. item.SellerName = mapUser.SellerName
  250. }
  251. item.CreateDateApi = time.Now()
  252. t1, _ := time.Parse("2006-01-02T15:04:05Z", v.CreateDate)
  253. item.CreateTime = t1.Add(+time.Hour * 8).Format(utils.FormatDateTime)
  254. //email=?,company_id=?,company_name=?,user_id=?,real_name=?
  255. if mapCelueHistoryIds[v.CelueHistoryId] == 0 {
  256. celueArticleHistoryItems = append(celueArticleHistoryItems, item)
  257. //写入记录到总的统计表
  258. record := new(models.CygxArticleHistoryRecordAll)
  259. record.ArticleId = articleId
  260. record.CelueHistoryId = item.CelueHistoryId
  261. record.CreateTime = item.CreateTime
  262. record.ModifyTime = time.Now()
  263. record.CreateDateApi = time.Now()
  264. if mapUserCompany[v.Mobile] != nil {
  265. mapUser := mapUserCompany[v.Mobile]
  266. record.Mobile = mapUser.Mobile
  267. record.CompanyName = mapUser.CompanyName
  268. record.CompanyStatus = mapUser.Status
  269. record.SellerName = mapUser.SellerName
  270. record.CompanyId = mapUser.CompanyId
  271. record.RealName = mapUser.RealName
  272. record.UserId = mapUser.UserId
  273. record.Email = mapUser.Email
  274. }
  275. record.Platfor = 2
  276. record.Source = "CELUE"
  277. if mapMobileArticleId[fmt.Sprint(item.Mobile, "_", item.ArticleId)] == articleId {
  278. record.IsDel = 1
  279. }
  280. allRecordItems = append(allRecordItems, record)
  281. }
  282. //count, err := models.GetCeLueArticleCountById(v.CelueHistoryId)
  283. //if err != nil && err.Error() != utils.ErrNoRow() {
  284. // return err
  285. //}
  286. //if count == 0 {
  287. // _, err := models.AddCeLueArticle(item, mapMobileArticleId)
  288. // if err != nil {
  289. // fmt.Println(err)
  290. // return err
  291. // }
  292. //}
  293. }
  294. if len(celueArticleHistoryItems) > 0 {
  295. e = models.AddCygxArticleHistoryRecordAllMulti(celueArticleHistoryItems, allRecordItems)
  296. if e != nil {
  297. err = errors.New("AddCygxArticleHistoryRecordAllMulti, Err: " + e.Error())
  298. return
  299. }
  300. }
  301. //处理同步过来的阅读记录所属用户
  302. //var condition string
  303. //condition = ` AND create_time > ` + "'" + startTime + "'"
  304. //listArticlePv, err := models.GetArticleHistoryRecordAllByMobileList(condition)
  305. //if err != nil {
  306. // fmt.Println("GetArticleHistoryRecordAllByMobileList ,Err" + err.Error())
  307. //}
  308. //for _, v := range listArticlePv {
  309. // if v.Mobile != "" {
  310. // user, err := models.GetWxUserItemByMobile(v.Mobile)
  311. // if err != nil && err.Error() != utils.ErrNoRow() {
  312. // fmt.Println("GetWxUserItemByUserId ,Err" + err.Error())
  313. // }
  314. // if user != nil {
  315. // err = models.UpdateCygxArticleHistoryRecordAll(user)
  316. // if err != nil {
  317. // fmt.Println("UpdateCygxArticleCollect ,Err" + err.Error())
  318. // }
  319. // }
  320. // }
  321. //}
  322. return
  323. }
  324. func GetArticleListByApi(cont context.Context) (err error) {
  325. defer func() {
  326. if err != nil {
  327. //fmt.Println("GetArticleListByApi Err:" + err.Error())
  328. go utils.SendAlarmMsg("同步策略平台数据失败", 2)
  329. go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "GetArticleListByApi ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  330. }
  331. }()
  332. listUpdateArticle, err := models.GetArticleCeluePushList()
  333. if err != nil && err.Error() != utils.ErrNoRow() {
  334. return err
  335. }
  336. //如果长度为零就不处理
  337. if len(listUpdateArticle) == 0 {
  338. return err
  339. }
  340. for _, v := range listUpdateArticle {
  341. // 这里直接go出去会出现并发,导致文章md5ID唯一索引限制报错
  342. err = HandleArticleListByApi(v.ArticleId)
  343. if err != nil {
  344. utils.FileLog.Info("HandleArticleListByApi", err.Error())
  345. }
  346. }
  347. return
  348. }
  349. // 获取策略平台推送过来的文章并更新
  350. func UpdateArticleByRedis() (err error) {
  351. for {
  352. utils.Rc.Brpop(utils.CYGX_ARTICLE_UPDATE_KEY, func(b []byte) {
  353. var log models.CygxArticleCeluePushRedis
  354. if err := json.Unmarshal(b, &log); err != nil {
  355. fmt.Println("json unmarshal wrong!")
  356. go utils.SendAlarmMsg("获取策略平台推送过来的文章并更新处理Redis队列消息失败:"+err.Error()+string(b), 2)
  357. }
  358. // 这里直接go出去会出现并发,导致文章md5ID唯一索引限制报错
  359. err = HandleArticleListByApi(log.ArticleId)
  360. if err != nil {
  361. utils.FileLog.Info("HandleArticleListByApi", err.Error())
  362. }
  363. })
  364. }
  365. }
  366. // 处理同步过来的文章
  367. func HandleArticleListByApi(artcleId int) (err error) {
  368. defer func() {
  369. if err != nil {
  370. go utils.SendAlarmMsg("处理同步过来的文章失败"+"HandleArticleListByApi ErrMsg:"+err.Error()+"artcleId:"+strconv.Itoa(artcleId), 2)
  371. }
  372. }()
  373. var clueApiUrl string
  374. clueApiUrl = fmt.Sprint(utils.ApiUrl, "articles/", artcleId)
  375. fmt.Println(clueApiUrl)
  376. authorization := utils.ApiAuthorization
  377. body, err := PublicGetDate(clueApiUrl, authorization)
  378. if err != nil {
  379. fmt.Println(err)
  380. return
  381. }
  382. var articleResultDate models.ArticleDetailResultApi
  383. err = json.Unmarshal(body, &articleResultDate)
  384. if err != nil {
  385. fmt.Println("Getres.PublicGetDate Err:", err.Error())
  386. return err
  387. }
  388. item := new(models.CygxShanghaiCompanyLog)
  389. item.CreateTime = time.Now()
  390. item.Url = clueApiUrl
  391. item.Body = ""
  392. item.Result = string(body)
  393. go models.AddCygxShanghaiCompanyLog(item)
  394. go models.UpdateCygxArticleCeluePush(artcleId)
  395. articleResult := articleResultDate.Data
  396. exitMap := make(map[int]int)
  397. categoryIdMap := make(map[int]int)
  398. classMap := make(map[int]int)
  399. reportMap := make(map[int]int)
  400. summaryMap := make(map[int]int)
  401. listMap, err := models.GetArticleApiMap()
  402. if err != nil {
  403. fmt.Println("GetlistMap Err:", err.Error())
  404. return err
  405. }
  406. openIdList, err := models.GetUserRecordListByMobile(4, utils.ArticleTaskClassMobile)
  407. if err != nil {
  408. fmt.Println(err)
  409. return err
  410. }
  411. fmt.Println(openIdList)
  412. //新旧分类 反向隐射,是否归类,是否是报告,是否是纪要库
  413. for _, v := range listMap {
  414. exitMap[v.SeriesId] = v.OldId
  415. categoryIdMap[v.SeriesId] = v.OldIdMap
  416. if v.IsClass == 1 {
  417. classMap[v.OldId] = 1
  418. }
  419. if v.IsReport == 1 {
  420. reportMap[v.OldId] = 1
  421. }
  422. if v.IsSummary == 1 {
  423. summaryMap[v.OldId] = 1
  424. }
  425. }
  426. var list []*models.Tactics2
  427. var listArticleData []*models.CygxArticleData
  428. var listAuthor []*models.CygxArticleAuthor
  429. //如果是英文报告 ,或者香港类型那么则不同步 publish_area :all 全部 ;hk 香港
  430. if articleResult.TypeId == 9 || articleResult.PublishArea != "all" {
  431. // 如果是英文报告,或者是香港报告就进行删除处理
  432. err = models.UpdateArticlePublish(artcleId, 0)
  433. if err != nil {
  434. fmt.Println("UpdateArticlePublish Err:", err.Error())
  435. return err
  436. }
  437. UpdateArticleResourceData(artcleId) //把数据写入 cygx_resource_data 表
  438. return
  439. }
  440. var cover string
  441. if articleResult.Series.Name == "图解市场" {
  442. cover = GetArticleCoverByChartFirst(articleResult.Content.Body)
  443. } else if articleResult.Series.Name == "行业深度" {
  444. cover = articleResult.Cover
  445. }
  446. fmt.Println(cover)
  447. //return
  448. //状态等于 2 跟 4 的进行同步 IsActive 为false 软删除的也不同步
  449. if exitMap[articleResult.SeriesId] > 0 && (articleResult.PublishStatus == 2 || articleResult.PublishStatus == 4) && articleResult.IsActive {
  450. articleResult.PublishDate = time.Date(articleResult.PublishDate.Year(), articleResult.PublishDate.Month(), articleResult.PublishDate.Day(), articleResult.PublishDate.Hour(), articleResult.PublishDate.Minute(), articleResult.PublishDate.Second(), articleResult.PublishDate.Nanosecond(), time.Local)
  451. item := new(models.Tactics2)
  452. itemArticleData := new(models.CygxArticleData)
  453. itemArticleData.ArticleId = articleResult.ArticleId
  454. itemArticleData.Cover = articleResult.Cover
  455. itemArticleData.CreateTime = time.Now()
  456. listArticleData = append(listArticleData, itemArticleData)
  457. itemAuthor := new(models.CygxArticleAuthor)
  458. item.ArticleId = articleResult.ArticleId
  459. item.Title = articleResult.Title
  460. item.TitleEn = articleResult.TitleEn
  461. item.File = articleResult.File
  462. if articleResult.Frequency == "日度" {
  463. item.UpdateFrequency = "daily"
  464. } else if articleResult.Frequency == "周度" {
  465. item.UpdateFrequency = "weekly"
  466. } else if articleResult.Frequency == "月度" {
  467. item.UpdateFrequency = "monthly"
  468. } else if articleResult.Frequency == "季度" {
  469. item.UpdateFrequency = "quarterly"
  470. } else if articleResult.Frequency == "年度" {
  471. item.UpdateFrequency = "yearly"
  472. } else {
  473. item.UpdateFrequency = "unknow"
  474. }
  475. item.CreateDate = articleResult.CreateDate
  476. item.PublishDate = articleResult.PublishDate.Add(time.Hour * 8)
  477. item.PublishStatus = 1
  478. item.Body = articleResult.Content.Body
  479. item.Abstract = articleResult.Content.Abstract
  480. item.Annotation = articleResult.Content.Annotation
  481. item.CategoryName = articleResult.Industry.Name
  482. item.CategoryId = categoryIdMap[articleResult.SeriesId]
  483. item.CategoryIdTwo = exitMap[articleResult.SeriesId]
  484. item.SubCategoryName = articleResult.Series.Name
  485. item.Cover = cover
  486. //{
  487. // //这两种情况特殊处理
  488. // if articleResult.Field.Name == "宏观利率" {
  489. // item.CategoryId = 10070
  490. // item.CategoryIdTwo = item.CategoryId
  491. // }
  492. // if articleResult.Field.Name == "信用" {
  493. // item.CategoryId = 10082
  494. // item.CategoryIdTwo = item.CategoryId
  495. // }
  496. //}
  497. if len(articleResult.Stock) > 0 {
  498. var stock string
  499. for _, vS := range articleResult.Stock {
  500. stock += vS + "/"
  501. }
  502. stock = strings.TrimRight(stock, "/")
  503. item.Stock = stock
  504. }
  505. item.FieldName = articleResult.Field.Name
  506. item.SeriesName = articleResult.Series.Name
  507. item.TypeName = articleResult.Type.Name
  508. item.UpdateDate = articleResult.UpdateDate
  509. item.CeLueFieldId = articleResult.Field.Id
  510. list = append(list, item)
  511. itemAuthor.ArticleId = articleResult.ArticleId
  512. itemAuthor.Name = articleResult.Author.Name
  513. itemAuthor.Mobile = articleResult.Author.PhoneNumber
  514. fmt.Println(articleResult.Cover)
  515. listAuthor = append(listAuthor, itemAuthor)
  516. } else {
  517. // 如果这篇文章没有发布,那么就不作处理。
  518. err = models.UpdateArticlePublish(artcleId, 0)
  519. if err != nil {
  520. fmt.Println("UpdateArticlePublish Err:", err.Error())
  521. return err
  522. }
  523. UpdateArticleResourceData(artcleId) //把数据写入 cygx_resource_data 表
  524. //go UpdateResourceData(artcleId, "article", "delete", time.Now().Format(utils.FormatDateTime))
  525. return
  526. }
  527. //同步作者
  528. for _, v := range listAuthor {
  529. var count int
  530. count, err = models.GetActivityAuthorCount(v.ArticleId, v.Mobile)
  531. if err != nil {
  532. fmt.Println("GetActivityAuthorCount Err:", err.Error())
  533. return err
  534. }
  535. if count == 0 {
  536. _, err = models.AddCygxActivityAuthor(v)
  537. if err != nil {
  538. fmt.Println("AddCygxActivityAuthor Err:", err.Error())
  539. return err
  540. }
  541. }
  542. }
  543. fmt.Println("同步文章条数:", len(list))
  544. listCustomArticle, err := models.GetCustomArticleId() //手动归类的文章,不替换文章类型
  545. if err != nil {
  546. fmt.Println("GetTacticsList Err:", err.Error())
  547. return err
  548. }
  549. listGetMatchTypeName, errMatch := models.GetMatchTypeNamenNotNull() //手动归类的文章,不替换文章类型
  550. if errMatch != nil {
  551. fmt.Println("GetTacticsList Err:", errMatch.Error())
  552. return err
  553. }
  554. //mapCategoryIdSet := make(map[int]int)
  555. //for _, vMatch := range listGetMatchTypeName {
  556. // mapCategoryIdSet[vMatch.CategoryId] = vMatch.CategoryIdSet
  557. //}
  558. fmt.Println("list len:", len(list))
  559. noSummaryArticleIds := "3454,3456,3457,3459,2449,2450,2453,2454,2459,2530,2583,2663,2670,2699,2715,2732,2748,2759,2399,2356,2870,3173,2978,2826,3470" //非纪要库类型的文章ID
  560. listNoSummaryArticleIds := strings.Split(noSummaryArticleIds, ",")
  561. for k, v := range list {
  562. //同步匹配类型
  563. matchTypeName := ""
  564. for _, vMatch := range listGetMatchTypeName {
  565. if v.CategoryId == vMatch.CategoryId {
  566. matchTypeName = vMatch.MatchTypeName
  567. }
  568. }
  569. //是否属于纪要库的数据
  570. //if _, has := summaryMap[v.CategoryId]; has {
  571. // v.IsSummary = 1
  572. //}
  573. if v.TypeName == "纪要" {
  574. v.IsSummary = 1
  575. }
  576. //排除不属于纪要库类型的文章
  577. for _, vArt := range listNoSummaryArticleIds {
  578. vArtInt, _ := strconv.Atoi(vArt)
  579. if v.ArticleId == vArtInt {
  580. v.IsSummary = 0
  581. }
  582. }
  583. if _, has := reportMap[v.CategoryIdTwo]; has {
  584. v.IsReport = 1
  585. if _, ok := classMap[v.CategoryIdTwo]; ok {
  586. v.IsClass = 1
  587. v.ReportType = 1 //是否属于行业报告
  588. } else {
  589. v.ReportType = 2 //是否属于产业报告
  590. }
  591. }
  592. v.Department = "弘则权益研究"
  593. //判断是否已经存在
  594. if v.ArticleId < 0 {
  595. fmt.Println("AddCygxArticle Err:")
  596. return err
  597. }
  598. var count int
  599. count, err = models.GetArticleCountById(v.ArticleId)
  600. if err != nil && err.Error() != utils.ErrNoRow() {
  601. fmt.Println("AddCygxArticle Err:", err.Error())
  602. return err
  603. }
  604. v.Body = strings.Replace(v.Body, "http://vmp.hzinsights.com", "https://vmp.hzinsights.com", -1)
  605. expertNumStr, expertContentStr, interviewDateStr, _, bodyReturn := BodyAnalysis2(v.Body)
  606. if strings.Index(v.Body, "报告全文(") > 0 && strings.Index(v.Body, "PDF格式报告下载.pdf") > 0 {
  607. v.Body = strings.Replace(v.Body, "报告全文(", "", -1)
  608. v.Body = strings.Replace(v.Body, "PDF格式报告下载.pdf", "", -1)
  609. v.Body = strings.Replace(v.Body, "):", "", -1)
  610. }
  611. var titleNew string
  612. titleNew = v.Title
  613. // 7资金流向 、11大类资产 、51每日复盘 、80医药周报、9估值研究
  614. if v.CategoryId == 7 || v.CategoryId == 11 || v.CategoryId == 51 || v.CategoryId == 9 {
  615. if v.UpdateFrequency == "daily" {
  616. var daystr string
  617. daystr = strconv.Itoa(v.PublishDate.Day())
  618. if len(daystr) == 1 {
  619. daystr = "0" + daystr
  620. }
  621. titleNew = v.Title + "(" + strconv.Itoa(v.PublishDate.Year())[2:len(strconv.Itoa(v.PublishDate.Year()))-0] + v.PublishDate.Format("01") + daystr + ")"
  622. } else if v.UpdateFrequency == "weekly" {
  623. titleNew = v.Title + utils.WeekByDate(v.PublishDate)
  624. }
  625. }
  626. //策略周度思考,医药周报,消费周报,消费周度思考,科技前言(周报)、智造精粹、智造本周荟 标题处理
  627. if v.CategoryId == 80 || v.CategoryId == 52 || v.CategoryId == 10074 || v.CategoryId == 1008 || v.CategoryId == 10071 || v.CategoryId == 10073 || v.CategoryId == 50 {
  628. titleNew = v.Title + utils.WeekByDate(v.PublishDate)
  629. }
  630. //类型是综述报告的,不区分是什么系列,统一归类到对应产业的路演精华
  631. if articleResult.Type.Name == "综述报告" {
  632. detailCategory, err := models.GetdetailByCategoryIdLyjh(v.CategoryIdTwo)
  633. if err != nil && err.Error() != utils.ErrNoRow() {
  634. return err
  635. }
  636. v.CategoryId = detailCategory.CategoryId
  637. matchTypeName = "路演精华"
  638. if v.IsClass == 0 {
  639. v.IsSummary = 0
  640. }
  641. }
  642. var chartPermissionId int
  643. var chartPermissionName string
  644. categoryDetail, _ := models.GetdetailByCategoryIdPush(v.CategoryId)
  645. if categoryDetail != nil {
  646. chartPermissionId = categoryDetail.ChartPermissionId
  647. chartPermissionName = categoryDetail.ChartPermissionName
  648. }
  649. if count > 0 {
  650. fmt.Println(k, v.ArticleId, "edit")
  651. articleIfoOld, e := models.GetArticleDetailTestById(v.ArticleId)
  652. if e != nil {
  653. err = e
  654. return
  655. }
  656. bodyTextOld, _ := GetReportContentTextSubNew(articleIfoOld.Body)
  657. bodyText, _ := GetReportContentTextSubNew(v.Body)
  658. var isCustom bool
  659. updateParams := make(map[string]interface{})
  660. //updateParams["Title"] = v.Title
  661. updateParams["Title"] = titleNew
  662. updateParams["TitleEn"] = v.TitleEn
  663. updateParams["UpdateFrequency"] = v.UpdateFrequency
  664. updateParams["CreateDate"] = v.CreateDate
  665. updateParams["PublishDate"] = v.PublishDate
  666. //updateParams["Body"] = html.EscapeString(v.Body)
  667. updateParams["Body"] = html.EscapeString(bodyReturn)
  668. updateParams["BodyText"] = bodyText
  669. updateParams["Abstract"] = html.EscapeString(v.Abstract)
  670. updateParams["CategoryName"] = v.CategoryName
  671. for _, vCustom := range listCustomArticle {
  672. if v.ArticleId == vCustom.ArticleId {
  673. fmt.Println("手动归类的文章:" + strconv.Itoa(v.ArticleId))
  674. isCustom = true
  675. }
  676. }
  677. if isCustom == false {
  678. updateParams["CategoryId"] = v.CategoryId
  679. updateParams["MatchTypeName"] = matchTypeName
  680. updateParams["IsSummary"] = v.IsSummary
  681. updateParams["IsReport"] = v.IsReport
  682. updateParams["ReportType"] = v.ReportType
  683. updateParams["SubCategoryName"] = v.SubCategoryName
  684. }
  685. //updateParams["CategoryId"] = v.CategoryId
  686. updateParams["PublishStatus"] = 1
  687. updateParams["ExpertBackground"] = expertContentStr
  688. updateParams["ExpertNumber"] = expertNumStr
  689. updateParams["InterviewDate"] = interviewDateStr
  690. //updateParams["IsClass"] = v.IsClass
  691. v.Department = "弘则权益研究"
  692. updateParams["Department"] = v.Department
  693. updateParams["FileLink"] = v.File
  694. updateParams["Stock"] = v.Stock
  695. updateParams["FieldName"] = v.FieldName
  696. updateParams["SeriesName"] = v.SeriesName
  697. updateParams["Annotation"] = v.Annotation
  698. updateParams["TypeName"] = v.TypeName
  699. updateParams["ModifyTimeByCl"] = v.UpdateDate
  700. updateParams["CeLueFieldId"] = v.CeLueFieldId
  701. updateParams["Cover"] = v.Cover
  702. updateParams["ChartPermissionId"] = chartPermissionId
  703. updateParams["ChartPermissionName"] = chartPermissionName
  704. whereParam := map[string]interface{}{"article_id": v.ArticleId}
  705. err = models.UpdateByExpr(models.CygxArticle{}, whereParam, updateParams)
  706. if err != nil {
  707. fmt.Println("UpdateByExpr Err:" + err.Error())
  708. return err
  709. }
  710. if len(bodyText)-len(bodyTextOld) > 100 {
  711. //fmt.Println("触发推送规则")
  712. //更新字数大于一百字触发推送规则
  713. go DoArticleOnenIdWxTemplateMsg(v.ArticleId)
  714. UpdateArticleResourceData(v.ArticleId) //把数据写入 cygx_resource_data 表
  715. //go UpdateResourceData(v.ArticleId, "article", "update", time.Now().Format(utils.FormatDateTime))
  716. } else {
  717. models.UpdatecygxResourceDatasearchTitle(v.Title, v.ArticleId)
  718. }
  719. } else {
  720. fmt.Println(k, v.ArticleId, "add")
  721. item := new(models.CygxArticle)
  722. articleIdInt := v.ArticleId
  723. item.ArticleId = articleIdInt
  724. //item.Title = v.Title
  725. item.Title = titleNew
  726. item.TitleEn = v.TitleEn
  727. item.UpdateFrequency = v.UpdateFrequency
  728. item.CreateDate = v.CreateDate
  729. item.PublishDate = v.PublishDate.Format(utils.FormatDateTime)
  730. //item.Body = html.EscapeString(v.Body)
  731. item.Body = html.EscapeString(bodyReturn)
  732. item.Abstract = html.EscapeString(v.Abstract)
  733. item.CategoryName = v.CategoryName
  734. item.SubCategoryName = v.SubCategoryName
  735. item.CategoryId = v.CategoryId
  736. item.CategoryIdTwo = v.CategoryIdTwo
  737. item.PublishStatus = 1
  738. item.ExpertBackground = expertContentStr
  739. item.ExpertNumber = expertNumStr
  740. item.InterviewDate = interviewDateStr
  741. item.Department = v.Department
  742. item.ArticleIdMd5 = utils.MD5(strconv.Itoa(articleIdInt))
  743. item.IsClass = v.IsClass
  744. item.IsSummary = v.IsSummary
  745. item.IsReport = v.IsReport
  746. item.ReportType = v.ReportType
  747. item.FileLink = v.File
  748. item.MatchTypeName = matchTypeName
  749. item.Stock = v.Stock
  750. item.FieldName = v.FieldName
  751. item.SeriesName = v.SeriesName
  752. item.Annotation = v.Annotation
  753. item.TypeName = v.TypeName
  754. item.ModifyTimeByCl = v.UpdateDate
  755. item.CeLueFieldId = v.CeLueFieldId
  756. item.Cover = v.Cover
  757. item.ChartPermissionId = chartPermissionId
  758. item.ChartPermissionName = chartPermissionName
  759. item.Cover = v.Cover
  760. item.CreateTime = time.Now()
  761. newId, err := models.AddCygxArticles(item)
  762. if err != nil {
  763. fmt.Println("AddCygxArticle Err:", err.Error())
  764. return err
  765. }
  766. //go UpdateResourceData(v.ArticleId, "article", "add", time.Now().Format(utils.FormatDateTime))
  767. UpdateArticleResourceData(v.ArticleId) //把数据写入 cygx_resource_data 表
  768. //fmt.Println(newId)
  769. //报告自动归类,以及推送相关模板消息
  770. if v.ReportType == 2 {
  771. var subjectStr string
  772. var industrialManagementIdStr string
  773. var industrialSubjectIdStr string
  774. var keyword1 string
  775. var keyword2 string
  776. var keyword3 string
  777. var keyword4 string
  778. sliceSubjects := strings.Split(v.Stock, "/")
  779. mapManagementForSubject := make(map[string]string)
  780. mapIndustrialId := make(map[int]int)
  781. if len(sliceSubjects) > 0 {
  782. for _, vSubject := range sliceSubjects {
  783. sliceKuohao := strings.Split(vSubject, "(") //过滤括号
  784. sliceXiahuaxian := strings.Split(sliceKuohao[0], "-") //过滤下划线
  785. subject := sliceXiahuaxian[0]
  786. subjectStr += "'" + subject + "',"
  787. }
  788. //获取该产业下所对应的行业图片
  789. detailCategory, errCategory := models.GetdetailByCategoryIdOne(v.CategoryId)
  790. if errCategory != nil {
  791. fmt.Println("GetdetailByCategoryIdOne Err:", err.Error())
  792. return err
  793. }
  794. subjectStr = strings.TrimRight(subjectStr, ",")
  795. if subjectStr != "" {
  796. listIndustrial, err := models.GetIndustrialManagementForSubjecName(subjectStr, detailCategory.ChartPermissionId)
  797. if err != nil {
  798. fmt.Println("GetIndustrialManagementForSubjecName Err:", err.Error())
  799. return err
  800. }
  801. subjectStr = strings.Replace(subjectStr, "','", "】【", -1)
  802. subjectStr = strings.Replace(subjectStr, "'", "", -1)
  803. subjectStr = "【" + subjectStr + "】"
  804. if len(listIndustrial) > 0 {
  805. for _, vIndustrial := range listIndustrial {
  806. if _, ok := mapIndustrialId[vIndustrial.IndustrialManagementId]; !ok {
  807. industrialManagementIdStr += strconv.Itoa(vIndustrial.IndustrialManagementId) + ","
  808. }
  809. mapIndustrialId[vIndustrial.IndustrialManagementId] = vIndustrial.IndustrialManagementId
  810. industrialSubjectIdStr += strconv.Itoa(vIndustrial.IndustrialSubjectId) + ","
  811. mapManagementForSubject[vIndustrial.IndustryName] += vIndustrial.SubjectName + "/"
  812. if vIndustrial.LayoutTime == "0000-00-00 00:00:00" {
  813. go models.UpdateIndustrialManagementLayoutTime(v.PublishDate.AddDate(0, 0, -10).Format(utils.FormatDateTime), vIndustrial.IndustrialManagementId)
  814. }
  815. }
  816. industrialManagementIdStr = strings.TrimRight(industrialManagementIdStr, ",")
  817. industrialSubjectIdStr = strings.TrimRight(industrialSubjectIdStr, ",")
  818. if industrialManagementIdStr != "" {
  819. err = models.ReportArticleClassificationEditNew(int(newId), industrialManagementIdStr, v.ArticleId, industrialSubjectIdStr)
  820. if err != nil {
  821. fmt.Println("ReportArticleClassificationEditNew Err:", err.Error())
  822. return err
  823. }
  824. }
  825. var peoductName string
  826. for mk, mv := range mapManagementForSubject {
  827. peoductName += "【" + mk + "--" + strings.TrimRight(mv, "/") + "】"
  828. }
  829. keyword1 = "新报告产业标签:【" + v.FieldName + "】,个股标签:" + subjectStr
  830. keyword2 = "已自动关联至以下产业和标的:" + peoductName
  831. keyword3 = v.Title
  832. keyword4 = v.PublishDate.Format(utils.FormatDateTime)
  833. SendWxMsgWithArticleClassToAdmin(keyword1, keyword2, keyword3, keyword4, openIdList, articleIdInt)
  834. for _, vIndustrial := range listIndustrial {
  835. IndustrialManagementTimeLineDateList3(vIndustrial.IndustrialManagementId) // 更新时间线报告内容
  836. }
  837. } else {
  838. keyword1 = "新报告产业标签:【" + v.FieldName + "】,个股标签:" + subjectStr
  839. keyword2 = "未归类"
  840. keyword3 = v.Title
  841. keyword4 = v.PublishDate.Format(utils.FormatDateTime)
  842. SendWxMsgWithArticleClassToAdmin(keyword1, keyword2, keyword3, keyword4, openIdList, articleIdInt)
  843. go models.UpdateIsClassFail(artcleId)
  844. }
  845. }
  846. }
  847. }
  848. go DoArticleOnenIdWxTemplateMsg(v.ArticleId)
  849. }
  850. if v.CategoryName == utils.ZHOU_QI_NAME {
  851. UpdateCygxZhouqiArticleMapTime(v.FieldName) //更新周期对应分类下所管理文章的更新时间
  852. }
  853. //【公司调研】系列纪要发布/更新后 end
  854. //查研观向6.9模板消息推送规则
  855. //但是如果报告的发布时间是在10天以前的,公众号就不推送了
  856. //if time.Now().AddDate(0, 0, -10).Before(articleResult.PublishDate.Add(time.Hour * 8)) {
  857. // go DoArticleOnenIdWxTemplateMsg(v.ArticleId)
  858. //}
  859. }
  860. go models.UpdateCygxArticleCeluePush(artcleId)
  861. AddComprehensiveArticle(artcleId) // 同步到综合页面的搜索引擎
  862. AddComprehensiveIndustrialSource("Hz", artcleId) // 同步到综合页面的搜索引擎(添加产业资源包)
  863. GetSummarytoEs(artcleId) // 同步到ES搜索引擎
  864. GetIndustrialManagementArticleNewPublishData()
  865. if len(listArticleData) > 0 {
  866. for _, v := range listArticleData {
  867. count, err := models.GetCygxArticleDataCount(v.ArticleId)
  868. if err != nil {
  869. fmt.Println("GetCygxArticleDataCount Err:", err.Error())
  870. return err
  871. }
  872. if count == 0 {
  873. _, err := models.AddCygxArticleData(v)
  874. if err != nil {
  875. fmt.Println("AddCygxArticleData Err:", err.Error())
  876. return err
  877. }
  878. }
  879. //} else {
  880. // updateParams := make(map[string]interface{})
  881. // updateParams["Cover"] = v.Cover
  882. // whereParam := map[string]interface{}{"article_id": v.ArticleId}
  883. // err = models.UpdateByExpr(models.CygxArticleData{}, whereParam, updateParams)
  884. // if err != nil {
  885. // fmt.Println("UpdateByExpr CygxArticleData Err:" + err.Error())
  886. // return err
  887. // }
  888. //}
  889. }
  890. }
  891. AddCygxReportMappingCategoryGroupByArticleId(artcleId)
  892. return err
  893. }
  894. //func init() {
  895. // DoArticleOnenIdWxTemplateMsg(8112)
  896. //}
  897. // 过滤策略平台报告,研选报告重复推送,以及权限勾选的推送
  898. func DoArticleOnenIdWxTemplateMsg(articleId int) (err error) {
  899. //策略平台同步过来的文章ID,做微信模板消息推送,同一篇报告12小时只推送一次
  900. cacheKey := fmt.Sprint(utils.CACHE_KEY_ARTICLE_ID_TEMPLATE, ":ArticleId", articleId)
  901. ttlTime := utils.Rc.GetRedisTTL(cacheKey)
  902. if ttlTime > 0 {
  903. fmt.Println(ttlTime)
  904. return err
  905. }
  906. var msg string
  907. defer func() {
  908. if err != nil {
  909. go utils.SendAlarmMsg("过滤策略平台报告,研选报告重复推送,以及权限勾选的推送失败 DoArticleOnenIdWxTemplateMsg ErrMsg:"+err.Error(), 2)
  910. fmt.Println(err)
  911. }
  912. if msg != "" {
  913. utils.FileLog.Info("发送模版消息失败,msg:%s", msg)
  914. }
  915. }()
  916. //获取文章详情
  917. articleInfo, err := models.GetArticleDetailById(articleId)
  918. if err != nil {
  919. return
  920. }
  921. articleInfo.Abstract, _ = GetReportContentTextSub(articleInfo.Abstract)
  922. var chartPermissionId int
  923. //固收的权限,暂时当策略的逻辑一样处理
  924. var chartPermissionName string
  925. reportMapDetail, err := models.GetdetailByCategoryIdPush(articleInfo.CategoryId)
  926. if err != nil {
  927. return err
  928. }
  929. if reportMapDetail.ChartPermissionId == utils.CE_LUE_ID {
  930. chartPermissionId = utils.CE_LUE_ID
  931. } else if reportMapDetail.ChartPermissionId == utils.GU_SHOU_ID_CHART {
  932. chartPermissionId = utils.GU_SHOU_ID_CHART
  933. } else {
  934. if reportMapDetail.PermissionType == 1 {
  935. chartPermissionName = reportMapDetail.ChartPermissionName + "(主观)"
  936. } else {
  937. chartPermissionName = reportMapDetail.ChartPermissionName + "(客观)"
  938. }
  939. chartPermissionDetail, err := models.GetChartPermissionByRemark(chartPermissionName)
  940. if err != nil {
  941. return err
  942. }
  943. chartPermissionId = chartPermissionDetail.ChartPermissionId
  944. }
  945. //文章关联的产业
  946. var industryName string
  947. var industrialManagementId string
  948. if chartPermissionId == utils.CE_LUE_ID || chartPermissionId == utils.GU_SHOU_ID_CHART {
  949. industryName = reportMapDetail.SubCategoryName
  950. } else {
  951. if articleInfo.ReportType == 2 {
  952. detailIndustryNameList, err := models.GetIndustrialManagemenDetailByAaticleID(articleId)
  953. if err != nil && err.Error() != utils.ErrNoRow() {
  954. return err
  955. }
  956. //如果不存在关联的产业则不推送
  957. if len(detailIndustryNameList) == 0 {
  958. return err
  959. }
  960. for _, v := range detailIndustryNameList {
  961. industryName += v.IndustryName + ","
  962. industrialManagementId += strconv.Itoa(v.IndustrialManagementId) + ","
  963. }
  964. industryName = strings.TrimRight(industryName, ",")
  965. industrialManagementId = strings.TrimRight(industrialManagementId, ",")
  966. //处理勾选了全部行业赛道的用户
  967. var allIn string
  968. allIn = DoXzsChooseSend(reportMapDetail.ChartPermissionName)
  969. if allIn != "" {
  970. allInList, err := models.GetCygxXzsChooseSendByAllIn(allIn)
  971. if err != nil && err.Error() != utils.ErrNoRow() {
  972. return err
  973. }
  974. if len(allInList) > 0 {
  975. var allInUserId string
  976. for _, v := range allInList {
  977. allInUserId += strconv.Itoa(v.UserId) + ","
  978. }
  979. allInUserId = strings.TrimRight(allInUserId, ",")
  980. userFllowIndustrialList, err := models.GetUserFllowIndustrialListByUserIdAndIndustrial(allInUserId, industrialManagementId)
  981. if err != nil && err.Error() != utils.ErrNoRow() {
  982. return err
  983. }
  984. mapUserFllowed := make(map[string]int)
  985. for _, v := range userFllowIndustrialList {
  986. //用户ID与其关注的产业组成唯一索引,判断是否存在,如果不存在就进行自动关注
  987. mapUserFllowed[fmt.Sprint("uid:", v.UserId, "_Industrial:", v.IndustrialManagementId)] = v.UserId
  988. }
  989. //获取这些allIN用户的的信息进行遍历生成需要插入的数据
  990. userList, err := models.GetWxUserListByUserIds(allInUserId)
  991. if err != nil && err.Error() != utils.ErrNoRow() {
  992. return err
  993. }
  994. var industryFllowItems []*models.CygxIndustryFllow
  995. for _, v := range detailIndustryNameList {
  996. for _, vUser := range userList {
  997. //如果用户这个用户没有关注这个产业就进行自动关注
  998. if mapUserFllowed[fmt.Sprint("uid:", vUser.UserId, "_Industrial:", v.IndustrialManagementId)] == 0 {
  999. item := new(models.CygxIndustryFllow)
  1000. item.IndustrialManagementId = v.IndustrialManagementId
  1001. item.UserId = vUser.UserId
  1002. item.Email = vUser.Email
  1003. item.Mobile = vUser.Mobile
  1004. item.RealName = vUser.RealName
  1005. item.CompanyId = vUser.CompanyId
  1006. item.CompanyName = vUser.CompanyName
  1007. item.Source = 3
  1008. item.Type = 1
  1009. item.CreateTime = time.Now()
  1010. item.ModifyTime = time.Now()
  1011. industryFllowItems = append(industryFllowItems, item)
  1012. }
  1013. }
  1014. }
  1015. err = models.AddCygxIndustryFllowMulti(industryFllowItems)
  1016. if err != nil {
  1017. go utils.SendAlarmMsg("用户关注全部赛道,出现新赛道批量添加失败 ,AddCygxIndustryFllowMulti ErrMsg:"+err.Error(), 2)
  1018. }
  1019. }
  1020. }
  1021. }
  1022. }
  1023. // 获取所有有权的用户的 openid
  1024. mapOpenidPower := make(map[int]string)
  1025. permissionIdList := make([]string, 0)
  1026. var condition string
  1027. var pars []interface{}
  1028. slicepermissionId := strings.Split(strconv.Itoa(chartPermissionId), ",")
  1029. if len(slicepermissionId) > 0 {
  1030. for _, v := range slicepermissionId {
  1031. if v == strconv.Itoa(utils.GU_SHOU_ID_CHART) {
  1032. v = strconv.Itoa(utils.CE_LUE_ID)
  1033. }
  1034. permissionIdList = append(permissionIdList, v)
  1035. }
  1036. condition += ` AND p.chart_permission_id IN (` + utils.GetOrmInReplace(len(slicepermissionId)) + ` ) `
  1037. pars = append(pars, permissionIdList)
  1038. }
  1039. openidPowerListMobile, err := models.GetCygxUserRecordPower(condition, pars)
  1040. if err != nil {
  1041. return err
  1042. }
  1043. var mobileArr []string
  1044. for _, v := range openidPowerListMobile {
  1045. mobileArr = append(mobileArr, v.Mobile)
  1046. }
  1047. openidPowerList, err := models.GetWxOpenIdByMobileSliceList(mobileArr)
  1048. if err != nil {
  1049. return err
  1050. }
  1051. //四大行业的行业综述报告,对所有开通了对应行业权限的客户都推送。
  1052. if articleInfo.ReportType == 1 && chartPermissionId != utils.CE_LUE_ID && chartPermissionId != utils.GU_SHOU_ID_CHART {
  1053. for _, item := range openidPowerList {
  1054. first := "您关注的赛道,有新的报告发布/更新,欢迎查看"
  1055. keyword1 := "【" + articleInfo.SubCategoryName + "】有报告发布/更新"
  1056. openIditem := new(models.OpenIdList)
  1057. openIditem.OpenId = item.OpenId
  1058. openIditem.UserId = item.UserId
  1059. SendWxMsgWithArticleUpdate(first, keyword1, articleInfo.Title, articleInfo.PublishDate, articleInfo.Abstract, openIditem, articleId)
  1060. }
  1061. return err
  1062. }
  1063. for _, v := range openidPowerList {
  1064. mapOpenidPower[v.UserId] = v.OpenId
  1065. }
  1066. //获取关注这个文章对应产业的用户的 openid
  1067. mapOpenidFllow := make(map[int]string)
  1068. if chartPermissionId != utils.CE_LUE_ID && chartPermissionId != utils.GU_SHOU_ID_CHART {
  1069. openidFllowListMobile, err := models.GetCygxUserFllowOpenid(articleId)
  1070. if err != nil {
  1071. return err
  1072. }
  1073. var mobileArrFllow []string
  1074. for _, v := range openidFllowListMobile {
  1075. mobileArrFllow = append(mobileArrFllow, v.Mobile)
  1076. }
  1077. openidFllowList, err := models.GetWxOpenIdByMobileSliceList(mobileArrFllow)
  1078. if err != nil {
  1079. return err
  1080. }
  1081. for _, v := range openidFllowList {
  1082. mapOpenidFllow[v.UserId] = v.OpenId
  1083. }
  1084. }
  1085. //获取拒绝接收推送的的用户的 openid
  1086. mapOpenidRefuset := make(map[int]string)
  1087. openidRefusetList, err := models.GetCygxUserRefusetOpenid()
  1088. if err != nil {
  1089. return err
  1090. }
  1091. for _, v := range openidRefusetList {
  1092. mapOpenidRefuset[v.UserId] = v.OpenId
  1093. }
  1094. //获取提交过推送规则的用户的 openid
  1095. mapUserIdChooseSend := make(map[int]int)
  1096. mapUserIdChooseSendTypeNoPermission := make(map[int]int)
  1097. chooseSendtList, err := models.GetCygxXzsChooseSend()
  1098. for _, v := range chooseSendtList {
  1099. mapUserIdChooseSend[v.UserId] = v.UserId
  1100. if v.IsRefuse == 1 {
  1101. continue
  1102. }
  1103. //获取选择主客观赛道的用户
  1104. if reportMapDetail.PermissionType == 1 {
  1105. if v.IsSubjective == 1 {
  1106. mapUserIdChooseSendTypeNoPermission[v.UserId] = v.UserId
  1107. }
  1108. } else {
  1109. if v.IsObjective == 1 {
  1110. mapUserIdChooseSendTypeNoPermission[v.UserId] = v.UserId
  1111. }
  1112. }
  1113. }
  1114. //已经推送了的openid
  1115. mapOpenidPushed := make(map[int]string)
  1116. keyword2 := articleInfo.Title
  1117. keyword3 := time.Now().Format(utils.FormatDateTime)
  1118. keyword4 := articleInfo.Abstract
  1119. mapAlreadySend := make(map[int]int)
  1120. //如果是策略就按照策略的推送方式来写
  1121. if reportMapDetail.ChartPermissionName == utils.CE_LUE_NAME || reportMapDetail.ChartPermissionName == utils.GU_SHOU_NAME {
  1122. //获取提交过推送规则的用户的 openid
  1123. mapUserFllowCeLue := make(map[int]string)
  1124. userFllowCeLueList, err := models.GetCygxUserFllowCeLueOpenid(articleInfo.CategoryId)
  1125. if err != nil {
  1126. return err
  1127. }
  1128. for _, v := range userFllowCeLueList {
  1129. mapUserFllowCeLue[v.UserId] = v.OpenId
  1130. }
  1131. for k, v := range mapOpenidPower {
  1132. if mapOpenidPushed[k] != "" {
  1133. continue
  1134. }
  1135. openIditem := new(models.OpenIdList)
  1136. openIditem.UserId = k
  1137. openIditem.OpenId = v
  1138. first := "您关注的赛道,有新的报告发布/更新,欢迎查看"
  1139. keyword1 := "【" + articleInfo.SubCategoryName + "】有报告发布/更新"
  1140. mapActivityUserId := make(map[int]string)
  1141. if articleInfo.ReportType == 2 {
  1142. sliceSubjectId, _ := models.GetSubjectIds(articleId)
  1143. if sliceSubjectId != "" {
  1144. appointmentList, err := models.GetCygxAppointmentSummaryBySubjectId(sliceSubjectId)
  1145. if err != nil {
  1146. fmt.Println("GetCygxAppointmentSummaryListBySubjectId Err:", err.Error())
  1147. return err
  1148. }
  1149. var userIds []string
  1150. if len(appointmentList) > 0 {
  1151. for _, v := range appointmentList {
  1152. //fmt.Println("mapActivityUserId:", v.UserId)
  1153. userIds = append(userIds, strconv.Itoa(v.UserId))
  1154. mapActivityUserId[v.UserId] += v.ActivityName + ","
  1155. }
  1156. userIdstr := strings.Join(userIds, ",")
  1157. openIdLists, err := models.GetOpenIdByUserIds(userIdstr)
  1158. if err != nil {
  1159. fmt.Println("GetOpenIdByUserIds Err:", err.Error())
  1160. return err
  1161. }
  1162. for _, item := range openIdLists {
  1163. first := "您预约的调研,有关联的纪要发布/更新了"
  1164. keyword1 := industryName
  1165. openIditem := new(models.OpenIdList)
  1166. openIditem.OpenId = item.OpenId
  1167. openIditem.UserId = item.UserId
  1168. if _, ok := mapAlreadySend[openIditem.UserId]; !ok {
  1169. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1170. delete(mapUserIdChooseSendTypeNoPermission, openIditem.UserId)
  1171. fmt.Println("AlreadySend:", openIditem.UserId)
  1172. mapAlreadySend[openIditem.UserId] = openIditem.UserId
  1173. }
  1174. }
  1175. }
  1176. }
  1177. //fmt.Println("openIditem.UserId:", openIditem.UserId)
  1178. //if _,ok := mapActivityUserId[openIditem.UserId]; ok{
  1179. // first = "您预约的调研,有关联的纪要发布/更新了"
  1180. // keyword1 = industryName
  1181. // SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1182. // delete(mapUserIdChooseSendTypeNoPermission, openIditem.UserId)
  1183. // mapAlreadySend[openIditem.UserId] = openIditem.UserId
  1184. //}
  1185. }
  1186. _, ok := mapAlreadySend[openIditem.UserId]
  1187. //如果有权限而且小助手没有提交过信息的 就做正常推送
  1188. if mapUserIdChooseSend[openIditem.UserId] == 0 && !ok {
  1189. fmt.Println("1:", articleId)
  1190. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1191. }
  1192. //如果小助手勾选了对应的权限信息 就做正常推送
  1193. if mapUserIdChooseSend[openIditem.UserId] != 0 && mapUserFllowCeLue[openIditem.UserId] != "" && !ok {
  1194. fmt.Println("2:", articleId)
  1195. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1196. }
  1197. }
  1198. } else {
  1199. for k, v := range mapOpenidPower {
  1200. if mapOpenidPushed[k] != "" {
  1201. continue
  1202. }
  1203. mapUserIdChooseSendType := make(map[int]int)
  1204. for _, vS := range chooseSendtList {
  1205. if vS.IsRefuse == 1 {
  1206. continue
  1207. }
  1208. //获取选择主客观赛道的用户
  1209. if reportMapDetail.PermissionType == 1 {
  1210. if vS.IsSubjective == 1 {
  1211. mapUserIdChooseSendType[vS.UserId] = vS.UserId
  1212. }
  1213. } else if reportMapDetail.PermissionType == 2 {
  1214. if vS.IsObjective == 1 {
  1215. mapUserIdChooseSendType[vS.UserId] = vS.UserId
  1216. }
  1217. } else {
  1218. if vS.IsSubjective == 1 || vS.IsObjective == 1 {
  1219. mapUserIdChooseSendType[vS.UserId] = vS.UserId
  1220. }
  1221. }
  1222. }
  1223. openIditem := new(models.OpenIdList)
  1224. openIditem.UserId = k
  1225. openIditem.OpenId = v
  1226. first := "您关注的赛道,有新的报告发布/更新,欢迎查看"
  1227. keyword1 := "【" + articleInfo.SubCategoryName + "】有报告发布/更新"
  1228. mapActivityUserId := make(map[int]string)
  1229. if articleInfo.ReportType == 2 {
  1230. keyword1 = fmt.Sprint("【", industryName, "】有报告发布/更新")
  1231. sliceSubjectId, _ := models.GetSubjectIds(articleId)
  1232. if sliceSubjectId != "" {
  1233. appointmentList, err := models.GetCygxAppointmentSummaryBySubjectId(sliceSubjectId)
  1234. if err != nil {
  1235. fmt.Println("GetCygxAppointmentSummaryListBySubjectId Err:", err.Error())
  1236. return err
  1237. }
  1238. var userIds []string
  1239. if len(appointmentList) > 0 {
  1240. for _, v := range appointmentList {
  1241. //fmt.Println("mapActivityUserId:", v.UserId)
  1242. userIds = append(userIds, strconv.Itoa(v.UserId))
  1243. mapActivityUserId[v.UserId] += v.ActivityName + ","
  1244. }
  1245. userIdstr := strings.Join(userIds, ",")
  1246. openIdLists, err := models.GetOpenIdByUserIds(userIdstr)
  1247. if err != nil {
  1248. fmt.Println("GetOpenIdByUserIds Err:", err.Error())
  1249. return err
  1250. }
  1251. for _, item := range openIdLists {
  1252. //first := "您预约的调研,有关联的纪要发布/更新了"
  1253. //keyword1 := industryName
  1254. openIditem := new(models.OpenIdList)
  1255. openIditem.OpenId = item.OpenId
  1256. openIditem.UserId = item.UserId
  1257. if _, ok := mapAlreadySend[openIditem.UserId]; !ok {
  1258. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1259. delete(mapUserIdChooseSendTypeNoPermission, openIditem.UserId)
  1260. fmt.Println("AlreadySend:", openIditem.UserId)
  1261. mapAlreadySend[openIditem.UserId] = openIditem.UserId
  1262. }
  1263. }
  1264. }
  1265. }
  1266. }
  1267. //如果有权限而且小助手没有提交过信息的 就做正常推送
  1268. _, ok := mapAlreadySend[openIditem.UserId]
  1269. if mapUserIdChooseSend[openIditem.UserId] == 0 && !ok {
  1270. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1271. delete(mapUserIdChooseSendTypeNoPermission, openIditem.UserId)
  1272. mapAlreadySend[openIditem.UserId] = openIditem.UserId
  1273. }
  1274. //如果小助手勾选了对应的权限信息 就做正常推送
  1275. if mapUserIdChooseSend[openIditem.UserId] != 0 && mapOpenidFllow[openIditem.UserId] != "" && mapUserIdChooseSendType[openIditem.UserId] > 0 && !ok {
  1276. SendWxMsgWithArticleUpdate(first, keyword1, keyword2, keyword3, keyword4, openIditem, articleId)
  1277. delete(mapUserIdChooseSendTypeNoPermission, openIditem.UserId)
  1278. mapAlreadySend[openIditem.UserId] = openIditem.UserId
  1279. }
  1280. }
  1281. }
  1282. utils.Rc.Put(cacheKey, articleId, time.Hour*12)
  1283. return
  1284. }
  1285. // GetSpecialArticleDetailUserPower 处理用户查看专项调研文章详情的权限
  1286. func GetSpecialArticleDetailUserPower(user *models.WxUserItem, articleInfo *models.ArticleDetail) (havePower bool, err error) {
  1287. userType, _, e := GetUserType(user.CompanyId)
  1288. if e != nil {
  1289. err = errors.New("GetSpecialUserType, Err: " + e.Error())
  1290. return
  1291. }
  1292. // 永续客户、大套餐客户可以查看行业升级套餐客户 权限
  1293. if userType == 1 || userType == 2 {
  1294. havePower = true
  1295. return
  1296. }
  1297. permissionStr, e := GetCompanyPermissionUpgrade(user.CompanyId)
  1298. if e != nil {
  1299. err = errors.New("GetCompanyPermissionUpgrade, Err: " + e.Error())
  1300. return
  1301. }
  1302. reportMapDetail, e := models.GetdetailByCategoryIdPush(articleInfo.CategoryId)
  1303. if e != nil {
  1304. err = errors.New("GetdetailByCategoryIdPush, Err: " + e.Error())
  1305. return
  1306. }
  1307. if reportMapDetail == nil {
  1308. err = errors.New("GetdetailByCategoryIdP,获取详情失败, ")
  1309. return
  1310. }
  1311. fmt.Println(permissionStr)
  1312. //如果没有对应的升级权限,则返回
  1313. if !strings.Contains(permissionStr, reportMapDetail.ChartPermissionName) {
  1314. return
  1315. } else {
  1316. havePower = true
  1317. }
  1318. return
  1319. }
  1320. // HandleArticleCategoryImg 预处理文章的封面图片
  1321. func HandleArticleCategoryImg(list []*models.HomeArticle) (items []*models.HomeArticle, err error) {
  1322. //研选的五张图片
  1323. detailResearch, e := models.GetConfigByCode("category_research_img_url")
  1324. if e != nil {
  1325. err = errors.New("获取研选的五张图片失败" + e.Error())
  1326. return
  1327. }
  1328. researchList := strings.Split(detailResearch.ConfigValue, "{|}")
  1329. //对应分类的所图片
  1330. detailCategoryUrl, err := models.GetConfigByCode("category_map_img_url")
  1331. if err != nil {
  1332. err = errors.New("获取对应分类的所图片失败" + err.Error())
  1333. return
  1334. }
  1335. categoryUrlList := strings.Split(detailCategoryUrl.ConfigValue, "{|}")
  1336. mapCategoryUrl := make(map[string]string)
  1337. var categoryId string
  1338. var imgUrlChart string
  1339. for _, v := range categoryUrlList {
  1340. vslice := strings.Split(v, "_")
  1341. categoryId = vslice[0]
  1342. imgUrlChart = vslice[len(vslice)-1]
  1343. mapCategoryUrl[categoryId] = imgUrlChart
  1344. }
  1345. mapChartPerssion := make(map[string]string)
  1346. reportMappingList, err := models.GetReportMappingStrategyAll()
  1347. if err != nil {
  1348. err = errors.New("GetReportMappingStrategyAll err" + err.Error())
  1349. return
  1350. }
  1351. for _, v := range reportMappingList {
  1352. mapChartPerssion[strconv.Itoa(v.CategoryId)] = v.ChartPermissionName
  1353. }
  1354. for k, v := range list {
  1355. list[k].Abstract, _ = GetReportContentTextSub(v.Abstract)
  1356. item := list[k]
  1357. //如果文章一开始的内容是图片,优先展示第一张图片
  1358. if list[k].Annotation == "" {
  1359. imgurl, _ := FixArticleImgUrl(html.UnescapeString(list[k].Body))
  1360. if imgurl != "" {
  1361. list[k].BodyHtml = imgurl
  1362. }
  1363. }
  1364. //newBody, _ := GetReportContentTextSubByarticle(item.Body, item.Annotation, item.ArticleId)
  1365. list[k].Annotation = ArticleAnnotation(item)
  1366. list[k].Body = ""
  1367. list[k].PublishDate = utils.StrTimeToTime(item.PublishDate).Format(utils.FormatDate) //时间字符串格式转时间格式
  1368. list[k].ChartPermissionName = mapChartPerssion[v.CategoryId]
  1369. //如果是研选系列的任意取五张图片的中的一张
  1370. if v.CategoryId == "0" || v.ArticleId > utils.SummaryArticleId {
  1371. knum := v.ArticleId % 5
  1372. list[k].ImgUrlPc = researchList[knum]
  1373. } else {
  1374. list[k].ImgUrlPc = mapCategoryUrl[v.CategoryId]
  1375. }
  1376. list[k].ArticleResponse = 4 // 默认展示核心观点
  1377. //ArticleResponse int `description:"报告类型 0:啥也不是,1研选报告,2:研选纪要,3:研选沙龙,4;核心观点"`
  1378. if list[k].ArticleId >= utils.SummaryArticleId {
  1379. list[k].HttpUrl = utils.StrategyPlatform + strconv.Itoa(v.ArticleId)
  1380. list[k].IsNeedJump = true
  1381. list[k].ArticleResponse = 1
  1382. //if v.IsReport == 1 {
  1383. // list[k].ArticleResponse = 1
  1384. //} else if v.ArticleTypeName == "纪要" {
  1385. // list[k].ArticleResponse = 2
  1386. //} else if v.ArticleTypeName == "观点" {
  1387. // list[k].ArticleResponse = 3
  1388. //} else if v.ArticleTypeName == "沙龙" {
  1389. // list[k].ArticleResponse = 3
  1390. //} else if v.ArticleTypeName == "路演精华" {
  1391. // list[k].ArticleResponse = 1
  1392. //}
  1393. }
  1394. list[k].Source = 1
  1395. //添加行业默认图片
  1396. if v.ImgUrlPc == "" {
  1397. if v.ChartPermissionName == utils.YI_YAO_NAME {
  1398. list[k].ImgUrlPc = utils.YI_YAO_OTHER_IMG
  1399. } else if v.ChartPermissionName == utils.XIAO_FEI_NAME {
  1400. list[k].ImgUrlPc = utils.XIAO_FEI_OTHER_IMG
  1401. } else if v.ChartPermissionName == utils.KE_JI_NAME {
  1402. list[k].ImgUrlPc = utils.KE_JI_OTHER_IMG
  1403. } else if v.ChartPermissionName == utils.ZHI_ZAO_NAME {
  1404. list[k].ImgUrlPc = utils.ZHI_ZAO_OTHER_IMG
  1405. }
  1406. }
  1407. //是不是研选报告
  1408. if v.ArticleTypeId > 0 {
  1409. list[k].IsResearch = true
  1410. }
  1411. }
  1412. articleIds := make([]int, 0)
  1413. for i := range list {
  1414. articleIds = append(articleIds, list[i].ArticleId)
  1415. }
  1416. articleMapPv := GetArticleHistoryByArticleId(articleIds) //文章Pv
  1417. // 报告关联产业信息
  1418. industryMap := make(map[int][]*models.IndustrialManagementIdInt, 0)
  1419. if len(articleIds) > 0 {
  1420. var industryCond string
  1421. var industryPars []interface{}
  1422. industryCond += ` AND mg.article_id IN (` + utils.GetOrmInReplace(len(articleIds)) + `)`
  1423. industryPars = append(industryPars, articleIds)
  1424. industryList, e := models.GetIndustrialListByarticleId(industryPars, industryCond)
  1425. if e != nil {
  1426. err = errors.New("GetIndustrialListByarticleId" + e.Error())
  1427. return
  1428. }
  1429. for i := range industryList {
  1430. v := industryList[i]
  1431. industryMap[v.ArticleId] = append(industryMap[v.ArticleId], &models.IndustrialManagementIdInt{
  1432. ArticleId: v.ArticleId,
  1433. IndustrialManagementId: v.IndustrialManagementId,
  1434. IndustryName: v.IndustryName,
  1435. ChartPermissionId: v.ChartPermissionId,
  1436. })
  1437. }
  1438. }
  1439. for k, v := range list {
  1440. if len(industryMap[v.ArticleId]) > 0 {
  1441. list[k].List = industryMap[v.ArticleId]
  1442. } else {
  1443. list[k].List = make([]*models.IndustrialManagementIdInt, 0)
  1444. }
  1445. v.Pv = articleMapPv[v.ArticleId]
  1446. if v.Pv > 999 {
  1447. list[k].Pv = 999
  1448. }
  1449. }
  1450. if len(list) == 0 {
  1451. list = make([]*models.HomeArticle, 0)
  1452. }
  1453. items = list
  1454. return
  1455. }
  1456. // 处理核心观点的展示规则
  1457. func ArticleAnnotation(item *models.HomeArticle) (annotation string) {
  1458. if item.ArticleId >= utils.SummaryArticleId {
  1459. item.Annotation = YxArticleAnnotation(item)
  1460. }
  1461. if item.Annotation != "" {
  1462. annotation = strings.Replace(item.Annotation, "<br>", "", -1)
  1463. annotation = strings.Replace(item.Annotation, "&nbsp;", "", -1)
  1464. } else {
  1465. return
  1466. }
  1467. bodyText, _ := GetReportContentTextSubNew(annotation)
  1468. if bodyText == "" {
  1469. return
  1470. }
  1471. if annotation != "" {
  1472. annotation = html.UnescapeString(annotation)
  1473. doc, _ := goquery.NewDocumentFromReader(strings.NewReader(annotation))
  1474. docText := doc.Text()
  1475. mapDoc := make(map[int]string)
  1476. var mapSort []int
  1477. p := doc.Find("p")
  1478. p.Each(func(tk int, pd *goquery.Selection) {
  1479. pdText := pd.Text()
  1480. pdText = strings.Replace(pdText, " ", "", -1)
  1481. if pdText != "" {
  1482. textLen := strings.Index(docText, pdText)
  1483. mapDoc[(strings.Index(docText, pdText))] = pdText
  1484. mapSort = append(mapSort, textLen)
  1485. }
  1486. })
  1487. li := doc.Find("li")
  1488. li.Each(func(tk int, li *goquery.Selection) {
  1489. liText := li.Text()
  1490. liText = strings.Replace(liText, " ", "", -1)
  1491. if liText != "" {
  1492. textLen := strings.Index(docText, liText)
  1493. mapDoc[(strings.Index(docText, liText))] = strconv.Itoa(tk+1) + "." + liText
  1494. mapSort = append(mapSort, textLen)
  1495. }
  1496. })
  1497. ul := doc.Find("ul")
  1498. ul.Each(func(tk int, ul *goquery.Selection) {
  1499. ulText := ul.Text()
  1500. ulText = strings.Replace(ulText, " ", "", -1)
  1501. if ulText != "" {
  1502. textLen := strings.Index(docText, ulText)
  1503. mapDoc[(strings.Index(docText, ulText))] = ulText
  1504. mapSort = append(mapSort, textLen)
  1505. }
  1506. })
  1507. if len(mapSort) == 0 {
  1508. return
  1509. } else {
  1510. //排序
  1511. sort.Ints(mapSort)
  1512. var annotationHtml string
  1513. for _, vSort := range mapSort {
  1514. for k, v := range mapDoc {
  1515. if k == vSort && v != "" {
  1516. annotationHtml += v + "<br>"
  1517. }
  1518. }
  1519. }
  1520. annotationHtml = strings.TrimRight(annotationHtml, "<br>")
  1521. annotationHtml = "<p>" + annotationHtml + "</p>"
  1522. annotation = annotationHtml
  1523. }
  1524. }
  1525. return
  1526. }
  1527. // 处理核心观点的展示规则
  1528. func AnnotationHtml(bodyText string) (annotation string) {
  1529. if bodyText == "" {
  1530. return
  1531. }
  1532. annotation = bodyText
  1533. annotation = html.UnescapeString(annotation)
  1534. doc, _ := goquery.NewDocumentFromReader(strings.NewReader(annotation))
  1535. docText := doc.Text()
  1536. mapDoc := make(map[int]string)
  1537. var mapSort []int
  1538. p := doc.Find("p")
  1539. p.Each(func(tk int, pd *goquery.Selection) {
  1540. pdText := pd.Text()
  1541. //pdText = strings.Replace(pdText, " ", "", -1)
  1542. if pdText != "" {
  1543. textLen := strings.Index(docText, pdText)
  1544. mapDoc[(strings.Index(docText, pdText))] = pdText
  1545. mapSort = append(mapSort, textLen)
  1546. }
  1547. })
  1548. li := doc.Find("li")
  1549. li.Each(func(tk int, li *goquery.Selection) {
  1550. liText := li.Text()
  1551. liText = strings.Replace(liText, " ", "", -1)
  1552. if liText != "" {
  1553. textLen := strings.Index(docText, liText)
  1554. mapDoc[(strings.Index(docText, liText))] = strconv.Itoa(tk+1) + "." + liText
  1555. mapSort = append(mapSort, textLen)
  1556. }
  1557. })
  1558. ul := doc.Find("ul")
  1559. ul.Each(func(tk int, ul *goquery.Selection) {
  1560. ulText := ul.Text()
  1561. ulText = strings.Replace(ulText, " ", "", -1)
  1562. if ulText != "" {
  1563. textLen := strings.Index(docText, ulText)
  1564. mapDoc[(strings.Index(docText, ulText))] = ulText
  1565. mapSort = append(mapSort, textLen)
  1566. }
  1567. })
  1568. section := doc.Find("section")
  1569. section.Each(func(tk int, section *goquery.Selection) {
  1570. sectionText := section.Text()
  1571. sectionText = strings.Replace(sectionText, " ", "", -1)
  1572. if sectionText != "" {
  1573. textLen := strings.Index(docText, sectionText)
  1574. mapDoc[(strings.Index(docText, sectionText))] = sectionText
  1575. mapSort = append(mapSort, textLen)
  1576. }
  1577. })
  1578. if len(mapSort) == 0 {
  1579. return
  1580. } else {
  1581. //排序
  1582. sort.Ints(mapSort)
  1583. var annotationHtml string
  1584. for _, vSort := range mapSort {
  1585. for k, v := range mapDoc {
  1586. if k == vSort && v != "" {
  1587. annotationHtml += v + "<br>"
  1588. }
  1589. }
  1590. }
  1591. annotationHtml = strings.TrimRight(annotationHtml, "<br>")
  1592. annotationHtml = "<p>" + annotationHtml + "</p>"
  1593. annotation = annotationHtml
  1594. }
  1595. return
  1596. }
  1597. // 处理产品内测展示规则
  1598. func ProductInteriorHtml(bodyText string) (annotation string) {
  1599. if bodyText == "" {
  1600. return
  1601. }
  1602. sliceBody := strings.Split(bodyText, "</p>")
  1603. annotation, _ = GetReportContentTextSub(sliceBody[0])
  1604. return
  1605. }
  1606. // 解析研选内容中的核心观点
  1607. func YxArticleAnnotation(article *models.HomeArticle) (annotation string) {
  1608. //如果不规范,就获取内容主体
  1609. if strings.Count(article.Body, "<hr") == 0 {
  1610. //如果内容不规范而且,还有图片,就把核心观点置空
  1611. if article.BodyHtml != "" {
  1612. return
  1613. }
  1614. annotation, _ = GetReportContentTextSub(article.Body)
  1615. return
  1616. }
  1617. body := strings.ReplaceAll(article.Body, "<strong>", "")
  1618. body = strings.ReplaceAll(body, "</strong>", "")
  1619. body = strings.ReplaceAll(body, "</ol>", "</div>")
  1620. body = strings.ReplaceAll(body, "<ol>", "<div>")
  1621. body = strings.ReplaceAll(body, "</li>", "</p>")
  1622. body = strings.ReplaceAll(body, "<li>", "<p>")
  1623. re, _ := regexp.Compile("<strong.*?>")
  1624. body = re.ReplaceAllString(body, "")
  1625. reLi, _ := regexp.Compile("<li.*?>")
  1626. body = reLi.ReplaceAllString(body, "")
  1627. var plus int
  1628. coreIndex := strings.Index(body, "核心观点:")
  1629. plus = 15
  1630. if coreIndex == -1 {
  1631. coreIndex = strings.Index(body, "核心观点:")
  1632. plus = 13
  1633. }
  1634. if coreIndex == -1 {
  1635. coreIndex = strings.Index(body, "核心观点")
  1636. plus = 12
  1637. }
  1638. if coreIndex == -1 {
  1639. coreIndex = strings.Index(body, "核心结论:")
  1640. plus = 15
  1641. }
  1642. if coreIndex == -1 {
  1643. coreIndex = strings.Index(body, "核心结论:")
  1644. plus = 13
  1645. }
  1646. if coreIndex == -1 {
  1647. coreIndex = strings.Index(body, "核心结论")
  1648. plus = 12
  1649. }
  1650. endIndex := strings.Index(body, "<hr")
  1651. if coreIndex != -1 && endIndex != -1 {
  1652. body = body[coreIndex+plus : endIndex]
  1653. }
  1654. annotation = body
  1655. return
  1656. }
  1657. // 通过接解析带有Md5的文章链接获取文章ID
  1658. func GetReportLinkToArticleid(reportLink string) (articleId int, err error) {
  1659. defer func() {
  1660. if err != nil {
  1661. go utils.SendAlarmMsg("通过接解析带有Md5的文章链接获取文章ID失败"+err.Error(), 2)
  1662. }
  1663. }()
  1664. var artMd5 string
  1665. //处理Md5的
  1666. strnum1 := strings.Index(reportLink, "id=")
  1667. if strnum1 > 0 {
  1668. sliceId := strings.Split(reportLink, "id=")
  1669. if len(sliceId) > 1 {
  1670. reportLink = sliceId[1]
  1671. sliceMd5Id := strings.Split(reportLink, "&")
  1672. artMd5 = sliceMd5Id[0]
  1673. }
  1674. if artMd5 != "" {
  1675. detail, errArt := models.GetArticleDetailByIdMd5(artMd5)
  1676. if errArt != nil && errArt.Error() != utils.ErrNoRow() {
  1677. err = errArt
  1678. return
  1679. }
  1680. if detail != nil {
  1681. articleId = detail.ArticleId
  1682. }
  1683. }
  1684. } else {
  1685. //处理活动的
  1686. linkList := strings.Split(reportLink, "/")
  1687. if linkList[len(linkList)-1] != "" {
  1688. linkArticleId, _ := strconv.Atoi(linkList[len(linkList)-1])
  1689. if linkArticleId > 0 {
  1690. articleInfo, errArt := models.GetArticleDetailById(linkArticleId)
  1691. if errArt != nil && errArt.Error() != utils.ErrNoRow() {
  1692. err = errArt
  1693. return
  1694. }
  1695. if articleInfo != nil {
  1696. articleId = articleInfo.ArticleId
  1697. }
  1698. }
  1699. }
  1700. }
  1701. return
  1702. }
  1703. // GetArticleStockMap 获取个股标签所对应的文章ID
  1704. func GetArticleStockMap() (mapResp map[string]int, err error) {
  1705. defer func() {
  1706. if err != nil {
  1707. go utils.SendAlarmMsg("获取个股标签所对应的文章ID失败"+err.Error(), 2)
  1708. }
  1709. }()
  1710. list, err := models.GetArticleStock()
  1711. if err != nil && err.Error() != utils.ErrNoRow() {
  1712. return
  1713. }
  1714. mapResp = make(map[string]int, 0)
  1715. if len(list) > 0 {
  1716. //一对一精准匹配
  1717. for _, v := range list {
  1718. sliceSubjects := strings.Split(v.Stock, "/")
  1719. if len(sliceSubjects) > 0 {
  1720. for _, vSubject := range sliceSubjects {
  1721. sliceKuohao := strings.Split(vSubject, "(") //过滤括号
  1722. sliceXiahuaxian := strings.Split(sliceKuohao[0], "-") //过滤下划线
  1723. subject := sliceXiahuaxian[0]
  1724. mapResp[subject] = v.ArticleId
  1725. }
  1726. }
  1727. }
  1728. }
  1729. return
  1730. }
  1731. // GetAiQianYanArtilceList 获取AI前沿几篇文章
  1732. func GetAiQianYanArtilceList(startSize, pageSize int) (items []*models.ReportArticle, total int, err error) {
  1733. defer func() {
  1734. if err != nil {
  1735. go utils.SendAlarmMsg("获取AI前沿几篇文章失败"+err.Error(), 2)
  1736. }
  1737. }()
  1738. var condition string
  1739. var pars []interface{}
  1740. condition += ` AND title LIKE '%AI前沿%' AND publish_status = 1 ORDER BY publish_date DESC `
  1741. articleList, e := models.GetCygxCygxArticleList(condition, pars, startSize, pageSize)
  1742. if e != nil {
  1743. err = errors.New("GetCygxCygxArticleList, Err: " + e.Error())
  1744. return
  1745. }
  1746. total, e = models.GetCygxArticleCount(condition, pars)
  1747. if e != nil {
  1748. err = errors.New("GetCygxArticleCount, Err: " + e.Error())
  1749. return
  1750. }
  1751. for _, v := range articleList {
  1752. item := new(models.ReportArticle)
  1753. item.ArticleId = v.ArticleId
  1754. item.Title = v.Title
  1755. item.Abstract = v.Abstract
  1756. item.Annotation = v.Annotation
  1757. item.PublishDate = v.PublishDate
  1758. item.Resource = 1
  1759. item.CategoryId = strconv.Itoa(v.CategoryId)
  1760. item.Body = v.Body
  1761. items = append(items, item)
  1762. }
  1763. return
  1764. }
  1765. // 获取文章内部图表的第一张封面图
  1766. func GetArticleCoverByChartFirst(body string) (cover string) {
  1767. var err error
  1768. defer func() {
  1769. if err != nil {
  1770. fmt.Println(err)
  1771. go utils.SendAlarmMsg("获取文章内部图表的第一张封面图失败,Err:"+err.Error()+body, 3)
  1772. }
  1773. }()
  1774. var chartUrl string
  1775. var hrefRegexp = regexp.MustCompile(utils.RegularUrl)
  1776. match := hrefRegexp.FindAllString(body, -1)
  1777. if match != nil {
  1778. for _, v := range match {
  1779. if strings.Contains(v, "charts/display") {
  1780. chartUrl = v
  1781. break
  1782. }
  1783. }
  1784. }
  1785. if chartUrl != "" {
  1786. urlSlice := strings.Split(chartUrl, "/")
  1787. lenurlSlice := len(urlSlice)
  1788. sourceId, _ := strconv.Atoi(urlSlice[lenurlSlice-1])
  1789. if sourceId == 0 {
  1790. sourceId, _ = strconv.Atoi(urlSlice[lenurlSlice-2])
  1791. }
  1792. if sourceId == 0 {
  1793. err = errors.New("新的图标链接无法解析" + chartUrl)
  1794. return
  1795. }
  1796. chartUrlApi := utils.ApiUrl + "charts/" + strconv.Itoa(sourceId)
  1797. authorization := utils.ApiAuthorization
  1798. body, e := PublicGetDate(chartUrlApi, authorization)
  1799. if e != nil {
  1800. err = errors.New("PublicGetDate" + err.Error())
  1801. return
  1802. }
  1803. var resultDate models.ChartResultDetailApi
  1804. e = json.Unmarshal(body, &resultDate)
  1805. if e != nil {
  1806. err = errors.New(" json.Unmarshal" + err.Error())
  1807. }
  1808. cover = resultDate.Data.Cover
  1809. }
  1810. return
  1811. }
  1812. // GetYxArticleIdMap 获取研选文章ID
  1813. func GetYxArticleIdMap(articleIds []int) (mapResp map[int]bool) {
  1814. var err error
  1815. defer func() {
  1816. if err != nil {
  1817. go utils.SendAlarmMsg("获取研选文章ID失败,GetYxArticleIdMap"+err.Error(), 2)
  1818. }
  1819. }()
  1820. var condition string
  1821. var pars []interface{}
  1822. condition = ` AND article_type_id > 0 `
  1823. if len(articleIds) > 0 {
  1824. condition += ` AND article_id IN (` + utils.GetOrmInReplace(len(articleIds)) + `)`
  1825. pars = append(pars, articleIds)
  1826. }
  1827. articleList, e := models.GetArticleList(condition, pars)
  1828. if e != nil {
  1829. err = errors.New("GetArticleList, Err: " + e.Error())
  1830. return
  1831. }
  1832. mapResp = make(map[int]bool, 0)
  1833. for _, v := range articleList {
  1834. mapResp[v.ArticleId] = true
  1835. }
  1836. return
  1837. }
  1838. // GetUserArticleHistoryRecord 获取用户已经阅读的文章ID
  1839. func GetUserArticleHistoryRecord(user *models.WxUserItem) (articleIdMap map[int]bool, err error) {
  1840. var condition string
  1841. var pars []interface{}
  1842. condition = " AND user_id = ? "
  1843. pars = append(pars, user.UserId)
  1844. articleIdMap = make(map[int]bool, 0)
  1845. list, err := models.GetArticleHistoryRecord(condition, pars)
  1846. if err != nil && err.Error() != utils.ErrNoRow() {
  1847. return
  1848. }
  1849. for _, v := range list {
  1850. articleIdMap[v.ArticleId] = true
  1851. }
  1852. return
  1853. }
  1854. // GetUserArticleHistoryIndustrialNewPublishDate 获取产业关联的的文章用户是否阅读
  1855. func GetUserArticleHistoryIndustrialNewPublishDate(user *models.WxUserItem, industrialIdArr []int, articleTypeIds, resource string) (mapIndustrialArticleHistory map[int]bool, err error) {
  1856. if len(industrialIdArr) == 0 {
  1857. return
  1858. }
  1859. var condition string
  1860. var pars []interface{}
  1861. articleIdMap, err := GetUserArticleHistoryRecord(user)
  1862. if err != nil {
  1863. return
  1864. }
  1865. if resource == utils.REOURCE_YX {
  1866. condition = ` AND a.publish_status = 1 AND a.article_type_id IN (` + articleTypeIds + `) `
  1867. } else {
  1868. condition = ` AND a.article_type_id = 0 AND a.article_id < ` + strconv.Itoa(utils.SummaryArticleId)
  1869. }
  1870. var startTime time.Time
  1871. if utils.StrTimeToTime(utils.OnlineTime).After(user.CreatedTime) {
  1872. startTime = utils.StrTimeToTime(utils.OnlineTime)
  1873. } else {
  1874. startTime = user.CreatedTime
  1875. }
  1876. condition += ` AND mg.industrial_management_id IN (` + utils.GetOrmInReplace(len(industrialIdArr)) + `) AND a.publish_date > ? `
  1877. pars = append(pars, industrialIdArr, startTime)
  1878. list, err := models.GetCygxIndustrialArticleGroupManagement(condition, pars)
  1879. if err != nil {
  1880. return
  1881. }
  1882. mapIndustrialArticle := make(map[int][]*models.CygxIndustrialArticleGroupManagementResp)
  1883. //mapIndustrialArticleMax := make(map[int]int, 0)
  1884. mapIndustrialArticleHistory = make(map[int]bool, 0)
  1885. //mapIndustrialArticleMaxTime := make(map[int]time.Time)
  1886. for _, v := range list {
  1887. item := new(models.CygxIndustrialArticleGroupManagementResp)
  1888. item.IndustrialManagementId = v.IndustrialManagementId
  1889. item.ArticleId = v.ArticleId
  1890. item.PublishDate = v.PublishDate
  1891. mapIndustrialArticle[v.IndustrialManagementId] = append(mapIndustrialArticle[v.IndustrialManagementId], item)
  1892. }
  1893. for k, v := range mapIndustrialArticle {
  1894. for _, item := range v {
  1895. if !articleIdMap[item.ArticleId] {
  1896. mapIndustrialArticleHistory[k] = true
  1897. }
  1898. }
  1899. }
  1900. return
  1901. }