At nonce to Piwik Pro inline script

Created on 13 February 2025, about 2 months ago

Problem/Motivation

On my website I have an inline script for adding analytics to my website via piwik pro. I add this directly in my twig template or can do it by attaching a library. Both cases will fail for my csp settings and it mentions adding a nonce attribute.

And that is where I'm stuck now. I see more issues about this but none of them point me in the right direction yet. But that is mostly my problem, I'm new to this csp header thing and find it hard to wrap my head around it. What I don't understand is how to add the right nonce on the right place and how this nonce will end up in my header.

Can some explain this to me for a code like this:

<script type="text/javascript">
      (function (window, document, dataLayerName, id) {
        window[dataLayerName] = window[dataLayerName] || [], window[dataLayerName].push({
          start: (new Date).getTime(),
          event: "stg.start"
        });
        var scripts = document.getElementsByTagName('script')[0], tags = document.createElement('script');

        function stgCreateCookie(a, b, c) {
          var d = "";
          if (c) {
            var e = new Date;
            e.setTime(e.getTime() + 24 * c * 60 * 60 * 1e3), d = "; expires=" + e.toUTCString()
          }
          document.cookie = a + "=" + b + d + "; path=/"
        }

        var isStgDebug = (window.location.href.match("stg_debug") || document.cookie.match("stg_debug")) && !window.location.href.match("stg_disable_debug");
        stgCreateCookie("stg_debug", isStgDebug ? 1 : "", isStgDebug ? 14 : -1);
        var qP = [];
        dataLayerName !== "dataLayer" && qP.push("data_layer_name=" + dataLayerName), isStgDebug && qP.push("stg_debug");
        var qPString = qP.length > 0 ? ("?" + qP.join("&")) : "";
        tags.async = !0, tags.src = "{{ analytics_url }}" + id + ".js" + qPString, scripts.parentNode.insertBefore(tags, scripts);
        !function (a, n, i) {
          a[n] = a[n] || {};
          for (var c = 0; c < i.length; c++) !function (i) {
            a[n][i] = a[n][i] || {}, a[n][i].api = a[n][i].api || function () {
              var a = [].slice.call(arguments, 0);
              "string" == typeof a[0] && window[dataLayerName].push({
                event: n + "." + i + ":" + a[0],
                parameters: [].slice.call(arguments, 1)
              })
            }
          }(i[c])
        }(window, "ppms", ["tm", "cm"]);
      })(window, document, 'dataLayer', '{{ analytics_id }}');
    </script>
💬 Support request
Status

Active

Version

2.0

Component

Code

Created by

🇳🇱Netherlands zebda

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @zebda
  • 🇨🇦Canada gapple

    First, a nonce shouldn't be required if you (1) insert the JavaScript snippet via a library, and (2) authorize any necessary external domains in script-src & script-src-elem.

    The Google Tag module and its merge request to support CSP may be a helpful reference Support Content Security Policy nonce Active . Any information the script needs for gtag configuration as well as a nonce value is passed via drupalSettings.
    It only uses a nonce to authorize its script loader to add scripts from additional third party domains, but it still needs to specify those third party domains as a fallback for compatibility, otherwise expected features may be blocked when a page is unable to use a nonce due to something else on the page being incompatible.
    If piwik doesn't load further scripts from a domain not already authorized in the policy, then it doesn't require a nonce.

    ----

    Using a nonce for an inline script is something that could use some improved documentation, but is not something I encourage 🙂.
    My, now quite old article, on converting inline JS for D8+ only briefly covers part of it, since the nonce feature didn't exist at the time.

    My preferred approach is:

    1. Place the snippet in a file, added to the page via a library
    2. Alter the snippet to add the nonce value to the script element that it inserts into the page:
            var scripts = document.getElementsByTagName('script')[0], tags = document.createElement('script');
            tags.setAttribute('nonce', drupalSettings.csp.nonce);
          
    3. If you're not hard-coding your {{ analytics_id }} and {{ analytics_url }} values in the js file, pass them to the front-end with drupalSettings in hook_page_attachments
    4. Make the library dependent on the csp.nonce library (which implicitly requires the core.drupalSettings library.

    The other option is to use a placeholder for the nonce attribute's value on the inline script and the attribute on the inserted script element so that every request receives a new nonce value. The placeholder can be retrieved from the Nonce Builder service in the relevant theme hook to add as a template variable, and the lazy builder for replacing the placeholder added the element's #attached property or to the page via hook_page_attachments.

  • 🇳🇱Netherlands zebda

    Thanks this is great! The nonce is added to the script, but not added to the header. How do I make sure it is also added to the csp header?

  • 🇨🇦Canada gapple

    The easiest way is to use attachments, either on a relevant render element or with hook_page_attachments(), to add the csp_nonce property with the necessary fallback values.
    If you're adding the JS through a library, this should include your analytics_url domain, and any others that piwik includes, and probably not Csp::POLICY_UNSAFE_INLINE.

      $element['#attached']['csp_nonce'] = [
        'script' => [Csp::POLICY_UNSAFE_INLINE],
      ];
    

    The nonce is not automatically added to the header by using the csp.nonce JS library as a dependency because of the need to specify fallback values.
    If some other functionality on your page requires 'unsafe-inline', then a nonce will not be added to the header, but any placeholders will still receive a nonce value.

  • 🇳🇱Netherlands zebda

    Thanks! I got it to work.

  • Automatically closed - issue fixed for 2 weeks with no activity.

Production build 0.71.5 2024