import java import semmle.code.java.dataflow.DataFlow class ShouldNotBeSunk extends StringLiteral { ShouldNotBeSunk() { this.getValue().matches("%not sunk%") } } class ShouldBeSunk extends StringLiteral { ShouldBeSunk() { this.getValue().matches("%sunk%") and not this instanceof ShouldNotBeSunk } } module Config implements DataFlow::ConfigSig { predicate isSource(DataFlow::Node n) { n.asExpr() instanceof ShouldBeSunk or n.asExpr() instanceof ShouldNotBeSunk } predicate isSink(DataFlow::Node n) { n.asExpr().(Argument).getCall().getCallee().getName() = "sink" } } module Flow = DataFlow::Global; predicate isSunk(StringLiteral sl) { Flow::flowFromExpr(sl) } query predicate shouldBeSunkButIsnt(ShouldBeSunk src) { not isSunk(src) } query predicate shouldntBeSunkButIs(ShouldNotBeSunk src) { isSunk(src) }