everything.go 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021
  1. package main
  2. import (
  3. "app.yhyue.com/moapp/jybase/redis"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/gogf/gf/util/gconv"
  7. "log"
  8. "net/url"
  9. "os"
  10. "path/filepath"
  11. "strings"
  12. "time"
  13. "app.yhyue.com/moapp/jybase/mail"
  14. "github.com/tealeg/xlsx"
  15. "app.yhyue.com/moapp/jybase/date"
  16. dates "app.yhyue.com/moapp/jybase/date"
  17. "app.yhyue.com/moapp/jybase/mongodb"
  18. "app.yhyue.com/moapp/jybase/common"
  19. )
  20. // 渠道定时任务
  21. func everythingSync() {
  22. lastEverythingTime := cfg.LastEverythingTime
  23. codeArr := []string{}
  24. if len(UserChannel) > 0 {
  25. for i := range UserChannel {
  26. codeArr = append(codeArr, fmt.Sprintf(`"%s"`, i))
  27. }
  28. }
  29. sql := fmt.Sprintf(`select * from user_source where channel_code in (%s) and create_time > "%s" order by create_time asc`, strings.Join(codeArr, ","), lastEverythingTime)
  30. //sql := fmt.Sprintf(`select * from user_source where id =662847`)
  31. log.Println("三方渠道数据定时任务开始", sql)
  32. data := ThirdParty.SelectBySql(sql)
  33. if data != nil && *data != nil && len(*data) > 0 {
  34. everythingSyncHandle(data, true)
  35. }
  36. sql = fmt.Sprintf(`select * from user_source where channel_code in (%s) and ishandle=1 order by create_time asc`, strings.Join(codeArr, ","))
  37. log.Println("三方渠道数据定时任务开始1", sql)
  38. data = ThirdParty.SelectBySql(sql)
  39. if data != nil && *data != nil && len(*data) > 0 {
  40. everythingSyncHandle(data, false)
  41. }
  42. log.Println("三方渠道数据定时任务结束")
  43. }
  44. func everythingSyncHandle(data *[]map[string]interface{}, isUpdate bool) {
  45. nowTime := time.Now().Format(dates.Date_Full_Layout)
  46. for _, v := range *data {
  47. id := gconv.Int64(v["id"])
  48. channel_code := gconv.String(v["channel_code"])
  49. changeMap := UserChannel[channel_code]
  50. channeName := changeMap["name"]
  51. channelCode := changeMap["clueCode"]
  52. user_id := common.ObjToString(v["user_id"])
  53. position_id := common.Int64All(v["position_id"])
  54. phone := common.ObjToString(v["phone"])
  55. state := common.IntAll(v["state"])
  56. user_mold, is_assign, is_transfer, last_login_time, registe_time, mailbox, source, clueId, order_type := 4, 0, 0, "", "", "", "", int64(0), ""
  57. owner, sales_lead_phone, sales_ent_name, sales_position, sales_dep, data_request, sales_leads_source := "", "", "", "", "", "", ""
  58. cData := TiDb.FindOne("dwd_f_crm_clue_info", map[string]interface{}{"phone": phone}, "", "")
  59. if cData != nil {
  60. is_assign = common.IntAll((*cData)["is_assign"])
  61. is_transfer = common.IntAll((*cData)["is_transfer"])
  62. clueId = common.Int64All((*cData)["id"])
  63. }
  64. udata := TiDb.FindOne("dwd_f_userbase_baseinfo", map[string]interface{}{"userid": user_id}, "", "")
  65. if udata != nil {
  66. userSource := common.ObjToString((*udata)["source"])
  67. if userSource == "0102" {
  68. continue
  69. }
  70. }
  71. //
  72. /*cuData := TiDbData.FindOne("customer", map[string]interface{}{"phone": phone}, "", "")
  73. if cuData != nil {
  74. owner = common.ObjToString((*cuData)["owner"])
  75. unique_id = common.ObjToString((*cuData)["unique_id"])
  76. belongTo = common.ObjToString((*cuData)["belongTo"])
  77. }*/
  78. vData := TiDb.FindOne("dwd_f_userbase_visit_info", map[string]interface{}{"userid": user_id}, "", "")
  79. if vData != nil {
  80. last_login_time = common.ObjToString((*vData)["date"])
  81. }
  82. dData := Mysql.Find("dataexport_order", map[string]interface{}{"user_id": user_id}, "", "", -1, -1)
  83. if dData != nil && len(*dData) > 0 {
  84. pMap := map[string]string{}
  85. pArr := []string{}
  86. for _, v := range *dData {
  87. product_type := common.ObjToString(v["product_type"])
  88. pMap[product_type] = "1"
  89. }
  90. for k := range pMap {
  91. pArr = append(pArr, k)
  92. }
  93. if len(pArr) > 0 {
  94. order_type = strings.Join(pArr, ",")
  95. }
  96. }
  97. uData, ok := Mgo.FindOne("user", map[string]interface{}{"_id": mongodb.StringTOBsonId(user_id)})
  98. if ok && uData != nil {
  99. l_registedate := common.Int64All((*uData)["l_registedate"])
  100. registe_time = time.Unix(l_registedate, 0).Format(dates.Date_Full_Layout)
  101. }
  102. if TiDb.Count("dwd_f_userbase_baseinfo", map[string]interface{}{
  103. "userid": user_id,
  104. }) <= 0 {
  105. ThirdParty.Update("user_source", map[string]interface{}{
  106. "id": id,
  107. }, map[string]interface{}{
  108. "ishandle": 1,
  109. })
  110. continue
  111. }
  112. sData, oks := Mgo.Find("saleLeads", map[string]interface{}{"userid": user_id}, `{"_id":-1}`, nil, false, 0, 1)
  113. if oks && sData != nil {
  114. if len(*sData) > 0 {
  115. sDatas := (*sData)[0]
  116. sales_lead_phone = common.ObjToString(sDatas["phone"])
  117. sales_position = common.ObjToString(sDatas["position"])
  118. sales_ent_name = common.ObjToString(sDatas["company"])
  119. sales_dep = common.ObjToString(sDatas["branch"])
  120. if sales_dep == "" {
  121. sales_dep = common.ObjToString(sDatas["department"])
  122. }
  123. data_request = common.ObjToString(sDatas["data_requirement"])
  124. mailbox = common.ObjToString(sDatas["mail"])
  125. source = common.ObjToString(sDatas["interest"])
  126. }
  127. }
  128. if state == 1 {
  129. sales_leads_source = fmt.Sprintf("%s登录", channeName)
  130. if is_assign == 1 || is_transfer == 1 || owner != "" {
  131. user_mold = 1
  132. } else {
  133. user_mold = 2
  134. }
  135. } else if state == 2 {
  136. sales_leads_source = fmt.Sprintf("%s注册", channeName)
  137. if is_assign == 1 || is_transfer == 1 || owner != "" {
  138. user_mold = 3
  139. } else {
  140. user_mold = 4
  141. }
  142. }
  143. if ThirdParty.Count("user_channel_info", map[string]interface{}{"user_id": user_id}) > 0 {
  144. ThirdParty.Update("user_channel_info", map[string]interface{}{"user_id": user_id}, map[string]interface{}{
  145. "user_mold": user_mold,
  146. "last_login_time": common.If(last_login_time != "", last_login_time, nil),
  147. "sales_lead_phone": sales_lead_phone,
  148. "sales_ent_name": sales_ent_name,
  149. "sales_position": sales_position,
  150. "sales_dep": sales_dep,
  151. "sales_leads_source": sales_leads_source,
  152. "mailbox": mailbox,
  153. "order_type": order_type,
  154. "data_request": data_request,
  155. "source": source,
  156. "update_time": nowTime,
  157. })
  158. } else {
  159. ThirdParty.Insert("user_channel_info", map[string]interface{}{
  160. "user_id": user_id,
  161. "registe_time": registe_time,
  162. "phone": phone,
  163. "user_mold": user_mold,
  164. "last_login_time": common.If(last_login_time != "", last_login_time, nil),
  165. "sales_lead_phone": sales_lead_phone,
  166. "sales_ent_name": sales_ent_name,
  167. "sales_position": sales_position,
  168. "sales_dep": sales_dep,
  169. "sales_leads_source": sales_leads_source,
  170. "order_type": order_type,
  171. "mailbox": mailbox,
  172. "data_request": data_request,
  173. "source": source,
  174. "update_time": nowTime,
  175. })
  176. }
  177. if clueId > 0 && is_assign != 1 && is_transfer != 1 {
  178. TiDb.Update("dwd_f_userbase_baseinfo", map[string]interface{}{"userid": user_id}, map[string]interface{}{"belong_to": channelCode})
  179. TiDb.Update("dwd_f_crm_clue_info", map[string]interface{}{"phone": phone}, map[string]interface{}{"is_assign": -1})
  180. TiDb.Insert("dwd_f_crm_clue_change_record", map[string]interface{}{
  181. "clue_id": clueId,
  182. "position_id": common.If(position_id > 0, position_id, -1),
  183. "change_type": "退出公海",
  184. "new_value": fmt.Sprintf("通过合作渠道%s", channeName) + fmt.Sprint(common.If(user_mold == 1 || user_mold == 2, "登录", "注册")),
  185. "createtime": nowTime,
  186. "BCPCID": common.GetRandom(32),
  187. "operator_id": -1,
  188. })
  189. }
  190. ThirdParty.Update("user_source", map[string]interface{}{
  191. "id": id,
  192. }, map[string]interface{}{
  193. "ishandle": 0,
  194. })
  195. }
  196. if isUpdate {
  197. cfg.LastEverythingTime = common.ObjToString((*data)[len(*data)-1]["create_time"])
  198. common.WriteSysConfig(&cfg)
  199. }
  200. }
  201. func saveEverything(user_id, phone, item, sourceName, sourceCode, belongTo string) bool {
  202. changeMap := UserChannel[belongTo]
  203. changeName := gconv.String(changeMap["name"])
  204. nowTime, isOk := time.Now().Format(dates.Date_Full_Layout), false
  205. user_mold, is_assign, is_transfer, last_login_time, registe_time, mailbox, source, order_type, sales_leads_source := 4, 0, 0, "", "", "", "", "", ""
  206. owner, sales_lead_phone, sales_ent_name, sales_position, sales_dep, data_request, clueId, position_id := "", "", "", "", "", "", int64(0), int64(0)
  207. cData := TiDb.FindOne("dwd_f_crm_clue_info", map[string]interface{}{"phone": phone}, "", "")
  208. if cData != nil {
  209. is_assign = common.IntAll((*cData)["is_assign"])
  210. is_transfer = common.IntAll((*cData)["is_transfer"])
  211. clueId = common.Int64All((*cData)["is_transfer"])
  212. position_id = common.Int64All((*cData)["position_id"])
  213. }
  214. cuData := TiDbData.FindOne("customer", map[string]interface{}{"phone": phone}, "", "")
  215. if cuData != nil {
  216. owner = common.ObjToString((*cuData)["owner"])
  217. }
  218. dData := Mysql.Find("dataexport_order", map[string]interface{}{"user_id": user_id}, "", "", -1, -1)
  219. if dData != nil && len(*dData) > 0 {
  220. pMap := map[string]string{}
  221. pArr := []string{}
  222. for _, v := range *dData {
  223. product_type := common.ObjToString(v["product_type"])
  224. pMap[product_type] = "1"
  225. }
  226. for k := range pMap {
  227. pArr = append(pArr, k)
  228. }
  229. if len(pArr) > 0 {
  230. order_type = strings.Join(pArr, ",")
  231. }
  232. }
  233. vData := TiDb.FindOne("dwd_f_userbase_visit_info", map[string]interface{}{"userid": user_id}, "", "")
  234. if vData != nil {
  235. last_login_time = common.ObjToString((*vData)["date"])
  236. }
  237. uData, ok := Mgo.FindOne("user", map[string]interface{}{"_id": mongodb.StringTOBsonId(user_id)})
  238. if ok && uData != nil {
  239. l_registedate := common.Int64All((*uData)["l_registedate"])
  240. registe_time = time.Unix(l_registedate, 0).Format(dates.Date_Full_Layout)
  241. }
  242. if item == "orders" {
  243. sales_leads_source = "订单未支付"
  244. } else if item == "users" || item == "xcxusers" {
  245. return false
  246. } else if item == "saleLeads" {
  247. sales_leads_source = sourceName
  248. } else {
  249. sales_leads_source = "用户留资"
  250. }
  251. query := map[string]interface{}{"userid": user_id}
  252. if sourceCode != "" {
  253. query["source"] = sourceCode
  254. }
  255. sData, oks := Mgo.Find("saleLeads", query, `{"_id":-1}`, nil, false, 0, 1)
  256. if oks && sData != nil {
  257. if len(*sData) > 0 {
  258. sDatas := (*sData)[0]
  259. sales_lead_phone = common.ObjToString(sDatas["phone"])
  260. sales_position = common.ObjToString(sDatas["position"])
  261. sales_ent_name = common.ObjToString(sDatas["company"])
  262. sales_dep = common.ObjToString(sDatas["branch"])
  263. if sales_dep == "" {
  264. sales_dep = common.ObjToString(sDatas["department"])
  265. }
  266. data_request = common.ObjToString(sDatas["data_requirement"])
  267. mailbox = common.ObjToString(sDatas["mail"])
  268. source = common.ObjToString(sDatas["interest"])
  269. }
  270. }
  271. if is_assign == 1 || is_transfer == 1 || owner != "" {
  272. user_mold = 1
  273. } else {
  274. user_mold = 2
  275. }
  276. /*if owner == "" && unique_id != "" && belongTo == "市场部" {
  277. token := getToken()
  278. if token != "" {
  279. sss := url.QueryEscape("合作渠道一切都好")
  280. urls := `https://a1.7x24cc.com/commonInte?flag=1008&account=N000000029739&accessToken=` + token + `&cusObj={"unique_id":"` + unique_id + `","empNo":"8049","owner":"8049","source":"` + sss + `"}&dbType=0001`
  281. bs, err := doGet(urls)
  282. if err != nil {
  283. log.Println("调用接口失败", unique_id, err)
  284. }
  285. resMap := common.StringToMap(string(bs))
  286. if resMap["success"] != nil && resMap["success"].(bool) {
  287. ok := TiDbData.Update("customer", map[string]interface{}{"unique_id": unique_id}, map[string]interface{}{"empNo": "8049", "owner": "8049", "source": "合作渠道一切都好"})
  288. if ok {
  289. log.Println("更新成功", unique_id)
  290. } else {
  291. log.Println("更新失败", unique_id)
  292. }
  293. } else {
  294. log.Println("调用接口失败!!", unique_id)
  295. }
  296. }
  297. }*/
  298. if ThirdParty.Count("user_channel_info", map[string]interface{}{"user_id": user_id}) > 0 {
  299. ThirdParty.Update("user_channel_info", map[string]interface{}{"user_id": user_id}, map[string]interface{}{
  300. "user_mold": user_mold,
  301. "last_login_time": common.If(last_login_time != "", last_login_time, nil),
  302. "sales_lead_phone": sales_lead_phone,
  303. "sales_ent_name": sales_ent_name,
  304. "sales_position": sales_position,
  305. "sales_dep": sales_dep,
  306. "sales_leads_source": sales_leads_source,
  307. "mailbox": mailbox,
  308. "data_request": data_request,
  309. "source": source,
  310. "order_type": order_type,
  311. "update_time": nowTime,
  312. })
  313. } else {
  314. ThirdParty.Insert("user_channel_info", map[string]interface{}{
  315. "user_id": user_id,
  316. "registe_time": registe_time,
  317. "phone": phone,
  318. "user_mold": user_mold,
  319. "last_login_time": common.If(last_login_time != "", last_login_time, nil),
  320. "sales_lead_phone": sales_lead_phone,
  321. "sales_ent_name": sales_ent_name,
  322. "sales_position": sales_position,
  323. "sales_dep": sales_dep,
  324. "sales_leads_source": sales_leads_source,
  325. "mailbox": mailbox,
  326. "data_request": data_request,
  327. "order_type": order_type,
  328. "source": source,
  329. "update_time": nowTime,
  330. })
  331. }
  332. if clueId > 0 && is_assign != 1 && is_transfer != 1 {
  333. TiDb.Update("dwd_f_crm_clue_info", map[string]interface{}{"phone": phone}, map[string]interface{}{"is_assign": -1})
  334. TiDb.Insert("dwd_f_crm_clue_change_record", map[string]interface{}{
  335. "clue_id": clueId,
  336. "position_id": common.If(position_id > 0, position_id, -1),
  337. "change_type": "退出公海",
  338. "new_value": fmt.Sprintf("通过合作渠道%s登录", changeName),
  339. "createtime": nowTime,
  340. "BCPCID": common.GetRandom(32),
  341. "operator_id": -1,
  342. })
  343. } else if is_assign == 1 || is_transfer == 1 {
  344. isOk = true
  345. }
  346. return isOk
  347. }
  348. // 大客户线索-发邮件
  349. func bigCustomer() {
  350. runOk := getRunOk()
  351. if !runOk {
  352. log.Println("不是工作日,任务暂停")
  353. return
  354. }
  355. log.Println("大客户线索定时任务开始")
  356. dataArr := []map[string]interface{}{}
  357. filterArr := []string{"-pc", "-app", "-wx", "-h5"}
  358. sourceMap := map[string]map[string]interface{}{} //根据留资维表:tidb/Jianyu_subjectdb/d_saleleads_code
  359. saleSource := TiDb.SelectBySql(`SELECT source,name,appoint_email FROM d_saleleads_code WHERE department LIKE '%大客户%' AND is_delete = 1`)
  360. if saleSource != nil && len(*saleSource) > 0 {
  361. for _, v := range *saleSource {
  362. saleSourceMap := v
  363. source := common.ObjToString(v["source"])
  364. name := common.ObjToString(v["name"])
  365. for _, s := range filterArr {
  366. name = strings.ReplaceAll(name, s, "")
  367. }
  368. saleSourceMap["name"] = name
  369. sourceMap[source] = saleSourceMap
  370. }
  371. }
  372. nowTime := time.Now().Format(date.Date_Full_Layout)
  373. bigSaleTime := cfg.BigSaleTime
  374. bigOrderTime := cfg.BigOrderTime
  375. if bigSaleTime == 0 {
  376. bigSaleTime = time.Now().Unix()
  377. }
  378. if bigOrderTime == "" {
  379. bigOrderTime = nowTime
  380. }
  381. data := FindBatchData("dk")
  382. saleleadsData, ok := Mgo.Find("saleLeads", map[string]interface{}{"createtime": map[string]interface{}{"$gt": bigSaleTime}}, `{"phone":1,"createtime":1}`, nil, false, -1, -1)
  383. if ok && saleleadsData != nil && len(*saleleadsData) > 0 {
  384. for _, v := range *saleleadsData {
  385. sources := common.ObjToString(v["source"])
  386. userid := common.ObjToString(v["userid"])
  387. uid := ""
  388. if !mongodb.IsObjectIdHex(userid) {
  389. userMapping := TiDb.FindOne("dwd_f_userbase_id_mapping", map[string]interface{}{"position_id": userid}, "", "")
  390. if userMapping != nil && len(*userMapping) > 0 {
  391. userid = common.ObjToString((*userMapping)["userid"])
  392. uid = common.ObjToString((*userMapping)["uid"])
  393. }
  394. } else {
  395. userMapping := TiDb.FindOne("dwd_f_userbase_id_mapping", map[string]interface{}{"userid": userid}, "", "")
  396. if userMapping != nil && len(*userMapping) > 0 {
  397. userid = common.ObjToString((*userMapping)["userid"])
  398. uid = common.ObjToString((*userMapping)["uid"])
  399. }
  400. }
  401. if sourceMap[sources] != nil {
  402. sourceData := sourceMap[sources]
  403. source := sourceData["name"]
  404. phone := common.ObjToString(v["phone"])
  405. if IsInternal(phone) {
  406. continue
  407. }
  408. key := fmt.Sprintf("%s_%s", source, phone)
  409. if _, ok := data[key]; ok {
  410. continue
  411. }
  412. data[key] = true
  413. company := common.ObjToString(v["company"])
  414. job := common.ObjToString(v["position"])
  415. username := common.ObjToString(v["name"])
  416. email := common.ObjToString(v["mail"])
  417. interest := common.ObjToString(v["interest"])
  418. data_requirement := common.ObjToString(v["data_requirement"])
  419. belongTo, usernickname := "大客户", ""
  420. userData := TiDb.FindOne("dwd_f_userbase_baseinfo", map[string]interface{}{"uid": uid}, "", "")
  421. if userData != nil {
  422. usernickname = common.ObjToString((*userData)["nickname"])
  423. bt := common.ObjToString((*userData)["belong_to"])
  424. if strings.HasPrefix(bt, "03") || bt == "0102" {
  425. continue
  426. }
  427. }
  428. dataArr = append(dataArr, map[string]interface{}{
  429. "createTime": nowTime,
  430. "lastUpdateTime": nowTime,
  431. "uid": uid,
  432. "userid": userid,
  433. "username": username,
  434. "usernickname": usernickname,
  435. "company": company,
  436. "job": job,
  437. "phone": phone,
  438. "email": email,
  439. "source": source,
  440. "belongTo": belongTo,
  441. "interest": interest,
  442. "data_requirement": data_requirement,
  443. "appoint_email": sourceData["appoint_email"],
  444. })
  445. }
  446. if gconv.Int64(v["createtime"]) > bigSaleTime {
  447. bigSaleTime = gconv.Int64(v["createtime"])
  448. }
  449. }
  450. }
  451. cfg.BigSaleTime = bigSaleTime
  452. orderDataSql := `select a.*,b.product_type as productType,b.filter as productFilter,b.final_price from dataexport_order a INNER JOIN
  453. jy_order_detail b on a.order_code=b.order_code
  454. and a.create_time>? and b.product_type in ("数据流量包","历史数据") and a.order_status IN (0,1) AND a.salesperson IS NULL AND a.is_backstage_order = 0 `
  455. orderData := Mysql.SelectBySql(orderDataSql, bigOrderTime)
  456. if orderData != nil && len(*orderData) > 0 {
  457. for _, v := range *orderData {
  458. bigOrderTime = gconv.Time(gconv.String(v["create_time"])).Format(date.Date_Full_Layout)
  459. phone := common.ObjToString(v["user_phone"])
  460. order_status := common.IntAll(v["order_status"])
  461. product_type := common.ObjToString(v["productType"])
  462. orderUserId := gconv.String(v["user_id"])
  463. query := map[string]interface{}{}
  464. if !mongodb.IsObjectIdHex(orderUserId) {
  465. userMapping := TiDb.FindOne("dwd_f_userbase_id_mapping", map[string]interface{}{"position_id": orderUserId}, "", "")
  466. if userMapping != nil && len(*userMapping) > 0 {
  467. uid := common.ObjToString((*userMapping)["uid"])
  468. query["uid"] = uid
  469. }
  470. } else {
  471. query["userid"] = orderUserId
  472. }
  473. userData := TiDb.FindOne("dwd_f_userbase_baseinfo", query, "", "")
  474. username, usernickname, userid, uid, source, payorderinfo, unpayorderinfo := "", "", "", "", "", "", ""
  475. if userData != nil {
  476. if phone == "" {
  477. phone = common.ObjToString((*userData)["phone"])
  478. }
  479. if IsInternal(phone) {
  480. continue
  481. }
  482. username = common.ObjToString((*userData)["name"])
  483. usernickname = common.ObjToString((*userData)["nickname"])
  484. uid = common.ObjToString((*userData)["uid"])
  485. userid = common.ObjToString((*userData)["userid"])
  486. bt := common.ObjToString((*userData)["belong_to"])
  487. if strings.HasPrefix(bt, "03") || bt == "0102" {
  488. continue
  489. }
  490. }
  491. if product_type == "历史数据" {
  492. if order_status == 0 {
  493. source = "线上历史数据导出未支付订单"
  494. unpayorderinfo = "历史数据"
  495. } else {
  496. source = "线上历史数据导出已支付订单"
  497. payorderinfo = "历史数据" + fmt.Sprint(common.IntAll(v["final_price"])/100)
  498. }
  499. } else {
  500. if order_status == 0 {
  501. source = "线上数据流量包未支付订单"
  502. unpayorderinfo = "数据流量包"
  503. } else {
  504. source = "线上数据流量包已支付订单"
  505. payorderinfo = "数据流量包" + fmt.Sprint(common.IntAll(v["final_price"])/100)
  506. }
  507. }
  508. key := fmt.Sprintf("%s_%s", source, phone)
  509. if _, ok := data[key]; ok {
  510. continue
  511. }
  512. data[key] = true
  513. productFilterMap := gconv.Map(v["productFilter"])
  514. data_count := productFilterMap["pNum"]
  515. dataArr = append(dataArr, map[string]interface{}{
  516. "createTime": nowTime,
  517. "lastUpdateTime": nowTime,
  518. "phone": phone,
  519. "username": username,
  520. "usernickname": usernickname,
  521. "uid": uid,
  522. "userid": userid,
  523. "company": v["company_name"],
  524. "email": v["user_mail"],
  525. "belongTo": "大客户",
  526. "source": source,
  527. "data_count": data_count,
  528. "unpayorderinfo": unpayorderinfo,
  529. "payorderinfo": payorderinfo, //增加客户需求
  530. "appoint_email": "",
  531. })
  532. }
  533. cfg.BigOrderTime = bigOrderTime
  534. }
  535. saveDataToRedis(data, "dk")
  536. xlsxArr := []string{"用户昵称", "姓名", "公司名称", "职位", "联系人电话", "购买条数", "用户邮箱", "已支付订单信息", "未支付订单类型", "销售线索来源", "具体来源", "数据需求"}
  537. if len(dataArr) > 0 {
  538. bigArr := map[string][]map[string]interface{}{}
  539. for _, v := range dataArr {
  540. minemail := ""
  541. if gconv.String(v["appoint_email"]) == "" {
  542. minemail, _ = EmailSelect("dk", "")
  543. } else {
  544. minemail = gconv.String(v["appoint_email"])
  545. }
  546. bigArr[minemail] = append(bigArr[minemail], v)
  547. }
  548. for k, vb := range bigArr {
  549. batch := FindBatch("dk")
  550. fileName, detailName := time.Now().Format(date.Date_Short_Layout)+"-"+batch+"大客户销售线索", ""
  551. xf := xlsx.NewFile()
  552. style := xlsx.NewStyle()
  553. style.Font.Size = 12
  554. style.Font.Bold = true
  555. style.Alignment.Vertical = "center"
  556. style.Alignment.Horizontal = "center"
  557. detailName = time.Now().Format(date.Date_Short_Layout) + "数据详情请查看附件"
  558. sh, _ := xf.AddSheet("线索数据")
  559. row1 := sh.AddRow()
  560. for _, x := range xlsxArr {
  561. cell := row1.AddCell()
  562. cell.SetString(x)
  563. cell.SetStyle(style)
  564. }
  565. for _, v := range vb {
  566. row := sh.AddRow()
  567. row.AddCell().SetString(gconv.String(v["usernickname"]))
  568. row.AddCell().SetString(gconv.String(v["username"]))
  569. row.AddCell().SetString(gconv.String(v["company"]))
  570. row.AddCell().SetString(gconv.String(v["job"]))
  571. row.AddCell().SetString(gconv.String(v["phone"]))
  572. row.AddCell().SetValue(gconv.Int64(v["data_count"]))
  573. row.AddCell().SetString(gconv.String(v["email"]))
  574. row.AddCell().SetString(gconv.String(v["payorderinfo"]))
  575. row.AddCell().SetString(gconv.String(v["unpayorderinfo"]))
  576. row.AddCell().SetString(gconv.String(v["source"]))
  577. row.AddCell().SetString(gconv.String(v["interest"]))
  578. row.AddCell().SetString(gconv.String(v["data_requirement"]))
  579. //增加客户需求
  580. }
  581. email := k
  582. //email = "wanghao@jianyu360.com"
  583. dir := "./xlsx/dk/" + fileName + ".xlsx"
  584. err := xf.Save(dir)
  585. if err != nil {
  586. log.Println("xls error", err, dir)
  587. } else {
  588. status := mail.GSendMail_q("剑鱼标讯", email, "", "", fileName, detailName, dir, fileName+".xlsx", &Gmail)
  589. if status {
  590. log.Println("send mail success", fileName, email)
  591. }
  592. }
  593. }
  594. }
  595. common.WriteSysConfig(&cfg)
  596. log.Println("大客户线索定时任务结束")
  597. }
  598. func IsInternal(phone string) bool {
  599. if DataAnalysisService.Count("dwd_f_userbase_insider", map[string]interface{}{
  600. "mobile": phone,
  601. }) > 0 {
  602. return true
  603. }
  604. return false
  605. }
  606. func eventReg() {
  607. lastEventRegTime := cfg.LastEventRegTime
  608. sql := fmt.Sprintf(`select * from exhibition_sign_up where is_del = 0 and update_time > "%s" order by update_time asc`, lastEventRegTime)
  609. log.Println("活动报名表定时任务开始", sql)
  610. data := Jyactivities.SelectBySql(sql)
  611. if data != nil && *data != nil && len(*data) > 0 {
  612. for _, v := range *data {
  613. ok1, ok2, _ := FormatData(v, "eventReg")
  614. if !ok1 {
  615. common.WriteSysConfig(&cfg)
  616. log.Println("线索卡点", "eventReg", v, lastEventRegTime)
  617. } else {
  618. if !ok2 {
  619. log.Println("用户分配已达上限", "eventReg", v, lastEventRegTime)
  620. common.WriteSysConfig(&cfg)
  621. }
  622. }
  623. cfg.LastEventRegTime = gconv.String(v["update_time"])
  624. }
  625. }
  626. common.WriteSysConfig(&cfg)
  627. log.Println("活动报名表定时任务结束")
  628. }
  629. func saveHlyj(belong_to, item, phone, name, sourceName, cluename, position, nowTime string, isGroup, isCommerce int) {
  630. if strings.HasPrefix(belong_to, "02") && item == "eventReg" {
  631. saveMap := map[string]interface{}{
  632. "unique_id": phone,
  633. "phone": phone,
  634. "username": name,
  635. "source": sourceName,
  636. "status999": "status5",
  637. "company": cluename,
  638. "job": position,
  639. "belongTo": "市场部",
  640. "createTime": nowTime,
  641. "lastUpdateTime": nowTime,
  642. }
  643. token := getToken()
  644. updateData := map[string]interface{}{
  645. "dbType": "0001",
  646. "customerList": []map[string]interface{}{saveMap},
  647. }
  648. dataByte, _ := json.Marshal(&updateData)
  649. url := `https://a1.7x24cc.com/commonInte?flag=1007&account=N000000029739&accessToken=` + token + `&json=` + url.QueryEscape(string(dataByte))
  650. bs, err := doGet(url)
  651. if err != nil {
  652. log.Println("调用接口失败")
  653. } else {
  654. resMap := common.StringToMap(string(bs))
  655. if resMap["success"] != nil && resMap["success"].(bool) {
  656. saveMap["company_nature"] = isGroup
  657. saveMap["company_verification"] = isCommerce
  658. TiDbData.Insert("customer", saveMap)
  659. } else {
  660. log.Println("新增线索失败")
  661. }
  662. }
  663. }
  664. }
  665. func FindBatch(moudle string) string {
  666. now := time.Now().Format("2006-01-02")
  667. key := fmt.Sprintf("batch_%s_%s", now, moudle)
  668. yesterday := time.Now().AddDate(0, 0, -1).Format("2006-01-02")
  669. yesterdayKey := fmt.Sprintf("batch_%s_%s", yesterday, moudle)
  670. if ok, _ := redis.Exists("newother", yesterdayKey); ok {
  671. //删除之前数据
  672. redis.Del("newother", yesterdayKey)
  673. }
  674. batch := redis.Incr("newother", key)
  675. return fmt.Sprintf("%04d", batch)
  676. }
  677. func FindBatchData(moudle string) map[string]interface{} {
  678. now := time.Now().Format("2006-01-02")
  679. //数据提取
  680. keyContent := fmt.Sprintf("data_%s_%s", now, moudle)
  681. data := redis.Get("newother", keyContent)
  682. if data == nil {
  683. return make(map[string]interface{})
  684. } else {
  685. return gconv.Map(data)
  686. }
  687. }
  688. // 特殊留资
  689. func SpecialSaleLeads() {
  690. /*if !isRunning() {
  691. log.Println("不是工作日,任务暂停")
  692. return
  693. }*/
  694. saleLeadsConfig := TiDb.Find("saleLeads_config", map[string]interface{}{
  695. "status": 0,
  696. }, "", "", -1, -1)
  697. log.Println("特殊留资线索定时任务开始")
  698. for _, v := range *saleLeadsConfig {
  699. log.Println(fmt.Sprintf("%s留资线索定时任务开始", gconv.String(v["groupName"])), gconv.String(v["lastTime"]))
  700. //留来源获取
  701. sourceMap := buildSourceMap(gconv.String(v["groupName"]))
  702. if len(sourceMap) == 0 {
  703. log.Println(fmt.Sprintf("%s留资线索定时任务结束", gconv.String(v["groupName"])))
  704. continue
  705. }
  706. dataArr, endtime := processLeads(v, sourceMap)
  707. //最后一次数据修改
  708. TiDb.Update("saleLeads_config", map[string]interface{}{
  709. "id": gconv.Int64(v["id"]),
  710. }, map[string]interface{}{
  711. "lastTime": endtime,
  712. })
  713. exportToExcel(dataArr, gconv.String(v["title"]), gconv.String(v["groupCode"]), gconv.String(v["mail"]))
  714. log.Println(fmt.Sprintf("%s留资线索定时任务结束", gconv.String(v["groupName"])))
  715. }
  716. log.Println("特殊留资线索定时任务结束")
  717. }
  718. // 判断是否在工作日
  719. func isRunning() bool {
  720. return getRunOk()
  721. }
  722. // 从数据库构建来源映射
  723. func buildSourceMap(department string) map[string]map[string]interface{} {
  724. sourceMap := map[string]map[string]interface{}{}
  725. saleSource := TiDb.SelectBySql(fmt.Sprintf(`SELECT source,name,appoint_email FROM d_saleleads_code WHERE department LIKE '%%%s%%' AND is_delete = 1`, department))
  726. if saleSource != nil {
  727. for _, v := range *saleSource {
  728. source := gconv.String(v["source"])
  729. name := gconv.String(v["name"])
  730. name = filterSourceName(name)
  731. appointEmail := gconv.String(v["appoint_email"])
  732. sourceMap[source] = map[string]interface{}{
  733. "name": name,
  734. "appointEmail": appointEmail,
  735. }
  736. }
  737. }
  738. return sourceMap
  739. }
  740. // 过滤来源名称
  741. func filterSourceName(name string) string {
  742. filterArr := []string{"-pc", "-app", "-wx", "-h5"}
  743. for _, s := range filterArr {
  744. name = strings.ReplaceAll(name, s, "")
  745. }
  746. return name
  747. }
  748. // 获取最后一条数据时间
  749. func getCommitteeTime(cfgTime int64) int64 {
  750. if cfgTime == 0 {
  751. return time.Now().Unix()
  752. }
  753. return cfgTime
  754. }
  755. // 处理留资信息
  756. func processLeads(saleLeadsMap map[string]interface{}, sourceMap map[string]map[string]interface{}) ([]map[string]interface{}, int64) {
  757. batch := gconv.String(saleLeadsMap["groupCode"])
  758. lastTime := gconv.Int64(saleLeadsMap["lastTime"])
  759. dataArr := []map[string]interface{}{}
  760. endtime := lastTime
  761. saleleadsData, ok := Mgo.Find("saleLeads", map[string]interface{}{"createtime": map[string]interface{}{"$gt": lastTime}}, `{"phone":1,"createtime":1}`, nil, false, -1, -1)
  762. if ok && saleleadsData != nil {
  763. data := FindBatchData(batch)
  764. for _, v := range *saleleadsData {
  765. lead := map[string]interface{}{}
  766. lead, endtime = processLead(v, sourceMap, lastTime, data)
  767. if lead != nil {
  768. dataArr = append(dataArr, lead)
  769. }
  770. }
  771. saveDataToRedis(data, batch)
  772. }
  773. return dataArr, endtime
  774. }
  775. // 处理单个线索
  776. func processLead(v map[string]interface{}, sourceMap map[string]map[string]interface{}, lastTime int64, data map[string]interface{}) (map[string]interface{}, int64) {
  777. sources := gconv.String(v["source"])
  778. nowTime := time.Now().Format(date.Date_Full_Layout)
  779. // 更新最后会时间
  780. if gconv.Int64(v["createtime"]) > lastTime {
  781. lastTime = gconv.Int64(v["createtime"])
  782. }
  783. sourceData, exist := sourceMap[sources]
  784. if !exist {
  785. return nil, lastTime
  786. }
  787. phone := gconv.String(v["phone"])
  788. if IsInternal(phone) {
  789. return nil, lastTime
  790. }
  791. // 获取用户信息
  792. userData := getUserData(v)
  793. if userData == nil || isUserInBlackList(userData) {
  794. return nil, lastTime
  795. }
  796. key := fmt.Sprintf("%s_%s_%s", sources, phone, common.ObjToString(v["interest"]))
  797. if _, exists := data[key]; exists {
  798. return nil, lastTime
  799. }
  800. data[key] = true
  801. return map[string]interface{}{
  802. "createTime": nowTime,
  803. "createtime": v["createtime"],
  804. "username": gconv.String(v["name"]),
  805. "company": gconv.String(v["company"]),
  806. "phone": phone,
  807. "source": gconv.String(sourceData["name"]),
  808. "belongTo": "咨询组/市场组", // 根据需要修改
  809. "interest": gconv.String(v["interest"]),
  810. "data_requirement": gconv.String(v["data_requirement"]),
  811. "branch": v["branch"],
  812. "job": v["position"],
  813. "email": gconv.String(v["mail"]),
  814. "appointEmail": gconv.String(sourceData["appointEmail"]),
  815. }, lastTime
  816. }
  817. // 获取用户数据
  818. func getUserData(v map[string]interface{}) map[string]interface{} {
  819. userid := gconv.String(v["userid"])
  820. userMapping := &map[string]interface{}{}
  821. if !mongodb.IsObjectIdHex(userid) {
  822. userMapping = TiDb.FindOne("dwd_f_userbase_id_mapping", map[string]interface{}{"position_id": userid}, "", "")
  823. } else {
  824. userMapping = TiDb.FindOne("dwd_f_userbase_id_mapping", map[string]interface{}{"userid": userid}, "", "")
  825. }
  826. if userMapping != nil && len(*userMapping) > 0 {
  827. return *userMapping
  828. }
  829. return nil
  830. }
  831. // 判断用户是否在黑名单
  832. func isUserInBlackList(userData map[string]interface{}) bool {
  833. bt := gconv.String(userData["belong_to"])
  834. return strings.HasPrefix(bt, "03") || bt == "0102"
  835. }
  836. // 判断是否是安博会线索
  837. func isABHLead(lead map[string]interface{}) bool {
  838. s_sourceid := gconv.String(lead["s_sourceid"])
  839. return s_sourceid == db.Sourceid
  840. }
  841. // 导出到 Excel
  842. func exportToExcel(dataArr []map[string]interface{}, title, batch, mails string) {
  843. if len(dataArr) == 0 {
  844. return
  845. }
  846. bigArr := map[string][]map[string]interface{}{}
  847. for _, v := range dataArr {
  848. appointEmail := gconv.String(v["appointEmail"])
  849. if appointEmail == "" {
  850. appointEmail, _ = EmailSelect(batch, mails)
  851. }
  852. bigArr[appointEmail] = append(bigArr[appointEmail], v)
  853. }
  854. for email, arr := range bigArr {
  855. data := FindBatch(batch)
  856. xlsxArr := []string{"姓名", "联系方式", "职位", "部门", "邮箱", "公司名称", "咨询需求", "销售线索来源", "具体来源", "留资时间"}
  857. fileName, detailName := fmt.Sprintf("%s %s-%s", title, time.Now().Format(date.Date_Short_Layout), data), "今日新增销售线索,请查收附件,及时跟进。"
  858. xf := xlsx.NewFile()
  859. style := xlsx.NewStyle()
  860. style.Font.Size = 12
  861. style.Font.Bold = true
  862. style.Alignment.Vertical = "center"
  863. style.Alignment.Horizontal = "center"
  864. sh, _ := xf.AddSheet("线索数据")
  865. row1 := sh.AddRow()
  866. for _, x := range xlsxArr {
  867. cell := row1.AddCell()
  868. cell.SetString(x)
  869. cell.SetStyle(style)
  870. }
  871. for _, v := range arr {
  872. row := sh.AddRow()
  873. row.AddCell().SetString(gconv.String(v["username"]))
  874. row.AddCell().SetString(gconv.String(v["phone"]))
  875. row.AddCell().SetString(gconv.String(v["job"]))
  876. row.AddCell().SetString(gconv.String(v["branch"]))
  877. row.AddCell().SetString(gconv.String(v["email"]))
  878. row.AddCell().SetString(gconv.String(v["company"]))
  879. row.AddCell().SetString(gconv.String(v["data_requirement"]))
  880. row.AddCell().SetString(gconv.String(v["source"]))
  881. row.AddCell().SetString(gconv.String(v["interest"]))
  882. row.AddCell().SetString(gconv.Time(gconv.Int64(v["createtime"])).Format(date.Date_Full_Layout))
  883. }
  884. dir := fmt.Sprintf("./xlsx/%s/%s.xlsx", batch, fileName)
  885. // 确保目录存在(自动创建)
  886. dirPath := filepath.Dir(dir) // 提取目录部分(如 "./xlsx/batch")
  887. if err := os.MkdirAll(dirPath, 0755); err != nil {
  888. log.Println("创建目录失败:", err)
  889. return
  890. }
  891. if err := xf.Save(dir); err != nil {
  892. log.Println("xls error", err, dir)
  893. return
  894. }
  895. sendEmail(fileName, detailName, dir, email)
  896. }
  897. }
  898. // 发送邮件
  899. func sendEmail(fileName, detailName, dir, email string) {
  900. gmail := &mail.GmailAuth{
  901. SmtpHost: db.Mail.SmtpHost,
  902. SmtpPort: db.Mail.SmtpPort,
  903. User: db.Mail.User,
  904. Pwd: db.Mail.Pwd,
  905. }
  906. if status := mail.GSendMail_q("剑鱼标讯", email, "", "", fileName, detailName, dir, fileName+".xlsx", gmail); status {
  907. log.Println("send mail success", fileName, email)
  908. }
  909. }
  910. // 保存数据到 Redis
  911. func saveDataToRedis(data map[string]interface{}, batch string) {
  912. keyContent := fmt.Sprintf("data_%s_%s", time.Now().Format("2006-01-02"), batch)
  913. redis.Put("newother", keyContent, data, 86400)
  914. }
  915. // 邮箱选择处理
  916. func EmailSelect(batch, mailStr string) (string, string) {
  917. name := ""
  918. minemail := ""
  919. bigData := TiDb.SelectBySql("select * from dwd_f_crm_clue_big_autodraw_record where type =? ", batch)
  920. if bigData != nil && len(*bigData) > 0 {
  921. //查看有没有新增人员 人名-邮箱,邮箱|人名-邮箱,
  922. mailMap := map[string]string{}
  923. if mailStr != "" {
  924. for _, s := range strings.Split(mailStr, "|") {
  925. if len(strings.Split(s, "-")) > 1 {
  926. arr := strings.Split(s, "-")
  927. personName := arr[0]
  928. mailMap[personName] = arr[1]
  929. }
  930. }
  931. for personName, mail := range mailMap {
  932. isOk := false
  933. for _, m := range *bigData {
  934. tablePersonName := gconv.String(m["name"])
  935. if tablePersonName == personName {
  936. isOk = true
  937. }
  938. }
  939. if !isOk {
  940. rData := TiDb.FindOne("dwd_f_crm_clue_big_autodraw_record", map[string]interface{}{
  941. "type": batch,
  942. }, "", "count desc")
  943. //这是一个新人
  944. TiDb.Insert("dwd_f_crm_clue_big_autodraw_record", map[string]interface{}{
  945. "name": personName,
  946. "count": common.Int64All((*rData)["count"]),
  947. "email": mail,
  948. "type": batch,
  949. })
  950. return minemail, name
  951. }
  952. }
  953. }
  954. mincount := common.IntAll((*bigData)[0]["count"])
  955. for _, vv := range *bigData {
  956. vcount := common.IntAll(vv["count"])
  957. vemail := gconv.String(vv["email"])
  958. vName := gconv.String(vv["name"])
  959. if vcount <= mincount {
  960. mincount = vcount
  961. minemail = vemail
  962. name = vName
  963. }
  964. }
  965. } else {
  966. for k, s := range strings.Split(mailStr, "|") {
  967. if len(strings.Split(s, "-")) > 1 {
  968. arr := strings.Split(s, "-")
  969. if k == 0 {
  970. minemail, name = arr[1], arr[0]
  971. }
  972. personName := arr[0]
  973. TiDb.Insert("dwd_f_crm_clue_big_autodraw_record", map[string]interface{}{
  974. "name": personName,
  975. "count": 0,
  976. "email": arr[1],
  977. "type": batch,
  978. })
  979. }
  980. }
  981. }
  982. TiDb.UpdateOrDeleteBySql(`update dwd_f_crm_clue_big_autodraw_record set count = count + 1 where name = ? and type=?`, name, batch)
  983. return minemail, name
  984. }