use atoi::atoi; use bytes::Bytes; use memchr::memrchr; use crate::error::Error; use crate::io::Decode; #[derive(Debug)] pub struct CommandComplete { /// The command tag. This is usually a single word that identifies which SQL command /// was completed. tag: Bytes, } impl Decode<'_> for CommandComplete { #[inline] fn decode_with(buf: Bytes, _: ()) -> Result { Ok(CommandComplete { tag: buf }) } } impl CommandComplete { /// Returns the number of rows affected. /// If the command does not return rows (e.g., "CREATE TABLE"), returns 0. pub fn rows_affected(&self) -> u64 { // Look backwards for the first SPACE memrchr(b' ', &self.tag) // This is either a word or the number of rows affected .and_then(|i| atoi(&self.tag[(i + 1)..])) .unwrap_or(0) } } #[test] fn test_decode_command_complete_for_insert() { const DATA: &[u8] = b"INSERT 0 1214\0"; let cc = CommandComplete::decode(Bytes::from_static(DATA)).unwrap(); assert_eq!(cc.rows_affected(), 1214); } #[test] fn test_decode_command_complete_for_begin() { const DATA: &[u8] = b"BEGIN\0"; let cc = CommandComplete::decode(Bytes::from_static(DATA)).unwrap(); assert_eq!(cc.rows_affected(), 0); } #[test] fn test_decode_command_complete_for_update() { const DATA: &[u8] = b"UPDATE 5\0"; let cc = CommandComplete::decode(Bytes::from_static(DATA)).unwrap(); assert_eq!(cc.rows_affected(), 5); } #[cfg(all(test, not(debug_assertions)))] #[bench] fn bench_decode_command_complete(b: &mut test::Bencher) { const DATA: &[u8] = b"INSERT 0 1214\0"; b.iter(|| { let _ = CommandComplete::decode(test::black_box(Bytes::from_static(DATA))); }); } #[cfg(all(test, not(debug_assertions)))] #[bench] fn bench_decode_command_complete_rows_affected(b: &mut test::Bencher) { const DATA: &[u8] = b"INSERT 0 1214\0"; let data = CommandComplete::decode(Bytes::from_static(DATA)).unwrap(); b.iter(|| { let _rows = test::black_box(&data).rows_affected(); }); }