user.rs 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. use crate::categories::CatStats;
  2. use derive_more::From;
  3. use pickledb::{PickleDb, PickleDbDumpPolicy, SerializationMethod};
  4. use radix_trie::Trie;
  5. use shellexpand::tilde;
  6. use std::collections::HashSet;
  7. use std::fmt;
  8. use std::path::PathBuf;
  9. use std::time::Duration;
  10. use thiserror::Error;
  11. /// Configuration for single user
  12. pub struct User {
  13. /// User id
  14. pub uid: i64,
  15. /// Categories statistics for the user
  16. pub catmap: CatStats,
  17. /// Available accounts for the user
  18. pub accounts: HashSet<String>,
  19. /// database with config
  20. db: PickleDb,
  21. }
  22. #[cfg(not(feature = "docker"))]
  23. pub const DEFAULT_DB_PATH: &str = "~/.config/receqif/";
  24. #[cfg(feature = "docker")]
  25. pub const DEFAULT_DB_PATH: &str = "/etc/receqif/";
  26. #[derive(Debug, Error, From)]
  27. pub enum UserError {
  28. #[error("Database error: {0}")]
  29. DbError(#[source] pickledb::error::Error),
  30. }
  31. impl Drop for User {
  32. fn drop(&mut self) {
  33. self.save_data().unwrap();
  34. }
  35. }
  36. impl fmt::Debug for User {
  37. fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
  38. f.debug_struct("User")
  39. .field("uid", &self.uid)
  40. .field("db", &format_args!("<PickleDb>"))
  41. .finish()
  42. }
  43. }
  44. impl User {
  45. pub fn new(uid: i64, dbfile: &Option<String>) -> Self {
  46. let ten_sec = Duration::from_secs(10);
  47. let path: String = match dbfile {
  48. Some(path) => path.to_string(),
  49. None => DEFAULT_DB_PATH.to_owned() + &uid.to_string() + ".db",
  50. };
  51. let confpath: &str = &tilde(&path);
  52. let confpath = PathBuf::from(confpath);
  53. let dbase = PickleDb::load(
  54. &confpath,
  55. PickleDbDumpPolicy::PeriodicDump(ten_sec),
  56. SerializationMethod::Json,
  57. );
  58. let db = match dbase {
  59. Ok(db) => db,
  60. Err(_) => PickleDb::new(
  61. &confpath,
  62. PickleDbDumpPolicy::PeriodicDump(ten_sec),
  63. SerializationMethod::Json,
  64. ),
  65. };
  66. let catmap: CatStats = match db.get("catmap") {
  67. Some(v) => v,
  68. None => Trie::new(),
  69. };
  70. let accounts = match db.get::<Vec<String>>("accounts") {
  71. Some(a) => HashSet::from_iter(a),
  72. None => HashSet::new(),
  73. };
  74. User {
  75. uid,
  76. catmap,
  77. accounts,
  78. db,
  79. }
  80. }
  81. pub fn accounts(&mut self, acc: Vec<String>) {
  82. self.accounts = HashSet::from_iter(acc);
  83. }
  84. pub fn save_data(&mut self) -> Result<(), UserError> {
  85. log::debug!("Saving user data");
  86. self.db
  87. .set("catmap", &self.catmap)
  88. .map_err(UserError::DbError)?;
  89. self.db
  90. .set("accounts", &self.accounts)
  91. .map_err(UserError::DbError)?;
  92. self.db.dump().map_err(UserError::DbError)?;
  93. Ok(())
  94. }
  95. pub fn new_account(&mut self, acc: String) {
  96. self.accounts.insert(acc);
  97. }
  98. }
  99. #[cfg(test)]
  100. mod tests {
  101. use super::*;
  102. use std::fs::create_dir_all;
  103. const TEST_DB_DIR: &str = "/tmp/receqif_test/";
  104. fn setup(db_suffix: &str) -> Result<User, Box<dyn std::error::Error>> {
  105. create_dir_all(TEST_DB_DIR)?;
  106. let temp_db_path = format!("{}test_user_{}.db", TEST_DB_DIR, db_suffix);
  107. let user = User::new(123, &Some(temp_db_path));
  108. Ok(user)
  109. }
  110. #[test]
  111. fn test_user_initialization() {
  112. let user = setup("init").expect("Failed to initialize user");
  113. assert!(user.accounts.is_empty());
  114. assert_eq!(user.catmap, Trie::new());
  115. }
  116. #[test]
  117. fn test_adding_account() {
  118. let mut user = setup("add_acc").expect("Failed to set up user for adding account");
  119. user.new_account("account".to_string());
  120. assert!(user.accounts.contains("account"));
  121. }
  122. #[test]
  123. fn test_saving_data() {
  124. let mut user = setup("save_data").expect("Failed to set up user for saving data");
  125. user.new_account("account".to_string());
  126. user.save_data().expect("Failed to save data");
  127. // Recreate the user object to verify data persistence
  128. let reloaded_user = setup("save_data").expect("Failed to reload user for verifying data");
  129. assert!(
  130. reloaded_user.accounts.contains("account"),
  131. "Account not found in reloaded user"
  132. );
  133. }
  134. }