-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcontent-security-policy.go
224 lines (193 loc) · 7.74 KB
/
content-security-policy.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
package helmet
import (
"fmt"
"net/http"
"strings"
)
// HeaderContentSecurityPolicy is the Content-Security-Policy HTTP security header.
const HeaderContentSecurityPolicy = "Content-Security-Policy"
// List of all Content-Security-Policy Fetch directives.
const (
DirectiveChildSrc CSPDirective = "child-src"
DirectiveConnectSrc CSPDirective = "connect-src"
DirectiveDefaultSrc CSPDirective = "default-src"
DirectiveFontSrc CSPDirective = "font-src"
DirectiveFrameSrc CSPDirective = "frame-src"
DirectiveImgSrc CSPDirective = "img-src"
DirectiveManifestSrc CSPDirective = "manifest-src"
DirectiveMediaSrc CSPDirective = "media-src"
DirectiveObjectSrc CSPDirective = "object-src"
DirectivePrefetchSrc CSPDirective = "prefetch-src"
DirectiveScriptSrc CSPDirective = "script-src"
DirectiveScriptSrcAttr CSPDirective = "script-src-attr"
DirectiveScriptSrcElem CSPDirective = "script-src-elem"
DirectiveStyleSrc CSPDirective = "style-src"
DirectiveStyleSrcAttr CSPDirective = "style-src-attr"
DirectiveStyleSrcElem CSPDirective = "style-src-elem"
DirectiveWorkerSrc CSPDirective = "worker-src"
)
// List of all Content-Security-Policy Document directives.
const (
DirectiveBaseURI CSPDirective = "base-uri"
DirectiveSandbox CSPDirective = "sandbox"
)
// List of all Content-Security-Policy Navigation directives.
const (
DirectiveFormAction CSPDirective = "form-action"
DirectiveFrameAncestors CSPDirective = "frame-ancestors"
DirectiveNavigateTo CSPDirective = "navigate-to"
)
// List of all Content-Security-Policy Reporting directives.
const (
DirectiveReportTo CSPDirective = "report-to"
DeprecatedDirectiveReportURI CSPDirective = "report-uri" // use 'report-to' Content-Security-Policy directive instead
)
// List of all Content-Security-Policy Other directives.
const (
DirectiveRequireSriFor CSPDirective = "require-sri-for"
DirectiveRequireTrustedTypesFor CSPDirective = "require-trusted-types-for"
DirectiveTrustedTypes CSPDirective = "trusted-types"
DirectiveUpgradeInsecureRequests CSPDirective = "upgrade-insecure-requests"
)
// List of all Content-Security-Policy Deprecated directives.
const (
DeprecatedDirectiveBlockAllMixedContent CSPDirective = "block-all-mixed-content"
DeprecatedDirectivePluginTypes CSPDirective = "plugin-types"
DeprecatedDirectiveReferrer CSPDirective = "referrer" // use 'Referrer-Policy' HTTP header instead
)
// List of all Content-Security-Policy sources.
const (
SourceWildcard CSPSource = "*"
SourceNone CSPSource = "'none'"
SourceSelf CSPSource = "'self'"
SourceHTTP CSPSource = "http:"
SourceHTTPS CSPSource = "https:"
SourceData CSPSource = "data:"
SourceMediastream CSPSource = "mediastream:"
SourceBlob CSPSource = "blob:"
SourceFilesystem CSPSource = "filesystem:"
SourceUnsafeAllowRedirects CSPSource = "'unsafe-allow-redirects'"
SourceUnsafeEval CSPSource = "'unsafe-eval'"
SourceUnsafeHashes CSPSource = "'unsafe-hashes'"
SourceUnsafeInline CSPSource = "'unsafe-inline'"
SourceStrictDynamic CSPSource = "'strict-dynamic'"
SourceReportSample CSPSource = "'report-sample'"
)
// List of all DeprecatedDirectiveReferrer values.
const (
DeprecatedReferrerNone CSPSource = "\"none\""
DeprecatedReferrerNoReferrer CSPSource = "\"no-referrer\""
DeprecatedReferrerNoneWhenDowngrade CSPSource = "\"none-when-downgrade\""
DeprecatedReferrerOrigin CSPSource = "\"origin\""
DeprecatedReferrerOriginWhenCrossOrigin CSPSource = "\"origin-when-cross-origin\""
DeprecatedReferrerUnsafeURL CSPSource = "\"unsafe-url\""
)
// List of all DirectiveSandbox values.
const (
SandboxAllowDownloads CSPSource = "allow-downloads"
SandboxAllowDownloadsWithoutUserActivation CSPSource = "allow-downloads-without-user-activation"
SandboxAllowForms CSPSource = "allow-forms"
SandboxAllowModals CSPSource = "allow-modals"
SandboxAllowOrientationLock CSPSource = "allow-orientation-lock"
SandboxAllowPointerLock CSPSource = "allow-pointer-lock"
SandboxAllowPopups CSPSource = "allow-popups"
SandboxAllowPopupsToEscapeSandbox CSPSource = "allow-popups-to-escape-sandbox"
SandboxAllowPresentation CSPSource = "allow-presentation"
SandboxAllowSameOrigin CSPSource = "allow-same-origin"
SandboxAllowScripts CSPSource = "allow-scripts"
SandboxAllowStorageAccessByUserActivatation CSPSource = "allow-storage-access-by-user-activation"
SandboxAllowTopNavigation CSPSource = "allow-top-navigation"
SandboxAllowTopNavigationByUserActivation CSPSource = "allow-top-navigation-by-user-activation"
)
// List of all DirectiveTrustedTypes values.
const (
TrustedTypesAllowDuplicates CSPSource = "allow-duplicates"
)
type (
// CSPDirective represents a Content-Security-Policy directive.
CSPDirective string
// CSPSource represents a Content-Security-Policy source.
CSPSource string
// ContentSecurityPolicy represents the Content-Security-Policy HTTP security header.
ContentSecurityPolicy struct {
policies map[CSPDirective][]CSPSource
cache string
}
)
// NewContentSecurityPolicy creates a new Content-Security-Policy.
func NewContentSecurityPolicy(policies map[CSPDirective][]CSPSource) *ContentSecurityPolicy {
if policies == nil {
return EmptyContentSecurityPolicy()
}
return &ContentSecurityPolicy{policies, ""}
}
// EmptyContentSecurityPolicy creates a blank slate Content-Security-Policy.
func EmptyContentSecurityPolicy() *ContentSecurityPolicy {
return NewContentSecurityPolicy(make(map[CSPDirective][]CSPSource))
}
// Add adds a directive and its sources.
func (csp *ContentSecurityPolicy) Add(directive CSPDirective, sources ...CSPSource) {
if len(directive) == 0 {
return
}
csp.cache = ""
csp.create(directive)
for _, source := range sources {
csp.policies[directive] = append(csp.policies[directive], source)
}
}
func (csp *ContentSecurityPolicy) create(directive CSPDirective) {
if len(directive) == 0 {
return
}
csp.cache = ""
if _, ok := csp.policies[directive]; !ok {
csp.policies[directive] = []CSPSource{}
}
}
// Remove removes a directive and its sources.
func (csp *ContentSecurityPolicy) Remove(directives ...CSPDirective) {
if len(directives) == 0 {
return
}
didRemove := false
for _, directive := range directives {
if _, ok := csp.policies[directive]; ok {
didRemove = true
delete(csp.policies, directive)
}
}
if didRemove {
csp.cache = ""
}
}
// String generates the Content-Security-Policy.
func (csp *ContentSecurityPolicy) String() string {
if csp.cache != "" {
return csp.cache
}
var policies = []string{}
for directive, sources := range csp.policies {
if len(sources) == 0 {
policies = append(policies, fmt.Sprintf("%s", directive))
} else {
sourcesAsStrings := []string{}
for _, source := range sources {
sourcesAsStrings = append(sourcesAsStrings, string(source))
}
policies = append(policies, fmt.Sprintf("%s %s", directive, strings.Join(sourcesAsStrings, " ")))
}
}
csp.cache = strings.Join(policies, "; ")
return csp.cache
}
// Empty returns whether the Content-Security-Policy is empty.
func (csp *ContentSecurityPolicy) Empty() bool {
return len(csp.policies) == 0
}
// Header adds the Content-Security-Policy HTTP security header to the given http.ResponseWriter.
func (csp *ContentSecurityPolicy) Header(w http.ResponseWriter) {
if !csp.Empty() {
w.Header().Set(HeaderContentSecurityPolicy, csp.String())
}
}