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

Rust: Handle path attributes in path resolution #19216

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions rust/ql/lib/codeql/files/FileSystem.qll
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,14 @@ class Container = Impl::Container;

class Folder = Impl::Folder;

/** Holds if `relativePath` needs to be appended to `f`. */
signature predicate appendSig(Folder f, string relativePath);

/** Provides the `append` predicate for appending a relative path onto a folder. */
module FolderAppend<appendSig/2 app> {
import Impl::FolderAppend<app/2>
}

/** A file. */
class File extends Container, Impl::File {
/** Holds if this file was extracted from ordinary source code. */
Expand Down
27 changes: 21 additions & 6 deletions rust/ql/lib/codeql/rust/internal/PathResolution.qll
Original file line number Diff line number Diff line change
Expand Up @@ -654,6 +654,11 @@ private predicate fileModule(SourceFile f, string name, Folder folder) {
)
}

private Meta getPathAttrMeta(Module m) {
result = m.getAnAttr().getMeta() and
result.getPath().getText() = "path"
}

/**
* Holds if `m` is a `mod name;` module declaration, where the corresponding
* module file needs to be looked up in `lookup` or one of its descandants.
Expand All @@ -662,12 +667,7 @@ private predicate modImport0(Module m, string name, Folder lookup) {
exists(File f, Folder parent, string fileName |
f = m.getFile() and
not m.hasItemList() and
// TODO: handle
// ```
// #[path = "foo.rs"]
// mod bar;
// ```
not m.getAnAttr().getMeta().getPath().getText() = "path" and
not exists(getPathAttrMeta(m)) and
name = m.getName().getText() and
parent = f.getParentContainer() and
fileName = f.getStem()
Expand Down Expand Up @@ -716,6 +716,16 @@ private predicate modImportNestedLookup(Module m, ModuleItemNode ancestor, Folde
)
}

private predicate pathAttrImport(Folder f, Module m, string relativePath) {
exists(Meta meta |
f = m.getFile().getParentContainer() and
meta = getPathAttrMeta(m) and
relativePath = meta.getExpr().(LiteralExpr).getTextValue().regexpCapture("\"(.+)\"", 1)
)
}

private predicate append(Folder f, string relativePath) { pathAttrImport(f, _, relativePath) }

/** Holds if `m` is a `mod name;` item importing file `f`. */
private predicate fileImport(Module m, SourceFile f) {
exists(string name, Folder parent |
Expand All @@ -729,6 +739,11 @@ private predicate fileImport(Module m, SourceFile f) {
// `m` is inside a nested module
modImportNestedLookup(m, m, parent)
)
or
exists(Folder folder, string relativePath |
pathAttrImport(folder, m, relativePath) and
f.getFile() = FolderAppend<append/2>::append(folder, relativePath)
)
}

/**
Expand Down
5 changes: 5 additions & 0 deletions rust/ql/test/library-tests/path-resolution/my2/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,3 +10,8 @@ pub use nested2::nested5::*; // $ item=I114
pub use nested2::nested7::nested8::{self}; // $ item=I118

pub mod my3;

#[path = "renamed.rs"]
mod mymod;

use mymod::f; // $ item=I1001
1 change: 1 addition & 0 deletions rust/ql/test/library-tests/path-resolution/my2/renamed.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
pub fn f() {} // I1001
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ mod
| main.rs:505:5:520:5 | mod m33 |
| my2/mod.rs:1:1:1:16 | mod nested2 |
| my2/mod.rs:12:1:12:12 | mod my3 |
| my2/mod.rs:14:1:15:10 | mod mymod |
| my2/nested2.rs:1:1:11:1 | mod nested3 |
| my2/nested2.rs:2:5:10:5 | mod nested4 |
| my2/nested2.rs:13:1:19:1 | mod nested5 |
Expand Down Expand Up @@ -299,12 +300,14 @@ resolvePath
| my2/mod.rs:10:9:10:24 | ...::nested7 | my2/nested2.rs:21:1:27:1 | mod nested7 |
| my2/mod.rs:10:9:10:33 | ...::nested8 | my2/nested2.rs:22:5:26:5 | mod nested8 |
| my2/mod.rs:10:37:10:40 | self | my2/nested2.rs:22:5:26:5 | mod nested8 |
| my2/mod.rs:17:5:17:9 | mymod | my2/mod.rs:14:1:15:10 | mod mymod |
| my2/mod.rs:17:5:17:12 | ...::f | my2/renamed.rs:1:1:1:13 | fn f |
| my2/my3/mod.rs:3:5:3:5 | g | my2/mod.rs:3:1:6:1 | fn g |
| my2/my3/mod.rs:4:5:4:5 | h | main.rs:50:1:69:1 | fn h |
| my2/my3/mod.rs:7:5:7:9 | super | my2/mod.rs:1:1:12:13 | SourceFile |
| my2/my3/mod.rs:7:5:7:9 | super | my2/mod.rs:1:1:17:30 | SourceFile |
| my2/my3/mod.rs:7:5:7:16 | ...::super | main.rs:1:1:550:2 | SourceFile |
| my2/my3/mod.rs:7:5:7:19 | ...::h | main.rs:50:1:69:1 | fn h |
| my2/my3/mod.rs:8:5:8:9 | super | my2/mod.rs:1:1:12:13 | SourceFile |
| my2/my3/mod.rs:8:5:8:9 | super | my2/mod.rs:1:1:17:30 | SourceFile |
| my2/my3/mod.rs:8:5:8:12 | ...::g | my2/mod.rs:3:1:6:1 | fn g |
| my.rs:3:5:3:10 | nested | my.rs:1:1:1:15 | mod nested |
| my.rs:3:5:3:13 | ...::g | my/nested.rs:19:1:22:1 | fn g |
Expand Down
44 changes: 44 additions & 0 deletions shared/util/codeql/util/FileSystem.qll
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,50 @@ module Make<InputSig Input> {
/** Gets the URL of this file. */
override string getURL() { result = "file://" + this.getAbsolutePath() + ":0:0:0:0" }
}

/** Holds if `relativePath` needs to be appended to `f`. */
signature predicate appendSig(Folder f, string relativePath);

/** Provides the `append` predicate for appending a relative path onto a folder. */
module FolderAppend<appendSig/2 app> {
pragma[nomagic]
private string getComponent(string relativePath, int i) {
app(_, relativePath) and
result = relativePath.replaceAll("\\", "/").regexpFind("[^/]+", i, _)
}

pragma[nomagic]
private Container appendStep(Folder f, string relativePath, int i) {
i = -1 and
app(f, relativePath) and
result = f
or
exists(Container mid, string comp |
mid = appendStep(f, relativePath, i - 1) and
comp = getComponent(relativePath, i) and
if comp = ".."
then result = mid.getParentContainer()
else
if comp = "."
then result = mid
else (
result = mid.getAChildContainer() and
result.getBaseName() = comp
)
)
}

/**
* Gets the file or folder obtained by appending `relativePath` onto `f`.
*/
pragma[nomagic]
Container append(Folder f, string relativePath) {
exists(int components |
components = (-1).maximum(max(int comp | exists(getComponent(relativePath, comp)) | comp)) and
result = appendStep(f, relativePath, components)
)
}
}
}

/** A file. */
Expand Down
Loading