-
Notifications
You must be signed in to change notification settings - Fork 4
/
mime.go
110 lines (93 loc) · 2.85 KB
/
mime.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// Content managed by Project Forge, see [projectforge.md] for details.
package cutil
import (
"encoding/xml"
"fmt"
"strings"
"github.com/pkg/errors"
"github.com/valyala/fasthttp"
"gopkg.in/yaml.v2"
"projectforge.dev/projectforge/app/util"
)
const (
mimeJSON = "application/json"
mimeXML = "text/xml"
mimeYAML = "application/x-yaml"
)
var AllowedHeaders = "*"
func WriteCORS(rc *fasthttp.RequestCtx) {
rc.Response.Header.Set("Access-Control-Allow-Headers", AllowedHeaders)
rc.Response.Header.Set("Access-Control-Allow-Method", "GET,POST,DELETE,PUT,PATCH,OPTIONS,HEAD")
rc.Response.Header.Set("Access-Control-Allow-Origin", "*")
rc.Response.Header.Set("Access-Control-Allow-Credentials", util.BoolTrue)
}
func RespondDebug(rc *fasthttp.RequestCtx, filename string, body any) (string, error) {
return RespondJSON(rc, filename, RequestCtxToMap(rc, body))
}
func RespondJSON(rc *fasthttp.RequestCtx, filename string, body any) (string, error) {
b := util.ToJSONBytes(body, true)
return RespondMIME(filename, mimeJSON, "json", b, rc)
}
type XMLResponse struct {
Result any `xml:"result"`
}
func RespondXML(rc *fasthttp.RequestCtx, filename string, body any) (string, error) {
body = XMLResponse{Result: body}
b, err := xml.Marshal(body)
if err != nil {
return "", errors.Wrapf(err, "can't serialize response of type [%T] to XML", body)
}
return RespondMIME(filename, mimeXML, "xml", b, rc)
}
func RespondYAML(rc *fasthttp.RequestCtx, filename string, body any) (string, error) {
b, err := yaml.Marshal(body)
if err != nil {
return "", err
}
return RespondMIME(filename, mimeYAML, "yaml", b, rc)
}
func RespondMIME(filename string, mime string, ext string, ba []byte, rc *fasthttp.RequestCtx) (string, error) {
rc.Response.Header.SetContentType(mime + "; charset=UTF-8")
if len(filename) > 0 {
if !strings.HasSuffix(filename, "."+ext) {
filename = filename + "." + ext
}
rc.Response.Header.Set("Content-Disposition", fmt.Sprintf(`attachment; filename=%q`, filename))
}
WriteCORS(rc)
if len(ba) == 0 {
return "", errors.New("no bytes available to write")
}
if _, err := rc.Write(ba); err != nil {
return "", errors.Wrap(err, "cannot write to response")
}
return "", nil
}
func GetContentType(rc *fasthttp.RequestCtx) string {
ret := string(rc.Request.Header.ContentType())
if idx := strings.Index(ret, ";"); idx > -1 {
ret = ret[0:idx]
}
t := string(rc.URI().QueryArgs().Peek("t"))
switch t {
case "debug":
return t
case "json":
return mimeJSON
case "xml":
return mimeXML
case "yaml", "yml":
return mimeYAML
default:
return strings.TrimSpace(ret)
}
}
func IsContentTypeJSON(c string) bool {
return c == "application/json" || c == "text/json"
}
func IsContentTypeXML(c string) bool {
return c == "application/xml" || c == mimeXML
}
func IsContentTypeYAML(c string) bool {
return c == "application/x-yaml" || c == "text/yaml"
}