UNPKG

14.2 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright Google LLC All Rights Reserved.
4 *
5 * Use of this source code is governed by an MIT-style license that can be
6 * found in the LICENSE file at https://angular.io/license
7 */
8/**
9 * @fileoverview
10 * A module to facilitate use of a Trusted Types policy within the JIT
11 * compiler. It lazily constructs the Trusted Types policy, providing helper
12 * utilities for promoting strings to Trusted Types. When Trusted Types are not
13 * available, strings are used as a fallback.
14 * @security All use of this module is security-sensitive and should go through
15 * security review.
16 */
17import { global } from '../util';
18/**
19 * The Trusted Types policy, or null if Trusted Types are not
20 * enabled/supported, or undefined if the policy has not been created yet.
21 */
22let policy;
23/**
24 * Returns the Trusted Types policy, or null if Trusted Types are not
25 * enabled/supported. The first call to this function will create the policy.
26 */
27function getPolicy() {
28 if (policy === undefined) {
29 policy = null;
30 if (global.trustedTypes) {
31 try {
32 policy =
33 global.trustedTypes.createPolicy('angular#unsafe-jit', {
34 createScript: (s) => s,
35 });
36 }
37 catch {
38 // trustedTypes.createPolicy throws if called with a name that is
39 // already registered, even in report-only mode. Until the API changes,
40 // catch the error not to break the applications functionally. In such
41 // cases, the code will fall back to using strings.
42 }
43 }
44 }
45 return policy;
46}
47/**
48 * Unsafely promote a string to a TrustedScript, falling back to strings when
49 * Trusted Types are not available.
50 * @security In particular, it must be assured that the provided string will
51 * never cause an XSS vulnerability if used in a context that will be
52 * interpreted and executed as a script by a browser, e.g. when calling eval.
53 */
54function trustedScriptFromString(script) {
55 return getPolicy()?.createScript(script) || script;
56}
57/**
58 * Unsafely call the Function constructor with the given string arguments.
59 * @security This is a security-sensitive function; any use of this function
60 * must go through security review. In particular, it must be assured that it
61 * is only called from the JIT compiler, as use in other code can lead to XSS
62 * vulnerabilities.
63 */
64export function newTrustedFunctionForJIT(...args) {
65 if (!global.trustedTypes) {
66 // In environments that don't support Trusted Types, fall back to the most
67 // straightforward implementation:
68 return new Function(...args);
69 }
70 // Chrome currently does not support passing TrustedScript to the Function
71 // constructor. The following implements the workaround proposed on the page
72 // below, where the Chromium bug is also referenced:
73 // https://github.com/w3c/webappsec-trusted-types/wiki/Trusted-Types-for-function-constructor
74 const fnArgs = args.slice(0, -1).join(',');
75 const fnBody = args[args.length - 1];
76 const body = `(function anonymous(${fnArgs}
77) { ${fnBody}
78})`;
79 // Using eval directly confuses the compiler and prevents this module from
80 // being stripped out of JS binaries even if not used. The global['eval']
81 // indirection fixes that.
82 const fn = global['eval'](trustedScriptFromString(body));
83 if (fn.bind === undefined) {
84 // Workaround for a browser bug that only exists in Chrome 83, where passing
85 // a TrustedScript to eval just returns the TrustedScript back without
86 // evaluating it. In that case, fall back to the most straightforward
87 // implementation:
88 return new Function(...args);
89 }
90 // To completely mimic the behavior of calling "new Function", two more
91 // things need to happen:
92 // 1. Stringifying the resulting function should return its source code
93 fn.toString = () => body;
94 // 2. When calling the resulting function, `this` should refer to `global`
95 return fn.bind(global);
96 // When Trusted Types support in Function constructors is widely available,
97 // the implementation of this function can be simplified to:
98 // return new Function(...args.map(a => trustedScriptFromString(a)));
99}
100//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"output_jit_trusted_types.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/output/output_jit_trusted_types.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH;;;;;;;;GAQG;AAEH,OAAO,EAAC,MAAM,EAAC,MAAM,SAAS,CAAC;AAmC/B;;;GAGG;AACH,IAAI,MAAwC,CAAC;AAE7C;;;GAGG;AACH,SAAS,SAAS;IAChB,IAAI,MAAM,KAAK,SAAS,EAAE;QACxB,MAAM,GAAG,IAAI,CAAC;QACd,IAAI,MAAM,CAAC,YAAY,EAAE;YACvB,IAAI;gBACF,MAAM;oBACD,MAAM,CAAC,YAAyC,CAAC,YAAY,CAAC,oBAAoB,EAAE;wBACnF,YAAY,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC;qBAC/B,CAAC,CAAC;aACR;YAAC,MAAM;gBACN,iEAAiE;gBACjE,uEAAuE;gBACvE,sEAAsE;gBACtE,mDAAmD;aACpD;SACF;KACF;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED;;;;;;GAMG;AACH,SAAS,uBAAuB,CAAC,MAAc;IAC7C,OAAO,SAAS,EAAE,EAAE,YAAY,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC;AACrD,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,wBAAwB,CAAC,GAAG,IAAc;IACxD,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE;QACxB,0EAA0E;QAC1E,kCAAkC;QAClC,OAAO,IAAI,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC;KAC9B;IAED,0EAA0E;IAC1E,4EAA4E;IAC5E,oDAAoD;IACpD,6FAA6F;IAC7F,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC3C,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACrC,MAAM,IAAI,GAAG,uBAAuB,MAAM;MACtC,MAAM;GACT,CAAC;IAEF,0EAA0E;IAC1E,yEAAyE;IACzE,0BAA0B;IAC1B,MAAM,EAAE,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,uBAAuB,CAAC,IAAI,CAAW,CAAa,CAAC;IAC/E,IAAI,EAAE,CAAC,IAAI,KAAK,SAAS,EAAE;QACzB,4EAA4E;QAC5E,sEAAsE;QACtE,qEAAqE;QACrE,kBAAkB;QAClB,OAAO,IAAI,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC;KAC9B;IAED,uEAAuE;IACvE,yBAAyB;IACzB,uEAAuE;IACvE,EAAE,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC;IACzB,0EAA0E;IAC1E,OAAO,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAEvB,2EAA2E;IAC3E,4DAA4D;IAC5D,qEAAqE;AACvE,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 */\n\n/**\n * @fileoverview\n * A module to facilitate use of a Trusted Types policy within the JIT\n * compiler. It lazily constructs the Trusted Types policy, providing helper\n * utilities for promoting strings to Trusted Types. When Trusted Types are not\n * available, strings are used as a fallback.\n * @security All use of this module is security-sensitive and should go through\n * security review.\n */\n\nimport {global} from '../util';\n\n/**\n * While Angular only uses Trusted Types internally for the time being,\n * references to Trusted Types could leak into our core.d.ts, which would force\n * anyone compiling against @angular/core to provide the @types/trusted-types\n * package in their compilation unit.\n *\n * Until https://github.com/microsoft/TypeScript/issues/30024 is resolved, we\n * will keep Angular's public API surface free of references to Trusted Types.\n * For internal and semi-private APIs that need to reference Trusted Types, the\n * minimal type definitions for the Trusted Types API provided by this module\n * should be used instead. They are marked as \"declare\" to prevent them from\n * being renamed by compiler optimization.\n *\n * Adapted from\n * https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/trusted-types/index.d.ts\n * but restricted to the API surface used within Angular.\n */\n\nexport declare interface TrustedScript {\n  __brand__: 'TrustedScript';\n}\n\nexport declare interface TrustedTypePolicyFactory {\n  createPolicy(policyName: string, policyOptions: {\n    createScript?: (input: string) => string,\n  }): TrustedTypePolicy;\n}\n\nexport declare interface TrustedTypePolicy {\n  createScript(input: string): TrustedScript;\n}\n\n\n/**\n * The Trusted Types policy, or null if Trusted Types are not\n * enabled/supported, or undefined if the policy has not been created yet.\n */\nlet policy: TrustedTypePolicy|null|undefined;\n\n/**\n * Returns the Trusted Types policy, or null if Trusted Types are not\n * enabled/supported. The first call to this function will create the policy.\n */\nfunction getPolicy(): TrustedTypePolicy|null {\n  if (policy === undefined) {\n    policy = null;\n    if (global.trustedTypes) {\n      try {\n        policy =\n            (global.trustedTypes as TrustedTypePolicyFactory).createPolicy('angular#unsafe-jit', {\n              createScript: (s: string) => s,\n            });\n      } catch {\n        // trustedTypes.createPolicy throws if called with a name that is\n        // already registered, even in report-only mode. Until the API changes,\n        // catch the error not to break the applications functionally. In such\n        // cases, the code will fall back to using strings.\n      }\n    }\n  }\n  return policy;\n}\n\n/**\n * Unsafely promote a string to a TrustedScript, falling back to strings when\n * Trusted Types are not available.\n * @security In particular, it must be assured that the provided string will\n * never cause an XSS vulnerability if used in a context that will be\n * interpreted and executed as a script by a browser, e.g. when calling eval.\n */\nfunction trustedScriptFromString(script: string): TrustedScript|string {\n  return getPolicy()?.createScript(script) || script;\n}\n\n/**\n * Unsafely call the Function constructor with the given string arguments.\n * @security This is a security-sensitive function; any use of this function\n * must go through security review. In particular, it must be assured that it\n * is only called from the JIT compiler, as use in other code can lead to XSS\n * vulnerabilities.\n */\nexport function newTrustedFunctionForJIT(...args: string[]): Function {\n  if (!global.trustedTypes) {\n    // In environments that don't support Trusted Types, fall back to the most\n    // straightforward implementation:\n    return new Function(...args);\n  }\n\n  // Chrome currently does not support passing TrustedScript to the Function\n  // constructor. The following implements the workaround proposed on the page\n  // below, where the Chromium bug is also referenced:\n  // https://github.com/w3c/webappsec-trusted-types/wiki/Trusted-Types-for-function-constructor\n  const fnArgs = args.slice(0, -1).join(',');\n  const fnBody = args[args.length - 1];\n  const body = `(function anonymous(${fnArgs}\n) { ${fnBody}\n})`;\n\n  // Using eval directly confuses the compiler and prevents this module from\n  // being stripped out of JS binaries even if not used. The global['eval']\n  // indirection fixes that.\n  const fn = global['eval'](trustedScriptFromString(body) as string) as Function;\n  if (fn.bind === undefined) {\n    // Workaround for a browser bug that only exists in Chrome 83, where passing\n    // a TrustedScript to eval just returns the TrustedScript back without\n    // evaluating it. In that case, fall back to the most straightforward\n    // implementation:\n    return new Function(...args);\n  }\n\n  // To completely mimic the behavior of calling \"new Function\", two more\n  // things need to happen:\n  // 1. Stringifying the resulting function should return its source code\n  fn.toString = () => body;\n  // 2. When calling the resulting function, `this` should refer to `global`\n  return fn.bind(global);\n\n  // When Trusted Types support in Function constructors is widely available,\n  // the implementation of this function can be simplified to:\n  // return new Function(...args.map(a => trustedScriptFromString(a)));\n}\n"]}
\No newline at end of file