www.invicti.com Open in urlscan Pro
2606:4700::6812:918  Public Scan

Submitted URL: https://www.netsparker.com/whitepaper-http-security-headers/#ReferrerPolicyHTTPHeader
Effective URL: https://www.invicti.com/white-papers/whitepaper-http-security-headers/
Submission: On September 13 via api from GB — Scanned from GB

Form analysis 2 forms found in the DOM

POST

<form method="POST" class="ns-form mktoForm mktoHasWidth mktoLayoutLeft" data-forminstance="form-1" data-mktoformid="1172" __bizdiag="-1387717392" __biza="W___" id="" novalidate="novalidate" data-styles-ready="true">
  <style type="text/css">
    .mktoForm .mktoButtonWrap.mktoSimple .mktoButton {
      color: #fff;
      border: 1px solid #75ae4c;
      padding: 0.4em 1em;
      font-size: 1em;
      background-color: #99c47c;
      background-image: -webkit-gradient(linear, left top, left bottom, from(#99c47c), to(#75ae4c));
      background-image: -webkit-linear-gradient(top, #99c47c, #75ae4c);
      background-image: -moz-linear-gradient(top, #99c47c, #75ae4c);
      background-image: linear-gradient(to bottom, #99c47c, #75ae4c);
    }

    .mktoForm .mktoButtonWrap.mktoSimple .mktoButton:hover {
      border: 1px solid #447f19;
    }

    .mktoForm .mktoButtonWrap.mktoSimple .mktoButton:focus {
      outline: none;
      border: 1px solid #447f19;
    }

    .mktoForm .mktoButtonWrap.mktoSimple .mktoButton:active {
      background-color: #75ae4c;
      background-image: -webkit-gradient(linear, left top, left bottom, from(#75ae4c), to(#99c47c));
      background-image: -webkit-linear-gradient(top, #75ae4c, #99c47c);
      background-image: -moz-linear-gradient(top, #75ae4c, #99c47c);
      background-image: linear-gradient(to bottom, #75ae4c, #99c47c);
    }
  </style>
  <div class="mktoFormRow">
    <div class="mktoFieldDescriptor mktoFormCol">
      <div class="mktoOffset"></div>
      <div class="mktoFieldWrap mktoRequiredField"><label for="Email" id="LblEmail" class="mktoLabel mktoHasWidth">
          <div class="mktoAsterix">*</div>Work Email
        </label>
        <div class="mktoGutter mktoHasWidth"></div><input id="Email" name="Email" placeholder="Work Email *" maxlength="255" aria-labelledby="LblEmail InstructEmail" type="email" class="mktoField mktoEmailField mktoHasWidth mktoRequired"
          aria-required="true"><span id="InstructEmail" tabindex="-1" class="mktoInstruction"></span>
        <div class="mktoClear"></div>
      </div>
      <div class="mktoClear"></div>
    </div>
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow marketo-checkbox-row">
    <div class="mktoFieldDescriptor mktoFormCol">
      <div class="mktoOffset"></div>
      <div class="mktoFieldWrap"><label for="Consent_Explicit_Opt_In__c" id="LblConsent_Explicit_Opt_In__c" class="mktoLabel mktoHasWidth">
          <div class="mktoAsterix">*</div>
        </label>
        <div class="mktoGutter mktoHasWidth"></div>
        <div class="mktoLogicalField mktoCheckboxList mktoHasWidth"><input name="Consent_Explicit_Opt_In__c" id="Consent_Explicit_Opt_In__c" type="checkbox" value="yes"
            aria-labelledby="LblConsent_Explicit_Opt_In__c InstructConsent_Explicit_Opt_In__c" class="mktoField"><label for="Consent_Explicit_Opt_In__c" id="LblConsent_Explicit_Opt_In__c"></label></div><span id="InstructConsent_Explicit_Opt_In__c"
          tabindex="-1" class="mktoInstruction"></span>
        <div class="mktoClear"></div>
      </div>
      <div class="mktoClear"></div>
    </div>
    <div class="mktoFormCol">
      <div class="mktoOffset mktoHasWidth"></div>
      <div class="mktoFieldWrap">
        <div class="mktoHtmlText mktoHasWidth">I'd like to receive product and industry updates <script type="text/javascript" src="chrome-extension://nkimffhpgcdokjcjnffnpdajfbapllpk/packages/adobe-engage/injector.js"></script>
        </div>
        <div class="mktoClear"></div>
      </div>
      <div class="mktoClear"></div>
    </div>
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow">
    <div class="mktoFormCol">
      <div class="mktoOffset mktoHasWidth"></div>
      <div class="mktoFieldWrap">
        <div class="mktoHtmlText mktoHasWidth marketo-privacy-line">Your information will be kept <a href="https://www.invicti.com/legal/privacy-policy/" target="_blank" title="Privacy Policy" id="">private</a>. <script type="text/javascript"
            src="chrome-extension://nkimffhpgcdokjcjnffnpdajfbapllpk/packages/adobe-engage/injector.js"></script>
        </div>
        <div class="mktoClear"></div>
      </div>
      <div class="mktoClear"></div>
    </div>
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="utm_term__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="utm_source__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="direct">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="utm_medium__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="direct">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="utm_campaign__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="utm_content__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="GCLID__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="GA_Client_ID__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="GA1.1.830425624.1694583774">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="vwo_variation__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="1">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="vwo_id__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="410">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="Landing_Page_URL__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="https://www.invicti.com/white-papers/whitepaper-http-security-headers/">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="Page_Prior_to_Form_URL__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="Form_Submit_URL__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="https://www.invicti.com/white-papers/whitepaper-http-security-headers/#ReferrerPolicyHTTPHeader">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="CountryCode" class="mktoField mktoFieldDescriptor mktoFormCol" value="GB">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoFormRow"><input type="hidden" name="Netsparker__c" class="mktoField mktoFieldDescriptor mktoFormCol" value="true">
    <div class="mktoClear"></div>
  </div>
  <div class="mktoButtonRow"><span class="mktoButtonWrap mktoSimple"><button type="submit" class="mktoButton btn btn--primary btn--full">Download</button></span></div><input type="hidden" name="formid" class="mktoField mktoFieldDescriptor"
    value="1172"><input type="hidden" name="munchkinId" class="mktoField mktoFieldDescriptor" value="923-HQB-396">
</form>

POST

<form method="POST" class="ns-form mktoForm mktoHasWidth mktoLayoutLeft" data-forminstance="form-1" data-mktoformid="1172" __bizdiag="-1387717392" __biza="W___" novalidate="novalidate"
  style="font-family: Helvetica, Arial, sans-serif; font-size: 13px; color: rgb(51, 51, 51); visibility: hidden; position: absolute; top: -500px; left: -1000px; width: 1600px;"></form>

Text Content

Netsparker is now Invicti
Get a demo
AppSec with Zero Noise Get a demo
Get a demo
 * Product
   * Overview
   * Features
 * Why Us?
   * Solutions
     * Industries
       * IT & Telecom
       * Government
       * Financial Services
       * Education
       * Healthcare
     * Roles
       * CTO & CISO
       * Engineering Manager
       * Security Engineer
       * DevSecOps
   * Comparison
   * Case studies
   * Customers
   * Testimonials
 * Plans
 * About Us
   * Our Story
   * In the news
   * Careers
   * Contact us
 * Resources
   * Blog
   * White Papers
   * Webinars
   * Resource Library
   * Invicti Learn
   * Partners
     * Channel
     * MSSP
   * Support

M
E
N
U
All White Papers
All White Papers


TABLE OF CONTENTS

 1.  Introduction
 2.  X-Frame-Options HTTP Header
     1. How to Prevent Clickjacking Attacks
     2. X-Frame Options Directives
     3. Important Points About the X-Frame-Options HTTP Header
     4. Which Browsers Support the X-Frame-Options HTTP Header?
 3.  X-XSS-Protection HTTP Header
     1. What Can Malicious Attackers Do When Exploiting a Reflected XSS
        Vulnerability?
     2. The Different X-XSS-Protection Header Directives
     3. Bypassing the XSS Blocking Mechanism
 4.  X-Content-Type-Options HTTP Header
     1. Text File Upload Example
 5.  X-Download-Options HTTP Header
     1. Cookie Theft Example
 6.  Content Security Policy (CSP) HTTP Header
     1. Content Security Policy Directives
     2. One Thing To Keep in Mind When Using Content Security Policy
 7.  HTTP Strict Transport Security (HSTS) HTTP Header
     1. Is There Any Way to Further Improve HTTPS Protection on Your Website?
     2. First Request and Preload
 8.  HTTP Public Key Pinning
     1. Certificate Authorities and Fraudulently Issued Certificates
     2. Setting up an HPKP Header
     3. HPKP Directives
     4. Which Browsers Support PKP Security Headers?
 9.  Expect-CT HTTP Header
     1. Certificate Transparency Logs
     2. How Do We Implement the Expect-CT Header?
 10. Referrer-Policy HTTP Header
     1. How the Referrer-Policy Header Works
     2. Referrer-Policy Directives
     3. Which Browsers Support Referrer-Policy Security Header?
 11. Proactivity is Vital!

DOWNLOAD THIS WHITE PAPER AS A PDF

*
Work Email




*




I'd like to receive product and industry updates



Your information will be kept private.

















Download

Your information will be kept private

White Paper


HTTP SECURITY HEADERS AND HOW THEY WORK


INTRODUCTION

This whitepaper explains how HTTP headers can be used in relation to web
application security. It highlights the most commonly used HTTP headers and
explains how each of them works in technical detail.

Headers are part of the HTTP specification, defining the metadata of the message
in both the HTTP request and response. While the HTTP message body is often
meant to be read by the user, metadata is processed exclusively by the web
browser and has been included in HTTP protocol since version 1.0.

In request messages, the metadata can hold the following information:

 * Language of the request
 * Cookies
 * Credentials for the website
 * Cache data

In response messages, the metadata can hold the following information:

 * Size and type of the content
 * Cache storage preferences
 * Server data
 * Time and date
 * Credentials to be set by the client

Security headers are HTTP response headers that define whether a set of security
precautions should be activated or deactivated on the web browser.


X-FRAME-OPTIONS HTTP HEADER

The X-Frame-Options Header is a security header suggested by Microsoft to avoid
the UI Redressing attacks that began with Clickjacking in 2009. It’s supported
by all major browsers.

UI Redressing attacks are based on loading web pages inside an iframe and
overlaying them with other UI elements. There are various types of UI
Redressing, such as hijacking keystrokes or extraction of content, each with its
own advantages for attackers.



The Clickjacking attack method works by loading the malicious website inside a
low opacity iframe and overlaying it with an innocuous looking button, checkbox
or link, which tricks the user into interacting with the vulnerable website
beneath. This forces the user to click the apparently safe UI element, which
triggers a set of actions on the embedded vulnerable website.



In this example, Amazon is loaded in a low opacity iframe and is therefore not
visible by the user. The user sees the Click Here button instead of
the Buy button below. When the user clicks on the Click Here button however,
only the Buy button on Amazon is actually clicked, which triggers a set of
actions on Amazon.

Since these interactions take place as if the victim was intentionally browsing
the website, the interaction triggered on Amazon will include the victim’s
credentials (such as Cookies) too. (Please note that this scenario is completely
imaginary and set in an environment where security mechanisms like
X-Frame-Options headers are unavailable.)


HOW TO PREVENT CLICKJACKING ATTACKS

Clickjacking is an attack that targets users as the weakest link in the online
security chain. Multiple methods, such as Frame Busting, have been implemented
to protect users from this attack. The most reliable method is the
X-Frame-Options header, which was added to Microsoft’s Internet Explorer
browsers in 2009.

In order to protect our users from attacks like Clickjacking, the best tactic is
to prevent malicious websites from framing our pages to render with iframes or
frames. We can use the X-Frame-Options security header to do this.


X-FRAME OPTIONS DIRECTIVES

There are three X-Frame-Options directives available:

X-Frame-Options: DENY | SAMEORIGIN | ALLOW-FROM URL

DirectiveDescriptionDENY:The page must not be embedded into another page within
an iframe or any similar HTML element.SAMEORIGIN:The website can only be
embedded in a site that’s paired in terms of scheme, hostname and port. For
example, https://www.example.com can only be loaded
through https://www.example.com, while https://www.attacker.com and
even http://example.com are not allowed to embed it.

For further information about Same-Origin Policy, see Introducing the
Same-Origin Policy Whitepaper.ALLOW-FROM URL:The website can only be framed by
the URL specified here.

There are two important points to remember with X-Frame-Options:
Chromium based browsers only partially support X-Frame-Options (the ALLOW-FROM
directive is unavailable)Using the ALLOW-FROM URL instruction, we can whitelist
only one domain and allow our website to be loaded in an iframe


IMPORTANT POINTS ABOUT THE X-FRAME-OPTIONS HTTP HEADER

 * The X-Frame-Options header must be present in the HTTP responses of all pages
 * Instead of X-Frame-Options, the Content-Security-Policy frame-ancestors
   directive can be used:

Content-Security-Policy: frame-ancestors 'none'; // No URL can load the page in an iframe.

Content-Security-Policy: frame-ancestors 'self'; // Serves the same function as the SAMEORIGIN parameter.

While you can whitelist only one URL in X-Frame-Options, with the CSP
frame-ancestors you can add multiple URLs that can embed your site in an iframe.


WHICH BROWSERS SUPPORT THE X-FRAME-OPTIONS HTTP HEADER?

Like many security features, the X-Frame-Options header was not always available
in browsers. Instead the need for such a header became apparent much later.
Therefore, not all browsers supported it in their first versions. However,
almost all modern browsers support X-Frame-Options, either fully or partially.
As mentioned above, the ALLOW-FROM directive isn’t supported by every browser.



Source: https://caniuse.com/#search=x-frame-options


X-XSS-PROTECTION HTTP HEADER

X-XSS-Protection allows developers to change the behavior of the Reflected
XSS (Cross-Site Scripting) security filters. These filters aim to detect
dangerous HTML input and either prevent the site from loading or remove
potentially malicious scripts.




WHAT CAN MALICIOUS ATTACKERS DO WHEN EXPLOITING A REFLECTED XSS VULNERABILITY?

Reflected XSS is a vulnerability that arises from the evaluation of user input
as script code in the page context.

Malicious actions – such as stealing users’ cookies, tracking keyboard strokes
or mouse moves, or issuing requests on behalf of the user – can all be carried
out with the help of XSS. This is how it works. Consider the following PHP code:

<p>Welcome <?php echo $_GET["name"];?></p>

By passing the following HTML and JavaScript code to the name parameter, the
application will embed it unfiltered on the page, which will display a
JavaScript alert window on the vulnerable website.



http://www.example.com?name=<script>alert(1);</script>


THE DIFFERENT X-XSS-PROTECTION HEADER DIRECTIVES

It is possible to change the behavior of the XSS filter in the web browser by
using various directives. In this section we explain what the different
directives are and what their purpose is.



X-XSS-Protection: 1

This is the default setting. It enables XSS filtering on the web browser and
blocks out potential XSS payloads from being executed on the page.

X-XSS-Protection: 1; mode=block;

This enables XSS filtering in the browser. It avoids potential execution of XSS
payloads by blocking the rendering of the page. When the XSS payload is
deployed, the visitor gets a blank page on the browser.

In Chromium based browsers, the XSS injection attempt can be reported to the URL
specified in the report directive.

X-XSS-Protection: 1; mode=block; report=https://domain.tld/folder/file.ext

The XSS filter is responsible for the detection of reflected script code. It is
triggered if potentially malicious HTML code is found in both the request and
response on the HTML page. While some directives will instruct the browser to
remove the malicious script in question, others prevent the rendering of the
page entirely.

For example:

Request URL:

http://www.example.com/?param=<script>alert(1);</script>

Response body:

…
<div>
<script>alert(1);</script>
&lt;/div>

However, XSS filters have been abused in the past in order to to block the
rendering of parts of an HTML page. Attackers can take advantage of the default
behavior of XSS filters that block any potentially dangerous code if it occurs
within the URL of the page.




BYPASSING THE XSS BLOCKING MECHANISM

XSS filters are also used by attackers to disable important HTML and JavaScript
code, for example, Frame Busting mechanisms. Recent years have shown that
innovative techniques have been developed that can deactivate scripts or steal
user data with the help of these filters.

Let’s look at an example of a Frame Busting mechanism. If the developer figures
out that the website was loaded within a frame, they can redirect the top window
to their website using this code:

<script>
if(top != self) {
  top.location = self.location;
}
</script>

The attacker can bypass this using the X-XSS-Protection mechanism that is active
by default:

<iframe src="http://www.victim.com/?v=<script>if">



Instead of blocking a Cross-Site Scripting attack, sometimes X-XSS-Protection:
1; (a default setting in Internet Explorer) has been used to bypass the XSS
blocking mechanisms. Those vulnerabilities were fixed, but it shows that every
feature you add to a browser can threaten the security of end users, even those
intended to enhance their security.

Researchers argue that the Content-Security-Policy headers in modern browsers
are sufficient to ensure mechanisms like X-XSS-Protection are no longer
necessary. Others, however, have suggested that the feature is not secure enough
and doesn’t add much value (see the discussion on Bugzilla).


X-CONTENT-TYPE-OPTIONS HTTP HEADER

This HTTP header is typically used to control the MIME Type Sniffing function in
web browsers. MIME Type Sniffing is a content evaluation function used by
browsers when the content type is not specified. Basically, if
the Content-Type header is blank or missing, the browser ‘sniffs’ the content
and attempts to display the source in the most appropriate way.

However, if used in conjunction with an upload functionality, this sniffing
process can pose some risks, so developers should be really careful how to use
this header. Below is an example highlighting the security risk.


TEXT FILE UPLOAD EXAMPLE

Let’s suppose that a user can upload a text file to a website. If the uploaded
file includes HTML, script tags or Javascript code, and we don’t specify
a Content-Type as we return it, this is what happens:

 * The browser will sniff the content
 * Decide that it’s a text/html type of file, and
 * Run the code inside

Even the image files that are uploaded to our websites should include
the Content-Type header when returned to the user. Otherwise, script and other
malicious code could be injected into the metadata of image files (EXIF data)
and be executed.

To prevent the browser from sniffing the page’s content and deciding on which
MIME type to use, use the X-Content-Type-Options header with
the nosniff directive:

X-Content-Type-Options: nosniff


X-DOWNLOAD-OPTIONS HTTP HEADER

The X-Download-Options header can be used download the requested data instead of
viewing it in the browser. The X-Download Options header is available in
browsers such as Internet Explorer 8 and above.

Therefore it is like an in-depth defense mechanism that is especially suited for
applications that allow users to upload content. Below is an example of an
attack that this header can help you avoid.


COOKIE THEFT EXAMPLE

Let’s imagine a user uploads a file called stealcookie.html to our website. You
can use the below header to force the file to be downloaded instead of executed:

Content-Disposition: attachment; filename=stealcookie.html

Though even if we force the download, the user will still be prompted to decide
if he or she should download the file or open the file. If the user opts to open
the file, the malicious code will be executed.



If the user clicks Open, the file would be parsed by the browser and displayed
on our website, enabling the malicious code in the file to access all other
pages on the same domain.

The only way to prevent this is to remove the option the user has to open the
file by adding the noopen directive to X-Download-Options HTTP header:

X-Download-Options: noopen

When this directive is used, the user can still save and open the file, but this
way the malicious code will be prevented from running on our website. Though it
will run on the user’s file system.




CONTENT SECURITY POLICY (CSP) HTTP HEADER

Introduced in November 2012, Content Security Policy presents an extra layer of
security against multiple vulnerabilities such as XSS, Clickjacking, Protocol
Downgrading and Frame Injection. It appears that CSP will become the most
significant tool for client side security in the near future, since it provides
a substitute for security headers, such as X-Frame-Options and X-XSS-Protection,
that aren’t enabled by default.

Here is a sample CSP header:

Content-Security-Policy: <policy-directive>; <policy-directive>

In CSP, we use a whitelist to define rules. With this approach, we can filter
out any resources that do not fit with our rules. All we have to do is to state
the resources within the Content-Security-Policy response header:

Content-Security-Policy: script-src 'self' https://apis.google.com

This CSP directive allows script loading only over our domain
and https://apis.google.com. Any other inline scripts will not run.


CONTENT SECURITY POLICY DIRECTIVES

Along with the CSP header, we can use the following directives to further limit
and define the use of resources.

DirectiveDescriptionbase-uri:The base HTML element contains the absolute URL
that is prepended to all the relative URLs on the page. This directive helps us
restrict the URLs that are allowed to be used in the base HTML element, and
therefore prevent Base Tag Hijacking attacks.child-src:This directive allows us
to define which websites are permitted to be loaded in frames located on the
page. We can use it as an extra precaution to protect our page from Frame
Injection attacks.connect-src:This directive restricts the resources that can be
loaded via script interfaces such as XHR or WebSockets. This prevents attackers
from stealing data from the site.font-src:This directive specifies the font
sources that can be loaded using @font-face. It is mostly used to prevent
attackers from sending extracted data back to their server using the @font-face
src directive.form-action:This directive specifies the URLs that can be used as
targets for form submissions. It can be used as an extra precaution to protect
pages from Form Tag Hijacking and Cross-Site Scripting
attacks.frame-ancestors:This directive specifies the sites that have the
authority to load the current page in a frame, iframe, object, embed, and applet
tag. It is a substitute for X-Frame-Options, since it can also help prevent
Clickjacking and UI Redressing attacks.img-src:This directive defines the
sources from which images can be loaded.media-src:This directive defines or
restricts the sources from which video and audio can be loaded.object-src:This
directive defines or restricts the sources from <object>, <embed>, and <applet>,
which helps preventing Cross-Site Scripting attacks.plugin-types:This directive
defines or restricts the plugin types that can be loaded.report-uri:This
directive specifies the URLs that will receive the report when a CSP directive
is violated.style-src:This directive defines or restricts the sources for CSS
files. This allows you to avoid data exfiltration via
CSS.upgrade-insecure-requests:This directive converts the HTTP requests to
HTTPS.

DEFAULT-SRC DIRECTIVE EXAMPLE

By default these directives are unrestrictive, meaning that if they are not
declared in the CSP header, any request will go through. So, if no value is
given for style-src, this will be interpreted as style-src: * and styles from
all sources will be allowed.

We can use the default-src directive to change this. The specified value will
override most directives ending with -src by setting a default value for them.
If we define default-src as http://www.example.com and don’t set a value
for font-src, the fonts can only be loaded from https://www.example.com.

However, default-src cannot override these directives:

 * base-uri
 * form-action
 * frame-ancestors
 * plugin-types
 * report-uri
 * sandbox

We can add more than one directive in one HTTP header by separating them with a
semicolon:

script-src https://host1.com https://host2.com; style-src https://www.example.com

Multiple values for a directive must be separated with a space:

script-src https://host1.com https://host2.com;


ONE THING TO KEEP IN MIND WHEN USING CONTENT SECURITY POLICY

Define CSP for each page set in the HTTP response. This will help you define the
optimal policy for each page and its specific needs.

You can learn more about Content-Security-Policy (CSP) on the Invicti Blog.


HTTP STRICT TRANSPORT SECURITY (HSTS) HTTP HEADER

As the name suggests, HSTS is a mechanism that forces browsers to use a secure
web connection.

In fact, we can call HSTS the ‘missing link’ in the secure web connection chain.
Why?

With end-to-end encryption, SSL lays the foundation for a secure and authorized
connection. In an ideal SSL implementation, we can ensure that the data is not
altered or monitored by a third party during transmission, or verify that the
person we’re communicating with is the intended recipient of the traffic.

I say ‘ideal’ because the partial implementations that some web applications
use, such as secure connections only in login and checkout pages, pose a huge
threat. The risks that arise with partial implementations are covered in detail
in HTTP Cookie Hijacking in the Wild (DEF CON 24 conference).

If we do not force all our pages to be transmitted over a secure connection,
attackers can easily forward their targets to surf the web on the unsafe
connection or convert the HTTPS traffic to HTTP by launching MITM attacks (such
as SSL Strip).

That’s not all. Expired or invalid certificates are highly threatening to your
secure connections. Your website’s reputation will also be damaged if your users
are confronted with such easily preventable problems.


IS THERE ANY WAY TO FURTHER IMPROVE HTTPS PROTECTION ON YOUR WEBSITE?

If you want to take full advantage of SSL, you have to convert all HTTP requests
to HTTPS, and load all images, scripts, style files and other files over a
secure connection. You also have to go as far as preventing the user from
dismissing certificate errors.

Doing all this manually is rather difficult, especially in case of a certificate
error when you have to prevent the user from accessing your website.

Thankfully, the HSTS specification published in November 2012 is the solution to
this problem. All we have to do is to set a security header
(Strict-Transport-Security) on our web page response. With this security header
we can direct our browser to:

 * Convert all requests to an HTTPS connection
 * In case of a certificate related error such as an expired certificate,
   prevent the user from browsing the website anyway
 * Cache this setting for a specified amount of time

STRICT TRANSPORT SECURITY HEADER EXAMPLE

Here is an example of how to use this header:

Strict-Transport-Security: max-age=31536000;

max-age:  This directive allows us to specify the amount of time (in seconds)
that the content of the header will be stored in the browser cache.

There are two more significant specifications in the HSTS implementation:
the includeSubdomains parameter and the preload parameter. We can enforce the
same strict, secure connection on all the subdomains of the website using
the includeSubdomains parameter.


FIRST REQUEST AND PRELOAD

We can only set the HSTS header over a secure connection. Just like Public Key
Pinning (which is another HTTPS security header), HSTS is based on Trust on
First Use (TOFU). This means that we have to manually redirect the users, on
their first HTTP request, to the HTTPS version of our website. In its response,
there will be an HSTS header, meaning that the browsers will store it in their
cache. The next time an unsafe (HTTP) connection arises, it will automatically
be redirected to a secure connection.

The obvious question is: could a hacker override this first HTTP request with a
MiTM attack?

Unfortunately, the answer is yes. However, this time, the HSTS preload list
comes to the rescue.

We need to add the preload directive to our HSTS header for the website to be
eligible for inclusion in the HSTS preload list. The browser will check whether
or not a site is included in the list and will simply refuse to load it over an
insecure connection.

ADDITIONAL REQUIREMENTS FOR THE HSTS PRELOAD LIST

Adding our website to the HSTS Preload List has some further requirements:

 1. You need to serve a valid certificate.
 2. You must redirect from HTTP to HTTPS on the same host, meaning that the
    HTTPS traffic for example.com should go through example.com (not through any
    other domain such as secure.example.com).
 3. You must support HTTPS for all subdomains, particularly the www subdomain.
 4. You have to set the following values for the HSTS header:

 * The max-age must be at least 31536000 seconds (1 year)
 * The includeSubDomains and preload directives must be specified

Strict-Transport-Security: max-age=31536000; includeSubDomains; preload


HTTP PUBLIC KEY PINNING

In order to understand HTTP Public Key Pinning (HPKP), let’s take a look at how
browsers implement the SSL handshake, which is conducted before establishing a
secure connection.

When our users try to access our site securely, we send them our public key that
both affirms that they are talking to the right party and will be used to
encrypt the shared key. We call this a ‘certificate’ and it holds information
like our website’s name, certificate expiration date, and the amount of bits in
the cryptographic key.


CERTIFICATE AUTHORITIES AND FRAUDULENTLY ISSUED CERTIFICATES

There’s one additional piece of information in the certificate. When the browser
receives our certificate, it checks if our certificate was signed by a trusted
certificate authority (CA). One of the pieces of information in our certificate
is the name of the CA, which will be used by the browser to validate our
certificate. A secure connection is established once it is confirmed as genuine.

However, several incidents have revealed that, regardless of the motive, one of
these certificate authorities can sign a certificate on behalf of a website
without notifying the website owner!

 * One example of this occured in the Netherlands in 2011 when the authorized
   certificate distributer DigiNotar was hacked. Around 500 certificates were
   fraudulently signed. Multiple Google services, including 300,000 Gmail
   accounts belonging to Iranian citizens, were monitored with these
   certificates.
 * A similar incident took place in 2008 and 2011, affecting a certificate
   authority called StartCom. Certificates for paypal.com and verisign.com were
   published. In 2011, attackers accessed the root key of StartCom, gaining the
   ability to produce, invalidate, or renew any StartCom certificates.
 * In 2011, due to an incident with a certificate authority called Turktrust, an
   intermediate certificate was issued to sign certificates on behalf of Google
   domains. One of them was spotted on December 26, 2012 by Google Chrome, and
   subsequently reported to Google. It’s noteworthy that the Chrome browser
   figured this out and reported it back to Google.

HOW TO PREVENT FRAUDULENTLY ISSUED CERTIFICATES

The question is: How can a browser automatically detect a fraudulently issued
certificate for a specific domain and even report this certificate back to
Google?

The answer to this question is a mechanism called HTTP Public Key
Pinning (HPKP). Ever since the introduction of this feature in Chrome 13,
websites can make their certificate fingerprints (hash values) known to the
browsers using the Public-Key-Pins response header.

Browsers store these hashes on the user’s computer. Every time the website is
visited, the browser generates a hash based on the certificate’s public key. If
the hash doesn’t match the stored value, the connection won’t be established,
and the incident is reported to a URL if the report-uri directive is set.

In conclusion, HTTP Public Key Pinning protects the users and websites from CAs
signing fake certificates.


SETTING UP AN HPKP HEADER

Public-Key-Pins is a security header. We can enable HPKP by sending it back with
every response over a secure connection (HTTPS):

Public-Key-Pins:
pin-sha256="d6qzRu9zOECb90Uez27xWltNsj0e1Md7GkYYkVoZWmM="; pin-sha256="E9CZ9INDbd+2eRQozYqqbQ2yXLVKB9+xcprMF+44U1g="; report-uri="http://example.com/pkp-report"; max-age=10000; includeSubDomains


HPKP DIRECTIVES

Various directives are available for the HPKP header, such as max-age and
report-uri. This is a complete list.

DirectiveDescriptionpin-sha256=”<sha256>”:This is the public key fingerprint of
our certificate encoded in base64. You can store hashes of multiple certificates
here as backups, in case one of the certificates expires.max-age:This is the
length of time (in seconds) this directive should be
cached.includeSubdomains:This states whether the pinning covers the subdomains
or not.report-uri:As outlined in the Google example above, in case of a
mismatch, this directive specifies the URL to which the reports of the HPKP
policy breach should be sent. The breach information will be sent to this URL
using the HTTP POST method.

You can set the HTTP Public Key Pinning mechanism to work in Report-Only mode,
which means the browser will still establish a secure connection if a mismatch
occurs, but will report the breach to the URL stated in the report-uri
directive:

Public-Key-Pins-Report-Only: pin-sha256="d6qzRu9zOECb90Uez27xWltNsj0e1Md7GkYYkVoZWmM="; pin-sha256="E9CZ9INDbd+2eRQozYqqbQ2yXLVKB9+xcprMF+44U1g="; report-uri="http://example.com/pkp-report"; max-age=10000; includeSubDomains

According to RFC 7469, you must set at least 2 pins. One is for your current
certificate, and the other is for your backup certificate. The given set of Pins
contains at least one Pin that does not refer to an SPKI in the certificate
chain.  (That is, the host must set a Backup Pin; see Section 4.3.)

Research published on EvenPaths concludes that there are HPKP implementation
errors, even in webpages belonging to large companies such as Facebook and
WhatsApp. This information was collected using the Alexa Top 1 Million domain
list. And while there are websites that did not pin any certificates in their
HPKP headers, there were also websites that pinned as many as 17.


WHICH BROWSERS SUPPORT PKP SECURITY HEADERS?

After reading the above about HPKP, get ready for a little surprise… Starting
with Google Chrome version 69, HPKP – the mechanism that helped Google to catch
fraudulently issued certificates in the wild – is no longer supported.



Source: https://caniuse.com/#search=Public%20Key%20Pinning

Even though this is a little surprising at first, it makes sense if you think
about it. If even well resourced companies such as Facebook (as outlined in the
ElevenPaths study) make mistakes in implementing this security measure, it’s
probably too complicated to be used reliably on a large scale.

However, there is a header that promises an equal amount of protection, with a
little extra effort.


EXPECT-CT HTTP HEADER

The header that will eventually replace HPKP is called Expect-CT. Even though
HPKP was a useful security feature, it by far wasn’t the only way to detect
certificates issued by rogue CAs or to prevent them from doing so. With security
mechanisms such as Certificate Authority Authorization (CAA) and Certificate
Transparency, we can still be notified of certificates issued on behalf of us
without our permission or knowledge and in some cases even prevent CAs from
issuing these certificates altogether.

But how would these protections work and can they really replace HPKP?


CERTIFICATE TRANSPARENCY LOGS

The answer is Certificate Transparency (CT). That’s the reason why the security
header we are going to talk about is called ‘Expect-CT’ (in other words ‘Expect
the certificate to be submitted to a Certificate Transparency Log’).

These Certificate Transparency Logs are publicly accessible and therefore
administrators can check them and search for their own domains. If there is a
certificate issued for their domain without their prior knowledge or
authorization, they can immediately take steps to protect their users.

Of course CAs aren’t motivated to add more complexity to the already heavily
regulated process of signing certificates. This is why Google, as the developer
of one of the most popular browsers, had to put its foot down and make the CT
log mandatory for new certificates issued from April 2018.

HOW CT LOGS WORK

Let’s have a quick look at the Certificate Transparency log mechanism. CAs, by
design, have to submit each issued certificate to this log. There are three
parts that need to come together in order for CT to work:

 1. Submission of the certificates to logs, which is the task of CAs
 2. Monitoring the logs
 3. Auditing, which is done by built-in auditors in browsers

In addition, it is advised that site owners add the Expect-CT header to their
responses. Browsers decide whether or not the certificates presented to them
follow the outlined rules or not. Along with the certificate, browsers check the
Signed Certificate Timestamp (SCT). This data contains the timestamp of when the
certificate was logged. The browser uses the SCT information to check if the
outlined conditions are met.

WHERE IS THE RISK?

If we have a certificate issued for our website before the obligatory start date
of April 2018 in Google Chrome, and the expiration date is set 10 years ahead,
Google cannot enforce the certificate transparency rules. Otherwise, this would
result in a large amount of perfectly valid certificates that are no longer
usable, even if they were issued before CT was invented. Since HPKP is
deprecated in Chrome, there would be no way of being notified of an illegally
issued certificate for our website.

However, if we set and enforce the Expect-CT header and use max-age to cache
this directive in the browser for a while, during the connection to our website,
certificates that do not meet the CT requirements will not be accepted even if
they are signed before April 2018. This way, we eliminate the risk of older
certificates being determined to be valid without our knowledge.


HOW DO WE IMPLEMENT THE EXPECT-CT HEADER?

First we have to make sure that our current certificate supports CT. We can do
so by generating an SSL Labs report:

https://www.ssllabs.com/ssltest/analyze.html?d=invicti.com

Next we can activate the Expect-CT header in the HTTP response by using a header
as below:

Expect-CT: enforce,max-age=30,report-uri="https://ABSOLUTE_REPORT_URL"

The Expect-CT header has three directives:

 * max-age: This is the duration (in seconds) of the data being stored in the
   browser cache.
 * report-uri: This is the URL to which the breach report will be sent. It must
   be an absolute URL (e.g. https://example.com/report) not a relative one (e.g.
   /report).
 * enforce: This indicates whether the connection should be established, if a
   certificate that doesn’t comply with CT is present.

You can also use this in report-only mode which doesn’t initially enforce the CT
requirement.

Expect-CT: max-age=30,report-uri="https://ABSOLUTE_REPORT_URL"

If everything is as expected, you can then choose to enforce.

Expect-CT: max-age=30,report-uri="https://ABSOLUTE_REPORT_URL", enforce


REFERRER-POLICY HTTP HEADER

Referer is a request header that is confusing on multiple levels. First of all
‘referer’ is misspelt. (The correct spelling is ‘referrer’.) Even though this is
an amusing fun fact, it also shows just how hard it is to even correct a simple
mistake such as a missing ‘r’ in an HTTP header field. Just imagine how much
harder it would be to correct a critical security vulnerability in a widespread
protocol!

But the misspelling is not the only reason why this header is often not properly
understood. Let’s take a look at how this header works.


HOW THE REFERRER-POLICY HEADER WORKS

You are the owner of website A and you want your visitors to check out website
B. You do this by placing a hyperlink to Website B on your homepage. If users
click on the link, their browser will automatically add the Referer header to
the request headers. It’s content will be the address of website A. This has the
advantage that Website B can see who linked to their site just by checking
the Referer header of each incoming request. The Referer header will be added to
requests made for style, image, script loads, and form submissions. The request
would look like this:

GET / HTTP/1.1
Host: B.com
Referer: A.com

You might want to hide the information in the Referer header for multiple
reasons, such as security and privacy.

As a response header, Referrer-Policy gives you the following options to help
control the Referer request header. Note how Referrer-Policy is written with a
double r (rr). Arguably, this just adds to the Referer/Referrer spelling
confusion even though it’s the correct way to write it.

The Referrer-Policy header can be set up in HTTP response messages as follows:

Referrer-Policy: no-referrer


REFERRER-POLICY DIRECTIVES

Here are all potential values the Referrer-Policy header can send.

DirectiveDescription” ” (empty):This indicates that the Referrer-Policy is not
set and that the directive to control referer can be set by an HTML element on
the page. This can be done by utilizing <meta> tags or specifying a policy for
individual HTML tags using the rel or referrerpolicy
attributes.Referrer-Policy:no-referrer:This will not add any Referer header even
if the redirected page has the same origin as the host.Referrer-Policy:
no-referrerno-referrer-when-downgrade:In case of a protocol downgrading (passing
from a more secure protocol to a less secure one, such as HTTPS to HTTP), the
Referer header will not be sent.Referrer-Policy: no-referrer-when-downgradeThis
is the default behavior of all browsers.same-origin:This only sends the Referer
header if the target site is of the same origin (scheme, domain, and port must
match). You can read more about this in Introducing the Same-origin Policy
Whitepaper.Referrer-Policy: same-originorigin:This truncates the path portion of
the URL in the Referer header. As mentioned above, the origin consists of the
scheme, domain, and port.Referrer-Policy: originDuring protocol-downgrading
(switching from HTTPS to HTTP), path information will be omitted and only origin
data will be sent to the HTTP website within the referer.strict-origin:This
value will ensure that the browser only sends the origin as the referrer when
the protocol security level stays the same (e.g. HTTPS and HTTPS), but that it
won’t send it to a website with a lower security level, such as from HTTPS to
HTTP.Referrer-Policy: strict-originorigin-when-cross-origin:If the target and
host websites have the same origin, the Referer header will include the full
url. If the two have different origins, only scheme and domain data will be
included in the Referer header.Referrer-Policy: origin-when-cross-originOrigin
data will also be sent to the requested HTTP site with the Referer header in
case of protocol downgrading.strict-origin-when-cross-origin:Using this option,
the origin in the Referer data will only be visible when the target and host
website share in the same protocol security level or the target site is of a
higher one.Referrer-Policy: strict-origin-when-cross-originunsafe-url:Browsers
will share the full URL in the Referer header in every request done from the
host to the target website.Referrer-Policy: unsafe-urlYou should note that with
this option, the full URL will be shared even from secure to unsafe connections.
This option opts you out of the default behavior of browsers (avoiding URL
visibility in protocol downgrading) and therefore should be used with caution.


WHICH BROWSERS SUPPORT REFERRER-POLICY SECURITY HEADER?

Yet again, this header is not supported equally well in all modern browsers. In
the image below you can see which browsers fully support Referrer-Policy and
which don’t. For further information, visit the link below the image.

Source: https://caniuse.com/#feat=referrer-policy




PROACTIVITY IS VITAL!

Security nowadays, contrary to popular belief, is not a defensive art. It is
mostly the art of preempting your challengers’ moves and being able to plan
ahead. Security adversaries are more creative than ever, revealing new tactics
every day. Thanks to HTTP security headers, it is possible to be a few steps
ahead, ensuring the security of our sites, our users and our data.

.

Authored by Ziyahan Albeniz
Reviewed by Sven Morgenroth
Translated by Umran Yildirimkaya



Invicti Security Corp
1000 N Lamar Blvd Suite 300
Austin, TX 78703, US

© Invicti 2023

 * RESOURCES
   * Features
   * Integrations
   * Plans
   * Case Studies
   * Advisories
   * Invicti Learn
 * USE CASES
   * Penetration Testing Software
   * Website Security Scanner
   * Ethical Hacking Software
   * Web Vulnerability Scanner
   * Comparisons
   * Online Application Scanner
 * WEB SECURITY
   * The Problem with False Positives
   * Why Pay for Web Scanners
   * SQL Injection Cheat Sheet
   * Getting Started with Web Security
   * Vulnerability Index
   * Using Content Security Policy to Secure Web Applications
 * COMPANY
   * About Us
   * Contact Us
   * Support
   * Careers
   * Resources
   * Partners

© Invicti 2023
 * Legal
 * Privacy Policy
 * California Privacy Rights
 * Terms of Use
 * Accessibility
 * Sitemap

By using this website you agree with our use of cookies to improve its
performance and enhance your experience. More information in our Privacy Policy.

OK