1
0

refactored project

This commit is contained in:
Ishan Jain 2022-07-26 13:46:11 +05:30
parent af086e6910
commit 1d570b229d

View File

@ -1,4 +1,4 @@
use anyhow::{bail, Result}; use anyhow::{bail, Error, Result};
use sqlite_starter_rust::{ use sqlite_starter_rust::{
header::PageHeader, record::parse_record, schema::Schema, varint::parse_varint, header::PageHeader, record::parse_record, schema::Schema, varint::parse_varint,
}; };
@ -51,8 +51,7 @@ fn main() -> Result<()> {
print!("number of tables: {}", schemas.len()); print!("number of tables: {}", schemas.len());
// Uncomment this block to pass the first stage Ok(())
// println!("number of tables: {}", schemas.len());
} }
".tables" => { ".tables" => {
@ -85,43 +84,34 @@ fn main() -> Result<()> {
{ {
print!("{} ", schema.name); print!("{} ", schema.name);
} }
Ok(())
} }
v => { v => {
let (columns, table) = read_column_and_table(v); let db_header = read_db_header(&database)?;
if v.contains("count(*)") {
count_rows_in_table(v, db_header, &database)
} else {
read_columns(v, db_header, &database)
}
}
}
}
fn read_columns(query: &str, db_header: DBHeader, database: &[u8]) -> Result<(), Error> {
let (columns, table) = read_column_and_table(query);
// Assume it's valid SQL // Assume it's valid SQL
let db_page_size = u16::from_be_bytes([database[16], database[17]]); let schema = db_header
.schemas
// Parse page header from database
let page_header = PageHeader::parse(&database[100..108])?;
// Obtain all cell pointers
let cell_pointers = database[108..]
.chunks_exact(2)
.take(page_header.number_of_cells.into())
.map(|bytes| u16::from_be_bytes(bytes.try_into().unwrap()));
// Obtain all records from column 5
#[allow(unused_variables)]
let schemas = cell_pointers.into_iter().map(|cell_pointer| {
let stream = &database[cell_pointer as usize..];
let (_, offset) = parse_varint(stream);
let (rowid, read_bytes) = parse_varint(&stream[offset..]);
parse_record(&stream[offset + read_bytes..], 5)
.map(|record| Schema::parse(record).expect("Invalid record"))
});
let schema = schemas
.into_iter() .into_iter()
.flatten()
.find(|schema| schema.table_name == table) .find(|schema| schema.table_name == table)
.unwrap(); .unwrap();
let column_map = find_column_positions(&schema.sql); let column_map = find_column_positions(&schema.sql);
let table_page_offset = db_page_size as usize * (schema.root_page as usize - 1); let table_page_offset = db_header.page_size as usize * (schema.root_page as usize - 1);
let page_header = let page_header =
PageHeader::parse(&database[table_page_offset..table_page_offset + 8]).unwrap(); PageHeader::parse(&database[table_page_offset..table_page_offset + 8]).unwrap();
@ -143,11 +133,62 @@ fn main() -> Result<()> {
println!("{}", String::from_utf8_lossy(&row[cpos])); println!("{}", String::from_utf8_lossy(&row[cpos]));
} }
}
}
Ok(()) Ok(())
} }
struct DBHeader {
page_size: u16,
schemas: Vec<Schema>,
}
fn read_db_header(database: &[u8]) -> Result<DBHeader, Error> {
let db_page_size = u16::from_be_bytes([database[16], database[17]]);
// Parse page header from database
let page_header = PageHeader::parse(&database[100..108])?;
// Obtain all cell pointers
let cell_pointers = database[108..]
.chunks_exact(2)
.take(page_header.number_of_cells.into())
.map(|bytes| u16::from_be_bytes(bytes.try_into().unwrap()));
// Obtain all records from column 5
#[allow(unused_variables)]
let schemas = cell_pointers.into_iter().map(|cell_pointer| {
let stream = &database[cell_pointer as usize..];
let (_, offset) = parse_varint(stream);
let (rowid, read_bytes) = parse_varint(&stream[offset..]);
parse_record(&stream[offset + read_bytes..], 5)
.map(|record| Schema::parse(record).expect("Invalid record"))
.unwrap()
});
Ok(DBHeader {
page_size: db_page_size,
schemas: schemas.collect(),
})
}
fn count_rows_in_table(query: &str, db_header: DBHeader, database: &[u8]) -> Result<(), Error> {
let (_, table) = read_column_and_table(query);
// Assume it's valid SQL
let schema = db_header
.schemas
.into_iter()
.find(|schema| schema.table_name == table)
.unwrap();
let table_page_offset = db_header.page_size as usize * (schema.root_page as usize - 1);
let page_header =
PageHeader::parse(&database[table_page_offset..table_page_offset + 8]).unwrap();
println!("{}", page_header.number_of_cells);
Ok(())
}
fn find_column_positions(schema: &str) -> HashMap<&str, usize> { fn find_column_positions(schema: &str) -> HashMap<&str, usize> {
let schema = schema.trim_start_matches(|c| c != '('); let schema = schema.trim_start_matches(|c| c != '(');
let schema = schema let schema = schema