product.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. package controllers
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_mini_crm_ht/models"
  6. "eta/eta_mini_crm_ht/models/request"
  7. "eta/eta_mini_crm_ht/models/response"
  8. "eta/eta_mini_crm_ht/services"
  9. "eta/eta_mini_crm_ht/utils"
  10. "github.com/go-sql-driver/mysql"
  11. "github.com/rdlucklib/rdluck_tools/paging"
  12. "github.com/shopspring/decimal"
  13. "strconv"
  14. "strings"
  15. "sync"
  16. "time"
  17. )
  18. type ProductController struct {
  19. BaseAuthController
  20. }
  21. // UnSetProductList
  22. // @Title 未设置的产品列表
  23. // @Description 未设置的产品列表
  24. // @Param PageSize query int true "每页数据条数"
  25. // @Param CurrentIndex query int true "当前页页码,从1开始"
  26. // @Param ClassifyIds query string true "二级分类id,可多选用英文,隔开"
  27. // @Param KeyWord query string true "报告标题/创建人"
  28. // @Param SortType query string true "排序方式"
  29. // @Success 200 {object} models.ReportAuthorResp
  30. // @router /unSetProductList [get]
  31. func (this *ProductController) UnSetProductList() {
  32. br := new(models.BaseResponse).Init()
  33. defer func() {
  34. this.Data["json"] = br
  35. this.ServeJSON()
  36. }()
  37. pageSize, _ := this.GetInt("PageSize")
  38. currentIndex, _ := this.GetInt("CurrentIndex")
  39. sortType := this.GetString("SortType")
  40. ProductType := this.GetString("ProductType")
  41. permissionIds := this.GetString("PermissionIds")
  42. KeyWord := this.GetString("KeyWord")
  43. var condition string
  44. var pars []interface{}
  45. if pageSize <= 0 {
  46. pageSize = utils.PageSize20
  47. }
  48. if currentIndex <= 0 {
  49. currentIndex = 1
  50. }
  51. if ProductType == "" {
  52. br.Msg = "产品类型不能为空"
  53. br.ErrMsg = "获取未设置产品列表失败,Err:产品类型为空"
  54. return
  55. }
  56. if KeyWord != "" {
  57. switch ProductType {
  58. case "report":
  59. condition += " AND title like '%?%'"
  60. case "media":
  61. condition += " AND media_name like '%?%'"
  62. }
  63. pars = append(pars, KeyWord)
  64. }
  65. sortCondition := " ORDER BY published_time "
  66. if sortType == "" {
  67. sortType = "DESC"
  68. }
  69. sortCondition = sortCondition + sortType
  70. var permissionIdsList []int
  71. if permissionIds != "" {
  72. permissionStr := strings.Split(permissionIds, ",")
  73. for _, permissionItem := range permissionStr {
  74. permissionId, _ := strconv.Atoi(permissionItem)
  75. permissionIdsList = append(permissionIdsList, permissionId)
  76. }
  77. }
  78. total, ids, err := services.GetUnsetProductCountByCondition(ProductType, permissionIdsList, condition, pars)
  79. if err != nil {
  80. br.Msg = "获取未设置产品列表失败"
  81. br.ErrMsg = "获取未设置产品列表失败,Err:" + err.Error()
  82. return
  83. }
  84. var list []*services.ProductView
  85. if len(ids) > 0 {
  86. startSize := utils.StartIndex(currentIndex, pageSize)
  87. list, err = services.GetUnsetProductByCondition(ProductType, ids, sortCondition, startSize, pageSize)
  88. if err != nil {
  89. br.Msg = "获取未设置产品列表失败"
  90. br.ErrMsg = "获取未设置产品列表失败,Err:" + err.Error()
  91. return
  92. }
  93. }
  94. var wg sync.WaitGroup
  95. wg.Add(len(list))
  96. for _, product := range list {
  97. go func(product *services.ProductView) {
  98. defer wg.Done()
  99. switch product.ProductType {
  100. case "report":
  101. product.RiskLevel, _, _ = services.GetRiskLevel("report", product.SourceId)
  102. product.ProductType = "报告"
  103. case "video":
  104. product.RiskLevel, _, _ = services.GetRiskLevel("media", product.SourceId)
  105. product.ProductType = "视频"
  106. case "audio":
  107. product.RiskLevel, _, _ = services.GetRiskLevel("media", product.SourceId)
  108. product.ProductType = "音频"
  109. }
  110. }(product)
  111. }
  112. wg.Wait()
  113. page := paging.GetPaging(currentIndex, pageSize, total)
  114. resp := new(response.ProductListResp)
  115. resp.List = list
  116. resp.Paging = page
  117. br.Ret = 200
  118. br.Success = true
  119. br.Data = resp
  120. br.Msg = "获取成功"
  121. }
  122. // AddProduct @Title 新增单品
  123. // @Description 新增单品
  124. // @Param File query file true "文件"
  125. // @Success 200 {object} models.ReportAuthorResp
  126. // @router /addProduct [post]
  127. func (this *ProductController) AddProduct() {
  128. br := new(models.BaseResponse).Init()
  129. defer func() {
  130. this.Data["json"] = br
  131. this.ServeJSON()
  132. }()
  133. var req request.ProductReq
  134. err := json.Unmarshal(this.Ctx.Input.RequestBody, &req)
  135. if err != nil {
  136. br.Msg = "参数解析异常!"
  137. br.ErrMsg = "参数解析失败,Err:" + err.Error()
  138. return
  139. }
  140. var product models.MerchantProduct
  141. if req.Type == "package" {
  142. if req.ValidDays <= 0 {
  143. br.Msg = "套餐有效期非法"
  144. br.ErrMsg = "套餐有效期非法,天数不能是负数"
  145. return
  146. }
  147. if req.ProductName == "" {
  148. br.Msg = "套餐名称不能为空"
  149. br.ErrMsg = "套餐名称不能为空"
  150. return
  151. }
  152. product.Title = req.ProductName
  153. product.ValidDays = req.ValidDays
  154. product.Description = req.Description
  155. product.CoverSrc = req.CoverSrc
  156. }
  157. switch req.Type {
  158. case "report":
  159. product.RiskLevel, product.Title, err = services.GetRiskLevel("report", req.SourceId)
  160. case "audio":
  161. product.RiskLevel, product.Title, err = services.GetRiskLevel("audio", req.SourceId)
  162. case "video":
  163. product.RiskLevel, product.Title, err = services.GetRiskLevel("video", req.SourceId)
  164. case "package":
  165. product.RiskLevel, _, err = services.GetRiskLevel("package", req.SourceId)
  166. default:
  167. br.Msg = "产品类型错误"
  168. br.ErrMsg = "获取产品列表失败,Err:产品类型错误"
  169. return
  170. }
  171. if err != nil {
  172. utils.FileLog.Error("新增单品失败", err.Error())
  173. br.Msg = "新增产品错误"
  174. if strings.Contains(err.Error(), "<QuerySeter> no row found") {
  175. br.Msg = "新增产品错误,产品信息不存在"
  176. } else {
  177. br.Msg = "新增产品错误" + err.Error()
  178. }
  179. return
  180. }
  181. if product.RiskLevel == "" {
  182. br.Msg = "新增产品错误"
  183. br.ErrMsg = "未获取到风险等级"
  184. return
  185. }
  186. if !checkProductRiskLevel(product.RiskLevel) {
  187. br.Msg = "产品风险等级不合法"
  188. br.ErrMsg = "产品风险等级不合法:" + product.RiskLevel
  189. return
  190. }
  191. if product.Title == "" {
  192. br.Msg = "产品名称不能为空"
  193. br.ErrMsg = "产品名称不能为空"
  194. return
  195. }
  196. var price decimal.Decimal
  197. price, err = decimal.NewFromString(req.Price)
  198. if err != nil {
  199. br.Msg = "产品价格格式不正确"
  200. br.ErrMsg = "产品价格格式不正确,err:" + err.Error() + "price:" + product.Price
  201. return
  202. }
  203. if price.Cmp(decimal.New(0, 0)) <= 0 {
  204. br.Msg = "产品价格不能小于0"
  205. br.ErrMsg = "产品价格不能小于0"
  206. return
  207. }
  208. product.SaleStatus = models.OnSale
  209. product.CreatedTime = time.Now()
  210. product.Price = req.Price
  211. product.SourceId = req.SourceId
  212. product.Type = models.MerchantProductType(req.Type)
  213. if product.Type == "" {
  214. br.Msg = "新增产品错误"
  215. br.ErrMsg = "产品类型为空"
  216. return
  217. }
  218. err = product.AddProduct()
  219. if err != nil {
  220. var mysqlErr *mysql.MySQLError
  221. if errors.As(err, &mysqlErr) {
  222. if mysqlErr.Number == 1062 {
  223. br.Msg = "该产品已设置付费,请刷新后重试"
  224. br.ErrMsg = "该产品已设置付费,请刷新后重试"
  225. } else {
  226. utils.FileLog.Error("新增产品失败,err:" + err.Error())
  227. br.Msg = "新增产品失败"
  228. br.ErrMsg = "新增产品失败,err:" + err.Error()
  229. }
  230. } else {
  231. utils.FileLog.Error("新增产品失败,err:" + err.Error())
  232. br.Msg = "新增产品失败"
  233. br.ErrMsg = "新增产品失败,err:" + err.Error()
  234. }
  235. return
  236. }
  237. br.Ret = 200
  238. br.Success = true
  239. br.Msg = "新增产品成功"
  240. return
  241. }
  242. // UpdateSaleStatus @Title 上下架产品
  243. // @Description 上下架产品
  244. // @Param File query file true "文件"
  245. // @Success 200 {object} models.ReportAuthorResp
  246. // @router /updateSaleStatus [post]
  247. func (this *ProductController) UpdateSaleStatus() {
  248. br := new(models.BaseResponse).Init()
  249. defer func() {
  250. this.Data["json"] = br
  251. this.ServeJSON()
  252. }()
  253. defer func() {
  254. this.Data["json"] = br
  255. this.ServeJSON()
  256. }()
  257. var req request.ProductSaleStatusReq
  258. err := json.Unmarshal(this.Ctx.Input.RequestBody, &req)
  259. if err != nil {
  260. br.Msg = "参数解析异常!"
  261. br.ErrMsg = "参数解析失败,Err:" + err.Error()
  262. return
  263. }
  264. if req.ProductId <= 0 {
  265. br.Msg = "产品编号非法!"
  266. br.ErrMsg = "产品编号非法,不能小于0"
  267. return
  268. }
  269. if req.SaleStatus != "onSale" && req.SaleStatus != "offSale" {
  270. br.Msg = "产品销售状态非法!"
  271. br.ErrMsg = "产品销售状态非法,未知的上下架类型:" + req.SaleStatus
  272. return
  273. }
  274. var saleStatus models.SaleStatus
  275. var messageStatus string
  276. switch req.SaleStatus {
  277. case "onSale":
  278. saleStatus = models.OnSale
  279. messageStatus = "上架"
  280. case "offSale":
  281. saleStatus = models.OffSale
  282. messageStatus = "下架"
  283. }
  284. product := models.MerchantProduct{
  285. Id: req.ProductId,
  286. SaleStatus: saleStatus,
  287. UpdatedTime: time.Now(),
  288. }
  289. err = product.UpdateProductSaleStatus()
  290. if err != nil {
  291. br.Msg = messageStatus + "失败"
  292. br.ErrMsg = messageStatus + "失败,err:" + err.Error()
  293. return
  294. }
  295. br.Msg = messageStatus + "成功"
  296. br.Ret = 200
  297. br.Success = true
  298. }
  299. // DeleteProduct @Title 删除产品
  300. // @Description 删除产品
  301. // @Param File query file true "文件"
  302. // @Success 200 {object} models.ReportAuthorResp
  303. // @router /deleteProduct [post]
  304. func (this *ProductController) DeleteProduct() {
  305. br := new(models.BaseResponse).Init()
  306. defer func() {
  307. this.Data["json"] = br
  308. this.ServeJSON()
  309. }()
  310. defer func() {
  311. this.Data["json"] = br
  312. this.ServeJSON()
  313. }()
  314. var req request.ProductSaleStatusReq
  315. err := json.Unmarshal(this.Ctx.Input.RequestBody, &req)
  316. if err != nil {
  317. br.Msg = "参数解析异常!"
  318. br.ErrMsg = "参数解析失败,Err:" + err.Error()
  319. return
  320. }
  321. if req.ProductId <= 0 {
  322. br.Msg = "产品编号非法!"
  323. br.ErrMsg = "产品编号非法,不能小于0"
  324. return
  325. }
  326. product := models.MerchantProduct{
  327. Id: req.ProductId,
  328. Deleted: true,
  329. UpdatedTime: time.Now(),
  330. }
  331. err = product.Delete()
  332. if err != nil {
  333. br.Msg = "删除产品失败"
  334. br.ErrMsg = "删除产品失败,err:" + err.Error()
  335. return
  336. }
  337. br.Msg = "删除产品成功"
  338. br.Ret = 200
  339. br.Success = true
  340. }
  341. //
  342. //// OnSale @Title 上架产品
  343. //// @Description 上架产品
  344. //// @Param File query file true "文件"
  345. //// @Success 200 {object} models.ReportAuthorResp
  346. //// @router /onSale [post]
  347. //func (this *ProductController) OnSale() {
  348. // br := new(models.BaseResponse).Init()
  349. // defer func() {
  350. // this.Data["json"] = br
  351. // this.ServeJSON()
  352. // }()
  353. // f, h, err := this.GetFile("File")
  354. // if err != nil {
  355. // br.Msg = "获取资源信息失败"
  356. // br.ErrMsg = "获取资源信息失败,Err:" + err.Error()
  357. // return
  358. // }
  359. // defer f.Close()
  360. // size, err := strconv.Atoi(utils.UPLOAD_IMG_SIZE)
  361. // if err != nil {
  362. // size = 100
  363. // }
  364. // if h.Size > 1024*1024*int64(size) {
  365. // br.Msg = fmt.Sprintf("图片大小不能超过%dK", size)
  366. // br.ErrMsg = "图片上传失败,Err:" + err.Error()
  367. // return
  368. // }
  369. // ext := path.Ext(h.Filename)
  370. // //if ext != ".mp3" {
  371. // // br.Msg = "音频格式不正确"
  372. // // br.ErrMsg = "音频上传失败,Err:" + err.Error()
  373. // // return
  374. // //}
  375. // dateDir := time.Now().Format("20060102")
  376. // uploadDir := utils.STATIC_DIR + "ht/audio" + dateDir
  377. // err = os.MkdirAll(uploadDir, utils.DIR_MOD)
  378. // if err != nil {
  379. // br.Msg = "存储目录创建失败"
  380. // br.ErrMsg = "存储目录创建失败,Err:" + err.Error()
  381. // return
  382. // }
  383. // randStr := utils.GetRandStringNoSpecialChar(28)
  384. // fileName := randStr + ext
  385. // fpath := uploadDir + "/" + fileName
  386. // err = this.SaveToFile("File", fpath)
  387. // if err != nil {
  388. // br.Msg = "图片上传失败"
  389. // br.ErrMsg = "图片上传失败,Err:" + err.Error()
  390. // return
  391. // }
  392. // audioUploadDir := utils.RESOURCE_DIR + "img/"
  393. // savePdfToOssPath := audioUploadDir + time.Now().Format("200601/20060102/")
  394. // audioName := utils.GetRandStringNoSpecialChar(28)
  395. // savePdfToOssPath += audioName + ext
  396. //
  397. // defer func() {
  398. // err = os.Remove(fpath)
  399. // fmt.Sprintf("删除文件失败:%v", err)
  400. // }()
  401. // ossClient := services.NewOssClient()
  402. // if ossClient == nil {
  403. // br.Msg = "图片上传失败"
  404. // br.ErrMsg = "初始化OSS服务失败"
  405. // return
  406. // }
  407. // mp3Url, err := ossClient.UploadFile("", fpath, savePdfToOssPath)
  408. // if err != nil {
  409. // br.Msg = "图片上传失败"
  410. // br.ErrMsg = "图片上传失败,Err:" + err.Error()
  411. // return
  412. // }
  413. // base := path.Base(h.Filename)
  414. // resp := new(response.MediaUploadResp)
  415. // resp.Url = mp3Url
  416. // resp.FileName = base
  417. // br.Data = resp
  418. // br.Msg = "上架成功"
  419. // br.Ret = 200
  420. // br.Success = true
  421. //}