response.go 2.9 KB

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