123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- package services
- import (
- "bytes"
- "encoding/json"
- "eta_gn/eta_report/models"
- "eta_gn/eta_report/services/alarm_msg"
- "eta_gn/eta_report/utils"
- "fmt"
- "io/ioutil"
- "net/http"
- "sync"
- "time"
- )
- const (
- PythonReportHtml2ImgApi = "/api/report/html2img" // 图片生成接口
- PythonReportHtml2ImgClearLocalApi = "/api/report/clear_local_file" // 清除本地图片文件接口
- )
- // CreateReportImgAndPdf 报告详情生成长图及PDF
- func CreateReportImgAndPdf(req Report2ImgQueueReq) {
- // 研报以后大概率也需要, 暂只处理智能研报的
- if req.ReportType != 2 {
- return
- }
- if req.ReportCode == "" {
- return
- }
- var err error
- defer func() {
- if err != nil {
- tips := fmt.Sprintf("报告详情转长图, ErrMsg: %s", err.Error())
- fmt.Println(tips)
- utils.FileLog.Info(tips)
- go alarm_msg.SendAlarmMsg(tips, 3)
- }
- }()
- // 先取报告转长图的host配置, 如果没有再取原有的报告分享配置(兼容只配了其中一个的客户)
- conf, e := models.GetBusinessConf()
- if e != nil {
- err = fmt.Errorf("获取商家配置失败, Err: %s", e.Error())
- return
- }
- reportViewHost := conf[models.BusinessConfReport2ImgUrl]
- if reportViewHost == "" {
- reportViewHost = conf[models.BusinessConfReportViewUrl]
- }
- if reportViewHost == "" {
- err = fmt.Errorf("报告分享域名未配置")
- return
- }
- // 校验报告是否被删
- reportOb := new(models.SmartReport)
- cond := ` AND report_code = ?`
- pars := make([]interface{}, 0)
- pars = append(pars, req.ReportCode)
- item, e := reportOb.GetItemByCondition(cond, pars)
- if e != nil {
- if e.Error() == utils.ErrNoRow() {
- return
- }
- err = fmt.Errorf("获取报告失败, Err: %s", e.Error())
- return
- }
- // 报告平均生成时间基本在3min以下, 设定一个6min的超时以免后续阻塞
- apiDone := make(chan bool, 1)
- apiError := make(chan error, 1)
- fileName := utils.GetRandStringNoSpecialChar(28)
- apiResult := make([]string, 0)
- go func() {
- var apiReq ReportHtml2ImgApiReq
- apiReq.ReportUrl = fmt.Sprintf("%s/%s?code=%s", reportViewHost, "smart_report_getImg", req.ReportCode)
- //if utils.RunMode == "debug" {
- // apiReq.ReportUrl = "https://ficc.hzinsights.com/reportshare_crm_report?code=4e38d30e656da5ae9d3a425109ce9e04"
- //}
- //fmt.Println(apiReq.ReportUrl)
- apiReq.FileName = fileName
- //fmt.Println("curl start")
- res, e := CurlReportHtml2ImgApi(apiReq)
- if e != nil {
- apiError <- fmt.Errorf("CurlReportHtml2ImgApi err: %s", e.Error())
- return
- }
- //fmt.Println("curl end")
- apiResult = res
- apiDone <- true
- }()
- select {
- case <-time.After(6 * time.Minute):
- err = fmt.Errorf("报告生成长图超时")
- return
- case e = <-apiError:
- err = e
- return
- case <-apiDone:
- fmt.Println("api done")
- }
- if len(apiResult) != 2 {
- err = fmt.Errorf("文件生成有误")
- return
- }
- // 上传IMG, PDF
- var imgUrl, pdfUrl string
- var errImg, errPdf error
- //uploadDir := "static/report_images/"
- //fileDir := ""
- wg := sync.WaitGroup{}
- wg.Add(2)
- go func() {
- defer func() {
- wg.Done()
- }()
- imgFileName := fileName + ".png"
- ossClient := NewOssClient()
- if ossClient == nil {
- errImg = fmt.Errorf("初始化OSS服务失败")
- return
- }
- fmt.Println("start UploadFile")
- si, e := ossClient.UploadFile(imgFileName, apiResult[0], "")
- if e != nil {
- fmt.Println("UploadFile,Err:" + e.Error())
- errImg = fmt.Errorf("文件上传失败, Err: %s", e.Error())
- return
- }
- fmt.Println("end UploadFile:" + imgFileName)
- //if utils.ObjectStorageClient == "minio" {
- // si, e := UploadMinIoToDir(imgFileName, apiResult[0], uploadDir, fileDir)
- // if e != nil {
- // errImg = e
- // return
- // }
- // imgUrl = si
- // return
- //}
- //// 默认OSS
- //si, e := UploadAliyunToDir(imgFileName, apiResult[0], uploadDir, fileDir)
- //if e != nil {
- // errImg = e
- // return
- //}
- imgUrl = si
- var clearReq ReportHtml2ImgApiClearLocalReq
- clearReq.FileName = imgFileName
- e = CurlReportHtml2ImgApiClearLocal(clearReq)
- if e != nil {
- errImg = e
- return
- }
- }()
- go func() {
- defer func() {
- wg.Done()
- }()
- pdfFileName := fileName + ".pdf"
- ossClient := NewOssClient()
- if ossClient == nil {
- errImg = fmt.Errorf("初始化OSS服务失败")
- return
- }
- sp, e := ossClient.UploadFile(pdfFileName, apiResult[1], "")
- if e != nil {
- errImg = fmt.Errorf("文件上传失败, Err: %s", e.Error())
- return
- }
- //if utils.ObjectStorageClient == "minio" {
- // sp, e := UploadMinIoToDir(pdfFileName, apiResult[1], uploadDir, fileDir)
- // if e != nil {
- // errPdf = e
- // return
- // }
- // pdfUrl = sp
- // return
- //}
- //sp, e := UploadAliyunToDir(pdfFileName, apiResult[1], uploadDir, fileDir)
- //if e != nil {
- // errPdf = e
- // return
- //}
- pdfUrl = sp
- var clearReq ReportHtml2ImgApiClearLocalReq
- clearReq.FileName = pdfFileName
- e = CurlReportHtml2ImgApiClearLocal(clearReq)
- if e != nil {
- errImg = e
- return
- }
- }()
- wg.Wait()
- if errImg != nil {
- err = fmt.Errorf("upload img err: %s", e.Error())
- return
- }
- if errPdf != nil {
- err = fmt.Errorf("upload pdf err: %s", e.Error())
- return
- }
- fmt.Println("imgUrl:" + imgUrl + ";pdfUrl:" + pdfUrl)
- // 更新报告链接
- item.DetailImgUrl = imgUrl
- item.DetailPdfUrl = pdfUrl
- item.ModifyTime = time.Now().Local()
- updateCols := []string{"DetailImgUrl", "DetailPdfUrl", "ModifyTime"}
- if e = item.Update(updateCols); e != nil {
- err = fmt.Errorf("更新报告链接失败, Err: %s", e.Error())
- return
- }
- return
- }
- // ReportHtml2ImgApiReq 报告生成图片接口请求体
- type ReportHtml2ImgApiReq struct {
- ReportUrl string `json:"report_url" description:"报告详情分享地址"`
- FileName string `json:"file_name" description:"生成的文件名"`
- OutputType string `json:"output_type" description:"生成类型: img/pdf, 为空则两种均生成"`
- }
- // ReportHtml2ImgApiResp 报告生成图片接口响应体
- type ReportHtml2ImgApiResp struct {
- Code int `json:"code" description:"状态码"`
- Msg string `json:"error" description:"提示信息"`
- Data []string `json:"data" description:"返回数据"`
- }
- // CurlReportHtml2ImgApi 请求报告生成图片接口
- func CurlReportHtml2ImgApi(params ReportHtml2ImgApiReq) (apiResp []string, err error) {
- if utils.Report2ImgServerUrl == "" {
- err = fmt.Errorf("服务地址为空")
- return
- }
- url := fmt.Sprint(utils.Report2ImgServerUrl, PythonReportHtml2ImgApi)
- //fmt.Println("url: ", url)
- jsonData, e := json.Marshal(params)
- if e != nil {
- err = fmt.Errorf("data json marshal err: %s", e.Error())
- return
- }
- //fmt.Println(string(jsonData))
- //fmt.Println("http post start")
- resp, e := http.Post(url, "application/json", bytes.NewBuffer(jsonData))
- if e != nil {
- err = fmt.Errorf("http post err: %s", e.Error())
- return
- }
- defer resp.Body.Close()
- //fmt.Println("http post end")
- b, e := ioutil.ReadAll(resp.Body)
- if e != nil {
- err = fmt.Errorf("resp body read err: %s", e.Error())
- return
- }
- if len(b) == 0 {
- err = fmt.Errorf("resp body is empty")
- return
- }
- result := new(ReportHtml2ImgApiResp)
- if e = json.Unmarshal(b, &result); e != nil {
- err = fmt.Errorf("result unmarshal err: %s\nresult: %s", e.Error(), string(b))
- return
- }
- if result.Code != 200 {
- err = fmt.Errorf("result: %s", string(b))
- return
- }
- apiResp = result.Data
- return
- }
- // ReportHtml2ImgApiClearLocalReq 清除本地文件接口请求体
- type ReportHtml2ImgApiClearLocalReq struct {
- FileName string `json:"file_name" description:"生成的文件名"`
- }
- // ReportHtml2ImgApiClearLocalResp 清除本地文件接口响应体
- type ReportHtml2ImgApiClearLocalResp struct {
- Code int `json:"code" description:"状态码"`
- Msg string `json:"error" description:"提示信息"`
- Data string `json:"data" description:"返回数据"`
- }
- // CurlReportHtml2ImgApiClearLocal 请求清除本地文件
- func CurlReportHtml2ImgApiClearLocal(params ReportHtml2ImgApiClearLocalReq) (err error) {
- if utils.Report2ImgServerUrl == "" {
- err = fmt.Errorf("服务地址为空")
- return
- }
- url := fmt.Sprint(utils.Report2ImgServerUrl, PythonReportHtml2ImgClearLocalApi)
- jsonData, e := json.Marshal(params)
- if e != nil {
- err = fmt.Errorf("data json marshal err: %s", e.Error())
- return
- }
- //fmt.Println(string(jsonData))
- resp, e := http.Post(url, "application/json", bytes.NewBuffer(jsonData))
- if e != nil {
- err = fmt.Errorf("http post err: %s", e.Error())
- return
- }
- defer resp.Body.Close()
- b, e := ioutil.ReadAll(resp.Body)
- if e != nil {
- err = fmt.Errorf("resp body read err: %s", e.Error())
- return
- }
- if len(b) == 0 {
- err = fmt.Errorf("resp body is empty")
- return
- }
- result := new(ReportHtml2ImgApiClearLocalResp)
- if e = json.Unmarshal(b, &result); e != nil {
- err = fmt.Errorf("result unmarshal err: %s\nresult: %s", e.Error(), string(b))
- return
- }
- if result.Code != 200 {
- err = fmt.Errorf("result: %s", string(b))
- return
- }
- return
- }
|