gin接收图片文件,websocet持续返回响应,解决多任务排队问题

背景

有一个需求是这样的,前端需要通过http请求的form-data上传图片文件,后端接收图片后调用AI接口执行命令,由于命令执行时间较长,需要持续返回当前任务在全局任务列表中的位置,以便前端即时更新排队信息。

思考

如果直接在gin的请求处理函数中开goroutine,并在goroutine中通过类似c.JSON的方法来返回响应,前端会无法收到响应。原因是当处理函数返回时,Gin 会自动关闭 HTTP 请求的响应,这意味着在处理函数返回后,后台的协程无法再向响应中写入数据,从而导致客户端收不到信息。(不用goroutine也不行,因为持续返回响应涉及到使用for循环,会阻塞通道,阻塞主协程,仍然无法返回响应),使用gin配合websocket解决。

于是解决方案就是,使用gin的处理函数接收图片后,把当前任务的id作为响应返回给前端。然后前端再用当前任务的id建立websocket请求,在websocket请求中,通常不会自动关闭,除非显式地由客户端或服务器关闭,所以就可以在ws的处理函数中开goroutine来持续返回响应。

func WsHandleConnect(s *melody.Session) {
	TaskId, _ := s.Get("task_id")
	// 将 TaskId 转换为字符串并解析为整数
	taskIdStr, _ := TaskId.(string)
	taskId, err := strconv.Atoi(taskIdStr)
	if err != nil {
		sendJSONResponse(s, Res{
			Code: 4000,
			Msg:  "Invalid Task ID",
			Data: nil,
		})
		s.Close()
		return
	}
	var task *model.Task
	// 遍历 TaskQueue 查找对应的任务
	for _, t := range global.TaskQueue {
		if t.ID == taskId {
			task = t
			break
		}
	}
	if task == nil {
		sendJSONResponse(s, Res{
			Code: 4000,
			Msg:  "Task ID not provided",
			Data: nil,
		})
		s.Close()
		return
	}
	logger.Log.Error("task:", task.ID)
	go func() {
		for {
			select {
			case result := <-task.Response:

				sendJSONResponse(s, Res{
					Code: global.SuccessCode,
					Msg:  result,
					Data: nil,
				})
				s.Close()
				return
			default:
				time.Sleep(3 * time.Second)
				if len(global.TaskQueue) == 0 {
					sendJSONResponse(s, Res{
						Code: global.SuccessCode,
						Msg:  "Task failed",
						Data: nil,
					})
					s.Close()
					return
				}
				position := service.FindTaskPosition(task.ID)
				sendJSONResponse(s, Res{
					Code: global.SuccessCode,
					Data: position,
				})

			}
		}
	}()
}

  • 9
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

idMiFeng

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值