Files
2025-10-27 15:16:12 +04:00

134 lines
4.3 KiB
Rust

use crate::common::TempDatabase;
#[test]
fn test_fail_drop_indexed_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b);");
let conn = tmp_db.connect_limbo();
conn.execute("CREATE INDEX i ON t (a)")?;
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(res.is_err(), "Expected error when dropping indexed column");
Ok(())
}
#[test]
fn test_fail_drop_unique_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a UNIQUE, b);");
let conn = tmp_db.connect_limbo();
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(res.is_err(), "Expected error when dropping UNIQUE column");
Ok(())
}
#[test]
fn test_fail_drop_compound_unique_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b, UNIQUE(a, b));");
let conn = tmp_db.connect_limbo();
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(
res.is_err(),
"Expected error when dropping column in compound UNIQUE"
);
Ok(())
}
#[test]
fn test_fail_drop_primary_key_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a PRIMARY KEY, b);");
let conn = tmp_db.connect_limbo();
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(
res.is_err(),
"Expected error when dropping PRIMARY KEY column"
);
Ok(())
}
#[test]
fn test_fail_drop_compound_primary_key_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b, PRIMARY KEY(a, b));");
let conn = tmp_db.connect_limbo();
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(
res.is_err(),
"Expected error when dropping column in compound PRIMARY KEY"
);
Ok(())
}
#[test]
fn test_fail_drop_partial_index_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b);");
let conn = tmp_db.connect_limbo();
conn.execute("CREATE INDEX i ON t (b) WHERE a > 0")?;
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(
res.is_err(),
"Expected error when dropping column referenced by partial index"
);
Ok(())
}
#[test]
fn test_fail_drop_view_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b);");
let conn = tmp_db.connect_limbo();
conn.execute("CREATE VIEW v AS SELECT a, b FROM t")?;
let res = conn.execute("ALTER TABLE t DROP COLUMN a");
assert!(
res.is_err(),
"Expected error when dropping column referenced by view"
);
Ok(())
}
// FIXME: this should rewrite the view to reference the new column name
#[test]
fn test_fail_rename_view_column() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite("CREATE TABLE t (a, b);");
let conn = tmp_db.connect_limbo();
conn.execute("CREATE VIEW v AS SELECT a, b FROM t")?;
let res = conn.execute("ALTER TABLE t RENAME a TO c");
assert!(
res.is_err(),
"Expected error when renaming column referenced by view"
);
Ok(())
}
#[test]
fn test_allow_drop_unreferenced_columns() -> anyhow::Result<()> {
let _ = env_logger::try_init();
let tmp_db = TempDatabase::new_with_rusqlite(
"CREATE TABLE t (pk INTEGER PRIMARY KEY, indexed INTEGER, viewed INTEGER, partial INTEGER, compound1 INTEGER, compound2 INTEGER, unused1 INTEGER, unused2 INTEGER, unused3 INTEGER);",
);
let conn = tmp_db.connect_limbo();
conn.execute("CREATE INDEX idx ON t(indexed)")?;
conn.execute("CREATE VIEW v AS SELECT viewed FROM t")?;
conn.execute("CREATE INDEX partial_idx ON t(compound1) WHERE partial > 0")?;
conn.execute("CREATE INDEX compound_idx ON t(compound1, compound2)")?;
// Should be able to drop unused columns
conn.execute("ALTER TABLE t DROP COLUMN unused1")?;
conn.execute("ALTER TABLE t DROP COLUMN unused2")?;
conn.execute("ALTER TABLE t DROP COLUMN unused3")?;
Ok(())
}