request_out.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. package rest_http
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "mime/multipart"
  8. "net/http"
  9. "net/textproto"
  10. "git.ali33.ru/fcg-xvii/rest"
  11. )
  12. func NewRequestOut(req rest.IRequest) *RequestOut {
  13. return &RequestOut{
  14. IRequest: req,
  15. }
  16. }
  17. // Response реализует объект ответа
  18. type RequestOut struct {
  19. Err rest.IErrorArgs
  20. rest.IRequest
  21. }
  22. func (s *RequestOut) RClose() {
  23. if s.IRequest != nil {
  24. s.IRequest.RClose()
  25. }
  26. }
  27. // Send отправляет запрос серверу
  28. func (s *RequestOut) Write(writer io.Writer) rest.IErrorArgs {
  29. w := writer.(http.ResponseWriter)
  30. //log.Println("SEND...")
  31. defer s.RClose()
  32. if s.Err != nil {
  33. w.Header().Set("Content-Type", "application/json")
  34. w.WriteHeader(500)
  35. w.Write([]byte(s.Err.Map().JSON()))
  36. return nil
  37. }
  38. // Если есть файлы, то используем multipart
  39. files := s.RFiles()
  40. if len(files) > 0 {
  41. var b bytes.Buffer
  42. writer := multipart.NewWriter(&b)
  43. w.Header().Set("Content-Type", writer.FormDataContentType())
  44. // Добавляем JSON-данные как часть
  45. partHeader := make(textproto.MIMEHeader)
  46. partHeader.Set("Content-Type", "application/json")
  47. partHeader.Set("Content-Disposition", `form-data; name="data"`)
  48. dataPart, err := writer.CreatePart(partHeader)
  49. if err != nil {
  50. return rest.ErrorMessage("ErrResponsePartDataCreate", err.Error())
  51. }
  52. if err := json.NewEncoder(dataPart).Encode(s.RData()); err != nil {
  53. return rest.ErrorMessage("ErrResponseDataJsonEncode", err.Error())
  54. }
  55. // Добавляем файлы
  56. for filename, file := range files {
  57. var part io.Writer
  58. if mimeFile, check := file.(*rest.MimeFile); check {
  59. h := make(textproto.MIMEHeader)
  60. h.Set("Content-Type", mimeFile.MimeType)
  61. h.Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, mimeFile.Name))
  62. if part, err = writer.CreatePart(h); err != nil {
  63. return rest.ErrorMessage("ErrResponsePartFileCreate", err.Error())
  64. }
  65. } else {
  66. if part, err = writer.CreateFormFile("file", filename); err != nil {
  67. return rest.ErrorMessage("ErrResponsePartFileCreate", err.Error())
  68. }
  69. }
  70. if _, err := io.Copy(part, file); err != nil {
  71. return rest.ErrorMessage("ErrResponsePartFileCopy", err.Error())
  72. }
  73. }
  74. // Закрываем multipart writer
  75. if err := writer.Close(); err != nil {
  76. return rest.ErrorMessage("ErrResponsePartFileClose", err.Error())
  77. }
  78. // Отправляем multipart response
  79. w.Header().Set("Content-Type", writer.FormDataContentType())
  80. w.Write(b.Bytes())
  81. } else {
  82. // Если нет файлов, просто отправляем JSON
  83. w.Header().Set("Content-Type", "application/json")
  84. if err := json.NewEncoder(w).Encode(s.RData()); err != nil {
  85. return rest.ErrorMessage("ErrResponseDataJsonEncode", err.Error())
  86. }
  87. }
  88. return nil
  89. }
  90. // Успрешный ответ
  91. func ResponseSuccess(req rest.IRequest) rest.IRequestOut {
  92. return &RequestOut{
  93. IRequest: req,
  94. }
  95. }
  96. func ResponseError(err rest.IErrorArgs) rest.IRequestOut {
  97. return &RequestOut{
  98. Err: err,
  99. }
  100. }