___TERMS_OF_SERVICE___ By creating or modifying this file you agree to Google Tag Manager's Community Template Gallery Developer Terms of Service available at https://developers.google.com/tag-manager/gallery-tos (or such other URL as Google may provide), as modified from time to time. ___INFO___ { "type": "CLIENT", "id": "cvt_temp_public_id", "version": 1, "securityGroups": [], "__wm": "VGVtcGxhdGUtQXV0aG9yX1ByZWZsaWdodC1SZXF1ZXN0LVNpbW8tQWhhdmE\u003d", "categories": [ "UTILITY" ], "displayName": "Preflight Request", "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "" }, "description": "Use this Client to respond to preflight (OPTIONS) requests with the appropriate Access-Control-* headers. Must be published - will not work in Debug mode.", "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "approvedOrigin", "displayName": "Allowed request origin", "simpleValueType": true, "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^(http(s)?://[^/]+|auto)$" ] } ], "help": "Enter the origin from where you want to approve requests.Set the value to \u003cstrong\u003eauto\u003c/strong\u003e in case you want to (attempt to) approve the request from any origin.", "defaultValue": "https://www.my.origin", "alwaysInSummary": true }, { "type": "TEXT", "name": "approvedMethods", "displayName": "Allowed request methods", "simpleValueType": true, "help": "Enter a comma-separated list of methods to approve the preflight request for, or set to * to approve all methods.", "defaultValue": "GET,POST", "valueValidators": [ { "type": "NON_EMPTY" } ], "alwaysInSummary": true }, { "type": "TEXT", "name": "approvedHeaders", "displayName": "Allowed request headers", "simpleValueType": true, "defaultValue": "content-type", "help": "Enter a comma-separated list of request headers to approve the preflight request for.", "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^[^*]+$" ], "errorMessage": "Asterisk (*) is not a valid value here. You must list the headers your request includes." } ], "alwaysInSummary": true } ] ___SANDBOXED_JS_FOR_SERVER___ const claimRequest = require('claimRequest'); const getRequestHeader = require('getRequestHeader'); const getRequestMethod = require('getRequestMethod'); const returnResponse = require('returnResponse'); const setResponseHeader = require('setResponseHeader'); const setResponseStatus = require('setResponseStatus'); // If not a preflight request, return if (getRequestMethod() !== 'OPTIONS') return; claimRequest(); // If origins is wildcarded, set the header to the request's origin if (data.approvedOrigin === 'auto') { setResponseHeader('Access-Control-Allow-Origin', getRequestHeader('origin')); } else { // Otherwise add an access header for each listed origin setResponseHeader('Access-Control-Allow-Origin', data.approvedOrigin); } setResponseHeader('Access-Control-Allow-Methods', data.approvedMethods); setResponseHeader('Access-Control-Allow-Headers', data.approvedHeaders); // The XHR needs .withCredentials set to true for the request to show up in Debug mode setResponseHeader('Access-Control-Allow-Credentials', 'true'); setResponseStatus(200); returnResponse(); ___SERVER_PERMISSIONS___ [ { "instance": { "key": { "publicId": "read_request", "versionId": "1" }, "param": [ { "key": "headerWhitelist", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "origin" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "method" } ] } ] } }, { "key": "headersAllowed", "value": { "type": 8, "boolean": true } }, { "key": "requestAccess", "value": { "type": 1, "string": "specific" } }, { "key": "headerAccess", "value": { "type": 1, "string": "specific" } }, { "key": "queryParameterAccess", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "access_response", "versionId": "1" }, "param": [ { "key": "writeResponseAccess", "value": { "type": 1, "string": "specific" } }, { "key": "writeHeaderAccess", "value": { "type": 1, "string": "specific" } }, { "key": "writeStatusAllowed", "value": { "type": 8, "boolean": true } }, { "key": "writeHeadersAllowed", "value": { "type": 8, "boolean": true } }, { "key": "writeHeaderWhitelist", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "access-control-allow-origin" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "access-control-allow-methods" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "access-control-allow-headers" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "access-control-allow-credentials" } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "return_response", "versionId": "1" }, "param": [] }, "isRequired": true } ] ___TESTS___ scenarios: - name: Do not run if request is not OPTIONS code: | mock('getRequestMethod', () => 'POST'); // Call runCode to run the template's code. runCode(mockData); assertApi('claimRequest').wasNotCalled(); - name: Returns successful response with basic settings code: | // Call runCode to run the template's code. runCode(mockData); assertApi('claimRequest').wasCalled(); assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Origin', 'https://www.domain.com'); assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Methods', 'GET,POST'); assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Headers', 'content-type'); assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Credentials', 'true'); assertApi('setResponseStatus').wasCalledWith(200); assertApi('returnResponse').wasCalled(); - name: Return successful response with * origin code: |- mock('getRequestHeader', h => { if (h === 'origin') return 'https://www.otherdomain.com'; }); mockData.approvedOrigin = 'auto'; // Call runCode to run the template's code. runCode(mockData); assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Origin', 'https://www.otherdomain.com'); setup: |- const mockData = { approvedOrigin: 'https://www.domain.com', approvedMethods: 'GET,POST', approvedHeaders: 'content-type' }; mock('getRequestMethod', () => 'OPTIONS'); ___NOTES___ Created on 02/09/2020, 14:42:43