mirror of
https://github.com/YGGverse/Yoda.git
synced 2025-01-15 17:20:08 +00:00
add memory cache for auth index
This commit is contained in:
parent
9530c37c59
commit
d9bf85884b
@ -41,17 +41,15 @@ impl Identity {
|
|||||||
///
|
///
|
||||||
/// https://geminiprotocol.net/docs/protocol-specification.gmi#client-certificates
|
/// https://geminiprotocol.net/docs/protocol-specification.gmi#client-certificates
|
||||||
pub fn gemini(&self, request: &str) -> Option<String> {
|
pub fn gemini(&self, request: &str) -> Option<String> {
|
||||||
if let Ok(auth_records) = self.gemini.auth.database.records(Some(request)) {
|
for profile_identity_gemini_id in self.gemini.auth.memory.starts_with(request) {
|
||||||
for auth_record in auth_records {
|
|
||||||
if let Ok(gemini_records) = self.gemini.database.records() {
|
if let Ok(gemini_records) = self.gemini.database.records() {
|
||||||
for gemini_record in gemini_records {
|
for gemini_record in gemini_records {
|
||||||
if gemini_record.id == auth_record.profile_identity_gemini_id {
|
if gemini_record.id == profile_identity_gemini_id {
|
||||||
return Some(gemini_record.pem);
|
return Some(gemini_record.pem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
None
|
None
|
||||||
} // @TODO apply protocol rules to selection
|
} // @TODO apply protocol rules to selection
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
mod database;
|
mod database;
|
||||||
|
mod memory;
|
||||||
|
|
||||||
use database::Database;
|
use database::Database;
|
||||||
|
use memory::Memory;
|
||||||
|
|
||||||
use sqlite::{Connection, Transaction};
|
use sqlite::{Connection, Transaction};
|
||||||
use std::{rc::Rc, sync::RwLock};
|
use std::{rc::Rc, sync::RwLock};
|
||||||
|
|
||||||
/// API for `profile_identity_gemini_id` + `url` auth pairs operations
|
/// API for `profile_identity_gemini_id` + `url` auth pairs operations
|
||||||
pub struct Auth {
|
pub struct Auth {
|
||||||
pub database: Rc<Database>,
|
// pub database: Rc<Database>,
|
||||||
|
pub memory: Rc<Memory>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Auth {
|
impl Auth {
|
||||||
@ -15,8 +18,31 @@ impl Auth {
|
|||||||
|
|
||||||
/// Create new `Self`
|
/// Create new `Self`
|
||||||
pub fn new(connection: Rc<RwLock<Connection>>) -> Self {
|
pub fn new(connection: Rc<RwLock<Connection>>) -> Self {
|
||||||
|
// Init children components
|
||||||
|
let database = Rc::new(Database::new(connection));
|
||||||
|
let memory = Rc::new(Memory::new());
|
||||||
|
|
||||||
|
// Build initial index
|
||||||
|
match database.records(None) {
|
||||||
|
Ok(records) => {
|
||||||
|
for record in records {
|
||||||
|
if record.is_active {
|
||||||
|
if memory
|
||||||
|
.add(record.url, record.profile_identity_gemini_id)
|
||||||
|
.is_err()
|
||||||
|
{
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(reason) => todo!("{reason}"),
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return new `Self`
|
||||||
Self {
|
Self {
|
||||||
database: Rc::new(Database::new(connection)),
|
// database,
|
||||||
|
memory,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,8 @@ use sqlite::{Connection, Error, Transaction};
|
|||||||
pub struct Table {
|
pub struct Table {
|
||||||
//pub id: i64,
|
//pub id: i64,
|
||||||
pub profile_identity_gemini_id: i64,
|
pub profile_identity_gemini_id: i64,
|
||||||
|
pub is_active: bool,
|
||||||
|
pub url: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Storage for `profile_identity_gemini_id` + `url` auth pairs
|
/// Storage for `profile_identity_gemini_id` + `url` auth pairs
|
||||||
@ -56,7 +58,9 @@ pub fn init(tx: &Transaction) -> Result<usize, Error> {
|
|||||||
pub fn select(tx: &Transaction, url: Option<&str>) -> Result<Vec<Table>, Error> {
|
pub fn select(tx: &Transaction, url: Option<&str>) -> Result<Vec<Table>, Error> {
|
||||||
let mut stmt = tx.prepare(
|
let mut stmt = tx.prepare(
|
||||||
"SELECT `id`,
|
"SELECT `id`,
|
||||||
`profile_identity_gemini_id`
|
`profile_identity_gemini_id`,
|
||||||
|
`is_active`,
|
||||||
|
`url`
|
||||||
|
|
||||||
FROM `profile_identity_gemini_auth`
|
FROM `profile_identity_gemini_auth`
|
||||||
WHERE `url` LIKE ?",
|
WHERE `url` LIKE ?",
|
||||||
@ -66,6 +70,8 @@ pub fn select(tx: &Transaction, url: Option<&str>) -> Result<Vec<Table>, Error>
|
|||||||
Ok(Table {
|
Ok(Table {
|
||||||
//id: row.get(0)?,
|
//id: row.get(0)?,
|
||||||
profile_identity_gemini_id: row.get(1)?,
|
profile_identity_gemini_id: row.get(1)?,
|
||||||
|
is_active: row.get(2)?,
|
||||||
|
url: row.get(3)?,
|
||||||
})
|
})
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
|
52
src/profile/identity/gemini/auth/memory.rs
Normal file
52
src/profile/identity/gemini/auth/memory.rs
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
mod error;
|
||||||
|
use error::Error;
|
||||||
|
|
||||||
|
use std::{cell::RefCell, collections::HashMap};
|
||||||
|
|
||||||
|
/// Reduce disk usage by cache Auth index in memory
|
||||||
|
pub struct Memory {
|
||||||
|
index: RefCell<HashMap<String, i64>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Memory {
|
||||||
|
// Constructors
|
||||||
|
|
||||||
|
/// Create new `Self`
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
index: RefCell::new(HashMap::new()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
|
||||||
|
/// Add new record with `url` as key and `profile_identity_gemini_id` as value
|
||||||
|
/// * validate record with same key does not exist yet
|
||||||
|
pub fn add(&self, url: String, profile_identity_gemini_id: i64) -> Result<(), Error> {
|
||||||
|
match self
|
||||||
|
.index
|
||||||
|
.borrow_mut()
|
||||||
|
.insert(url, profile_identity_gemini_id)
|
||||||
|
{
|
||||||
|
Some(_) => Err(Error::Overwrite), // @TODO prevent?
|
||||||
|
None => Ok(()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* @TODO update feature
|
||||||
|
/// Cleanup index
|
||||||
|
pub fn clear(&self, url: &str) {
|
||||||
|
self.index.borrow_mut().clear()
|
||||||
|
} */
|
||||||
|
|
||||||
|
/// Search for `profile_identity_gemini_id` by `url` starts with given substring
|
||||||
|
pub fn starts_with(&self, prefix: &str) -> Vec<i64> {
|
||||||
|
let mut result = Vec::new();
|
||||||
|
for (url, &profile_identity_gemini_id) in self.index.borrow().iter() {
|
||||||
|
if url.starts_with(prefix) {
|
||||||
|
result.push(profile_identity_gemini_id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result
|
||||||
|
}
|
||||||
|
}
|
4
src/profile/identity/gemini/auth/memory/error.rs
Normal file
4
src/profile/identity/gemini/auth/memory/error.rs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
#[derive(Debug)]
|
||||||
|
pub enum Error {
|
||||||
|
Overwrite,
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user