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

[red-knot] Respect typeshed's VERSIONS file when resolving stdlib modules #12141

Merged
merged 62 commits into from
Jul 5, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
7380511
Everything finally compiles
AlexWaygood Jun 29, 2024
75a140a
Move `ModuleName` into a new file
AlexWaygood Jun 29, 2024
8cd568c
Specify that the `Db` must allow typeshed versions to be queried
AlexWaygood Jun 29, 2024
eb6a377
The supported Python version is set at the same time as the search paths
AlexWaygood Jun 29, 2024
87419ce
`TypeshedVersions` returns a three-variant enum
AlexWaygood Jul 1, 2024
5d30cf8
Silly implementation for the `todo!()`s
AlexWaygood Jul 1, 2024
5c1f4e8
Implementation that actually looks at the target version we set
AlexWaygood Jul 1, 2024
1a82d21
Address easy review comments
AlexWaygood Jul 1, 2024
d91626c
Get rid of a lot of boilerplate
AlexWaygood Jul 2, 2024
1dc038e
Cleanup
AlexWaygood Jul 2, 2024
9e1ae5e
Merge branch 'main' into custom-typeshed-versions
AlexWaygood Jul 2, 2024
583e503
Make some public methods private
AlexWaygood Jul 2, 2024
906d801
Inline some simple helper methods
AlexWaygood Jul 2, 2024
ed088e3
Use Salsa queries to load `TypeshedVersions`
AlexWaygood Jul 2, 2024
dbe8826
fix test
AlexWaygood Jul 2, 2024
0a0a334
Address easy review comments
AlexWaygood Jul 3, 2024
ad2e987
Cleanup VERSIONS-parsing and add a todo for better error handling
AlexWaygood Jul 3, 2024
78c56f7
Move more logic internal to `path.rs`
AlexWaygood Jul 3, 2024
6723c7f
Get rid of the inner structs
AlexWaygood Jul 3, 2024
9ba9ac1
Bye bye `DoubleEndedIterator`
AlexWaygood Jul 3, 2024
9c556ba
Add tests for some helpers
AlexWaygood Jul 3, 2024
9c672f8
Simplify `ModulePartIterator`
AlexWaygood Jul 3, 2024
ad3bd7a
more tests
AlexWaygood Jul 3, 2024
21e60fe
fix Windows?
AlexWaygood Jul 3, 2024
76f5b3f
fix Windows???
AlexWaygood Jul 3, 2024
116f8f9
Add tests for `relativize_path`
AlexWaygood Jul 4, 2024
5ecc0f3
Add tests for `is_directory()` and `is_regular_package()`
AlexWaygood Jul 4, 2024
dde1393
Merge branch 'main' into custom-typeshed-versions
AlexWaygood Jul 4, 2024
c541dd0
Delete cruft from older tests
AlexWaygood Jul 4, 2024
2f0d349
Add tests for the resolver as a whole with `VERSIONS`
AlexWaygood Jul 4, 2024
9bbe5a8
More unit tests in `path.rs`
AlexWaygood Jul 4, 2024
e505b68
Merge branch 'main' into custom-typeshed-versions
AlexWaygood Jul 4, 2024
36b9288
Reduce use of `is_none()` in tests
AlexWaygood Jul 5, 2024
2f0d562
Add more todos for `into_ordered_search_paths()`
AlexWaygood Jul 5, 2024
86ee754
Merge branch 'main' into custom-typeshed-versions
AlexWaygood Jul 5, 2024
ea58e3a
Typeshed versions are looked up from the cache once per module resolu…
AlexWaygood Jul 5, 2024
8dda2e3
Move test-only methods into `test` submodules
AlexWaygood Jul 5, 2024
7efd84c
Elide some lifetimes
AlexWaygood Jul 5, 2024
553ce67
Delete and streamline some trait implementations
AlexWaygood Jul 5, 2024
9fda47a
Remove unnecessary `*`s
AlexWaygood Jul 5, 2024
b819680
Make `ModuleResolutionPathBuf::push()` assert invariants on release b…
AlexWaygood Jul 5, 2024
7eebb75
Reduce nesting in `ModuleName::from_components()`
AlexWaygood Jul 5, 2024
470f0c6
Document `TypeshedVersionsQueryResult` variants
AlexWaygood Jul 5, 2024
d4c11ee
Docs for `query_module()`
AlexWaygood Jul 5, 2024
b0c49c2
Add a `create_resolver_test()` helper
AlexWaygood Jul 5, 2024
55538bb
Fix `Debug` implementations
AlexWaygood Jul 5, 2024
a65c4dc
Get rid of `stdlib_path_test_case()` helper
AlexWaygood Jul 5, 2024
884c57b
Share more code between some methods in `path.rs`
AlexWaygood Jul 5, 2024
13e741e
Split up some tests in `resolver.rs`
AlexWaygood Jul 5, 2024
70cd43b
Reduce diff in `resolver.rs` tests
AlexWaygood Jul 5, 2024
82a9974
Split up a big test in `versions.rs`
AlexWaygood Jul 5, 2024
5f5ad70
Simplify tests in `path.rs`
AlexWaygood Jul 5, 2024
640d2fa
Small cleanup in `path.rs`
AlexWaygood Jul 5, 2024
4c7a106
Getting the target version is no longer a separate Salsa query
AlexWaygood Jul 5, 2024
777ec7d
`db` is now always the first argument
AlexWaygood Jul 5, 2024
f8e900e
Get rid of unneeded new dependency
AlexWaygood Jul 5, 2024
8fc3a7d
So many lifetimes
AlexWaygood Jul 5, 2024
b848415
Rename some things
AlexWaygood Jul 5, 2024
daf64b4
Audit docs in `resolver.rs`
AlexWaygood Jul 5, 2024
47c3467
Make assertions in `push()` make a little more sense
AlexWaygood Jul 5, 2024
401779b
Encapsulate all state passed around in a single struct
AlexWaygood Jul 5, 2024
38128e0
Add a TODO for better distinctions between absolute and relative paths
AlexWaygood Jul 5, 2024
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
Prev Previous commit
Next Next commit
Remove unnecessary *s
  • Loading branch information
AlexWaygood committed Jul 5, 2024
commit 9fda47ad45bf0f8e9cde05fb4ee4ecad02096bb5
16 changes: 7 additions & 9 deletions crates/red_knot_module_resolver/src/db.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,7 @@ pub(crate) mod tests {

use salsa::DebugWithDb;

use ruff_db::file_system::{
FileSystem, FileSystemPath, FileSystemPathBuf, MemoryFileSystem, OsFileSystem,
};
use ruff_db::file_system::{FileSystem, FileSystemPathBuf, MemoryFileSystem, OsFileSystem};
use ruff_db::vfs::Vfs;

use crate::resolver::{set_module_resolution_settings, ModuleResolutionSettings};
Expand Down Expand Up @@ -221,15 +219,15 @@ pub(crate) mod tests {

let db = TestDb::new();

let src = FileSystemPath::new("src").to_path_buf();
let site_packages = FileSystemPath::new("site_packages").to_path_buf();
let custom_typeshed = FileSystemPath::new("typeshed").to_path_buf();
let src = FileSystemPathBuf::from("src");
let site_packages = FileSystemPathBuf::from("site_packages");
let custom_typeshed = FileSystemPathBuf::from("typeshed");

let fs = db.memory_file_system();

fs.create_directory_all(&*src)?;
fs.create_directory_all(&*site_packages)?;
fs.create_directory_all(&*custom_typeshed)?;
fs.create_directory_all(&src)?;
fs.create_directory_all(&site_packages)?;
fs.create_directory_all(&custom_typeshed)?;
fs.write_file(custom_typeshed.join("stdlib/VERSIONS"), VERSIONS_DATA)?;

// Regular package on py38+
Expand Down
12 changes: 12 additions & 0 deletions crates/red_knot_module_resolver/src/path.rs
Original file line number Diff line number Diff line change
Expand Up @@ -490,6 +490,18 @@ impl PartialEq<ModuleResolutionPathRef<'_>> for FileSystemPath {
}
}

impl PartialEq<FileSystemPathBuf> for ModuleResolutionPathRef<'_> {
fn eq(&self, other: &FileSystemPathBuf) -> bool {
self == &**other
}
}

impl PartialEq<ModuleResolutionPathRef<'_>> for FileSystemPathBuf {
fn eq(&self, other: &ModuleResolutionPathRef<'_>) -> bool {
&**self == other
}
}

#[cfg(test)]
mod tests {
use insta::assert_debug_snapshot;
Expand Down
82 changes: 41 additions & 41 deletions crates/red_knot_module_resolver/src/resolver.rs
Original file line number Diff line number Diff line change
Expand Up @@ -386,7 +386,7 @@ mod tests {
let foo_module_name = ModuleName::new_static("foo").unwrap();
let foo_path = src.join("foo.py");
db.memory_file_system()
.write_file(&*foo_path, "print('Hello, world!')")?;
.write_file(&foo_path, "print('Hello, world!')")?;

let foo_module = resolve_module(&db, foo_module_name.clone()).unwrap();

Expand All @@ -396,10 +396,10 @@ mod tests {
);

assert_eq!("foo", foo_module.name());
assert_eq!(*src, foo_module.search_path());
assert_eq!(&src, &foo_module.search_path());
assert_eq!(ModuleKind::Module, foo_module.kind());

assert_eq!(*foo_path, *foo_module.file().path(&db));
assert_eq!(&foo_path, foo_module.file().path(&db));
assert_eq!(
Some(foo_module),
path_to_module(&db, &VfsPath::FileSystem(foo_path))
Expand Down Expand Up @@ -464,8 +464,8 @@ mod tests {
});
let search_path = resolved_module.search_path();
assert_eq!(
*custom_typeshed.join("stdlib"),
search_path,
&custom_typeshed.join("stdlib"),
&search_path,
"Search path for {module_name} was unexpectedly {search_path:?}"
);
assert!(
Expand Down Expand Up @@ -513,8 +513,8 @@ mod tests {
});
let search_path = resolved_module.search_path();
assert_eq!(
*custom_typeshed.join("stdlib"),
search_path,
&custom_typeshed.join("stdlib"),
&search_path,
"Search path for {module_name} was unexpectedly {search_path:?}"
);
assert!(
Expand Down Expand Up @@ -547,11 +547,11 @@ mod tests {
Some(&functools_module),
resolve_module(&db, functools_module_name).as_ref()
);
assert_eq!(*src, functools_module.search_path());
assert_eq!(&src, &functools_module.search_path());
assert_eq!(ModuleKind::Module, functools_module.kind());
assert_eq!(
*first_party_functools_path,
*functools_module.file().path(&db)
&first_party_functools_path,
functools_module.file().path(&db)
);

assert_eq!(
Expand All @@ -570,13 +570,13 @@ mod tests {
let foo_path = foo_dir.join("__init__.py");

db.memory_file_system()
.write_file(&*foo_path, "print('Hello, world!')")?;
.write_file(&foo_path, "print('Hello, world!')")?;

let foo_module = resolve_module(&db, ModuleName::new_static("foo").unwrap()).unwrap();

assert_eq!("foo", foo_module.name());
assert_eq!(*src, foo_module.search_path());
assert_eq!(*foo_path, *foo_module.file().path(&db));
assert_eq!(&src, &foo_module.search_path());
assert_eq!(&foo_path, foo_module.file().path(&db));

assert_eq!(
Some(&foo_module),
Expand All @@ -597,16 +597,16 @@ mod tests {
let foo_init = foo_dir.join("__init__.py");

db.memory_file_system()
.write_file(&*foo_init, "print('Hello, world!')")?;
.write_file(&foo_init, "print('Hello, world!')")?;

let foo_py = src.join("foo.py");
db.memory_file_system()
.write_file(&*foo_py, "print('Hello, world!')")?;
.write_file(&foo_py, "print('Hello, world!')")?;

let foo_module = resolve_module(&db, ModuleName::new_static("foo").unwrap()).unwrap();

assert_eq!(*src, foo_module.search_path());
assert_eq!(*foo_init, *foo_module.file().path(&db));
assert_eq!(&src, &foo_module.search_path());
assert_eq!(&foo_init, foo_module.file().path(&db));
assert_eq!(ModuleKind::Package, foo_module.kind());

assert_eq!(
Expand All @@ -625,12 +625,12 @@ mod tests {
let foo_stub = src.join("foo.pyi");
let foo_py = src.join("foo.py");
db.memory_file_system()
.write_files([(&*foo_stub, "x: int"), (&*foo_py, "print('Hello, world!')")])?;
.write_files([(&foo_stub, "x: int"), (&foo_py, "print('Hello, world!')")])?;

let foo = resolve_module(&db, ModuleName::new_static("foo").unwrap()).unwrap();

assert_eq!(*src, foo.search_path());
assert_eq!(*foo_stub, *foo.file().path(&db));
assert_eq!(&src, &foo.search_path());
assert_eq!(&foo_stub, foo.file().path(&db));

assert_eq!(
Some(foo),
Expand All @@ -650,16 +650,16 @@ mod tests {
let baz = bar.join("baz.py");

db.memory_file_system().write_files([
(&*foo.join("__init__.py"), ""),
(&*bar.join("__init__.py"), ""),
(&*baz, "print('Hello, world!')"),
(&foo.join("__init__.py"), ""),
(&bar.join("__init__.py"), ""),
(&baz, "print('Hello, world!')"),
])?;

let baz_module =
resolve_module(&db, ModuleName::new_static("foo.bar.baz").unwrap()).unwrap();

assert_eq!(*src, baz_module.search_path());
assert_eq!(*baz, *baz_module.file().path(&db));
assert_eq!(&src, &baz_module.search_path());
assert_eq!(&baz, baz_module.file().path(&db));

assert_eq!(
Some(baz_module),
Expand Down Expand Up @@ -790,8 +790,8 @@ mod tests {

let foo_module = resolve_module(&db, ModuleName::new_static("foo").unwrap()).unwrap();

assert_eq!(*src, foo_module.search_path());
assert_eq!(*foo_src, *foo_module.file().path(&db));
assert_eq!(&src, &foo_module.search_path());
assert_eq!(&foo_src, foo_module.file().path(&db));

assert_eq!(
Some(foo_module),
Expand Down Expand Up @@ -820,9 +820,9 @@ mod tests {
let temp_dir = tempfile::tempdir()?;
let root = FileSystemPath::from_std_path(temp_dir.path()).unwrap();

let src = root.join(&*src);
let site_packages = root.join(&*site_packages);
let custom_typeshed = root.join(&*custom_typeshed);
let src = root.join(src);
let site_packages = root.join(site_packages);
let custom_typeshed = root.join(custom_typeshed);

let foo = src.join("foo.py");
let bar = src.join("bar.py");
Expand Down Expand Up @@ -853,12 +853,12 @@ mod tests {

assert_ne!(foo_module, bar_module);

assert_eq!(*src, foo_module.search_path());
assert_eq!(&src, &foo_module.search_path());
assert_eq!(&foo, foo_module.file().path(&db));

// `foo` and `bar` shouldn't resolve to the same file

assert_eq!(*src, bar_module.search_path());
assert_eq!(&src, &bar_module.search_path());
assert_eq!(&bar, bar_module.file().path(&db));
assert_eq!(&foo, foo_module.file().path(&db));

Expand All @@ -884,17 +884,17 @@ mod tests {
let bar_path = src.join("bar.py");

db.memory_file_system()
.write_files([(&*foo_path, "x = 1"), (&*bar_path, "y = 2")])?;
.write_files([(&foo_path, "x = 1"), (&bar_path, "y = 2")])?;

let foo_module_name = ModuleName::new_static("foo").unwrap();
let foo_module = resolve_module(&db, foo_module_name.clone()).unwrap();

let bar = system_path_to_file(&db, &*bar_path).expect("bar.py to exist");
let bar = system_path_to_file(&db, &bar_path).expect("bar.py to exist");

db.clear_salsa_events();

// Delete `bar.py`
db.memory_file_system().remove_file(&*bar_path)?;
db.memory_file_system().remove_file(&bar_path)?;
bar.touch(&mut db);

// Re-query the foo module. The foo module should still be cached because `bar.py` isn't relevant
Expand Down Expand Up @@ -922,9 +922,9 @@ mod tests {
assert_eq!(resolve_module(&db, foo_module_name.clone()), None);

// Now write the foo file
db.memory_file_system().write_file(&*foo_path, "x = 1")?;
db.memory_file_system().write_file(&foo_path, "x = 1")?;
VfsFile::touch_path(&mut db, &VfsPath::FileSystem(foo_path.clone()));
let foo_file = system_path_to_file(&db, &*foo_path).expect("foo.py to exist");
let foo_file = system_path_to_file(&db, &foo_path).expect("foo.py to exist");

let foo_module = resolve_module(&db, foo_module_name).expect("Foo module to resolve");
assert_eq!(foo_file, foo_module.file());
Expand All @@ -940,21 +940,21 @@ mod tests {
let foo_init_path = src.join("foo/__init__.py");

db.memory_file_system()
.write_files([(&*foo_path, "x = 1"), (&*foo_init_path, "x = 2")])?;
.write_files([(&foo_path, "x = 1"), (&foo_init_path, "x = 2")])?;

let foo_module_name = ModuleName::new_static("foo").unwrap();
let foo_module = resolve_module(&db, foo_module_name.clone()).expect("foo module to exist");

assert_eq!(*foo_init_path, *foo_module.file().path(&db));
assert_eq!(&foo_init_path, foo_module.file().path(&db));

// Delete `foo/__init__.py` and the `foo` folder. `foo` should now resolve to `foo.py`
db.memory_file_system().remove_file(&*foo_init_path)?;
db.memory_file_system().remove_file(&foo_init_path)?;
db.memory_file_system()
.remove_directory(foo_init_path.parent().unwrap())?;
VfsFile::touch_path(&mut db, &VfsPath::FileSystem(foo_init_path));

let foo_module = resolve_module(&db, foo_module_name).expect("Foo module to resolve");
assert_eq!(*foo_path, *foo_module.file().path(&db));
assert_eq!(&foo_path, foo_module.file().path(&db));

Ok(())
}
Expand Down