//! Stores a LIVE SELECT query definition on the table use crate::key::category::Categorise; use crate::key::category::Category; use derive::Key; use serde::{Deserialize, Serialize}; use uuid::Uuid; /// Lv is used to track a live query and is cluster independent, i.e. it is tied with a ns/db/tb combo without the cl. /// The live statement includes the node id, so lq can be derived purely from an lv. /// /// The value of the lv is the statement. #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Key)] #[non_exhaustive] pub struct Lq<'a> { __: u8, _a: u8, pub ns: &'a str, _b: u8, pub db: &'a str, _c: u8, pub tb: &'a str, _d: u8, _e: u8, _f: u8, #[serde(with = "uuid::serde::compact")] pub lq: Uuid, } pub fn new<'a>(ns: &'a str, db: &'a str, tb: &'a str, lq: Uuid) -> Lq<'a> { Lq::new(ns, db, tb, lq) } pub fn prefix(ns: &str, db: &str, tb: &str) -> Vec { let mut k = super::all::new(ns, db, tb).encode().unwrap(); k.extend_from_slice(b"!lq\x00"); k } pub fn suffix(ns: &str, db: &str, tb: &str) -> Vec { let mut k = super::all::new(ns, db, tb).encode().unwrap(); k.extend_from_slice(b"!lq\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\x00"); k } impl Categorise for Lq<'_> { fn categorise(&self) -> Category { Category::TableLiveQuery } } impl<'a> Lq<'a> { pub fn new(ns: &'a str, db: &'a str, tb: &'a str, lq: Uuid) -> Self { Self { __: b'/', _a: b'*', ns, _b: b'*', db, _c: b'*', tb, _d: b'!', _e: b'l', _f: b'q', lq, } } } #[cfg(test)] mod tests { #[test] fn key() { use super::*; #[rustfmt::skip] let live_query_id = Uuid::from_bytes([1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16]); let val = Lq::new("testns", "testdb", "testtb", live_query_id); let enc = Lq::encode(&val).unwrap(); assert_eq!( enc, b"/*testns\x00*testdb\x00*testtb\x00!lq\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f\x10" ); let dec = Lq::decode(&enc).unwrap(); assert_eq!(val, dec); } #[test] fn prefix() { let val = super::prefix("testns", "testdb", "testtb"); assert_eq!(val, b"/*testns\x00*testdb\x00*testtb\x00!lq\x00") } #[test] fn suffix() { let val = super::suffix("testns", "testdb", "testtb"); assert_eq!(val, b"/*testns\x00*testdb\x00*testtb\x00!lq\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\x00") } }