import * as crypto from "crypto"; import { ftlValuesGlobalName } from "./ftlValuesGlobalName"; export function replaceImportsFromStaticInJsCode(params: { jsCode: string; urlOrigin: undefined | string }): { fixedJsCode: string } { /* NOTE: When we have urlOrigin defined it means that we are building with --external-assets so we have to make sur that the fixed js code will run inside and outside keycloak. When urlOrigin isn't defined we can assume the fixedJsCode will always run in keycloak context. */ const { jsCode, urlOrigin } = params; const fixedJsCode = jsCode .replace( /([a-zA-Z]+)\.([a-zA-Z]+)=function\(([a-zA-Z]+)\){return"static\/js\/"/g, (...[, n, u, e]) => ` ${n}[(function(){ ${ urlOrigin === undefined ? ` Object.defineProperty(${n}, "p", { get: function() { return window.${ftlValuesGlobalName}.url.resourcesPath; }, set: function (){} }); ` : ` var p= ""; Object.defineProperty(${n}, "p", { get: function() { return ("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + p; }, set: function (value){ p = value;} }); ` } return "${u}"; })()] = function(${e}) { return "${urlOrigin === undefined ? "/build/" : ""}static/js/"`, ) .replace(/([a-zA-Z]+\.[a-zA-Z]+)\+"static\//g, (...[, group]) => urlOrigin === undefined ? `window.${ftlValuesGlobalName}.url.resourcesPath + "/build/static/` : `("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + ${group} + "static/`, ) //TODO: Write a test case for this .replace(/".chunk.css",([a-zA-Z])+=([a-zA-Z]+\.[a-zA-Z]+)\+([a-zA-Z]+),/, (...[, group1, group2, group3]) => urlOrigin === undefined ? `".chunk.css",${group1} = window.${ftlValuesGlobalName}.url.resourcesPath + "/build/" + ${group3},` : `".chunk.css",${group1} = ("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + ${group2} + ${group3},`, ); return { fixedJsCode }; } export function replaceImportsInInlineCssCode(params: { cssCode: string; urlPathname: string; urlOrigin: undefined | string }): { fixedCssCode: string; } { const { cssCode, urlPathname, urlOrigin } = params; const fixedCssCode = cssCode.replace( urlPathname === "/" ? /url\(["']?\/([^/][^)"']+)["']?\)/g : new RegExp(`url\\(["']?${urlPathname}([^)"']+)["']?\\)`, "g"), (...[, group]) => `url(${urlOrigin === undefined ? "${url.resourcesPath}/build/" + group : params.urlOrigin + urlPathname + group})`, ); return { fixedCssCode }; } export function replaceImportsInCssCode(params: { cssCode: string }): { fixedCssCode: string; cssGlobalsToDefine: Record; } { const { cssCode } = params; const cssGlobalsToDefine: Record = {}; new Set(cssCode.match(/url\(["']?\/[^/][^)"']+["']?\)[^;}]*/g) ?? []).forEach( match => (cssGlobalsToDefine["url" + crypto.createHash("sha256").update(match).digest("hex").substring(0, 15)] = match), ); let fixedCssCode = cssCode; Object.keys(cssGlobalsToDefine).forEach( cssVariableName => //NOTE: split/join pattern ~ replace all (fixedCssCode = fixedCssCode.split(cssGlobalsToDefine[cssVariableName]).join(`var(--${cssVariableName})`)), ); return { fixedCssCode, cssGlobalsToDefine }; } export function generateCssCodeToDefineGlobals(params: { cssGlobalsToDefine: Record; urlPathname: string }): { cssCodeToPrependInHead: string; } { const { cssGlobalsToDefine, urlPathname } = params; return { "cssCodeToPrependInHead": [ ":root {", ...Object.keys(cssGlobalsToDefine) .map(cssVariableName => [ `--${cssVariableName}:`, cssGlobalsToDefine[cssVariableName].replace( new RegExp(`url\\(${urlPathname.replace(/\//g, "\\/")}`, "g"), "url(${url.resourcesPath}/build/", ), ].join(" "), ) .map(line => ` ${line};`), "}", ].join("\n"), }; }