diff --git a/crates/red_knot_python_semantic/src/module_resolver/path.rs b/crates/red_knot_python_semantic/src/module_resolver/path.rs index b91831de46342c..e177af4d99f550 100644 --- a/crates/red_knot_python_semantic/src/module_resolver/path.rs +++ b/crates/red_knot_python_semantic/src/module_resolver/path.rs @@ -9,12 +9,11 @@ use ruff_db::files::{system_path_to_file, vendored_path_to_file, File, FileError use ruff_db::system::{System, SystemPath, SystemPathBuf}; use ruff_db::vendored::{VendoredPath, VendoredPathBuf}; +use super::state::ResolverContext; +use super::typeshed::{typeshed_versions, TypeshedVersionsParseError, TypeshedVersionsQueryResult}; use crate::db::Db; use crate::module_name::ModuleName; -use super::state::ResolverState; -use super::typeshed::{TypeshedVersionsParseError, TypeshedVersionsQueryResult}; - /// A path that points to a Python module. /// /// A `ModulePath` is made up of two elements: @@ -60,7 +59,7 @@ impl ModulePath { } #[must_use] - pub(crate) fn is_directory(&self, resolver: &ResolverState) -> bool { + pub(crate) fn is_directory(&self, resolver: &ResolverContext) -> bool { let ModulePath { search_path, relative_path, @@ -74,7 +73,7 @@ impl ModulePath { == Err(FileError::IsADirectory) } SearchPathInner::StandardLibraryCustom(stdlib_root) => { - match query_stdlib_version(Some(stdlib_root), relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => false, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => { @@ -84,7 +83,7 @@ impl ModulePath { } } SearchPathInner::StandardLibraryVendored(stdlib_root) => { - match query_stdlib_version(None, relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => false, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => resolver @@ -96,7 +95,7 @@ impl ModulePath { } #[must_use] - pub(crate) fn is_regular_package(&self, resolver: &ResolverState) -> bool { + pub(crate) fn is_regular_package(&self, resolver: &ResolverContext) -> bool { let ModulePath { search_path, relative_path, @@ -113,7 +112,7 @@ impl ModulePath { .is_ok() } SearchPathInner::StandardLibraryCustom(search_path) => { - match query_stdlib_version(Some(search_path), relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => false, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => system_path_to_file( @@ -124,7 +123,7 @@ impl ModulePath { } } SearchPathInner::StandardLibraryVendored(search_path) => { - match query_stdlib_version(None, relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => false, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => resolver @@ -136,7 +135,7 @@ impl ModulePath { } #[must_use] - pub(crate) fn to_file(&self, resolver: &ResolverState) -> Option { + pub(crate) fn to_file(&self, resolver: &ResolverContext) -> Option { let db = resolver.db.upcast(); let ModulePath { search_path, @@ -150,7 +149,7 @@ impl ModulePath { system_path_to_file(db, search_path.join(relative_path)).ok() } SearchPathInner::StandardLibraryCustom(stdlib_root) => { - match query_stdlib_version(Some(stdlib_root), relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => None, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => { @@ -159,7 +158,7 @@ impl ModulePath { } } SearchPathInner::StandardLibraryVendored(stdlib_root) => { - match query_stdlib_version(None, relative_path, resolver) { + match query_stdlib_version(relative_path, resolver) { TypeshedVersionsQueryResult::DoesNotExist => None, TypeshedVersionsQueryResult::Exists | TypeshedVersionsQueryResult::MaybeExists => { @@ -273,19 +272,15 @@ fn stdlib_path_to_module_name(relative_path: &Utf8Path) -> Option { #[must_use] fn query_stdlib_version( - custom_stdlib_root: Option<&SystemPath>, relative_path: &Utf8Path, - resolver: &ResolverState, + context: &ResolverContext, ) -> TypeshedVersionsQueryResult { let Some(module_name) = stdlib_path_to_module_name(relative_path) else { return TypeshedVersionsQueryResult::DoesNotExist; }; - let ResolverState { - db, - typeshed_versions, - target_version, - } = resolver; - typeshed_versions.query_module(*db, &module_name, custom_stdlib_root, *target_version) + let ResolverContext { db, target_version } = context; + + typeshed_versions(*db).query_module(&module_name, *target_version) } /// Enumeration describing the various ways in which validation of a search path might fail. @@ -293,7 +288,7 @@ fn query_stdlib_version( /// If validation fails for a search path derived from the user settings, /// a message must be displayed to the user, /// as type checking cannot be done reliably in these circumstances. -#[derive(Debug, PartialEq, Eq)] +#[derive(Debug)] pub(crate) enum SearchPathValidationError { /// The path provided by the user was not a directory NotADirectory(SystemPathBuf), @@ -304,13 +299,12 @@ pub(crate) enum SearchPathValidationError { NoStdlibSubdirectory(SystemPathBuf), /// The typeshed path provided by the user is a directory, - /// but no `stdlib/VERSIONS` file exists. - /// (This is only relevant for stdlib search paths.) - NoVersionsFile(SystemPathBuf), - - /// `stdlib/VERSIONS` is a directory. + /// but `stdlib/VERSIONS` could not be read /// (This is only relevant for stdlib search paths.) - VersionsIsADirectory(SystemPathBuf), + FailedToReadVersionsFile { + path: SystemPathBuf, + error: std::io::Error, + }, /// The path provided by the user is a directory, /// and a `stdlib/VERSIONS` file exists, but it fails to parse. @@ -325,8 +319,9 @@ impl fmt::Display for SearchPathValidationError { Self::NoStdlibSubdirectory(path) => { write!(f, "The directory at {path} has no `stdlib/` subdirectory") } - Self::NoVersionsFile(path) => write!(f, "Expected a file at {path}/stdlib/VERSIONS"), - Self::VersionsIsADirectory(path) => write!(f, "{path}/stdlib/VERSIONS is a directory."), + Self::FailedToReadVersionsFile { path, error } => { + write!(f, "Failed to read the '{path}' file: {error}") + } Self::VersionsParseError(underlying_error) => underlying_error.fmt(f), } } @@ -407,13 +402,14 @@ impl SearchPath { } /// Create a new standard-library search path pointing to a custom directory on disk - pub(crate) fn custom_stdlib(db: &dyn Db, typeshed: SystemPathBuf) -> SearchPathResult { + pub(crate) fn custom_stdlib(db: &dyn Db, typeshed: &SystemPath) -> SearchPathResult { let system = db.system(); - if !system.is_directory(&typeshed) { + if !system.is_directory(typeshed) { return Err(SearchPathValidationError::NotADirectory( typeshed.to_path_buf(), )); } + let stdlib = Self::directory_path(system, typeshed.join("stdlib")).map_err(|err| match err { SearchPathValidationError::NotADirectory(path) => { @@ -421,18 +417,7 @@ impl SearchPath { } err => err, })?; - let typeshed_versions = - system_path_to_file(db.upcast(), stdlib.join("VERSIONS")).map_err(|err| match err { - FileError::NotFound => SearchPathValidationError::NoVersionsFile(typeshed), - FileError::IsADirectory => { - SearchPathValidationError::VersionsIsADirectory(typeshed) - } - })?; - super::typeshed::parse_typeshed_versions(db, typeshed_versions) - .as_ref() - .map_err(|validation_error| { - SearchPathValidationError::VersionsParseError(validation_error.clone()) - })?; + Ok(Self(Arc::new(SearchPathInner::StandardLibraryCustom( stdlib, )))) @@ -661,7 +646,7 @@ mod tests { .build(); assert_eq!( - SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()) + SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()) .unwrap() .to_module_path() .with_py_extension(), @@ -669,7 +654,7 @@ mod tests { ); assert_eq!( - &SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()) + &SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()) .unwrap() .join("foo") .with_pyi_extension(), @@ -780,7 +765,7 @@ mod tests { let TestCase { db, stdlib, .. } = TestCaseBuilder::new() .with_custom_typeshed(MockedTypeshed::default()) .build(); - SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()) + SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()) .unwrap() .to_module_path() .push("bar.py"); @@ -792,7 +777,7 @@ mod tests { let TestCase { db, stdlib, .. } = TestCaseBuilder::new() .with_custom_typeshed(MockedTypeshed::default()) .build(); - SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()) + SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()) .unwrap() .to_module_path() .push("bar.rs"); @@ -824,7 +809,7 @@ mod tests { .with_custom_typeshed(MockedTypeshed::default()) .build(); - let root = SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()).unwrap(); + let root = SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()).unwrap(); // Must have a `.pyi` extension or no extension: let bad_absolute_path = SystemPath::new("foo/stdlib/x.py"); @@ -872,8 +857,7 @@ mod tests { .with_custom_typeshed(typeshed) .with_target_version(target_version) .build(); - let stdlib = - SearchPath::custom_stdlib(&db, stdlib.parent().unwrap().to_path_buf()).unwrap(); + let stdlib = SearchPath::custom_stdlib(&db, stdlib.parent().unwrap()).unwrap(); (db, stdlib) } @@ -898,7 +882,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let asyncio_regular_package = stdlib_path.join("asyncio"); assert!(asyncio_regular_package.is_directory(&resolver)); @@ -926,7 +910,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let xml_namespace_package = stdlib_path.join("xml"); assert!(xml_namespace_package.is_directory(&resolver)); @@ -948,7 +932,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let functools_module = stdlib_path.join("functools.pyi"); assert!(functools_module.to_file(&resolver).is_some()); @@ -964,7 +948,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let collections_regular_package = stdlib_path.join("collections"); assert_eq!(collections_regular_package.to_file(&resolver), None); @@ -980,7 +964,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let importlib_namespace_package = stdlib_path.join("importlib"); assert_eq!(importlib_namespace_package.to_file(&resolver), None); @@ -1001,7 +985,7 @@ mod tests { }; let (db, stdlib_path) = py38_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py38); + let resolver = ResolverContext::new(&db, TargetVersion::Py38); let non_existent = stdlib_path.join("doesnt_even_exist"); assert_eq!(non_existent.to_file(&resolver), None); @@ -1029,7 +1013,7 @@ mod tests { }; let (db, stdlib_path) = py39_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py39); + let resolver = ResolverContext::new(&db, TargetVersion::Py39); // Since we've set the target version to Py39, // `collections` should now exist as a directory, according to VERSIONS... @@ -1058,7 +1042,7 @@ mod tests { }; let (db, stdlib_path) = py39_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py39); + let resolver = ResolverContext::new(&db, TargetVersion::Py39); // The `importlib` directory now also exists let importlib_namespace_package = stdlib_path.join("importlib"); @@ -1082,7 +1066,7 @@ mod tests { }; let (db, stdlib_path) = py39_typeshed_test_case(TYPESHED); - let resolver = ResolverState::new(&db, TargetVersion::Py39); + let resolver = ResolverContext::new(&db, TargetVersion::Py39); // The `xml` package no longer exists on py39: let xml_namespace_package = stdlib_path.join("xml"); diff --git a/crates/red_knot_python_semantic/src/module_resolver/resolver.rs b/crates/red_knot_python_semantic/src/module_resolver/resolver.rs index 5d66aec36ce69d..093b41f729cb0c 100644 --- a/crates/red_knot_python_semantic/src/module_resolver/resolver.rs +++ b/crates/red_knot_python_semantic/src/module_resolver/resolver.rs @@ -1,16 +1,16 @@ -use rustc_hash::{FxBuildHasher, FxHashSet}; -use std::borrow::Cow; -use std::iter::FusedIterator; - use ruff_db::files::{File, FilePath, FileRootKind}; use ruff_db::system::{DirectoryEntry, SystemPath, SystemPathBuf}; use ruff_db::vendored::VendoredPath; +use rustc_hash::{FxBuildHasher, FxHashSet}; +use std::borrow::Cow; +use std::iter::FusedIterator; use super::module::{Module, ModuleKind}; use super::path::{ModulePath, SearchPath, SearchPathValidationError}; -use super::state::ResolverState; +use super::state::ResolverContext; use crate::db::Db; use crate::module_name::ModuleName; +use crate::module_resolver::typeshed::{vendored_typeshed_verisons, TypeshedVersions}; use crate::{Program, SearchPathSettings}; /// Resolves a module name to a module. @@ -120,7 +120,7 @@ pub(crate) fn search_paths(db: &dyn Db) -> SearchPathIterator { Program::get(db).search_paths(db).iter(db) } -#[derive(Debug, PartialEq, Eq, Default)] +#[derive(Debug, PartialEq, Eq)] pub(crate) struct SearchPaths { /// Search paths that have been statically determined purely from reading Ruff's configuration settings. /// These shouldn't ever change unless the config settings themselves change. @@ -133,6 +133,8 @@ pub(crate) struct SearchPaths { /// in terms of module-resolution priority until we've discovered the editable installs /// for the first `site-packages` path site_packages: Vec, + + typeshed_versions: Cow<'static, TypeshedVersions>, } impl SearchPaths { @@ -159,7 +161,7 @@ impl SearchPaths { let mut static_paths = vec![]; for path in extra_paths { - tracing::debug!("Adding static extra search-path '{path}'"); + tracing::debug!("Adding extra search-path '{path}'"); let search_path = SearchPath::extra(system, path)?; files.try_add_root( @@ -170,22 +172,43 @@ impl SearchPaths { static_paths.push(search_path); } - tracing::debug!("Adding static search path '{src_root}'"); + tracing::debug!("Adding search path '{src_root}'"); static_paths.push(SearchPath::first_party(system, src_root)?); - static_paths.push(if let Some(custom_typeshed) = custom_typeshed { - tracing::debug!("Adding static custom-sdtlib search-path '{custom_typeshed}'"); + let (typeshed_versions, stdlib_path) = if let Some(custom_typeshed) = custom_typeshed { + tracing::debug!("Adding custom-sdtlib search-path '{custom_typeshed}'"); - let search_path = SearchPath::custom_stdlib(db, custom_typeshed)?; + let search_path = SearchPath::custom_stdlib(db, &custom_typeshed)?; files.try_add_root( db.upcast(), search_path.as_system_path().unwrap(), FileRootKind::LibrarySearchPath, ); - search_path + + let stdlib_root = search_path.as_system_path().unwrap(); + let versions_path = stdlib_root.join("VERSIONS"); + + let versions_content = system.read_to_string(&versions_path).map_err(|error| { + SearchPathValidationError::FailedToReadVersionsFile { + path: versions_path, + error, + } + })?; + + let parsed: TypeshedVersions = versions_content + .parse() + .map_err(SearchPathValidationError::VersionsParseError)?; + + (Cow::Owned(parsed), search_path) } else { - SearchPath::vendored_stdlib() - }); + tracing::debug!("Using vendored stdlib"); + ( + Cow::Borrowed(vendored_typeshed_verisons()), + SearchPath::vendored_stdlib(), + ) + }; + + static_paths.push(stdlib_path); let mut site_packages: Vec<_> = Vec::with_capacity(site_packages_paths.len()); @@ -221,16 +244,21 @@ impl SearchPaths { Ok(SearchPaths { static_paths, site_packages, + typeshed_versions, }) } - pub(crate) fn iter<'a>(&'a self, db: &'a dyn Db) -> SearchPathIterator<'a> { + pub(super) fn iter<'a>(&'a self, db: &'a dyn Db) -> SearchPathIterator<'a> { SearchPathIterator { db, static_paths: self.static_paths.iter(), dynamic_paths: None, } } + + pub(super) fn typeshed_versions(&self) -> &TypeshedVersions { + &self.typeshed_versions + } } /// Collect all dynamic search paths. For each `site-packages` path: @@ -248,6 +276,7 @@ pub(crate) fn dynamic_resolution_paths(db: &dyn Db) -> Vec { let SearchPaths { static_paths, site_packages, + typeshed_versions: _, } = Program::get(db).search_paths(db); let mut dynamic_paths = Vec::new(); @@ -479,7 +508,7 @@ struct ModuleNameIngredient<'db> { fn resolve_name(db: &dyn Db, name: &ModuleName) -> Option<(SearchPath, File, ModuleKind)> { let program = Program::get(db); let target_version = program.target_version(db); - let resolver_state = ResolverState::new(db, target_version); + let resolver_state = ResolverContext::new(db, target_version); let (_, minor_version) = target_version.as_tuple(); let is_builtin_module = ruff_python_stdlib::sys::is_builtin_module(minor_version, name.as_str()); @@ -543,7 +572,7 @@ fn resolve_name(db: &dyn Db, name: &ModuleName) -> Option<(SearchPath, File, Mod fn resolve_package<'a, 'db, I>( module_search_path: &SearchPath, components: I, - resolver_state: &ResolverState<'db>, + resolver_state: &ResolverContext<'db>, ) -> Result where I: Iterator, diff --git a/crates/red_knot_python_semantic/src/module_resolver/state.rs b/crates/red_knot_python_semantic/src/module_resolver/state.rs index 1b16e13e400916..ca0beaac6601c9 100644 --- a/crates/red_knot_python_semantic/src/module_resolver/state.rs +++ b/crates/red_knot_python_semantic/src/module_resolver/state.rs @@ -1,22 +1,16 @@ use ruff_db::vendored::VendoredFileSystem; -use super::typeshed::LazyTypeshedVersions; use crate::db::Db; use crate::TargetVersion; -pub(crate) struct ResolverState<'db> { +pub(crate) struct ResolverContext<'db> { pub(crate) db: &'db dyn Db, - pub(crate) typeshed_versions: LazyTypeshedVersions<'db>, pub(crate) target_version: TargetVersion, } -impl<'db> ResolverState<'db> { +impl<'db> ResolverContext<'db> { pub(crate) fn new(db: &'db dyn Db, target_version: TargetVersion) -> Self { - Self { - db, - typeshed_versions: LazyTypeshedVersions::new(), - target_version, - } + Self { db, target_version } } pub(crate) fn vendored(&self) -> &VendoredFileSystem { diff --git a/crates/red_knot_python_semantic/src/module_resolver/typeshed/mod.rs b/crates/red_knot_python_semantic/src/module_resolver/typeshed/mod.rs index 97cac75fa62e09..0411eb3c974ee9 100644 --- a/crates/red_knot_python_semantic/src/module_resolver/typeshed/mod.rs +++ b/crates/red_knot_python_semantic/src/module_resolver/typeshed/mod.rs @@ -1,6 +1,6 @@ pub use self::vendored::vendored_typeshed_stubs; pub(super) use self::versions::{ - parse_typeshed_versions, LazyTypeshedVersions, TypeshedVersionsParseError, + typeshed_versions, vendored_typeshed_verisons, TypeshedVersions, TypeshedVersionsParseError, TypeshedVersionsQueryResult, }; diff --git a/crates/red_knot_python_semantic/src/module_resolver/typeshed/versions.rs b/crates/red_knot_python_semantic/src/module_resolver/typeshed/versions.rs index 778dedf7d6dbec..d6b1b23b9319bb 100644 --- a/crates/red_knot_python_semantic/src/module_resolver/typeshed/versions.rs +++ b/crates/red_knot_python_semantic/src/module_resolver/typeshed/versions.rs @@ -1,4 +1,3 @@ -use std::cell::OnceCell; use std::collections::BTreeMap; use std::fmt; use std::num::{NonZeroU16, NonZeroUsize}; @@ -6,78 +5,12 @@ use std::ops::{RangeFrom, RangeInclusive}; use std::str::FromStr; use once_cell::sync::Lazy; -use ruff_db::system::SystemPath; use rustc_hash::FxHashMap; -use ruff_db::files::{system_path_to_file, File}; - use super::vendored::vendored_typeshed_stubs; use crate::db::Db; use crate::module_name::ModuleName; -use crate::TargetVersion; - -#[derive(Debug)] -pub(crate) struct LazyTypeshedVersions<'db>(OnceCell<&'db TypeshedVersions>); - -impl<'db> LazyTypeshedVersions<'db> { - #[must_use] - pub(crate) fn new() -> Self { - Self(OnceCell::new()) - } - - /// Query whether a module exists at runtime in the stdlib on a certain Python version. - /// - /// Simply probing whether a file exists in typeshed is insufficient for this question, - /// as a module in the stdlib may have been added in Python 3.10, but the typeshed stub - /// will still be available (either in a custom typeshed dir or in our vendored copy) - /// even if the user specified Python 3.8 as the target version. - /// - /// For top-level modules and packages, the VERSIONS file can always provide an unambiguous answer - /// as to whether the module exists on the specified target version. However, VERSIONS does not - /// provide comprehensive information on all submodules, meaning that this method sometimes - /// returns [`TypeshedVersionsQueryResult::MaybeExists`]. - /// See [`TypeshedVersionsQueryResult`] for more details. - #[must_use] - pub(crate) fn query_module( - &self, - db: &'db dyn Db, - module: &ModuleName, - stdlib_root: Option<&SystemPath>, - target_version: TargetVersion, - ) -> TypeshedVersionsQueryResult { - let versions = self.0.get_or_init(|| { - let versions_path = if let Some(system_path) = stdlib_root { - system_path.join("VERSIONS") - } else { - return &VENDORED_VERSIONS; - }; - let Ok(versions_file) = system_path_to_file(db.upcast(), &versions_path) else { - todo!( - "Still need to figure out how to handle VERSIONS files being deleted \ - from custom typeshed directories! Expected a file to exist at {versions_path}" - ) - }; - // TODO(Alex/Micha): If VERSIONS is invalid, - // this should invalidate not just the specific module resolution we're currently attempting, - // but all type inference that depends on any standard-library types. - // Unwrapping here is not correct... - parse_typeshed_versions(db, versions_file).as_ref().unwrap() - }); - versions.query_module(module, PyVersion::from(target_version)) - } -} - -#[salsa::tracked(return_ref)] -pub(crate) fn parse_typeshed_versions( - db: &dyn Db, - versions_file: File, -) -> Result { - // TODO: Handle IO errors - let file_content = versions_file - .read_to_string(db.upcast()) - .unwrap_or_default(); - file_content.parse() -} +use crate::{Program, TargetVersion}; static VENDORED_VERSIONS: Lazy = Lazy::new(|| { TypeshedVersions::from_str( @@ -88,6 +21,14 @@ static VENDORED_VERSIONS: Lazy = Lazy::new(|| { .unwrap() }); +pub(crate) fn vendored_typeshed_verisons() -> &'static TypeshedVersions { + &VENDORED_VERSIONS +} + +pub(crate) fn typeshed_versions(db: &dyn Db) -> &TypeshedVersions { + Program::get(db).search_paths(db).typeshed_versions() +} + #[derive(Debug, PartialEq, Eq, Clone)] pub(crate) struct TypeshedVersionsParseError { line_number: Option, @@ -164,7 +105,7 @@ impl fmt::Display for TypeshedVersionsParseErrorKind { } } -#[derive(Debug, PartialEq, Eq)] +#[derive(Debug, PartialEq, Eq, Clone)] pub(crate) struct TypeshedVersions(FxHashMap); impl TypeshedVersions { @@ -174,11 +115,13 @@ impl TypeshedVersions { } #[must_use] - fn query_module( + pub(crate) fn query_module( &self, module: &ModuleName, - target_version: PyVersion, + target_version: TargetVersion, ) -> TypeshedVersionsQueryResult { + let target_version = PyVersion::from(target_version); + if let Some(range) = self.exact(module) { if range.contains(target_version) { TypeshedVersionsQueryResult::Exists @@ -476,27 +419,27 @@ mod tests { assert!(versions.contains_exact(&asyncio)); assert_eq!( - versions.query_module(&asyncio, TargetVersion::Py310.into()), + versions.query_module(&asyncio, TargetVersion::Py310), TypeshedVersionsQueryResult::Exists ); assert!(versions.contains_exact(&asyncio_staggered)); assert_eq!( - versions.query_module(&asyncio_staggered, TargetVersion::Py38.into()), + versions.query_module(&asyncio_staggered, TargetVersion::Py38), TypeshedVersionsQueryResult::Exists ); assert_eq!( - versions.query_module(&asyncio_staggered, TargetVersion::Py37.into()), + versions.query_module(&asyncio_staggered, TargetVersion::Py37), TypeshedVersionsQueryResult::DoesNotExist ); assert!(versions.contains_exact(&audioop)); assert_eq!( - versions.query_module(&audioop, TargetVersion::Py312.into()), + versions.query_module(&audioop, TargetVersion::Py312), TypeshedVersionsQueryResult::Exists ); assert_eq!( - versions.query_module(&audioop, TargetVersion::Py313.into()), + versions.query_module(&audioop, TargetVersion::Py313), TypeshedVersionsQueryResult::DoesNotExist ); } @@ -588,15 +531,15 @@ foo: 3.8- # trailing comment assert!(parsed_versions.contains_exact(&bar)); assert_eq!( - parsed_versions.query_module(&bar, TargetVersion::Py37.into()), + parsed_versions.query_module(&bar, TargetVersion::Py37), TypeshedVersionsQueryResult::Exists ); assert_eq!( - parsed_versions.query_module(&bar, TargetVersion::Py310.into()), + parsed_versions.query_module(&bar, TargetVersion::Py310), TypeshedVersionsQueryResult::Exists ); assert_eq!( - parsed_versions.query_module(&bar, TargetVersion::Py311.into()), + parsed_versions.query_module(&bar, TargetVersion::Py311), TypeshedVersionsQueryResult::DoesNotExist ); } @@ -608,15 +551,15 @@ foo: 3.8- # trailing comment assert!(parsed_versions.contains_exact(&foo)); assert_eq!( - parsed_versions.query_module(&foo, TargetVersion::Py37.into()), + parsed_versions.query_module(&foo, TargetVersion::Py37), TypeshedVersionsQueryResult::DoesNotExist ); assert_eq!( - parsed_versions.query_module(&foo, TargetVersion::Py38.into()), + parsed_versions.query_module(&foo, TargetVersion::Py38), TypeshedVersionsQueryResult::Exists ); assert_eq!( - parsed_versions.query_module(&foo, TargetVersion::Py311.into()), + parsed_versions.query_module(&foo, TargetVersion::Py311), TypeshedVersionsQueryResult::Exists ); } @@ -628,15 +571,15 @@ foo: 3.8- # trailing comment assert!(parsed_versions.contains_exact(&bar_baz)); assert_eq!( - parsed_versions.query_module(&bar_baz, TargetVersion::Py37.into()), + parsed_versions.query_module(&bar_baz, TargetVersion::Py37), TypeshedVersionsQueryResult::Exists ); assert_eq!( - parsed_versions.query_module(&bar_baz, TargetVersion::Py39.into()), + parsed_versions.query_module(&bar_baz, TargetVersion::Py39), TypeshedVersionsQueryResult::Exists ); assert_eq!( - parsed_versions.query_module(&bar_baz, TargetVersion::Py310.into()), + parsed_versions.query_module(&bar_baz, TargetVersion::Py310), TypeshedVersionsQueryResult::DoesNotExist ); } @@ -648,15 +591,15 @@ foo: 3.8- # trailing comment assert!(!parsed_versions.contains_exact(&bar_eggs)); assert_eq!( - parsed_versions.query_module(&bar_eggs, TargetVersion::Py37.into()), + parsed_versions.query_module(&bar_eggs, TargetVersion::Py37), TypeshedVersionsQueryResult::MaybeExists ); assert_eq!( - parsed_versions.query_module(&bar_eggs, TargetVersion::Py310.into()), + parsed_versions.query_module(&bar_eggs, TargetVersion::Py310), TypeshedVersionsQueryResult::MaybeExists ); assert_eq!( - parsed_versions.query_module(&bar_eggs, TargetVersion::Py311.into()), + parsed_versions.query_module(&bar_eggs, TargetVersion::Py311), TypeshedVersionsQueryResult::DoesNotExist ); } @@ -668,11 +611,11 @@ foo: 3.8- # trailing comment assert!(!parsed_versions.contains_exact(&spam)); assert_eq!( - parsed_versions.query_module(&spam, TargetVersion::Py37.into()), + parsed_versions.query_module(&spam, TargetVersion::Py37), TypeshedVersionsQueryResult::DoesNotExist ); assert_eq!( - parsed_versions.query_module(&spam, TargetVersion::Py313.into()), + parsed_versions.query_module(&spam, TargetVersion::Py313), TypeshedVersionsQueryResult::DoesNotExist ); } diff --git a/crates/red_knot_python_semantic/src/program.rs b/crates/red_knot_python_semantic/src/program.rs index e677fa4f29849a..1bd37f5e0705ef 100644 --- a/crates/red_knot_python_semantic/src/program.rs +++ b/crates/red_knot_python_semantic/src/program.rs @@ -11,7 +11,6 @@ use crate::Db; pub struct Program { pub target_version: TargetVersion, - #[default] #[return_ref] pub(crate) search_paths: SearchPaths, } @@ -28,9 +27,8 @@ impl Program { let search_paths = SearchPaths::from_settings(db, search_paths) .with_context(|| "Invalid search path settings")?; - Ok(Program::builder(settings.target_version) + Ok(Program::builder(settings.target_version, search_paths) .durability(Durability::HIGH) - .search_paths(search_paths) .new(db)) }