Skip to content

Commit 9f9857b

Browse files
committed
QL4QL: Add query suggestion use of inline test expectations
1 parent bd9eb7d commit 9f9857b

18 files changed

+146
-28
lines changed

ql/ql/src/codeql_ql/ast/Yaml.qll

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
/**
2+
* Provides classes for working with YAML data.
3+
*
4+
* YAML documents are represented as abstract syntax trees whose nodes
5+
* are either YAML values or alias nodes referring to another YAML value.
6+
*/
7+
8+
private import codeql.yaml.Yaml as LibYaml
9+
10+
private module YamlSig implements LibYaml::InputSig {
11+
import codeql.Locations
12+
13+
class LocatableBase extends @yaml_locatable {
14+
Location getLocation() { yaml_locations(this, result) }
15+
16+
string toString() { none() }
17+
}
18+
19+
class NodeBase extends LocatableBase, @yaml_node {
20+
NodeBase getChildNode(int i) { yaml(result, _, this, i, _, _) }
21+
22+
string getTag() { yaml(this, _, _, _, result, _) }
23+
24+
string getAnchor() { yaml_anchors(this, result) }
25+
26+
override string toString() { yaml(this, _, _, _, _, result) }
27+
}
28+
29+
class ScalarNodeBase extends NodeBase, @yaml_scalar_node {
30+
int getStyle() { yaml_scalars(this, result, _) }
31+
32+
string getValue() { yaml_scalars(this, _, result) }
33+
}
34+
35+
class CollectionNodeBase extends NodeBase, @yaml_collection_node { }
36+
37+
class MappingNodeBase extends CollectionNodeBase, @yaml_mapping_node { }
38+
39+
class SequenceNodeBase extends CollectionNodeBase, @yaml_sequence_node { }
40+
41+
class AliasNodeBase extends NodeBase, @yaml_alias_node {
42+
string getTarget() { yaml_aliases(this, result) }
43+
}
44+
45+
class ParseErrorBase extends LocatableBase, @yaml_error {
46+
string getMessage() { yaml_errors(this, result) }
47+
}
48+
}
49+
50+
import LibYaml::Make<YamlSig>
51+
52+
/** A `.qlref` YAML document. */
53+
class QlRefDocument extends YamlDocument {
54+
QlRefDocument() { this.getFile().getExtension() = "qlref" }
55+
56+
/** Holds if this `.qlref` file uses inline test expectations. */
57+
predicate usesInlineExpectations() {
58+
exists(YamlMapping n, YamlScalar value |
59+
n.getDocument() = this and
60+
value.getValue().matches("%InlineExpectations%")
61+
|
62+
value = n.lookup("postprocess")
63+
or
64+
value = n.lookup("postprocess").(YamlSequence).getElement(_)
65+
)
66+
}
67+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
/**
2+
* @name Query test without inline test expectations
3+
* @description Using inline test expectations is a best practice for writing query tests.
4+
* @kind problem
5+
* @problem.severity warning
6+
* @id ql/qlref-inline-expectations
7+
* @precision high
8+
*/
9+
10+
import ql
11+
import codeql_ql.ast.Yaml
12+
13+
from QlRefDocument f
14+
where not f.usesInlineExpectations()
15+
select f, "Query test does not use inline test expectations."

ql/ql/src/utils/test/InlineExpectationsTest.qll

+1-28
Original file line numberDiff line numberDiff line change
@@ -3,33 +3,6 @@
33
* See `shared/util/codeql/util/test/InlineExpectationsTest.qll`
44
*/
55

6-
private import ql as QL
76
private import codeql.util.test.InlineExpectationsTest
8-
9-
private module Impl implements InlineExpectationsTestSig {
10-
private import codeql_ql.ast.internal.TreeSitter as TS
11-
12-
private newtype TExpectationComment = MkExpectationComment(TS::QL::LineComment comment)
13-
14-
/**
15-
* Represents a line comment.
16-
*/
17-
class ExpectationComment extends TExpectationComment {
18-
TS::QL::LineComment comment;
19-
20-
ExpectationComment() { this = MkExpectationComment(comment) }
21-
22-
/** Returns the contents of the given comment, _without_ the preceding comment marker (`//`). */
23-
string getContents() { result = comment.getValue().suffix(2) }
24-
25-
/** Gets a textual representation of this element. */
26-
string toString() { result = comment.toString() }
27-
28-
/** Gets the location of this comment. */
29-
Location getLocation() { result = comment.getLocation() }
30-
}
31-
32-
class Location = QL::Location;
33-
}
34-
7+
private import internal.InlineExpectationsTestImpl
358
import Make<Impl>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
/**
2+
* @kind test-postprocess
3+
*/
4+
5+
private import ql
6+
private import codeql.util.test.InlineExpectationsTest as T
7+
private import internal.InlineExpectationsTestImpl
8+
import T::TestPostProcessing
9+
import T::TestPostProcessing::Make<Impl, Input>
10+
11+
private module Input implements T::TestPostProcessing::InputSig<Impl> {
12+
string getRelativeUrl(Location location) {
13+
exists(File f, int startline, int startcolumn, int endline, int endcolumn |
14+
location.hasLocationInfo(_, startline, startcolumn, endline, endcolumn) and
15+
f = location.getFile()
16+
|
17+
result =
18+
f.getRelativePath() + ":" + startline + ":" + startcolumn + ":" + endline + ":" + endcolumn
19+
)
20+
}
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
private import ql as QL
2+
private import codeql.util.test.InlineExpectationsTest
3+
4+
module Impl implements InlineExpectationsTestSig {
5+
private import codeql_ql.ast.internal.TreeSitter as TS
6+
7+
private newtype TExpectationComment = MkExpectationComment(TS::QL::LineComment comment)
8+
9+
/**
10+
* Represents a line comment.
11+
*/
12+
class ExpectationComment extends TExpectationComment {
13+
TS::QL::LineComment comment;
14+
15+
ExpectationComment() { this = MkExpectationComment(comment) }
16+
17+
/** Returns the contents of the given comment, _without_ the preceding comment marker (`//`). */
18+
string getContents() { result = comment.getValue().suffix(2) }
19+
20+
/** Gets a textual representation of this element. */
21+
string toString() { result = comment.toString() }
22+
23+
/** Gets the location of this comment. */
24+
Location getLocation() { result = comment.getLocation() }
25+
}
26+
27+
class Location = QL::Location;
28+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
import ql
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
| QlRefInlineExpectations.qlref:1:1:1:40 | queries ... ions.ql | Query test does not use inline test expectations. |
2+
| Test3.qlref:1:1:1:39 | query: ... ists.ql | Query test does not use inline test expectations. |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
queries/style/QlRefInlineExpectations.ql

ql/ql/test/queries/style/QlRefInlineExpectations/Test1.expected

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
query: queries/style/OmittableExists.ql
2+
postprocess: utils/test/InlineExpectationsTestQuery.ql

ql/ql/test/queries/style/QlRefInlineExpectations/Test2.expected

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
query: queries/style/OmittableExists.ql
2+
postprocess:
3+
- utils/test/InlineExpectationsTestQuery.ql

ql/ql/test/queries/style/QlRefInlineExpectations/Test3.expected

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
query: queries/style/OmittableExists.ql

ql/tools/pre-finalize.cmd

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
type NUL && "%CODEQL_DIST%\codeql" database index-files ^
44
--include=**/qlpack.yml ^
5+
--include-extension=.qlref ^
56
--size-limit=5m ^
67
--language yaml ^
78
-- ^

ql/tools/pre-finalize.sh

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ set -eu
44

55
"$CODEQL_DIST/codeql" database index-files \
66
"--include=**/qlpack.yml" \
7+
--include-extension=.qlref \
78
--size-limit=5m \
89
--language yaml \
910
-- \

ql/tools/qltest.cmd

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ IF %ERRORLEVEL% NEQ 0 exit /b %ERRORLEVEL%
1616
type NUL && "%CODEQL_DIST%\codeql.exe" database index-files ^
1717
--prune=**/*.testproj ^
1818
--include-extension=.yml ^
19+
--include-extension=.qlref ^
1920
--size-limit=5m ^
2021
--language=yaml ^
2122
--working-dir=. ^

ql/tools/qltest.sh

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ set -eu
1515
exec "${CODEQL_DIST}/codeql" database index-files \
1616
--prune="**/*.testproj" \
1717
--include-extension=.yml \
18+
--include-extension=.qlref \
1819
--size-limit=5m \
1920
--language=yaml \
2021
--working-dir=.\

0 commit comments

Comments
 (0)