Rust implementation of NextGraph, a Decentralized and local-first web 3.0 ecosystem https://nextgraph.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
nextgraph-rs/p2p-broker/src/broker_store/invitation.rs

192 lines
5.9 KiB

// Copyright (c) 2022-2023 Niko Bonnieure, Par le Peuple, NextGraph.org developers
// All rights reserved.
// Licensed under the Apache License, Version 2.0
// <LICENSE-APACHE2 or http://www.apache.org/licenses/LICENSE-2.0>
// or the MIT license <LICENSE-MIT or http://opensource.org/licenses/MIT>,
// at your option. All files in the project carrying such
// notice may not be copied, modified, or distributed except
// according to those terms.
//! User account
use std::collections::hash_map::DefaultHasher;
use std::hash::Hash;
use std::hash::Hasher;
use std::time::SystemTime;
use p2p_net::errors::ProtocolError;
use p2p_net::types::*;
use p2p_repo::kcv_store::KCVStore;
use p2p_repo::store::*;
use p2p_repo::types::SymKey;
use p2p_repo::types::Timestamp;
use p2p_repo::utils::now_timestamp;
use serde_bare::from_slice;
use serde_bare::to_vec;
pub struct Invitation<'a> {
/// User ID
id: [u8; 32],
store: &'a dyn KCVStore,
}
impl<'a> Invitation<'a> {
const PREFIX: u8 = b"i"[0];
// propertie's invitation suffixes
const TYPE: u8 = b"t"[0];
//const EXPIRE: u8 = b"e"[0];
const PREFIX_EXPIRE: u8 = b"e"[0];
// propertie's expiry suffixes
const INVITATION: u8 = b"i"[0];
const ALL_PROPERTIES: [u8; 1] = [Self::TYPE];
const SUFFIX_FOR_EXIST_CHECK: u8 = Self::TYPE;
pub fn open(id: &[u8; 32], store: &'a dyn KCVStore) -> Result<Invitation<'a>, StorageError> {
let opening = Invitation {
id: id.clone(),
store,
};
if !opening.exists() {
return Err(StorageError::NotFound);
}
Ok(opening)
}
pub fn create(
id: &InvitationCode,
expiry: u32,
memo: &Option<String>,
store: &'a dyn KCVStore,
) -> Result<Invitation<'a>, StorageError> {
let (code_type, code) = match id {
InvitationCode::Unique(c) => (0u8, c.slice()),
InvitationCode::Multi(c) => (1u8, c.slice()),
InvitationCode::Admin(c) => (2u8, c.slice()),
};
let acc = Invitation {
id: code.clone(),
store,
};
if acc.exists() {
return Err(StorageError::BackendError);
}
let mut value = to_vec(&(code_type, expiry, memo.clone()))?;
store.write_transaction(&|tx| {
tx.put(Self::PREFIX, &to_vec(code)?, Some(Self::TYPE), &value)?;
Ok(())
})?;
Ok(acc)
}
pub fn get_all_invitations(
store: &'a dyn KCVStore,
mut admin: bool,
mut unique: bool,
mut multi: bool,
) -> Result<Vec<(InvitationCode, u32, Option<String>)>, StorageError> {
let size = to_vec(&[0u8; 32])?.len();
let mut res: Vec<(InvitationCode, u32, Option<String>)> = vec![];
if !admin && !unique && !multi {
admin = true;
unique = true;
multi = true;
}
for invite in store.get_all_keys_and_values(Self::PREFIX, size, None)? {
if invite.0.len() == size + 2 {
let code: [u8; 32] = from_slice(&invite.0[1..invite.0.len() - 1])?;
if invite.0[size + 1] == Self::TYPE {
let code_type: (u8, u32, Option<String>) = from_slice(&invite.1)?;
let inv_code = match code_type {
(0, ex, memo) => {
if unique {
Some((InvitationCode::Unique(SymKey::ChaCha20Key(code)), ex, memo))
} else {
None
}
}
(1, ex, memo) => {
if multi {
Some((InvitationCode::Multi(SymKey::ChaCha20Key(code)), ex, memo))
} else {
None
}
}
(2, ex, memo) => {
if admin {
Some((InvitationCode::Admin(SymKey::ChaCha20Key(code)), ex, memo))
} else {
None
}
}
_ => panic!("invalid code type value"),
};
if inv_code.is_some() {
res.push(inv_code.unwrap());
}
}
}
}
Ok(res)
}
pub fn exists(&self) -> bool {
self.store
.get(
Self::PREFIX,
&to_vec(&self.id).unwrap(),
Some(Self::SUFFIX_FOR_EXIST_CHECK),
)
.is_ok()
}
pub fn id(&self) -> [u8; 32] {
self.id
}
pub fn get_type(&self) -> Result<u8, ProtocolError> {
let type_ser = self
.store
.get(Self::PREFIX, &to_vec(&self.id)?, Some(Self::TYPE))?;
let t: (u8, u32, Option<String>) = from_slice(&type_ser)?;
if t.1 < now_timestamp() {
return Err(ProtocolError::Expired);
}
Ok(t.0)
}
pub fn is_expired(&self) -> Result<bool, StorageError> {
let expire_ser = self
.store
.get(Self::PREFIX, &to_vec(&self.id)?, Some(Self::TYPE))?;
let expire: (u8, u32, Option<String>) = from_slice(&expire_ser)?;
if expire.1 < now_timestamp() {
return Ok(true);
}
Ok(false)
}
pub fn del(&self) -> Result<(), StorageError> {
self.store.write_transaction(&|tx| {
tx.del_all(Self::PREFIX, &to_vec(&self.id)?, &Self::ALL_PROPERTIES)?;
Ok(())
})
}
}
#[cfg(test)]
mod test {
use p2p_repo::store::*;
use p2p_repo::types::*;
use p2p_repo::utils::*;
use std::fs;
use stores_lmdb::kcv_store::LmdbKCVStore;
use tempfile::Builder;
use crate::broker_store::account::Account;
#[test]
pub fn test_invitation() {}
}