Apps on the Shopify App Store must set the proper Content Security Policy `frame-ancestors` directive to avoid clickjacking attacks. If the Content Security Policy `frame-ancestors` directive is missing or set incorrectly when you submit your app to the Shopify App Store, then your app might be rejected. You'll be required to address this before re-submitting your app for review.

> Tip:
> To learn more about clickjacking, refer to [Portswigger's Web Academy](https://portswigger.net/web-security/clickjacking) or [OWASP Clickjacking](https://owasp.org/www-community/attacks/Clickjacking). To learn more about the `frame-ancestors` directive, refer to [MDN Web Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy/frame-ancestors).

## Embedded Apps

If your app is an [embedded app](/docs/apps/build/app-surfaces#embedded-app-pages), then you need to make sure that your app is only frameable by the authenticated shop domain. Set the `frame-ancestors` directive dynamically based on the _current shop domain_ and the _Shopify admin domain_. Setting this directive guarantees that your app can be framed only within the shop admin.

For example, if the shop `shopify-dev.myshopify.com` installs your app, then the response headers from your app when being rendered by this shop will contain the following `frame-ancestors` declaration:

```text
Content-Security-Policy: frame-ancestors https://shopify-dev.myshopify.com https://admin.shopify.com;
```

You can include other declarations in your `Content-Security-Policy` header besides `frame-ancestors`.

> Note:
> The `frame-ancestors` declaration must be different for every shop, and these headers must be present in any routes that render HTML content.

## Apps that aren't embedded

If your app isn't embedded, then we recommend setting the `frame-ancestors` directive to `'none'` in order to disallow framing.

```
Content-Security-Policy: frame-ancestors 'none';
```

## Troubleshooting

Apps can fail to meet the iframe protection requirement in the following ways:

* The app isn't embedded, but is configured as an embedded app in the Partner Dashboard
* The app is embedded, but isn't following the expected `frame-ancestors` guidelines

The following scenarios explain how to correct these issues.

### The app shouldn't be embedded, but is configured as an embedded app in the Partner Dashboard

If your app isn't embedded in the shop admin, then it shouldn't be configured as embedded in the Partner Dashboard.

Follow these steps to check the configuration of your app:

1. Log in to your [Partner Dashboard](https://partners.shopify.com/organizations).
2. Click **Apps**.
3. Click the name of your app.
4. Click **Configuration**.
5. In the **Embedded in Shopify admin** section, set the value for **Embed app in Shopify admin** to **False**.
6. Click **Save and release**.

### The app is embedded, but isn't following the expected `frame-ancestors` guidelines

This scenario uses the following example values:

* `Fraud Filter` as the app
* `cambridgetestshop-staging.myshopify.com` as the shop

To validate whether `Fraud Filter` implements the expected headers, follow these steps:

1. Log in to the `cambridgetestshop-staging` shop.
2. Click **Apps**.

    <img
      src="/assets/apps/security/apps_sidebar.png"
      alt="Screenshot showing how to navigate to the apps page"
      width="250"
    />

3. Right-click anywhere on the page and select **Inspect**.

    <img
      src="/assets/apps/security/inspect_page.png"
      alt="Screenshot showing how to inspect a web page"
      width="250"
    />

4. Select the **Network** tab.

    <img
      src="/assets/apps/security/network_tab.png"
      alt="Screenshot showing how to select the network tab in the inspector"
      width="600"
    />

5. Load your app by clicking on its name. The content of the **Network** tab will start to change.

    <img
      src="/assets/apps/security/fraud_filter_app.png"
      alt="Screenshot showing the fraud filter app within the apps page"
      width="600"
    />

    <img
      src="/assets/apps/security/requests_network_inspector.png"
      alt="Screenshot showing requests in the network inspector"
      width="600"
    />

6. Click **Doc** in the **Network** tab to filter requests for documents.

    <img
      src="/assets/apps/security/doc_filter.png"
      alt="Screenshot showing how to filter requests for documents in the network inspector"
      width="600"
    />

7. Click the document to load a panel with more details.
    - If there's more than one document, then select the last one in the list.

    <img
      src="/assets/apps/security/filtered_docs.png"
      alt="Screenshot showing filtered requests for documents in the network inspector"
      width="600"
    />

8. Check the **Request URL**. The URL should be from the app's domain. In this scenario, the **Request URL** is from the _Fraud Filter_ app domain.

    <img
      src="/assets/apps/security/fraud_filter_response.png"
      alt="Screenshot showing the filtered request for fraud filter"
      width="600"
    />

9. Check that the "frame-ancestors" directive is included in the "Content-Security-Policy" header.

    <img
      src="/assets/apps/security/fraud_filter_response_headers.png"
      alt="Screenshot showing the response headers for fraud filter"
      width="600"
    />

In this scenario, the directive is correctly included for the `cambridgetestshop-staging.myshopify.com` shop.