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 | import { DOCUMENT } from '@angular/common';
|
---|
9 | import { forwardRef, Inject, Injectable, Injector, SecurityContext, ɵ_sanitizeHtml as _sanitizeHtml, ɵ_sanitizeUrl as _sanitizeUrl, ɵallowSanitizationBypassAndThrow as allowSanitizationBypassOrThrow, ɵbypassSanitizationTrustHtml as bypassSanitizationTrustHtml, ɵbypassSanitizationTrustResourceUrl as bypassSanitizationTrustResourceUrl, ɵbypassSanitizationTrustScript as bypassSanitizationTrustScript, ɵbypassSanitizationTrustStyle as bypassSanitizationTrustStyle, ɵbypassSanitizationTrustUrl as bypassSanitizationTrustUrl, ɵgetSanitizationBypassType as getSanitizationBypassType, ɵunwrapSafeValue as unwrapSafeValue } from '@angular/core';
|
---|
10 | import * as i0 from "@angular/core";
|
---|
11 | export { SecurityContext };
|
---|
12 | /**
|
---|
13 | * DomSanitizer helps preventing Cross Site Scripting Security bugs (XSS) by sanitizing
|
---|
14 | * values to be safe to use in the different DOM contexts.
|
---|
15 | *
|
---|
16 | * For example, when binding a URL in an `<a [href]="someValue">` hyperlink, `someValue` will be
|
---|
17 | * sanitized so that an attacker cannot inject e.g. a `javascript:` URL that would execute code on
|
---|
18 | * the website.
|
---|
19 | *
|
---|
20 | * In specific situations, it might be necessary to disable sanitization, for example if the
|
---|
21 | * application genuinely needs to produce a `javascript:` style link with a dynamic value in it.
|
---|
22 | * Users can bypass security by constructing a value with one of the `bypassSecurityTrust...`
|
---|
23 | * methods, and then binding to that value from the template.
|
---|
24 | *
|
---|
25 | * These situations should be very rare, and extraordinary care must be taken to avoid creating a
|
---|
26 | * Cross Site Scripting (XSS) security bug!
|
---|
27 | *
|
---|
28 | * When using `bypassSecurityTrust...`, make sure to call the method as early as possible and as
|
---|
29 | * close as possible to the source of the value, to make it easy to verify no security bug is
|
---|
30 | * created by its use.
|
---|
31 | *
|
---|
32 | * It is not required (and not recommended) to bypass security if the value is safe, e.g. a URL that
|
---|
33 | * does not start with a suspicious protocol, or an HTML snippet that does not contain dangerous
|
---|
34 | * code. The sanitizer leaves safe values intact.
|
---|
35 | *
|
---|
36 | * @security Calling any of the `bypassSecurityTrust...` APIs disables Angular's built-in
|
---|
37 | * sanitization for the value passed in. Carefully check and audit all values and code paths going
|
---|
38 | * into this call. Make sure any user data is appropriately escaped for this security context.
|
---|
39 | * For more detail, see the [Security Guide](https://g.co/ng/security).
|
---|
40 | *
|
---|
41 | * @publicApi
|
---|
42 | */
|
---|
43 | export class DomSanitizer {
|
---|
44 | }
|
---|
45 | DomSanitizer.ɵprov = i0.ɵɵdefineInjectable({ factory: function DomSanitizer_Factory() { return i0.ɵɵinject(DomSanitizerImpl); }, token: DomSanitizer, providedIn: "root" });
|
---|
46 | DomSanitizer.decorators = [
|
---|
47 | { type: Injectable, args: [{ providedIn: 'root', useExisting: forwardRef(() => DomSanitizerImpl) },] }
|
---|
48 | ];
|
---|
49 | export function domSanitizerImplFactory(injector) {
|
---|
50 | return new DomSanitizerImpl(injector.get(DOCUMENT));
|
---|
51 | }
|
---|
52 | export class DomSanitizerImpl extends DomSanitizer {
|
---|
53 | constructor(_doc) {
|
---|
54 | super();
|
---|
55 | this._doc = _doc;
|
---|
56 | }
|
---|
57 | sanitize(ctx, value) {
|
---|
58 | if (value == null)
|
---|
59 | return null;
|
---|
60 | switch (ctx) {
|
---|
61 | case SecurityContext.NONE:
|
---|
62 | return value;
|
---|
63 | case SecurityContext.HTML:
|
---|
64 | if (allowSanitizationBypassOrThrow(value, "HTML" /* Html */)) {
|
---|
65 | return unwrapSafeValue(value);
|
---|
66 | }
|
---|
67 | return _sanitizeHtml(this._doc, String(value)).toString();
|
---|
68 | case SecurityContext.STYLE:
|
---|
69 | if (allowSanitizationBypassOrThrow(value, "Style" /* Style */)) {
|
---|
70 | return unwrapSafeValue(value);
|
---|
71 | }
|
---|
72 | return value;
|
---|
73 | case SecurityContext.SCRIPT:
|
---|
74 | if (allowSanitizationBypassOrThrow(value, "Script" /* Script */)) {
|
---|
75 | return unwrapSafeValue(value);
|
---|
76 | }
|
---|
77 | throw new Error('unsafe value used in a script context');
|
---|
78 | case SecurityContext.URL:
|
---|
79 | const type = getSanitizationBypassType(value);
|
---|
80 | if (allowSanitizationBypassOrThrow(value, "URL" /* Url */)) {
|
---|
81 | return unwrapSafeValue(value);
|
---|
82 | }
|
---|
83 | return _sanitizeUrl(String(value));
|
---|
84 | case SecurityContext.RESOURCE_URL:
|
---|
85 | if (allowSanitizationBypassOrThrow(value, "ResourceURL" /* ResourceUrl */)) {
|
---|
86 | return unwrapSafeValue(value);
|
---|
87 | }
|
---|
88 | throw new Error('unsafe value used in a resource URL context (see https://g.co/ng/security#xss)');
|
---|
89 | default:
|
---|
90 | throw new Error(`Unexpected SecurityContext ${ctx} (see https://g.co/ng/security#xss)`);
|
---|
91 | }
|
---|
92 | }
|
---|
93 | bypassSecurityTrustHtml(value) {
|
---|
94 | return bypassSanitizationTrustHtml(value);
|
---|
95 | }
|
---|
96 | bypassSecurityTrustStyle(value) {
|
---|
97 | return bypassSanitizationTrustStyle(value);
|
---|
98 | }
|
---|
99 | bypassSecurityTrustScript(value) {
|
---|
100 | return bypassSanitizationTrustScript(value);
|
---|
101 | }
|
---|
102 | bypassSecurityTrustUrl(value) {
|
---|
103 | return bypassSanitizationTrustUrl(value);
|
---|
104 | }
|
---|
105 | bypassSecurityTrustResourceUrl(value) {
|
---|
106 | return bypassSanitizationTrustResourceUrl(value);
|
---|
107 | }
|
---|
108 | }
|
---|
109 | DomSanitizerImpl.ɵprov = i0.ɵɵdefineInjectable({ factory: function DomSanitizerImpl_Factory() { return domSanitizerImplFactory(i0.ɵɵinject(i0.INJECTOR)); }, token: DomSanitizerImpl, providedIn: "root" });
|
---|
110 | DomSanitizerImpl.decorators = [
|
---|
111 | { type: Injectable, args: [{ providedIn: 'root', useFactory: domSanitizerImplFactory, deps: [Injector] },] }
|
---|
112 | ];
|
---|
113 | DomSanitizerImpl.ctorParameters = () => [
|
---|
114 | { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT,] }] }
|
---|
115 | ];
|
---|
116 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dom_sanitization_service.js","sourceRoot":"","sources":["../../../../../../../packages/platform-browser/src/security/dom_sanitization_service.ts"],"names":[],"mappings":"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","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\nimport {DOCUMENT} from '@angular/common';\nimport {forwardRef, Inject, Injectable, Injector, Sanitizer, SecurityContext, ɵ_sanitizeHtml as _sanitizeHtml, ɵ_sanitizeUrl as _sanitizeUrl, ɵallowSanitizationBypassAndThrow as allowSanitizationBypassOrThrow, ɵbypassSanitizationTrustHtml as bypassSanitizationTrustHtml, ɵbypassSanitizationTrustResourceUrl as bypassSanitizationTrustResourceUrl, ɵbypassSanitizationTrustScript as bypassSanitizationTrustScript, ɵbypassSanitizationTrustStyle as bypassSanitizationTrustStyle, ɵbypassSanitizationTrustUrl as bypassSanitizationTrustUrl, ɵBypassType as BypassType, ɵgetSanitizationBypassType as getSanitizationBypassType, ɵunwrapSafeValue as unwrapSafeValue} from '@angular/core';\n\nexport {SecurityContext};\n\n\n\n/**\n * Marker interface for a value that's safe to use in a particular context.\n *\n * @publicApi\n */\nexport interface SafeValue {}\n\n/**\n * Marker interface for a value that's safe to use as HTML.\n *\n * @publicApi\n */\nexport interface SafeHtml extends SafeValue {}\n\n/**\n * Marker interface for a value that's safe to use as style (CSS).\n *\n * @publicApi\n */\nexport interface SafeStyle extends SafeValue {}\n\n/**\n * Marker interface for a value that's safe to use as JavaScript.\n *\n * @publicApi\n */\nexport interface SafeScript extends SafeValue {}\n\n/**\n * Marker interface for a value that's safe to use as a URL linking to a document.\n *\n * @publicApi\n */\nexport interface SafeUrl extends SafeValue {}\n\n/**\n * Marker interface for a value that's safe to use as a URL to load executable code from.\n *\n * @publicApi\n */\nexport interface SafeResourceUrl extends SafeValue {}\n\n/**\n * DomSanitizer helps preventing Cross Site Scripting Security bugs (XSS) by sanitizing\n * values to be safe to use in the different DOM contexts.\n *\n * For example, when binding a URL in an `<a [href]=\"someValue\">` hyperlink, `someValue` will be\n * sanitized so that an attacker cannot inject e.g. a `javascript:` URL that would execute code on\n * the website.\n *\n * In specific situations, it might be necessary to disable sanitization, for example if the\n * application genuinely needs to produce a `javascript:` style link with a dynamic value in it.\n * Users can bypass security by constructing a value with one of the `bypassSecurityTrust...`\n * methods, and then binding to that value from the template.\n *\n * These situations should be very rare, and extraordinary care must be taken to avoid creating a\n * Cross Site Scripting (XSS) security bug!\n *\n * When using `bypassSecurityTrust...`, make sure to call the method as early as possible and as\n * close as possible to the source of the value, to make it easy to verify no security bug is\n * created by its use.\n *\n * It is not required (and not recommended) to bypass security if the value is safe, e.g. a URL that\n * does not start with a suspicious protocol, or an HTML snippet that does not contain dangerous\n * code. The sanitizer leaves safe values intact.\n *\n * @security Calling any of the `bypassSecurityTrust...` APIs disables Angular's built-in\n * sanitization for the value passed in. Carefully check and audit all values and code paths going\n * into this call. Make sure any user data is appropriately escaped for this security context.\n * For more detail, see the [Security Guide](https://g.co/ng/security).\n *\n * @publicApi\n */\n@Injectable({providedIn: 'root', useExisting: forwardRef(() => DomSanitizerImpl)})\nexport abstract class DomSanitizer implements Sanitizer {\n  /**\n   * Sanitizes a value for use in the given SecurityContext.\n   *\n   * If value is trusted for the context, this method will unwrap the contained safe value and use\n   * it directly. Otherwise, value will be sanitized to be safe in the given context, for example\n   * by replacing URLs that have an unsafe protocol part (such as `javascript:`). The implementation\n   * is responsible to make sure that the value can definitely be safely used in the given context.\n   */\n  abstract sanitize(context: SecurityContext, value: SafeValue|string|null): string|null;\n\n  /**\n   * Bypass security and trust the given value to be safe HTML. Only use this when the bound HTML\n   * is unsafe (e.g. contains `<script>` tags) and the code should be executed. The sanitizer will\n   * leave safe HTML intact, so in most situations this method should not be used.\n   *\n   * **WARNING:** calling this method with untrusted user data exposes your application to XSS\n   * security risks!\n   */\n  abstract bypassSecurityTrustHtml(value: string): SafeHtml;\n\n  /**\n   * Bypass security and trust the given value to be safe style value (CSS).\n   *\n   * **WARNING:** calling this method with untrusted user data exposes your application to XSS\n   * security risks!\n   */\n  abstract bypassSecurityTrustStyle(value: string): SafeStyle;\n\n  /**\n   * Bypass security and trust the given value to be safe JavaScript.\n   *\n   * **WARNING:** calling this method with untrusted user data exposes your application to XSS\n   * security risks!\n   */\n  abstract bypassSecurityTrustScript(value: string): SafeScript;\n\n  /**\n   * Bypass security and trust the given value to be a safe style URL, i.e. a value that can be used\n   * in hyperlinks or `<img src>`.\n   *\n   * **WARNING:** calling this method with untrusted user data exposes your application to XSS\n   * security risks!\n   */\n  abstract bypassSecurityTrustUrl(value: string): SafeUrl;\n\n  /**\n   * Bypass security and trust the given value to be a safe resource URL, i.e. a location that may\n   * be used to load executable code from, like `<script src>`, or `<iframe src>`.\n   *\n   * **WARNING:** calling this method with untrusted user data exposes your application to XSS\n   * security risks!\n   */\n  abstract bypassSecurityTrustResourceUrl(value: string): SafeResourceUrl;\n}\n\nexport function domSanitizerImplFactory(injector: Injector) {\n  return new DomSanitizerImpl(injector.get(DOCUMENT));\n}\n\n@Injectable({providedIn: 'root', useFactory: domSanitizerImplFactory, deps: [Injector]})\nexport class DomSanitizerImpl extends DomSanitizer {\n  constructor(@Inject(DOCUMENT) private _doc: any) {\n    super();\n  }\n\n  override sanitize(ctx: SecurityContext, value: SafeValue|string|null): string|null {\n    if (value == null) return null;\n    switch (ctx) {\n      case SecurityContext.NONE:\n        return value as string;\n      case SecurityContext.HTML:\n        if (allowSanitizationBypassOrThrow(value, BypassType.Html)) {\n          return unwrapSafeValue(value);\n        }\n        return _sanitizeHtml(this._doc, String(value)).toString();\n      case SecurityContext.STYLE:\n        if (allowSanitizationBypassOrThrow(value, BypassType.Style)) {\n          return unwrapSafeValue(value);\n        }\n        return value as string;\n      case SecurityContext.SCRIPT:\n        if (allowSanitizationBypassOrThrow(value, BypassType.Script)) {\n          return unwrapSafeValue(value);\n        }\n        throw new Error('unsafe value used in a script context');\n      case SecurityContext.URL:\n        const type = getSanitizationBypassType(value);\n        if (allowSanitizationBypassOrThrow(value, BypassType.Url)) {\n          return unwrapSafeValue(value);\n        }\n        return _sanitizeUrl(String(value));\n      case SecurityContext.RESOURCE_URL:\n        if (allowSanitizationBypassOrThrow(value, BypassType.ResourceUrl)) {\n          return unwrapSafeValue(value);\n        }\n        throw new Error(\n            'unsafe value used in a resource URL context (see https://g.co/ng/security#xss)');\n      default:\n        throw new Error(`Unexpected SecurityContext ${ctx} (see https://g.co/ng/security#xss)`);\n    }\n  }\n\n  override bypassSecurityTrustHtml(value: string): SafeHtml {\n    return bypassSanitizationTrustHtml(value);\n  }\n  override bypassSecurityTrustStyle(value: string): SafeStyle {\n    return bypassSanitizationTrustStyle(value);\n  }\n  override bypassSecurityTrustScript(value: string): SafeScript {\n    return bypassSanitizationTrustScript(value);\n  }\n  override bypassSecurityTrustUrl(value: string): SafeUrl {\n    return bypassSanitizationTrustUrl(value);\n  }\n  override bypassSecurityTrustResourceUrl(value: string): SafeResourceUrl {\n    return bypassSanitizationTrustResourceUrl(value);\n  }\n}\n"]} |
---|