server_stream_handler.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. package handler
  2. import (
  3. "context"
  4. "errors"
  5. "strings"
  6. context2 "git.ikuban.com/server/kratos-utils/v2/transport/http/context"
  7. "git.ikuban.com/server/kratos-utils/v2/transport/middleware"
  8. "github.com/go-kratos/kratos/v2/transport/http"
  9. "google.golang.org/grpc"
  10. "google.golang.org/grpc/metadata"
  11. )
  12. func serverStreamHandler(srv any, stream grpc.StreamDesc, option *middleware.Option) http.HandlerFunc {
  13. return func(ctx http.Context) error {
  14. http.SetOperation(ctx, option.Path)
  15. dec := func(in any) error {
  16. if err := ctx.Bind(&in); err != nil {
  17. return err
  18. }
  19. if err := ctx.BindQuery(&in); err != nil {
  20. return err
  21. }
  22. return nil
  23. }
  24. httpCtx := ctx
  25. h := ctx.Middleware(func(ctx context.Context, _ interface{}) (interface{}, error) {
  26. streamCtx := context2.NewStreamContext(ctx)
  27. err := stream.Handler(srv, newServerStream(streamCtx, dec, httpCtx.Response()))
  28. return nil, err
  29. })
  30. newCtx := middleware.NewOptionContext(ctx, option)
  31. _, err := h(newCtx, nil)
  32. if err != nil {
  33. return err
  34. }
  35. return nil
  36. }
  37. }
  38. var _ grpc.ServerStream = (*serverStream)(nil)
  39. type serverStream struct {
  40. ctx context2.StreamContext
  41. dec func(interface{}) error
  42. w http.ResponseWriter
  43. metadata metadata.MD
  44. isSendHeader bool
  45. }
  46. func newServerStream(ctx context2.StreamContext, dec func(interface{}) error, w http.ResponseWriter) grpc.ServerStream {
  47. s := &serverStream{
  48. ctx: ctx,
  49. dec: dec,
  50. w: w,
  51. }
  52. return s
  53. }
  54. // 设置header,SendHeader主动发送或者第一次SendMsg时发送
  55. func (s *serverStream) SetHeader(md metadata.MD) error {
  56. if md == nil {
  57. return nil
  58. }
  59. s.metadata = metadata.Join(s.metadata, md)
  60. return nil
  61. }
  62. // 主动发送header (只调一次)
  63. func (s *serverStream) SendHeader(md metadata.MD) error {
  64. if s.isSendHeader {
  65. return errors.New("header has been sent")
  66. }
  67. s.isSendHeader = true
  68. if err := s.SetHeader(md); err != nil {
  69. return err
  70. }
  71. // 没有指定响应头则设置默认响应头
  72. if s.metadata.Len() == 0 {
  73. err := s.SetHeader(metadata.MD{
  74. "Content-Type": []string{"text/event-stream"},
  75. "Cache-Control": []string{"no-cache"},
  76. "Connection": []string{"keep-alive"},
  77. "X-Accel-Buffering": []string{"no"},
  78. })
  79. if err != nil {
  80. return err
  81. }
  82. }
  83. for k, v := range s.metadata {
  84. if len(v) == 0 {
  85. continue
  86. }
  87. s.w.Header().Set(k, strings.Join(v, "; "))
  88. }
  89. return nil
  90. }
  91. func (s *serverStream) SetTrailer(md metadata.MD) {
  92. return
  93. }
  94. func (s *serverStream) Context() context.Context {
  95. return s.ctx
  96. }
  97. func (s *serverStream) SendMsg(m interface{}) error {
  98. if !s.isSendHeader {
  99. // 设置流式响应 headers
  100. err := s.SendHeader(nil)
  101. if err != nil {
  102. return err
  103. }
  104. }
  105. switch data := m.(type) {
  106. case []byte:
  107. _, err := s.w.Write(data)
  108. if err != nil {
  109. return err
  110. }
  111. case string:
  112. _, err := s.w.Write([]byte(data))
  113. if err != nil {
  114. return err
  115. }
  116. default:
  117. // 使用自定义序列化逻辑
  118. serialized, err := s.ctx.Serialize(data)
  119. if err != nil {
  120. return err
  121. }
  122. _, err = s.w.Write(append(serialized, byte('\n')))
  123. if err != nil {
  124. return err
  125. }
  126. }
  127. s.w.(http.Flusher).Flush()
  128. return nil
  129. }
  130. func (s *serverStream) RecvMsg(m any) error {
  131. return s.dec(m)
  132. }