mirror of
https://github.com/github/codeql.git
synced 2026-04-20 06:24:03 +02:00
167 lines
6.3 KiB
Plaintext
167 lines
6.3 KiB
Plaintext
/**
|
|
* @kind path-problem
|
|
*/
|
|
|
|
import codeql.ruby.AST
|
|
import codeql.ruby.ApiGraphs
|
|
import codeql.ruby.dataflow.FlowSummary
|
|
import codeql.ruby.TaintTracking
|
|
import codeql.ruby.dataflow.internal.FlowSummaryImpl
|
|
import codeql.ruby.frameworks.data.ModelsAsData
|
|
import TestUtilities.InlineFlowTest
|
|
import PathGraph
|
|
|
|
query predicate invalidSpecComponent(SummarizedCallable sc, string s, string c) {
|
|
(sc.propagatesFlow(s, _, _) or sc.propagatesFlow(_, s, _)) and
|
|
Private::External::invalidSpecComponent(s, c)
|
|
}
|
|
|
|
query predicate warning = ModelOutput::getAWarning/0;
|
|
|
|
private class SummarizedCallableIdentity extends SummarizedCallable {
|
|
SummarizedCallableIdentity() { this = "identity" }
|
|
|
|
override MethodCall getACall() { result.getMethodName() = this }
|
|
|
|
override predicate propagatesFlow(string input, string output, boolean preservesValue) {
|
|
input = "Argument[0]" and
|
|
output = "ReturnValue" and
|
|
preservesValue = true
|
|
}
|
|
}
|
|
|
|
private class SummarizedCallableApplyBlock extends SummarizedCallable {
|
|
SummarizedCallableApplyBlock() { this = "apply_block" }
|
|
|
|
override MethodCall getACall() { result.getMethodName() = this }
|
|
|
|
override predicate propagatesFlow(string input, string output, boolean preservesValue) {
|
|
input = "Argument[0]" and
|
|
output = "Argument[block].Parameter[0]" and
|
|
preservesValue = true
|
|
or
|
|
input = "Argument[block].ReturnValue" and
|
|
output = "ReturnValue" and
|
|
preservesValue = true
|
|
}
|
|
}
|
|
|
|
private class SummarizedCallableApplyLambda extends SummarizedCallable {
|
|
SummarizedCallableApplyLambda() { this = "apply_lambda" }
|
|
|
|
override MethodCall getACall() { result.getMethodName() = this }
|
|
|
|
override predicate propagatesFlow(string input, string output, boolean preservesValue) {
|
|
input = "Argument[1]" and
|
|
output = "Argument[0].Parameter[0]" and
|
|
preservesValue = true
|
|
or
|
|
input = "Argument[0].ReturnValue" and
|
|
output = "ReturnValue" and
|
|
preservesValue = true
|
|
}
|
|
}
|
|
|
|
private class StepsFromModel extends ModelInput::SummaryModelCsv {
|
|
override predicate row(string row) {
|
|
row =
|
|
[
|
|
"any;Method[set_value];Argument[0];Argument[self].Field[@value];value",
|
|
"any;Method[get_value];Argument[self].Field[@value];ReturnValue;value",
|
|
"Foo!;Method[firstArg];Argument[0];ReturnValue;taint",
|
|
"Foo!;Method[secondArg];Argument[1];ReturnValue;taint",
|
|
"Foo!;Method[onlyWithoutBlock].WithoutBlock;Argument[0];ReturnValue;taint",
|
|
"Foo!;Method[onlyWithBlock].WithBlock;Argument[0];ReturnValue;taint",
|
|
"Foo!;Method[blockArg].Argument[block].Parameter[0].Method[preserveTaint];Argument[0];ReturnValue;taint",
|
|
"Foo!;Method[namedArg];Argument[foo:];ReturnValue;taint",
|
|
"Foo!;Method[anyArg];Argument[any];ReturnValue;taint",
|
|
"Foo!;Method[anyNamedArg];Argument[any-named];ReturnValue;taint",
|
|
"Foo!;Method[anyPositionFromOne];Argument[1..];ReturnValue;taint",
|
|
"Foo!;Method[intoNamedCallback];Argument[0];Argument[foo:].Parameter[0];taint",
|
|
"Foo!;Method[intoNamedParameter];Argument[0];Argument[0].Parameter[foo:];taint",
|
|
"Foo!;Method[startInNamedCallback].Argument[foo:].Parameter[0].Method[preserveTaint];Argument[0];ReturnValue;taint",
|
|
"Foo!;Method[startInNamedParameter].Argument[0].Parameter[foo:].Method[preserveTaint];Argument[0];ReturnValue;taint",
|
|
"Foo;Method[flowToAnyArg];Argument[0];Argument[any];taint",
|
|
"Foo;Method[flowToSelf];Argument[0];Argument[self];taint",
|
|
"any;Method[matchedByName];Argument[0];ReturnValue;taint",
|
|
"any;Method[matchedByNameRcv];Argument[self];ReturnValue;taint",
|
|
"any;Method[withElementOne];Argument[self].WithElement[1];ReturnValue;value",
|
|
"any;Method[withExactlyElementOne];Argument[self].WithElement[1!];ReturnValue;value",
|
|
"any;Method[withoutElementOne];Argument[self].WithoutElement[1];Argument[self];value",
|
|
"any;Method[withoutExactlyElementOne];Argument[self].WithoutElement[1!];Argument[self];value",
|
|
"any;Method[readElementOne];Argument[self].Element[1];ReturnValue;value",
|
|
"any;Method[readExactlyElementOne];Argument[self].Element[1!];ReturnValue;value",
|
|
"any;Method[withoutElementOneAndTwo];Argument[self].WithoutElement[1].WithoutElement[2].WithElement[any];Argument[self];value",
|
|
]
|
|
}
|
|
}
|
|
|
|
private class TypeFromModel extends ModelInput::TypeModelCsv {
|
|
override predicate row(string row) {
|
|
row =
|
|
[
|
|
"~FooOrBar;Foo;", //
|
|
"~FooOrBar;Bar;", //
|
|
"~FooOrBar;~FooOrBar;Method[next].ReturnValue",
|
|
]
|
|
}
|
|
}
|
|
|
|
private class TypeFromCodeQL extends ModelInput::TypeModel {
|
|
override DataFlow::Node getASource(string type) {
|
|
type = "~FooOrBar" and
|
|
result.getConstantValue().getString() = "magic_string"
|
|
}
|
|
|
|
override API::Node getAnApiNode(string type) {
|
|
type = "~FooOrBar" and
|
|
result = API::getTopLevelMember("Alias").getMember(["Foo", "Bar"])
|
|
}
|
|
}
|
|
|
|
private class InvalidTypeModel extends ModelInput::TypeModelCsv {
|
|
override predicate row(string row) {
|
|
row =
|
|
[
|
|
"TooManyColumns;;Member[Foo].Instance;too;many;columns", //
|
|
"TooFewColumns", //
|
|
"Foo;Foo;Method[foo].Arg[0]", //
|
|
"Foo;Foo;Method[foo].Argument[0-1]", //
|
|
"Foo;Foo;Method[foo].Argument[*]", //
|
|
"Foo;Foo;Method[foo].Argument", //
|
|
"Foo;Foo;Method[foo].Member", //
|
|
]
|
|
}
|
|
}
|
|
|
|
private class SinkFromModel extends ModelInput::SinkModelCsv {
|
|
override predicate row(string row) {
|
|
row =
|
|
[
|
|
"~FooOrBar;Method[method].Argument[0];test-sink", //
|
|
"Foo!;Method[sinkAnyArg].Argument[any];test-sink", //
|
|
"Foo!;Method[sinkAnyNamedArg].Argument[any-named];test-sink", //
|
|
"Foo!;Method[getSinks].ReturnValue.Element[any].Method[mySink].Argument[0];test-sink", //
|
|
"Foo!;Method[arraySink].Argument[0].Element[any];test-sink", //
|
|
"Foo!;Method[secondArrayElementIsSink].Argument[0].Element[1];test-sink", //
|
|
"FuzzyLib!;Fuzzy.Method[fuzzyCall].Argument[0];test-sink"
|
|
]
|
|
}
|
|
}
|
|
|
|
module CustomConfig implements DataFlow::ConfigSig {
|
|
predicate isSource(DataFlow::Node source) { DefaultFlowConfig::isSource(source) }
|
|
|
|
predicate isSink(DataFlow::Node sink) {
|
|
DefaultFlowConfig::isSink(sink)
|
|
or
|
|
sink = ModelOutput::getASinkNode("test-sink").asSink()
|
|
}
|
|
}
|
|
|
|
import FlowTest<CustomConfig, CustomConfig>
|
|
|
|
from PathNode source, PathNode sink
|
|
where flowPath(source, sink)
|
|
select sink, source, sink, "$@", source, source.toString()
|