Website Spec
Security Recommended Updated 2026-05-29

Content Security Policy (CSP)

A CSP tells browsers which sources of script, style, image, and frame content to trust. A good policy stops most XSS and data-exfiltration attacks dead.

What it is

Content Security Policy is a response header that restricts which resources a page may load and execute. Level 3 is the current specification. The browser enforces the policy; any script, style, image, frame, or fetch from a source the policy does not allow is blocked, and a report is sent if a reporting endpoint is configured.

Content-Security-Policy: default-src 'self'; script-src 'self' 'nonce-r4nd0m' 'strict-dynamic'; object-src 'none'; base-uri 'none'; frame-ancestors 'none'; report-to csp-endpoint

Why it matters

The single biggest class of web vulnerabilities is cross-site scripting. A solid CSP turns a successful XSS injection from “attacker runs JavaScript in your origin” into “browser blocks the script and logs a violation”. CSP also limits where data can be exfiltrated to, prevents your site from being framed, and disables dangerous legacy features like inline event handlers.

How to implement

Build a strict, nonce-based policy. The recommended pattern from Google’s strict CSP guidance:

Content-Security-Policy:
  default-src 'self';
  script-src 'nonce-{random}' 'strict-dynamic' https: 'unsafe-inline';
  object-src 'none';
  base-uri 'none';
  frame-ancestors 'none';
  require-trusted-types-for 'script';
  report-to csp-endpoint

Key directives:

  • default-src 'self' — fallback for all fetch directives. Allow your own origin by default.
  • script-src — controls JavaScript. Use a per-response nonce; the strict-dynamic keyword lets a trusted script load further trusted scripts.
  • style-src — controls CSS. Same nonce model where possible.
  • img-src, font-src, connect-src, media-src — list the third parties you actually use.
  • frame-ancestors 'none' — replaces X-Frame-Options. See Clickjacking protection.
  • object-src 'none' — kills Flash and plugin embeds.
  • base-uri 'none' — blocks <base> tag injection attacks.
  • report-to — endpoint that receives violation reports as JSON.

Generate a fresh nonce per response and embed it in every inline <script> tag.

Common mistakes

  • unsafe-inline and unsafe-eval. Either one neutralises most of the protection. Use nonces or hashes instead.
  • Wildcards like script-src * or https: alone. Almost as bad as no policy at all.
  • No frame-ancestors. Leaves clickjacking open.
  • Forgetting the nonce on server-rendered inline scripts. The browser will block them and the page breaks.
  • Shipping a report-only policy forever. Use Content-Security-Policy-Report-Only to test, then switch to enforcing.

Verification

  • curl -sI https://example.com | grep -i content-security-policy should return the header.
  • Run the page through Google CSP Evaluator.
  • Open DevTools → Console. CSP violations appear as Refused to load … errors.
  • Wire up a reporting endpoint and watch for unexpected blocks before tightening.

Related topics

Sources & further reading

Search
esc close navigate open