use std::ffi::{c_char, CStr}; use std::ptr; use std::str; use crate::ffi; use crate::{Connection, Error, Name, Result, Statement}; /// Information about a column of a SQLite query. #[cfg(feature = "column_decltype")] #[derive(Debug)] pub struct Column<'stmt> { name: &'stmt str, decl_type: Option<&'stmt str>, } #[cfg(feature = "column_decltype")] impl Column<'_> { /// Returns the name of the column. #[inline] #[must_use] pub fn name(&self) -> &str { self.name } /// Returns the type of the column (`None` for expression). #[inline] #[must_use] pub fn decl_type(&self) -> Option<&str> { self.decl_type } } /// Metadata about the origin of a column of a SQLite query #[cfg(feature = "column_metadata")] #[derive(Debug)] pub struct ColumnMetadata<'stmt> { name: &'stmt str, database_name: Option<&'stmt str>, table_name: Option<&'stmt str>, origin_name: Option<&'stmt str>, } #[cfg(feature = "column_metadata")] impl ColumnMetadata<'_> { #[inline] #[must_use] /// Returns the name of the column in the query results pub fn name(&self) -> &str { self.name } #[inline] #[must_use] /// Returns the database name from which the column originates pub fn database_name(&self) -> Option<&str> { self.database_name } #[inline] #[must_use] /// Returns the table name from which the column originates pub fn table_name(&self) -> Option<&str> { self.table_name } #[inline] #[must_use] /// Returns the column name from which the column originates pub fn origin_name(&self) -> Option<&str> { self.origin_name } } impl Statement<'_> { /// Get all the column names in the result set of the prepared statement. /// /// If associated DB schema can be altered concurrently, you should make /// sure that current statement has already been stepped once before /// calling this method. pub fn column_names(&self) -> Vec<&str> { let n = self.column_count(); let mut cols = Vec::with_capacity(n); for i in 0..n { let s = self.column_name_unwrap(i); cols.push(s); } cols } /// Return the number of columns in the result set returned by the prepared /// statement. /// /// If associated DB schema can be altered concurrently, you should make /// sure that current statement has already been stepped once before /// calling this method. #[inline] pub fn column_count(&self) -> usize { self.stmt.column_count() } /// Check that column name reference lifetime is limited: /// /// > The returned string pointer is valid... /// /// `column_name` reference can become invalid if `stmt` is reprepared /// (because of schema change) when `query_row` is called. So we assert /// that a compilation error happens if this reference is kept alive: /// ```compile_fail /// use rusqlite::{Connection, Result}; /// fn main() -> Result<()> { /// let db = Connection::open_in_memory()?; /// let mut stmt = db.prepare("SELECT 1 as x")?; /// let column_name = stmt.column_name(0)?; /// let x = stmt.query_row([], |r| r.get::<_, i64>(0))?; // E0502 /// assert_eq!(1, x); /// assert_eq!("x", column_name); /// Ok(()) /// } /// ``` #[inline] pub(super) fn column_name_unwrap(&self, col: usize) -> &str { // Just panic if the bounds are wrong for now, we never call this // without checking first. self.column_name(col).expect("Column out of bounds") } /// Returns the name assigned to a particular column in the result set /// returned by the prepared statement. /// /// If associated DB schema can be altered concurrently, you should make /// sure that current statement has already been stepped once before /// calling this method. /// /// ## Failure /// /// Returns an `Error::InvalidColumnIndex` if `idx` is outside the valid /// column range for this row. /// /// # Panics /// /// Panics when column name is not valid UTF-8. #[inline] pub fn column_name(&self, col: usize) -> Result<&str> { self.stmt .column_name(col) // clippy::or_fun_call (nightly) vs clippy::unnecessary-lazy-evaluations (stable) .ok_or(Error::InvalidColumnIndex(col)) .map(|slice| { slice .to_str() .expect("Invalid UTF-8 sequence in column name") }) } /// Returns the column index in the result set for a given column name. /// /// If there is no AS clause then the name of the column is unspecified and /// may change from one release of SQLite to the next. /// /// If associated DB schema can be altered concurrently, you should make /// sure that current statement has already been stepped once before /// calling this method. /// /// # Failure /// /// Will return an `Error::InvalidColumnName` when there is no column with /// the specified `name`. #[inline] pub fn column_index(&self, name: &str) -> Result { let bytes = name.as_bytes(); let n = self.column_count(); for i in 0..n { // Note: `column_name` is only fallible if `i` is out of bounds, // which we've already checked. if bytes.eq_ignore_ascii_case(self.stmt.column_name(i).unwrap().to_bytes()) { return Ok(i); } } Err(Error::InvalidColumnName(String::from(name))) } /// Returns a slice describing the columns of the result of the query. /// /// If associated DB schema can be altered concurrently, you should make /// sure that current statement has already been stepped once before /// calling this method. #[cfg(feature = "column_decltype")] pub fn columns(&self) -> Vec> { let n = self.column_count(); let mut cols = Vec::with_capacity(n); for i in 0..n { let name = self.column_name_unwrap(i); let slice = self.stmt.column_decltype(i); let decl_type = slice.map(|s| { s.to_str() .expect("Invalid UTF-8 sequence in column declaration") }); cols.push(Column { name, decl_type }); } cols } /// Returns the names of the database, table, and row from which /// each column of this query's results originate. /// /// Computed or otherwise derived columns will have None values for these fields. #[cfg(feature = "column_metadata")] pub fn columns_with_metadata(&self) -> Vec> { let n = self.column_count(); let mut col_mets = Vec::with_capacity(n); for i in 0..n { let name = self.column_name_unwrap(i); let db_slice = self.stmt.column_database_name(i); let tbl_slice = self.stmt.column_table_name(i); let origin_slice = self.stmt.column_origin_name(i); col_mets.push(ColumnMetadata { name, database_name: db_slice.map(|s| { s.to_str() .expect("Invalid UTF-8 sequence in column db name") }), table_name: tbl_slice.map(|s| { s.to_str() .expect("Invalid UTF-8 sequence in column table name") }), origin_name: origin_slice.map(|s| { s.to_str() .expect("Invalid UTF-8 sequence in column origin name") }), }) } col_mets } /// Extract metadata of column at specified index /// /// Returns: /// - database name /// - table name /// - original column name /// - declared data type /// - name of default collation sequence /// - True if column has a NOT NULL constraint /// - True if column is part of the PRIMARY KEY /// - True if column is AUTOINCREMENT /// /// See [Connection::column_metadata] #[cfg(feature = "column_metadata")] #[allow(clippy::type_complexity)] pub fn column_metadata( &self, col: usize, ) -> Result< Option<( &CStr, &CStr, &CStr, Option<&CStr>, Option<&CStr>, bool, bool, bool, )>, > { let db_name = self.stmt.column_database_name(col); let table_name = self.stmt.column_table_name(col); let origin_name = self.stmt.column_origin_name(col); if db_name.is_none() || table_name.is_none() || origin_name.is_none() { return Ok(None); } let (data_type, coll_seq, not_null, primary_key, auto_inc) = self.conn .column_metadata(db_name, table_name.unwrap(), origin_name.unwrap())?; Ok(Some(( db_name.unwrap(), table_name.unwrap(), origin_name.unwrap(), data_type, coll_seq, not_null, primary_key, auto_inc, ))) } } impl Connection { /// Check if `table_name`.`column_name` exists. /// /// `db_name` is main, temp, the name in ATTACH, or `None` to search all databases. pub fn column_exists( &self, db_name: Option, table_name: N, column_name: N, ) -> Result { self.exists(db_name, table_name, Some(column_name)) } /// Check if `table_name` exists. /// /// `db_name` is main, temp, the name in ATTACH, or `None` to search all databases. pub fn table_exists(&self, db_name: Option, table_name: N) -> Result { self.exists(db_name, table_name, None) } /// Extract metadata of column at specified index /// /// Returns: /// - declared data type /// - name of default collation sequence /// - True if column has a NOT NULL constraint /// - True if column is part of the PRIMARY KEY /// - True if column is AUTOINCREMENT #[allow(clippy::type_complexity)] pub fn column_metadata( &self, db_name: Option, table_name: N, column_name: N, ) -> Result<(Option<&CStr>, Option<&CStr>, bool, bool, bool)> { let cs = db_name.as_ref().map(N::as_cstr).transpose()?; let db_name = cs.as_ref().map(|s| s.as_ptr()).unwrap_or(ptr::null()); let table_name = table_name.as_cstr()?; let column_name = column_name.as_cstr()?; let mut data_type: *const c_char = ptr::null_mut(); let mut coll_seq: *const c_char = ptr::null_mut(); let mut not_null = 0; let mut primary_key = 0; let mut auto_inc = 0; self.decode_result(unsafe { ffi::sqlite3_table_column_metadata( self.handle(), db_name, table_name.as_ptr(), column_name.as_ptr(), &mut data_type, &mut coll_seq, &mut not_null, &mut primary_key, &mut auto_inc, ) })?; Ok(( if data_type.is_null() { None } else { Some(unsafe { CStr::from_ptr(data_type) }) }, if coll_seq.is_null() { None } else { Some(unsafe { CStr::from_ptr(coll_seq) }) }, not_null != 0, primary_key != 0, auto_inc != 0, )) } fn exists( &self, db_name: Option, table_name: N, column_name: Option, ) -> Result { let cs = db_name.as_ref().map(N::as_cstr).transpose()?; let db_name = cs.as_ref().map(|s| s.as_ptr()).unwrap_or(ptr::null()); let table_name = table_name.as_cstr()?; let cn = column_name.as_ref().map(N::as_cstr).transpose()?; let column_name = cn.as_ref().map(|s| s.as_ptr()).unwrap_or(ptr::null()); let r = unsafe { ffi::sqlite3_table_column_metadata( self.handle(), db_name, table_name.as_ptr(), column_name, ptr::null_mut(), ptr::null_mut(), ptr::null_mut(), ptr::null_mut(), ptr::null_mut(), ) }; match r { ffi::SQLITE_OK => Ok(true), ffi::SQLITE_ERROR => Ok(false), _ => self.db.borrow().decode_result(r).map(|_| false), } } } #[cfg(test)] mod test { use crate::{Connection, Result}; #[test] #[cfg(feature = "column_decltype")] fn test_columns() -> Result<()> { use super::Column; let db = Connection::open_in_memory()?; let query = db.prepare("SELECT * FROM sqlite_master")?; let columns = query.columns(); let column_names: Vec<&str> = columns.iter().map(Column::name).collect(); assert_eq!( column_names.as_slice(), &["type", "name", "tbl_name", "rootpage", "sql"] ); let column_types: Vec> = columns .iter() .map(|col| col.decl_type().map(str::to_lowercase)) .collect(); assert_eq!( &column_types[..3], &[ Some("text".to_owned()), Some("text".to_owned()), Some("text".to_owned()), ] ); Ok(()) } #[test] #[cfg(feature = "column_metadata")] fn test_columns_with_metadata() -> Result<()> { let db = Connection::open_in_memory()?; let query = db.prepare("SELECT *, 1 FROM sqlite_master")?; let col_mets = query.columns_with_metadata(); assert_eq!(col_mets.len(), 6); for col in col_mets.iter().take(5) { assert_eq!(&col.database_name(), &Some("main")); assert_eq!(&col.table_name(), &Some("sqlite_master")); } assert!(col_mets[5].database_name().is_none()); assert!(col_mets[5].table_name().is_none()); assert!(col_mets[5].origin_name().is_none()); let col_origins: Vec> = col_mets.iter().map(|col| col.origin_name()).collect(); assert_eq!( &col_origins[..5], &[ Some("type"), Some("name"), Some("tbl_name"), Some("rootpage"), Some("sql"), ] ); Ok(()) } #[test] fn test_column_name_in_error() -> Result<()> { use crate::{types::Type, Error}; let db = Connection::open_in_memory()?; db.execute_batch( "BEGIN; CREATE TABLE foo(x INTEGER, y TEXT); INSERT INTO foo VALUES(4, NULL); END;", )?; let mut stmt = db.prepare("SELECT x as renamed, y FROM foo")?; let mut rows = stmt.query([])?; let row = rows.next()?.unwrap(); match row.get::<_, String>(0).unwrap_err() { Error::InvalidColumnType(idx, name, ty) => { assert_eq!(idx, 0); assert_eq!(name, "renamed"); assert_eq!(ty, Type::Integer); } e => { panic!("Unexpected error type: {e:?}"); } } match row.get::<_, String>("y").unwrap_err() { Error::InvalidColumnType(idx, name, ty) => { assert_eq!(idx, 1); assert_eq!(name, "y"); assert_eq!(ty, Type::Null); } e => { panic!("Unexpected error type: {e:?}"); } } Ok(()) } /// `column_name` reference should stay valid until `stmt` is reprepared (or /// reset) even if DB schema is altered (SQLite documentation is /// ambiguous here because it says reference "is valid until (...) the next /// call to `sqlite3_column_name()` or `sqlite3_column_name16()` on the same /// column.". We assume that reference is valid if only /// `sqlite3_column_name()` is used): #[test] #[cfg(feature = "modern_sqlite")] fn test_column_name_reference() -> Result<()> { let db = Connection::open_in_memory()?; db.execute_batch("CREATE TABLE y (x);")?; let stmt = db.prepare("SELECT x FROM y;")?; let column_name = stmt.column_name(0)?; assert_eq!("x", column_name); db.execute_batch("ALTER TABLE y RENAME COLUMN x TO z;")?; // column name is not refreshed until statement is re-prepared let same_column_name = stmt.column_name(0)?; assert_eq!(same_column_name, column_name); Ok(()) } #[test] #[cfg(feature = "column_metadata")] fn stmt_column_metadata() -> Result<()> { let db = Connection::open_in_memory()?; let query = db.prepare("SELECT *, 1 FROM sqlite_schema")?; let (db_name, table_name, col_name, data_type, coll_seq, not_null, primary_key, auto_inc) = query.column_metadata(0)?.unwrap(); assert_eq!(db_name, crate::MAIN_DB); assert_eq!(table_name, c"sqlite_master"); assert_eq!(col_name, c"type"); assert_eq!(data_type, Some(c"TEXT")); assert_eq!(coll_seq, Some(c"BINARY")); assert!(!not_null); assert!(!primary_key); assert!(!auto_inc); assert!(query.column_metadata(5)?.is_none()); Ok(()) } #[test] fn column_exists() -> Result<()> { let db = Connection::open_in_memory()?; assert!(db.column_exists(None, c"sqlite_schema", c"type")?); assert!(db.column_exists(Some(crate::TEMP_DB), c"sqlite_schema", c"type")?); assert!(!db.column_exists(Some(crate::MAIN_DB), c"sqlite_temp_schema", c"type")?); Ok(()) } #[test] fn table_exists() -> Result<()> { let db = Connection::open_in_memory()?; assert!(db.table_exists(None, c"sqlite_schema")?); assert!(db.table_exists(Some(crate::TEMP_DB), c"sqlite_schema")?); assert!(!db.table_exists(Some(crate::MAIN_DB), c"sqlite_temp_schema")?); Ok(()) } #[test] fn column_metadata() -> Result<()> { let db = Connection::open_in_memory()?; let (data_type, coll_seq, not_null, primary_key, auto_inc) = db.column_metadata(None, c"sqlite_schema", c"type")?; assert_eq!(data_type, Some(c"TEXT")); assert_eq!(coll_seq, Some(c"BINARY")); assert!(!not_null); assert!(!primary_key); assert!(!auto_inc); assert!(db.column_metadata(None, c"sqlite_schema", c"foo").is_err()); Ok(()) } }