Files
turso/tests/integration/query_processing/test_multi_thread.rs
Pere Diaz Bou 6dd8f6561d clippy
2025-07-21 12:53:56 +02:00

186 lines
6.6 KiB
Rust

use std::sync::{atomic::AtomicUsize, Arc};
use turso_core::StepResult;
use crate::common::{maybe_setup_tracing, TempDatabase};
#[test]
fn test_schema_change() {
let tmp_db = TempDatabase::new_empty(false);
let conn1 = tmp_db.connect_limbo();
conn1.execute("CREATE TABLE t(x, y, z)").unwrap();
conn1
.execute("INSERT INTO t VALUES (1, 2, 3), (10, 20, 30)")
.unwrap();
let conn2 = tmp_db.connect_limbo();
let mut stmt = conn2.prepare("SELECT x, z FROM t").unwrap();
conn1.execute("ALTER TABLE t DROP COLUMN x").unwrap();
let row = loop {
match stmt.step() {
Ok(turso_core::StepResult::Row) => {
let row = stmt.row().unwrap();
break row;
}
Ok(turso_core::StepResult::IO) => {
stmt.run_once().unwrap();
}
_ => panic!("unexpected step result"),
}
};
println!("{:?} {:?}", row.get_value(0), row.get_value(1));
}
#[test]
#[ignore]
fn test_create_multiple_connections() -> anyhow::Result<()> {
maybe_setup_tracing();
let tries = 1;
for _ in 0..tries {
let tmp_db = Arc::new(TempDatabase::new_empty(false));
{
let conn = tmp_db.connect_limbo();
conn.execute("CREATE TABLE t(x)").unwrap();
}
let mut threads = Vec::new();
for i in 0..10 {
let tmp_db_ = tmp_db.clone();
threads.push(std::thread::spawn(move || {
let conn = tmp_db_.connect_limbo();
'outer: loop {
let mut stmt = conn
.prepare(format!("INSERT INTO t VALUES ({i})").as_str())
.unwrap();
tracing::info!("inserting row {}", i);
loop {
match stmt.step().unwrap() {
StepResult::Row => {
panic!("unexpected row result");
}
StepResult::IO => {
stmt.run_once().unwrap();
}
StepResult::Done => {
tracing::info!("inserted row {}", i);
break 'outer;
}
StepResult::Interrupt => {
panic!("unexpected step result");
}
StepResult::Busy => {
// repeat until we can insert it
tracing::info!("busy {}, repeating", i);
break;
}
}
}
}
}));
}
for thread in threads {
thread.join().unwrap();
}
let conn = tmp_db.connect_limbo();
let mut stmt = conn.prepare("SELECT * FROM t").unwrap();
let mut rows = Vec::new();
loop {
match stmt.step().unwrap() {
StepResult::Row => {
let row = stmt.row().unwrap();
rows.push(row.get::<i64>(0).unwrap());
}
StepResult::IO => {
stmt.run_once().unwrap();
}
StepResult::Done => {
break;
}
StepResult::Interrupt => {
panic!("unexpected step result");
}
StepResult::Busy => {
panic!("unexpected busy result on select");
}
}
}
rows.sort();
assert_eq!(rows, (0..10).collect::<Vec<_>>());
}
Ok(())
}
#[test]
#[ignore]
fn test_reader_writer() -> anyhow::Result<()> {
let tries = 10;
for _ in 0..tries {
let tmp_db = Arc::new(TempDatabase::new_empty(false));
{
let conn = tmp_db.connect_limbo();
conn.execute("CREATE TABLE t(x)").unwrap();
}
let mut threads = Vec::new();
let number_of_writers = 100;
let current_written_rows = Arc::new(AtomicUsize::new(0));
{
let tmp_db = tmp_db.clone();
let current_written_rows = current_written_rows.clone();
threads.push(std::thread::spawn(move || {
let conn = tmp_db.connect_limbo();
for i in 0..number_of_writers {
conn.execute(format!("INSERT INTO t VALUES ({i})").as_str())
.unwrap();
current_written_rows.fetch_add(1, std::sync::atomic::Ordering::Relaxed);
}
}));
}
{
let current_written_rows = current_written_rows.clone();
threads.push(std::thread::spawn(move || {
let conn = tmp_db.connect_limbo();
loop {
let current_written_rows =
current_written_rows.load(std::sync::atomic::Ordering::Relaxed);
if current_written_rows == number_of_writers {
break;
}
let mut stmt = conn.prepare("SELECT * FROM t").unwrap();
let mut rows = Vec::new();
loop {
match stmt.step().unwrap() {
StepResult::Row => {
let row = stmt.row().unwrap();
let x = row.get::<i64>(0).unwrap();
rows.push(x);
}
StepResult::IO => {
stmt.run_once().unwrap();
}
StepResult::Done => {
rows.sort();
for i in 0..current_written_rows {
let i = i as i64;
assert!(
rows.contains(&i),
"row {i} not found in {rows:?}. current_written_rows: {current_written_rows}",
);
}
break;
}
StepResult::Interrupt | StepResult::Busy => {
panic!("unexpected step result");
}
}
}
}
}));
}
for thread in threads {
thread.join().unwrap();
}
}
Ok(())
}