responses.go 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. package httpx
  2. import (
  3. "encoding/json"
  4. "net/http"
  5. "sync"
  6. "github.com/zeromicro/go-zero/core/logx"
  7. "github.com/zeromicro/go-zero/rest/internal/header"
  8. )
  9. var (
  10. errorHandler func(error) (int, interface{})
  11. lock sync.RWMutex
  12. )
  13. // Error writes err into w.
  14. func Error(w http.ResponseWriter, err error, fns ...func(w http.ResponseWriter, err error)) {
  15. lock.RLock()
  16. handler := errorHandler
  17. lock.RUnlock()
  18. if handler == nil {
  19. if len(fns) > 0 {
  20. fns[0](w, err)
  21. } else {
  22. http.Error(w, err.Error(), http.StatusBadRequest)
  23. }
  24. return
  25. }
  26. code, body := handler(err)
  27. if body == nil {
  28. w.WriteHeader(code)
  29. return
  30. }
  31. e, ok := body.(error)
  32. if ok {
  33. http.Error(w, e.Error(), code)
  34. } else {
  35. WriteJson(w, code, body)
  36. }
  37. }
  38. // Ok writes HTTP 200 OK into w.
  39. func Ok(w http.ResponseWriter) {
  40. w.WriteHeader(http.StatusOK)
  41. }
  42. // OkJson writes v into w with 200 OK.
  43. func OkJson(w http.ResponseWriter, v interface{}) {
  44. WriteJson(w, http.StatusOK, v)
  45. }
  46. // SetErrorHandler sets the error handler, which is called on calling Error.
  47. func SetErrorHandler(handler func(error) (int, interface{})) {
  48. lock.Lock()
  49. defer lock.Unlock()
  50. errorHandler = handler
  51. }
  52. // WriteJson writes v as json string into w with code.
  53. func WriteJson(w http.ResponseWriter, code int, v interface{}) {
  54. bs, err := json.Marshal(v)
  55. if err != nil {
  56. http.Error(w, err.Error(), http.StatusInternalServerError)
  57. return
  58. }
  59. w.Header().Set(ContentType, header.JsonContentType)
  60. w.WriteHeader(code)
  61. if n, err := w.Write(bs); err != nil {
  62. // http.ErrHandlerTimeout has been handled by http.TimeoutHandler,
  63. // so it's ignored here.
  64. if err != http.ErrHandlerTimeout {
  65. logx.Errorf("write response failed, error: %s", err)
  66. }
  67. } else if n < len(bs) {
  68. logx.Errorf("actual bytes: %d, written bytes: %d", len(bs), n)
  69. }
  70. }