error_handler.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. package s3err
  2. import (
  3. "bytes"
  4. "encoding/xml"
  5. "fmt"
  6. "net/http"
  7. "strconv"
  8. "strings"
  9. "time"
  10. "github.com/aws/aws-sdk-go/private/protocol/xml/xmlutil"
  11. "github.com/gorilla/mux"
  12. "github.com/seaweedfs/seaweedfs/weed/glog"
  13. )
  14. type mimeType string
  15. const (
  16. mimeNone mimeType = ""
  17. MimeXML mimeType = "application/xml"
  18. )
  19. func WriteAwsXMLResponse(w http.ResponseWriter, r *http.Request, statusCode int, result interface{}) {
  20. var bytesBuffer bytes.Buffer
  21. err := xmlutil.BuildXML(result, xml.NewEncoder(&bytesBuffer))
  22. if err != nil {
  23. WriteErrorResponse(w, r, ErrInternalError)
  24. return
  25. }
  26. WriteResponse(w, r, statusCode, bytesBuffer.Bytes(), MimeXML)
  27. }
  28. func WriteXMLResponse(w http.ResponseWriter, r *http.Request, statusCode int, response interface{}) {
  29. WriteResponse(w, r, statusCode, EncodeXMLResponse(response), MimeXML)
  30. }
  31. func WriteEmptyResponse(w http.ResponseWriter, r *http.Request, statusCode int) {
  32. WriteResponse(w, r, statusCode, []byte{}, mimeNone)
  33. PostLog(r, statusCode, ErrNone)
  34. }
  35. func WriteErrorResponse(w http.ResponseWriter, r *http.Request, errorCode ErrorCode) {
  36. vars := mux.Vars(r)
  37. bucket := vars["bucket"]
  38. object := vars["object"]
  39. if strings.HasPrefix(object, "/") {
  40. object = object[1:]
  41. }
  42. apiError := GetAPIError(errorCode)
  43. errorResponse := getRESTErrorResponse(apiError, r.URL.Path, bucket, object)
  44. WriteXMLResponse(w, r, apiError.HTTPStatusCode, errorResponse)
  45. PostLog(r, apiError.HTTPStatusCode, errorCode)
  46. }
  47. func getRESTErrorResponse(err APIError, resource string, bucket, object string) RESTErrorResponse {
  48. return RESTErrorResponse{
  49. Code: err.Code,
  50. BucketName: bucket,
  51. Key: object,
  52. Message: err.Description,
  53. Resource: resource,
  54. RequestID: fmt.Sprintf("%d", time.Now().UnixNano()),
  55. }
  56. }
  57. // Encodes the response headers into XML format.
  58. func EncodeXMLResponse(response interface{}) []byte {
  59. var bytesBuffer bytes.Buffer
  60. bytesBuffer.WriteString(xml.Header)
  61. e := xml.NewEncoder(&bytesBuffer)
  62. e.Encode(response)
  63. return bytesBuffer.Bytes()
  64. }
  65. func setCommonHeaders(w http.ResponseWriter, r *http.Request) {
  66. w.Header().Set("x-amz-request-id", fmt.Sprintf("%d", time.Now().UnixNano()))
  67. w.Header().Set("Accept-Ranges", "bytes")
  68. // Handle CORS headers for requests with Origin header
  69. if r.Header.Get("Origin") != "" {
  70. // Use mux.Vars to detect bucket-specific requests more reliably
  71. vars := mux.Vars(r)
  72. bucket := vars["bucket"]
  73. isBucketRequest := bucket != ""
  74. if !isBucketRequest {
  75. // Service-level request (like OPTIONS /) - apply static CORS if none set
  76. if w.Header().Get("Access-Control-Allow-Origin") == "" {
  77. w.Header().Set("Access-Control-Allow-Origin", "*")
  78. w.Header().Set("Access-Control-Allow-Methods", "*")
  79. w.Header().Set("Access-Control-Allow-Headers", "*")
  80. w.Header().Set("Access-Control-Expose-Headers", "*")
  81. w.Header().Set("Access-Control-Allow-Credentials", "true")
  82. }
  83. } else {
  84. // Bucket-specific request - preserve existing CORS headers or set default
  85. // This handles cases where CORS middleware set headers but auth failed
  86. if w.Header().Get("Access-Control-Allow-Origin") == "" {
  87. // No CORS headers were set by middleware, so this request doesn't match any CORS rule
  88. // According to CORS spec, we should not set CORS headers for non-matching requests
  89. // However, if the bucket has CORS config but request doesn't match,
  90. // we still should not set headers here as it would be incorrect
  91. }
  92. // If CORS headers were already set by middleware, preserve them
  93. }
  94. }
  95. }
  96. func WriteResponse(w http.ResponseWriter, r *http.Request, statusCode int, response []byte, mType mimeType) {
  97. setCommonHeaders(w, r)
  98. if response != nil {
  99. w.Header().Set("Content-Length", strconv.Itoa(len(response)))
  100. }
  101. if mType != mimeNone {
  102. w.Header().Set("Content-Type", string(mType))
  103. }
  104. w.WriteHeader(statusCode)
  105. if response != nil {
  106. glog.V(4).Infof("status %d %s: %s", statusCode, mType, string(response))
  107. _, err := w.Write(response)
  108. if err != nil {
  109. glog.V(0).Infof("write err: %v", err)
  110. }
  111. w.(http.Flusher).Flush()
  112. }
  113. }
  114. // If none of the http routes match respond with MethodNotAllowed
  115. func NotFoundHandler(w http.ResponseWriter, r *http.Request) {
  116. glog.V(0).Infof("unsupported %s %s", r.Method, r.RequestURI)
  117. WriteErrorResponse(w, r, ErrMethodNotAllowed)
  118. }