response.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package rest
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "errors"
  6. "io"
  7. "log"
  8. "mime/multipart"
  9. "net/http"
  10. "net/textproto"
  11. mjson "git.ali33.ru/fcg-xvii/go-tools/json"
  12. )
  13. func NewResponse() *Response {
  14. return &Response{
  15. data: make(mjson.Map),
  16. files: make(map[string]io.ReadCloser),
  17. }
  18. }
  19. type Response struct {
  20. data mjson.Map
  21. files map[string]io.ReadCloser
  22. err error
  23. errArgs mjson.Map
  24. code int
  25. }
  26. func (s *Response) KeySet(key string, val any) {
  27. s.data[key] = val
  28. }
  29. func (s *Response) FileSet(name string, file io.ReadCloser) {
  30. s.files[name] = file
  31. }
  32. func (s *Response) SetError(code int, err error) {
  33. s.code = code
  34. s.err = err
  35. }
  36. func (s *Response) Close() {
  37. for _, file := range s.files {
  38. file.Close()
  39. }
  40. }
  41. func (s *Response) Send(w http.ResponseWriter) error {
  42. log.Println("SEND...")
  43. defer s.Close()
  44. if s.err != nil {
  45. w.Header().Set("Content-Type", "application/json")
  46. w.WriteHeader(s.code)
  47. body := mjson.Map{
  48. "error": s.err.Error(),
  49. "args": s.errArgs,
  50. }
  51. w.Write(body.JSON())
  52. return nil
  53. }
  54. // Если есть файлы, то используем multipart
  55. if len(s.files) > 0 {
  56. var b bytes.Buffer
  57. writer := multipart.NewWriter(&b)
  58. // Добавляем JSON-данные как часть
  59. partHeader := make(textproto.MIMEHeader)
  60. partHeader.Set("Content-Type", "application/json")
  61. partHeader.Set("Content-Disposition", `form-data; name="data"`)
  62. dataPart, err := writer.CreatePart(partHeader)
  63. if err != nil {
  64. return err
  65. }
  66. if err := json.NewEncoder(dataPart).Encode(s.data); err != nil {
  67. return err
  68. }
  69. // Добавляем файлы
  70. for filename, file := range s.files {
  71. part, err := writer.CreateFormFile("file", filename)
  72. if err != nil {
  73. return err
  74. }
  75. if _, err := io.Copy(part, file); err != nil {
  76. return err
  77. }
  78. }
  79. // Закрываем multipart writer
  80. if err := writer.Close(); err != nil {
  81. return err
  82. }
  83. // Отправляем multipart response
  84. w.Header().Set("Content-Type", writer.FormDataContentType())
  85. w.Write(b.Bytes())
  86. } else {
  87. // Если нет файлов, просто отправляем JSON
  88. w.Header().Set("Content-Type", "application/json")
  89. return json.NewEncoder(w).Encode(s.data)
  90. }
  91. return nil
  92. }
  93. ///////////////////////////////////////////
  94. func ResponseError(err string, args mjson.Map, code int) *Response {
  95. return &Response{
  96. code: code,
  97. err: errors.New(err),
  98. errArgs: args,
  99. }
  100. }
  101. func ResponseErrorMessage(err, message string, code int) *Response {
  102. return &Response{
  103. code: code,
  104. err: errors.New(err),
  105. errArgs: mjson.Map{
  106. "message": message,
  107. },
  108. }
  109. }
  110. ///////////////////////////////////////////
  111. func ResponseErrKeyNotExists(key string) *Response {
  112. return ResponseError(
  113. ErrKeyNotExists,
  114. mjson.Map{
  115. "key": key,
  116. },
  117. 500,
  118. )
  119. }
  120. func ResponseErrKeyInvalidType(key, kType string) *Response {
  121. return ResponseError(
  122. ErrKeyInvalidType,
  123. mjson.Map{
  124. "key": key,
  125. "type": kType,
  126. },
  127. 500,
  128. )
  129. }
  130. func ResponseNotFound(data mjson.Map) *Response {
  131. return &Response{
  132. code: 404,
  133. err: errors.New("ErrNotFound"),
  134. errArgs: data,
  135. }
  136. }