//! Test that a `tracked` fn on a `salsa::input` //! compiles and executes successfully. use std::path::{Path, PathBuf}; use expect_test::expect; use salsa::plumbing::{AsId, FromId}; use test_log::test; #[salsa::interned(debug)] struct InternedBoxed<'db> { data: Box, } #[salsa::interned(debug)] struct InternedString<'db> { data: String, } #[salsa::interned(debug)] struct InternedPair<'db> { data: (InternedString<'db>, InternedString<'db>), } #[salsa::interned(debug)] struct InternedTwoFields<'db> { data1: String, data2: String, } #[salsa::interned(debug)] struct InternedVec<'db> { data1: Vec, } #[salsa::interned(debug)] struct InternedPathBuf<'db> { data1: PathBuf, } #[salsa::interned(no_lifetime, debug)] struct InternedStringNoLifetime { data: String, } #[derive(Debug, Eq, PartialEq, Hash, Clone)] struct Foo; #[salsa::interned(debug)] struct InternedFoo<'db> { data: Foo, } #[derive(Clone, Copy, Hash, Debug, PartialEq, Eq, PartialOrd, Ord)] struct SalsaIdWrapper(salsa::Id); impl AsId for SalsaIdWrapper { fn as_id(&self) -> salsa::Id { self.0 } } impl FromId for SalsaIdWrapper { fn from_id(id: salsa::Id) -> Self { SalsaIdWrapper(id) } } #[salsa::interned(id = SalsaIdWrapper, debug)] struct InternedStringWithCustomId<'db> { data: String, } #[salsa::interned(id = SalsaIdWrapper, no_lifetime, debug)] struct InternedStringWithCustomIdAndNoLifetime<'db> { data: String, } #[salsa::tracked] fn intern_stuff(db: &dyn salsa::Database) -> String { let s1 = InternedString::new(db, "Hello, ".to_string()); let s2 = InternedString::new(db, "World, "); let s3 = InternedPair::new(db, (s1, s2)); format!("{s3:?}") } #[test] fn execute() { let db = salsa::DatabaseImpl::new(); expect![[r#" "InternedPair { data: (InternedString { data: \"Hello, \" }, InternedString { data: \"World, \" }) }" "#]].assert_debug_eq(&intern_stuff(&db)); } #[test] fn interning_returns_equal_keys_for_equal_data() { let db = salsa::DatabaseImpl::new(); let s1 = InternedString::new(&db, "Hello, ".to_string()); let s2 = InternedString::new(&db, "World, ".to_string()); let s1_2 = InternedString::new(&db, "Hello, "); let s2_2 = InternedString::new(&db, "World, "); assert_eq!(s1, s1_2); assert_eq!(s2, s2_2); } #[test] fn interning_returns_equal_keys_for_equal_data_multi_field() { let db = salsa::DatabaseImpl::new(); let s1 = InternedTwoFields::new(&db, "Hello, ".to_string(), "World"); let s2 = InternedTwoFields::new(&db, "World, ", "Hello".to_string()); let s1_2 = InternedTwoFields::new(&db, "Hello, ", "World"); let s2_2 = InternedTwoFields::new(&db, "World, ", "Hello"); let new = InternedTwoFields::new(&db, "Hello, World", ""); assert_eq!(s1, s1_2); assert_eq!(s2, s2_2); assert_ne!(s1, s2_2); assert_ne!(s1, new); } #[test] fn interning_boxed() { let db = salsa::DatabaseImpl::new(); assert_eq!( InternedBoxed::new(&db, "Hello"), InternedBoxed::new(&db, Box::from("Hello")) ); } #[test] fn interned_structs_have_public_ingredients() { use salsa::plumbing::AsId; let db = salsa::DatabaseImpl::new(); let s = InternedString::new(&db, String::from("Hello, world!")); let underlying_id = s.0; let data = InternedString::ingredient(&db).data(&db, underlying_id.as_id()); assert_eq!(data, &(String::from("Hello, world!"),)); } #[test] fn interning_vec() { let db = salsa::DatabaseImpl::new(); let s1 = InternedVec::new(&db, ["Hello, ".to_string(), "World".to_string()].as_slice()); let s2 = InternedVec::new(&db, ["Hello, ", "World"].as_slice()); let s3 = InternedVec::new(&db, vec!["Hello, ".to_string(), "World".to_string()]); let s4 = InternedVec::new(&db, ["Hello, ", "World"].as_slice()); let s5 = InternedVec::new(&db, ["Hello, ", "World", "Test"].as_slice()); let s6 = InternedVec::new(&db, ["Hello, ", "World", ""].as_slice()); let s7 = InternedVec::new(&db, ["Hello, "].as_slice()); assert_eq!(s1, s2); assert_eq!(s1, s3); assert_eq!(s1, s4); assert_ne!(s1, s5); assert_ne!(s1, s6); assert_ne!(s5, s6); assert_ne!(s6, s7); } #[test] fn interning_path_buf() { let db = salsa::DatabaseImpl::new(); let s1 = InternedPathBuf::new(&db, PathBuf::from("test_path".to_string())); let s2 = InternedPathBuf::new(&db, Path::new("test_path")); let s3 = InternedPathBuf::new(&db, Path::new("test_path/")); let s4 = InternedPathBuf::new(&db, Path::new("test_path/a")); assert_eq!(s1, s2); assert_eq!(s1, s3); assert_ne!(s1, s4); } #[test] fn interning_without_lifetimes() { let db = salsa::DatabaseImpl::new(); let s1 = InternedStringNoLifetime::new(&db, "Hello, ".to_string()); let s2 = InternedStringNoLifetime::new(&db, "World, ".to_string()); let s1_2 = InternedStringNoLifetime::new(&db, "Hello, "); let s2_2 = InternedStringNoLifetime::new(&db, "World, "); assert_eq!(s1, s1_2); assert_eq!(s2, s2_2); } #[test] fn interning_with_custom_ids() { let db = salsa::DatabaseImpl::new(); let s1 = InternedStringWithCustomId::new(&db, "Hello, ".to_string()); let s2 = InternedStringWithCustomId::new(&db, "World, ".to_string()); let s1_2 = InternedStringWithCustomId::new(&db, "Hello, "); let s2_2 = InternedStringWithCustomId::new(&db, "World, "); assert_eq!(s1, s1_2); assert_eq!(s2, s2_2); } #[test] fn interning_with_custom_ids_and_no_lifetime() { let db = salsa::DatabaseImpl::new(); let s1 = InternedStringWithCustomIdAndNoLifetime::new(&db, "Hello, ".to_string()); let s2 = InternedStringWithCustomIdAndNoLifetime::new(&db, "World, ".to_string()); let s1_2 = InternedStringWithCustomIdAndNoLifetime::new(&db, "Hello, "); let s2_2 = InternedStringWithCustomIdAndNoLifetime::new(&db, "World, "); assert_eq!(s1, s1_2); assert_eq!(s2, s2_2); } #[test] fn interning_reference() { let db = salsa::DatabaseImpl::new(); let s1 = InternedFoo::new(&db, Foo); let s2 = InternedFoo::new(&db, &Foo); assert_eq!(s1, s2); }