ngx-preboot
Version:
Record server view events and play back to Angular client view
76 lines • 11.7 kB
JavaScript
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
import { APP_BOOTSTRAP_LISTENER, ApplicationRef, Inject, InjectionToken, Optional, PLATFORM_ID } from '@angular/core';
import { DOCUMENT, isPlatformBrowser, isPlatformServer } from '@angular/common';
import { filter, take } from 'rxjs/operators';
import { EventReplayer } from './api/event.replayer';
import { PREBOOT_NONCE } from './common/tokens';
import { getInlineDefinition, getInlineInvocation } from './api/inline.preboot.code';
import { validateOptions } from './api';
const PREBOOT_SCRIPT_CLASS = 'preboot-inline-script';
export const PREBOOT_OPTIONS = new InjectionToken('PrebootOptions');
function createScriptFromCode(doc, nonce, inlineCode) {
const script = doc.createElement('script');
if (nonce) {
script.nonce = nonce;
}
script.className = PREBOOT_SCRIPT_CLASS;
script.textContent = inlineCode;
return script;
}
export function PREBOOT_FACTORY(doc, prebootOpts, nonce, platformId, appRef, eventReplayer) {
return () => {
validateOptions(prebootOpts);
if (isPlatformServer(platformId)) {
const inlineCodeDefinition = getInlineDefinition(prebootOpts);
const scriptWithDefinition = createScriptFromCode(doc, nonce, inlineCodeDefinition);
const inlineCodeInvocation = getInlineInvocation();
const existingScripts = doc.getElementsByClassName(PREBOOT_SCRIPT_CLASS);
// Check to see if preboot scripts are already inlined before adding them
// to the DOM. If they are, update the nonce to be current.
if (existingScripts.length === 0) {
const baseList = [];
const appRootSelectors = baseList.concat(prebootOpts.appRoot);
doc.head.appendChild(scriptWithDefinition);
appRootSelectors
.map((selector) => ({
selector,
appRootElem: doc.querySelector(selector),
}))
.forEach(({ selector, appRootElem }) => {
if (!appRootElem) {
console.log(`No server node found for selector: ${selector}`);
return;
}
const scriptWithInvocation = createScriptFromCode(doc, nonce, inlineCodeInvocation);
appRootElem.insertBefore(scriptWithInvocation, appRootElem.firstChild);
});
}
else if (existingScripts.length > 0 && nonce) {
existingScripts[0].nonce = nonce;
}
}
if (isPlatformBrowser(platformId)) {
const replay = prebootOpts.replay != null ? prebootOpts.replay : true;
if (replay) {
appRef.isStable
.pipe(filter((stable) => stable), take(1))
.subscribe(() => {
eventReplayer.replayAll();
});
}
}
};
}
export const PREBOOT_PROVIDER = {
provide: APP_BOOTSTRAP_LISTENER,
useFactory: PREBOOT_FACTORY,
deps: [DOCUMENT, PREBOOT_OPTIONS, [new Optional(), new Inject(PREBOOT_NONCE)], PLATFORM_ID, ApplicationRef, EventReplayer],
multi: true,
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider.js","sourceRoot":"../../src/lib/","sources":["provider.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AACH,OAAO,EAAE,sBAAsB,EAAE,cAAc,EAAE,MAAM,EAAE,cAAc,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AACtH,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAChF,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AAE9C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EAAE,mBAAmB,EAAE,mBAAmB,EAAE,MAAM,2BAA2B,CAAC;AAErF,OAAO,EAAE,eAAe,EAAE,MAAM,OAAO,CAAC;AAExC,MAAM,oBAAoB,GAAG,uBAAuB,CAAC;AACrD,MAAM,CAAC,MAAM,eAAe,GAAG,IAAI,cAAc,CAAiB,gBAAgB,CAAC,CAAC;AAEpF,SAAS,oBAAoB,CAAC,GAAa,EAAE,KAAoB,EAAE,UAAkB;IACnF,MAAM,MAAM,GAAG,GAAG,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;IAC3C,IAAI,KAAK,EAAE;QACR,MAAc,CAAC,KAAK,GAAG,KAAK,CAAC;KAC/B;IACD,MAAM,CAAC,SAAS,GAAG,oBAAoB,CAAC;IACxC,MAAM,CAAC,WAAW,GAAG,UAAU,CAAC;IAEhC,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,eAAe,CAC7B,GAAa,EACb,WAA2B,EAC3B,KAAoB,EACpB,UAAkB,EAClB,MAAsB,EACtB,aAA4B;IAE5B,OAAO,GAAG,EAAE;QACV,eAAe,CAAC,WAAW,CAAC,CAAC;QAE7B,IAAI,gBAAgB,CAAC,UAAU,CAAC,EAAE;YAChC,MAAM,oBAAoB,GAAG,mBAAmB,CAAC,WAAW,CAAC,CAAC;YAC9D,MAAM,oBAAoB,GAAG,oBAAoB,CAAC,GAAG,EAAE,KAAK,EAAE,oBAAoB,CAAC,CAAC;YACpF,MAAM,oBAAoB,GAAG,mBAAmB,EAAE,CAAC;YAEnD,MAAM,eAAe,GAAG,GAAG,CAAC,sBAAsB,CAAC,oBAAoB,CAAC,CAAC;YAEzE,yEAAyE;YACzE,2DAA2D;YAC3D,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE;gBAChC,MAAM,QAAQ,GAAa,EAAE,CAAC;gBAC9B,MAAM,gBAAgB,GAAG,QAAQ,CAAC,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAC9D,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAoB,CAAC,CAAC;gBAC3C,gBAAgB;qBACb,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;oBAClB,QAAQ;oBACR,WAAW,EAAE,GAAG,CAAC,aAAa,CAAC,QAAQ,CAAC;iBACzC,CAAC,CAAC;qBACF,OAAO,CAAC,CAAC,EAAE,QAAQ,EAAE,WAAW,EAAE,EAAE,EAAE;oBACrC,IAAI,CAAC,WAAW,EAAE;wBAChB,OAAO,CAAC,GAAG,CAAC,sCAAsC,QAAQ,EAAE,CAAC,CAAC;wBAC9D,OAAO;qBACR;oBACD,MAAM,oBAAoB,GAAG,oBAAoB,CAAC,GAAG,EAAE,KAAK,EAAE,oBAAoB,CAAC,CAAC;oBACpF,WAAW,CAAC,YAAY,CAAC,oBAAoB,EAAE,WAAW,CAAC,UAAU,CAAC,CAAC;gBACzE,CAAC,CAAC,CAAC;aACN;iBAAM,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,IAAI,KAAK,EAAE;gBAC7C,eAAe,CAAC,CAAC,CAAS,CAAC,KAAK,GAAG,KAAK,CAAC;aAC3C;SACF;QACD,IAAI,iBAAiB,CAAC,UAAU,CAAC,EAAE;YACjC,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,IAAI,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC;YACtE,IAAI,MAAM,EAAE;gBACV,MAAM,CAAC,QAAQ;qBACZ,IAAI,CACH,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,EAC1B,IAAI,CAAC,CAAC,CAAC,CACR;qBACA,SAAS,CAAC,GAAG,EAAE;oBACd,aAAa,CAAC,SAAS,EAAE,CAAC;gBAC5B,CAAC,CAAC,CAAC;aACN;SACF;IACH,CAAC,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,MAAM,gBAAgB,GAAG;IAC9B,OAAO,EAA8B,sBAAsB;IAC3D,UAAU,EAAE,eAAe;IAC3B,IAAI,EAAE,CAAC,QAAQ,EAAE,eAAe,EAAE,CAAC,IAAI,QAAQ,EAAE,EAAE,IAAI,MAAM,CAAC,aAAa,CAAC,CAAC,EAAE,WAAW,EAAE,cAAc,EAAE,aAAa,CAAC;IAC1H,KAAK,EAAE,IAAI;CACZ,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport { APP_BOOTSTRAP_LISTENER, ApplicationRef, Inject, InjectionToken, Optional, PLATFORM_ID } from '@angular/core';\nimport { DOCUMENT, isPlatformBrowser, isPlatformServer } from '@angular/common';\nimport { filter, take } from 'rxjs/operators';\n\nimport { EventReplayer } from './api/event.replayer';\nimport { PREBOOT_NONCE } from './common/tokens';\nimport { getInlineDefinition, getInlineInvocation } from './api/inline.preboot.code';\nimport { PrebootOptions } from './common/preboot.interfaces';\nimport { validateOptions } from './api';\n\nconst PREBOOT_SCRIPT_CLASS = 'preboot-inline-script';\nexport const PREBOOT_OPTIONS = new InjectionToken<PrebootOptions>('PrebootOptions');\n\nfunction createScriptFromCode(doc: Document, nonce: string | null, inlineCode: string) {\n  const script = doc.createElement('script');\n  if (nonce) {\n    (script as any).nonce = nonce;\n  }\n  script.className = PREBOOT_SCRIPT_CLASS;\n  script.textContent = inlineCode;\n\n  return script;\n}\n\nexport function PREBOOT_FACTORY(\n  doc: Document,\n  prebootOpts: PrebootOptions,\n  nonce: string | null,\n  platformId: Object,\n  appRef: ApplicationRef,\n  eventReplayer: EventReplayer\n) {\n  return () => {\n    validateOptions(prebootOpts);\n\n    if (isPlatformServer(platformId)) {\n      const inlineCodeDefinition = getInlineDefinition(prebootOpts);\n      const scriptWithDefinition = createScriptFromCode(doc, nonce, inlineCodeDefinition);\n      const inlineCodeInvocation = getInlineInvocation();\n\n      const existingScripts = doc.getElementsByClassName(PREBOOT_SCRIPT_CLASS);\n\n      // Check to see if preboot scripts are already inlined before adding them\n      // to the DOM. If they are, update the nonce to be current.\n      if (existingScripts.length === 0) {\n        const baseList: string[] = [];\n        const appRootSelectors = baseList.concat(prebootOpts.appRoot);\n        doc.head.appendChild(scriptWithDefinition);\n        appRootSelectors\n          .map((selector) => ({\n            selector,\n            appRootElem: doc.querySelector(selector),\n          }))\n          .forEach(({ selector, appRootElem }) => {\n            if (!appRootElem) {\n              console.log(`No server node found for selector: ${selector}`);\n              return;\n            }\n            const scriptWithInvocation = createScriptFromCode(doc, nonce, inlineCodeInvocation);\n            appRootElem.insertBefore(scriptWithInvocation, appRootElem.firstChild);\n          });\n      } else if (existingScripts.length > 0 && nonce) {\n        (existingScripts[0] as any).nonce = nonce;\n      }\n    }\n    if (isPlatformBrowser(platformId)) {\n      const replay = prebootOpts.replay != null ? prebootOpts.replay : true;\n      if (replay) {\n        appRef.isStable\n          .pipe(\n            filter((stable) => stable),\n            take(1)\n          )\n          .subscribe(() => {\n            eventReplayer.replayAll();\n          });\n      }\n    }\n  };\n}\n\nexport const PREBOOT_PROVIDER = {\n  provide: <InjectionToken<() => void>>APP_BOOTSTRAP_LISTENER,\n  useFactory: PREBOOT_FACTORY,\n  deps: [DOCUMENT, PREBOOT_OPTIONS, [new Optional(), new Inject(PREBOOT_NONCE)], PLATFORM_ID, ApplicationRef, EventReplayer],\n  multi: true,\n};\n"]}