// Copyright (c) 2022-2023 Niko Bonnieure, Par le Peuple, NextGraph.org developers // All rights reserved. // Licensed under the Apache License, Version 2.0 // // or the MIT license , // at your option. All files in the project carrying such // notice may not be copied, modified, or distributed except // according to those terms. //! Overlay use p2p_net::types::*; use p2p_repo::kcv_store::KCVStore; use p2p_repo::store::*; use p2p_repo::types::*; use p2p_repo::utils::now_timestamp; use serde::{Deserialize, Serialize}; use serde_bare::{from_slice, to_vec}; // TODO: versioning V0 #[derive(Clone, Debug, Serialize, Deserialize, PartialEq)] pub struct OverlayMeta { pub users: u32, pub last_used: Timestamp, } pub struct Overlay<'a> { /// Overlay ID id: OverlayId, store: &'a dyn KCVStore, } impl<'a> Overlay<'a> { const PREFIX: u8 = b"o"[0]; // propertie's suffixes const SECRET: u8 = b"s"[0]; const PEER: u8 = b"p"[0]; const TOPIC: u8 = b"t"[0]; const META: u8 = b"m"[0]; const REPO: u8 = b"r"[0]; const ALL_PROPERTIES: [u8; 5] = [ Self::SECRET, Self::PEER, Self::TOPIC, Self::META, Self::REPO, ]; const SUFFIX_FOR_EXIST_CHECK: u8 = Self::SECRET; pub fn open(id: &OverlayId, store: &'a dyn KCVStore) -> Result, StorageError> { let opening = Overlay { id: id.clone(), store, }; if !opening.exists() { return Err(StorageError::NotFound); } Ok(opening) } pub fn create( id: &OverlayId, secret: &SymKey, repo: Option, store: &'a dyn KCVStore, ) -> Result, StorageError> { let acc = Overlay { id: id.clone(), store, }; if acc.exists() { return Err(StorageError::BackendError); } store.write_transaction(&|tx| { tx.put( Self::PREFIX, &to_vec(&id)?, Some(Self::SECRET), &to_vec(&secret)?, )?; if repo.is_some() { tx.put( Self::PREFIX, &to_vec(&id)?, Some(Self::REPO), &to_vec(&repo.unwrap())?, )?; } let meta = OverlayMeta { users: 1, last_used: now_timestamp(), }; tx.put( Self::PREFIX, &to_vec(&id)?, Some(Self::META), &to_vec(&meta)?, )?; Ok(()) })?; Ok(acc) } 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) -> OverlayId { self.id } pub fn add_peer(&self, peer: &PeerId) -> Result<(), StorageError> { if !self.exists() { return Err(StorageError::BackendError); } self.store.put( Self::PREFIX, &to_vec(&self.id)?, Some(Self::PEER), to_vec(peer)?, ) } pub fn remove_peer(&self, peer: &PeerId) -> Result<(), StorageError> { self.store.del_property_value( Self::PREFIX, &to_vec(&self.id)?, Some(Self::PEER), to_vec(peer)?, ) } pub fn has_peer(&self, peer: &PeerId) -> Result<(), StorageError> { self.store.has_property_value( Self::PREFIX, &to_vec(&self.id)?, Some(Self::PEER), to_vec(peer)?, ) } pub fn add_topic(&self, topic: &TopicId) -> Result<(), StorageError> { if !self.exists() { return Err(StorageError::BackendError); } self.store.put( Self::PREFIX, &to_vec(&self.id)?, Some(Self::TOPIC), to_vec(topic)?, ) } pub fn remove_topic(&self, topic: &TopicId) -> Result<(), StorageError> { self.store.del_property_value( Self::PREFIX, &to_vec(&self.id)?, Some(Self::TOPIC), to_vec(topic)?, ) } pub fn has_topic(&self, topic: &TopicId) -> Result<(), StorageError> { self.store.has_property_value( Self::PREFIX, &to_vec(&self.id)?, Some(Self::TOPIC), to_vec(topic)?, ) } pub fn secret(&self) -> Result { match self .store .get(Self::PREFIX, &to_vec(&self.id)?, Some(Self::SECRET)) { Ok(secret) => Ok(from_slice::(&secret)?), Err(e) => Err(e), } } pub fn metadata(&self) -> Result { match self .store .get(Self::PREFIX, &to_vec(&self.id)?, Some(Self::META)) { Ok(meta) => Ok(from_slice::(&meta)?), Err(e) => Err(e), } } pub fn set_metadata(&self, meta: &OverlayMeta) -> Result<(), StorageError> { if !self.exists() { return Err(StorageError::BackendError); } self.store.replace( Self::PREFIX, &to_vec(&self.id)?, Some(Self::META), to_vec(meta)?, ) } pub fn repo(&self) -> Result { match self .store .get(Self::PREFIX, &to_vec(&self.id)?, Some(Self::REPO)) { Ok(repo) => Ok(from_slice::(&repo)?), Err(e) => Err(e), } } pub fn del(&self) -> Result<(), StorageError> { self.store .del_all(Self::PREFIX, &to_vec(&self.id)?, &Self::ALL_PROPERTIES) } }