___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, "__wm": "VGVtcGxhdGUtQXV0aG9yX3JlQ0FQVENIQXYzLVNpbW8tQWhhdmE\u003d", "categories": [ "UTILITY" ], "securityGroups": [], "displayName": "reCAPTCHA v3", "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "\u003d\u003d" }, "description": "Template for loading the reCAPTCHA v3 JavaScript as well as for verifying reCAPTCHA requests and generating a bot score.", "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "siteKey", "displayName": "Site key", "simpleValueType": true, "help": "Set the *site* key to verify valid requests for api.js.", "valueValidators": [ { "type": "NON_EMPTY" } ], "alwaysInSummary": true }, { "type": "TEXT", "name": "secretKey", "displayName": "Secret key", "simpleValueType": true, "help": "Set the *secret* key to validate communication between the server and reCAPTCHA.", "valueValidators": [ { "type": "NON_EMPTY" } ], "alwaysInSummary": true }, { "type": "CHECKBOX", "name": "setCookie", "checkboxText": "Write bot score in a cookie", "simpleValueType": true, "defaultValue": true }, { "type": "CHECKBOX", "name": "returnResponse", "checkboxText": "Return the bot score in the HTTP response", "simpleValueType": true, "defaultValue": false }, { "type": "CHECKBOX", "name": "setEncodingHeader", "checkboxText": "Set Content-Encoding header (required for Cloud Run)", "simpleValueType": true, "help": "If you check this, a Content-Encoding header with the value \"gzip\" will be sent back in the response to the browser." }, { "type": "GROUP", "name": "cookieSettings", "displayName": "Cookie settings", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "TEXT", "name": "cookieName", "displayName": "Cookie name", "simpleValueType": true, "defaultValue": "__r_b_s", "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "TEXT", "name": "maxAge", "displayName": "Cookie lifetime", "simpleValueType": true, "valueUnit": "seconds", "defaultValue": 2592000, "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "NON_NEGATIVE_NUMBER" } ], "help": "Set the cookie lifetime in seconds (30 days by default). If you set this to 0, a session cookie is created." }, { "type": "CHECKBOX", "name": "httpOnly", "checkboxText": "HttpOnly cookie", "simpleValueType": true, "defaultValue": true } ], "enablingConditions": [ { "paramName": "setCookie", "paramValue": true, "type": "EQUALS" } ] } ] ___SANDBOXED_JS_FOR_SERVER___ // Load APIs const claimRequest = require('claimRequest'); const encodeUriComponent = require('encodeUriComponent'); const getRemoteAddress = require('getRemoteAddress'); const getRequestHeader = require('getRequestHeader'); const getRequestMethod = require('getRequestMethod'); const getRequestQueryParameter = require('getRequestQueryParameter'); const getRequestPath = require('getRequestPath'); const getTimestampMillis = require('getTimestampMillis'); const JSON = require('JSON'); const logToConsole = require('logToConsole'); const makeString = require('makeString'); const returnResponse = require('returnResponse'); const sendHttpGet = require('sendHttpGet'); const sendHttpRequest = require('sendHttpRequest'); const setCookie = require('setCookie'); const setResponseBody = require('setResponseBody'); const setResponseHeader = require('setResponseHeader'); const setResponseStatus = require('setResponseStatus'); const templateDataStorage = require('templateDataStorage'); // Helpers const now = getTimestampMillis(); const oneDayAgo = getTimestampMillis() - 1000 * 60 * 60 * 24; const requestPath = getRequestPath(); const encode = str => { return encodeUriComponent(makeString(str || '')); }; const log = msg => { logToConsole('RECAPTCHA: ' + msg); }; const fail = msg => { log(msg); proxyResponse(msg, {}, 500); }; const proxyResponse = (response, headers, statusCode) => { setResponseStatus(statusCode); setResponseBody(response); for (const key in headers) { setResponseHeader(key, headers[key]); } setResponseHeader('access-control-allow-credentials', 'true'); setResponseHeader('access-control-allow-origin', getRequestHeader('origin')); if (data.setEncodingHeader) { setResponseHeader('Content-Encoding', 'gzip'); } returnResponse(); }; // Claim requests to /recaptcha/ if (requestPath === '/recaptcha' || requestPath === '/recaptcha/' || (requestPath === '/recaptcha/api.js' && getRequestQueryParameter('render') === data.siteKey)) { log('Claiming valid request.'); claimRequest(); // Process request for api.js if (requestPath === '/recaptcha/api.js') { log('Processing request for api.js'); if (!templateDataStorage.getItemCopy('__recaptcha_js') || templateDataStorage.getItemCopy('__recaptcha_ts') < oneDayAgo) { log('Fetching api.js from Google servers.'); sendHttpGet('https://www.google.com/recaptcha/api.js?render=' + data.siteKey, (statusCode, headers, body) => { if (statusCode === 200) { templateDataStorage.setItemCopy('__recaptcha_js', body); templateDataStorage.setItemCopy('__recaptcha_headers', headers); templateDataStorage.setItemCopy('__recaptcha_ts', now); } proxyResponse(body, headers, statusCode); }, {timeout: 5000}); } else { log('Loading api.js from template data cache.'); proxyResponse( templateDataStorage.getItemCopy('__recaptcha_js'), templateDataStorage.getItemCopy('__recaptcha_headers'), 200 ); } // Process requests for the reCAPTCHA API } else { log('Verifying user token and fetching bot score.'); const token = getRequestQueryParameter('token'); if (!token) return fail('Missing token from request.'); const postBody = 'secret=' + encode(data.secretKey) + '&response=' + encode(token) + '&remoteip=' + encode(getRemoteAddress()); sendHttpRequest('https://www.google.com/recaptcha/api/siteverify', (statusCode, headers, body) => { const response = JSON.parse(body || '{}'); if (response.success) { if (data.setCookie) { const cookieOptions = { httpOnly: data.httpOnly, domain: 'auto', secure: true }; if (data.maxAge !== '0') cookieOptions['max-age'] = data.maxAge; setCookie( data.cookieName, makeString(response.score), cookieOptions ); } const responseMsg = data.returnResponse ? JSON.stringify({status: 'success', score: response.score}) : 'success'; proxyResponse(responseMsg, {}, 200); } else { fail(JSON.stringify({status: 'failure', errorCodes: data['error-codes']})); } }, {method: 'POST', timeout: 2000, headers: {'content-type': 'application/x-www-form-urlencoded'}}, postBody); } } ___SERVER_PERMISSIONS___ [ { "instance": { "key": { "publicId": "read_request", "versionId": "1" }, "param": [ { "key": "queryParametersAllowed", "value": { "type": 8, "boolean": true } }, { "key": "headerWhitelist", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "forwarded" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "x-forwarded-for" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "origin" } ] } ] } }, { "key": "remoteAddressAllowed", "value": { "type": 8, "boolean": true } }, { "key": "headersAllowed", "value": { "type": 8, "boolean": true } }, { "key": "pathAllowed", "value": { "type": 8, "boolean": true } }, { "key": "queryParameterAccess", "value": { "type": 1, "string": "specific" } }, { "key": "requestAccess", "value": { "type": 1, "string": "specific" } }, { "key": "queryParameterWhitelist", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "queryParameter" } ], "mapValue": [ { "type": 1, "string": "render" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "queryParameter" } ], "mapValue": [ { "type": 1, "string": "token" } ] } ] } }, { "key": "headerAccess", "value": { "type": 1, "string": "specific" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "return_response", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "debug" } } ] }, "isRequired": true }, { "instance": { "key": { "publicId": "set_cookies", "versionId": "1" }, "param": [ { "key": "allowedCookies", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "name" }, { "type": 1, "string": "domain" }, { "type": 1, "string": "path" }, { "type": 1, "string": "secure" }, { "type": 1, "string": "session" } ], "mapValue": [ { "type": 1, "string": "*" }, { "type": 1, "string": "*" }, { "type": 1, "string": "*" }, { "type": 1, "string": "any" }, { "type": 1, "string": "any" } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "access_template_storage", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "access_response", "versionId": "1" }, "param": [ { "key": "writeResponseAccess", "value": { "type": 1, "string": "any" } }, { "key": "writeHeaderAccess", "value": { "type": 1, "string": "specific" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "send_http", "versionId": "1" }, "param": [ { "key": "allowedUrls", "value": { "type": 1, "string": "specific" } }, { "key": "urls", "value": { "type": 2, "listItem": [ { "type": 1, "string": "https://www.google.com/recaptcha/*" } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: Request not claimed with invalid site key code: |- mock('getRequestPath', () => '/recaptcha/api.js?render=invalid_' + mockData.siteKey); runCode(mockData); assertApi('claimRequest').wasNotCalled(); setup: |- const mockData = { siteKey: 'siteKey', secretKey: 'secretKey', setCookie: true, returnResponse: true, cookieName: 'cookieName', maxAge: '1', httpOnly: true }; ___NOTES___ Created on 09/03/2021, 14:26:06