Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

JS: Add Angular2 DOM sources #18458

Merged
merged 11 commits into from
Jan 17, 2025
Original file line number Diff line number Diff line change
Expand Up @@ -186,7 +186,7 @@ private boolean isAngularTemplateAttributeName(String name) {

/** Attribute names that look valid in HTML or in one of the template languages we support, like Vue and Angular. */
private static final Pattern VALID_ATTRIBUTE_NAME =
Pattern.compile("[*:@]?\\[?\\(?[\\w:_\\-.]+\\]?\\)?");
Pattern.compile("[*:@]?\\[?\\(?[\\w:_\\-.]+\\)?\\]?");

/** List of HTML attributes whose value is interpreted as JavaScript. */
private static final Pattern JS_ATTRIBUTE =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ public class Main {
* A version identifier that should be updated every time the extractor changes in such a way that
* it may produce different tuples for the same file under the same {@link ExtractorConfig}.
*/
public static final String EXTRACTOR_VERSION = "2024-10-29";
public static final String EXTRACTOR_VERSION = "2025-01-09";

public static final Pattern NEWLINE = Pattern.compile("\n");

Expand Down
46 changes: 28 additions & 18 deletions javascript/ql/lib/semmle/javascript/DOM.qll
Original file line number Diff line number Diff line change
Expand Up @@ -388,23 +388,33 @@ module DOM {
}
}

/**
* Gets a reference to a DOM event.
*/
private DataFlow::SourceNode domEventSource() {
// e.g. <form onSubmit={e => e.target}/>
exists(JsxAttribute attr | attr.getName().matches("on%") |
result = attr.getValue().flow().getABoundFunctionValue(0).getParameter(0)
)
or
// node.addEventListener("submit", e => e.target)
result = domValueRef().getAMethodCall("addEventListener").getABoundCallbackParameter(1, 0)
or
// node.onSubmit = (e => e.target);
exists(DataFlow::PropWrite write | write = domValueRef().getAPropertyWrite() |
write.getPropertyName().matches("on%") and
result = write.getRhs().getAFunctionValue().getParameter(0)
)
/** A data flow node that is a source of DOM events. */
class DomEventSource extends DataFlow::Node instanceof DomEventSource::Range { }

/** Companion module to the `DomEventSource` class. */
module DomEventSource {
/**
* A data flow node that should be considered a source of DOM events.
*/
abstract class Range extends DataFlow::Node { }

private class DefaultRange extends Range {
DefaultRange() {
// e.g. <form onSubmit={e => e.target}/>
exists(JsxAttribute attr | attr.getName().matches("on%") |
this = attr.getValue().flow().getABoundFunctionValue(0).getParameter(0)
)
or
// node.addEventListener("submit", e => e.target)
this = domValueRef().getAMethodCall("addEventListener").getABoundCallbackParameter(1, 0)
or
// node.onSubmit = (e => e.target);
exists(DataFlow::PropWrite write | write = domValueRef().getAPropertyWrite() |
write.getPropertyName().matches("on%") and
this = write.getRhs().getAFunctionValue().getParameter(0)
)
}
}
}

/** Gets a data flow node that refers directly to a value from the DOM. */
Expand All @@ -419,7 +429,7 @@ module DOM {
result = domValueRef().getAMethodCall(["item", "namedItem"])
or
t.startInProp("target") and
result = domEventSource()
result instanceof DomEventSource
or
t.startInProp(DataFlow::PseudoProperties::arrayElement()) and
result = domElementCollection()
Expand Down
21 changes: 21 additions & 0 deletions javascript/ql/lib/semmle/javascript/frameworks/Angular2.qll
Original file line number Diff line number Diff line change
Expand Up @@ -554,4 +554,25 @@ module Angular2 {
this = API::Node::ofType("@angular/core", "ElementRef").getMember("nativeElement").asSource()
}
}

/**
* A source of DOM events originating from the `$event` variable in an event handler installed in an Angular template.
*/
private class DomEventSources extends DOM::DomEventSource::Range {
DomEventSources() {
exists(HTML::Element elm, string attributeName |
elm = any(ComponentClass cls).getATemplateElement() and
// Ignore instantiations of known element (mainly focus on native DOM elements)
not elm = any(ComponentClass cls).getATemplateInstantiation() and
not elm.getName().matches("ng-%") and
this =
elm.getAttributeByName(attributeName)
.getCodeInAttribute()
.(TemplateTopLevel)
.getAVariableUse("$event") and
attributeName.matches("(%)") and // event handler attribute
not attributeName.matches("(ng%)") // exclude NG events which aren't necessarily DOM events
)
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -232,6 +232,15 @@ module XssThroughDom {
)
}
}

/**
* An object containing input values from an Angular form, accessed through an `NgForm` object.
*/
class AngularFormSource extends Source {
AngularFormSource() {
this = API::Node::ofType("@angular/forms", "NgForm").getMember("value").asSource()
}
}
}

/**
Expand Down Expand Up @@ -261,4 +270,16 @@ module XssThroughDom {
this = getSelectionCall(DataFlow::TypeTracker::end()).getAMethodCall("toString")
}
}

/**
* A source of DOM input originating from an Angular two-way data binding.
*/
private class AngularNgModelSource extends Source {
AngularNgModelSource() {
exists(Angular2::ComponentClass component, string fieldName |
fieldName = component.getATemplateElement().getAttributeByName("[(ngModel)]").getValue() and
this = component.getFieldInputNode(fieldName)
)
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
---
category: majorAnalysis
---
* The `js/xss-through-dom` query now recognises sources of DOM input originating from Angular templates.
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
edges
| angular.ts:12:5:12:23 | field: string = ""; | angular.ts:33:24:33:33 | this.field | provenance | |
| angular.ts:29:24:29:33 | form.value | angular.ts:29:24:29:37 | form.value.foo | provenance | |
| forms.js:8:23:8:28 | values | forms.js:9:31:9:36 | values | provenance | |
| forms.js:9:31:9:36 | values | forms.js:9:31:9:40 | values.foo | provenance | |
| forms.js:11:24:11:29 | values | forms.js:12:31:12:36 | values | provenance | |
Expand Down Expand Up @@ -42,6 +44,12 @@ edges
| xss-through-dom.js:154:25:154:27 | msg | xss-through-dom.js:155:27:155:29 | msg | provenance | |
| xss-through-dom.js:159:34:159:52 | $("textarea").val() | xss-through-dom.js:154:25:154:27 | msg | provenance | |
nodes
| angular.ts:12:5:12:23 | field: string = ""; | semmle.label | field: string = ""; |
| angular.ts:16:24:16:41 | event.target.value | semmle.label | event.target.value |
| angular.ts:20:24:20:35 | target.value | semmle.label | target.value |
| angular.ts:29:24:29:33 | form.value | semmle.label | form.value |
| angular.ts:29:24:29:37 | form.value.foo | semmle.label | form.value.foo |
| angular.ts:33:24:33:33 | this.field | semmle.label | this.field |
| forms.js:8:23:8:28 | values | semmle.label | values |
| forms.js:9:31:9:36 | values | semmle.label | values |
| forms.js:9:31:9:40 | values.foo | semmle.label | values.foo |
Expand Down Expand Up @@ -124,6 +132,10 @@ nodes
| xss-through-dom.js:159:34:159:52 | $("textarea").val() | semmle.label | $("textarea").val() |
subpaths
#select
| angular.ts:16:24:16:41 | event.target.value | angular.ts:16:24:16:41 | event.target.value | angular.ts:16:24:16:41 | event.target.value | $@ is reinterpreted as HTML without escaping meta-characters. | angular.ts:16:24:16:41 | event.target.value | DOM text |
| angular.ts:20:24:20:35 | target.value | angular.ts:20:24:20:35 | target.value | angular.ts:20:24:20:35 | target.value | $@ is reinterpreted as HTML without escaping meta-characters. | angular.ts:20:24:20:35 | target.value | DOM text |
| angular.ts:29:24:29:37 | form.value.foo | angular.ts:29:24:29:33 | form.value | angular.ts:29:24:29:37 | form.value.foo | $@ is reinterpreted as HTML without escaping meta-characters. | angular.ts:29:24:29:33 | form.value | DOM text |
| angular.ts:33:24:33:33 | this.field | angular.ts:12:5:12:23 | field: string = ""; | angular.ts:33:24:33:33 | this.field | $@ is reinterpreted as HTML without escaping meta-characters. | angular.ts:12:5:12:23 | field: string = ""; | DOM text |
| forms.js:9:31:9:40 | values.foo | forms.js:8:23:8:28 | values | forms.js:9:31:9:40 | values.foo | $@ is reinterpreted as HTML without escaping meta-characters. | forms.js:8:23:8:28 | values | DOM text |
| forms.js:12:31:12:40 | values.bar | forms.js:11:24:11:29 | values | forms.js:12:31:12:40 | values.bar | $@ is reinterpreted as HTML without escaping meta-characters. | forms.js:11:24:11:29 | values | DOM text |
| forms.js:25:23:25:34 | values.email | forms.js:24:15:24:20 | values | forms.js:25:23:25:34 | values.email | $@ is reinterpreted as HTML without escaping meta-characters. | forms.js:24:15:24:20 | values | DOM text |
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { Component } from "@angular/core";
import { NgForm } from "@angular/forms";

@Component({
template: `
<input type="text" (input)="setInput1($event)"></input>
<input type="text" (input)="setInput2($event.target)"></input>
<input type="text" [(ngModel)]="field"></input>
`
})
export class Foo {
field: string = "";
safeField: string = "";

setInput1(event) {
document.write(event.target.value); // NOT OK
}

setInput2(target) {
document.write(target.value); // NOT OK
}

setOtherInput(e) {
document.write(e.target.value); // OK
document.write(e.value); // OK
}

blah(form: NgForm) {
document.write(form.value.foo); // NOT OK
}

useField() {
document.write(this.field); // NOT OK
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe add a negative example?
Add a field name otherField and test that it's safe.

Similarly a setOtherInput(..) method where the parameter is safe to use.

document.write(this.safeField); // OK
}
}