This commit is contained in:
John Smith 2023-05-08 10:40:13 -04:00
parent 76b8d569dc
commit 43a2c0b699
4 changed files with 181 additions and 162 deletions

View File

@ -20,9 +20,8 @@ impl StorageManager {
rpc_processor: RPCProcessor, rpc_processor: RPCProcessor,
key: TypedKey, key: TypedKey,
subkey: ValueSubkey, subkey: ValueSubkey,
last_value: Option<SignedValueData>,
last_descriptor: Option<SignedValueDescriptor>,
safety_selection: SafetySelection, safety_selection: SafetySelection,
last_subkey_result: SubkeyResult,
) -> Result<SubkeyResult, VeilidAPIError> { ) -> Result<SubkeyResult, VeilidAPIError> {
let routing_table = rpc_processor.routing_table(); let routing_table = rpc_processor.routing_table();
@ -38,15 +37,15 @@ impl StorageManager {
}; };
// Make do-get-value answer context // Make do-get-value answer context
let schema = if let Some(d) = &last_descriptor { let schema = if let Some(d) = &last_subkey_result.descriptor {
Some(d.schema()?) Some(d.schema()?)
} else { } else {
None None
}; };
let context = Arc::new(Mutex::new(DoGetValueContext { let context = Arc::new(Mutex::new(DoGetValueContext {
value: last_value, value: last_subkey_result.value,
value_count: 0, value_count: 0,
descriptor: last_descriptor.clone(), descriptor: last_subkey_result.descriptor.clone(),
schema, schema,
})); }));
@ -54,7 +53,7 @@ impl StorageManager {
let call_routine = |next_node: NodeRef| { let call_routine = |next_node: NodeRef| {
let rpc_processor = rpc_processor.clone(); let rpc_processor = rpc_processor.clone();
let context = context.clone(); let context = context.clone();
let last_descriptor = last_descriptor.clone(); let last_descriptor = last_subkey_result.descriptor.clone();
async move { async move {
let vres = rpc_processor let vres = rpc_processor
.clone() .clone()

View File

@ -57,8 +57,8 @@ impl StorageManager {
flush_record_stores_task: TickTask::new(FLUSH_RECORD_STORES_INTERVAL_SECS), flush_record_stores_task: TickTask::new(FLUSH_RECORD_STORES_INTERVAL_SECS),
} }
} }
fn new_inner() -> StorageManagerInner { fn new_inner(unlocked_inner: Arc<StorageManagerUnlockedInner>) -> StorageManagerInner {
StorageManagerInner::default() StorageManagerInner::new(unlocked_inner)
} }
fn local_limits_from_config(config: VeilidConfig) -> RecordStoreLimits { fn local_limits_from_config(config: VeilidConfig) -> RecordStoreLimits {
@ -96,15 +96,16 @@ impl StorageManager {
table_store: TableStore, table_store: TableStore,
block_store: BlockStore, block_store: BlockStore,
) -> StorageManager { ) -> StorageManager {
let this = StorageManager { let unlocked_inner = Arc::new(Self::new_unlocked_inner(
unlocked_inner: Arc::new(Self::new_unlocked_inner(
config, config,
crypto, crypto,
protected_store, protected_store,
table_store, table_store,
block_store, block_store,
)), ));
inner: Arc::new(AsyncMutex::new(Self::new_inner())), let this = StorageManager {
unlocked_inner: unlocked_inner.clone(),
inner: Arc::new(AsyncMutex::new(Self::new_inner(unlocked_inner))),
}; };
this.setup_tasks(); this.setup_tasks();
@ -169,7 +170,7 @@ impl StorageManager {
self.cancel_tasks().await; self.cancel_tasks().await;
// Release the storage manager // Release the storage manager
*inner = Self::new_inner(); *inner = Self::new_inner(self.unlocked_inner.clone());
debug!("finished storage manager shutdown"); debug!("finished storage manager shutdown");
} }
@ -179,22 +180,6 @@ impl StorageManager {
inner.rpc_processor = opt_rpc_processor inner.rpc_processor = opt_rpc_processor
} }
/// # DHT Key = Hash(ownerKeyKind) of: [ ownerKeyValue, schema ]
fn get_key<D>(vcrypto: CryptoSystemVersion, record: &Record<D>) -> TypedKey
where
D: Clone + RkyvArchive + RkyvSerialize<RkyvSerializer>,
for<'t> <D as RkyvArchive>::Archived: CheckBytes<RkyvDefaultValidator<'t>>,
<D as RkyvArchive>::Archived: RkyvDeserialize<D, SharedDeserializeMap>,
{
let compiled = record.descriptor().schema_data();
let mut hash_data = Vec::<u8>::with_capacity(PUBLIC_KEY_LENGTH + 4 + compiled.len());
hash_data.extend_from_slice(&vcrypto.kind().0);
hash_data.extend_from_slice(&record.owner().bytes);
hash_data.extend_from_slice(compiled);
let hash = vcrypto.generate_hash(&hash_data);
TypedKey::new(vcrypto.kind(), hash)
}
async fn lock(&self) -> Result<AsyncMutexGuardArc<StorageManagerInner>, VeilidAPIError> { async fn lock(&self) -> Result<AsyncMutexGuardArc<StorageManagerInner>, VeilidAPIError> {
let inner = asyncmutex_lock_arc!(&self.inner); let inner = asyncmutex_lock_arc!(&self.inner);
if !inner.initialized { if !inner.initialized {
@ -203,6 +188,7 @@ impl StorageManager {
Ok(inner) Ok(inner)
} }
/// Create a local record from scratch with a new owner key, open it, and return the opened descriptor
pub async fn create_record( pub async fn create_record(
&self, &self,
kind: CryptoKind, kind: CryptoKind,
@ -211,59 +197,32 @@ impl StorageManager {
) -> Result<DHTRecordDescriptor, VeilidAPIError> { ) -> Result<DHTRecordDescriptor, VeilidAPIError> {
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
// Get cryptosystem // Create a new owned local record from scratch
let Some(vcrypto) = self.unlocked_inner.crypto.get(kind) else { let (key, owner) = inner
apibail_generic!("unsupported cryptosystem"); .create_new_owned_local_record(kind, schema, safety_selection)
}; .await?;
// Get local record store // Now that the record is made we should always succeed to open the existing record
let Some(local_record_store) = inner.local_record_store.as_mut() else { // The initial writer is the owner of the record
apibail_not_initialized!(); inner
}; .open_existing_record(key, Some(owner), safety_selection)
.map(|r| r.unwrap())
// Compile the dht schema
let schema_data = schema.compile();
// New values require a new owner key
let owner = vcrypto.generate_keypair();
// Make a signed value descriptor for this dht value
let signed_value_descriptor = SignedValueDescriptor::make_signature(
owner.key,
schema_data,
vcrypto.clone(),
owner.secret,
)?;
// Add new local value record
let cur_ts = get_aligned_timestamp();
let local_record_detail = LocalRecordDetail { safety_selection };
let record =
Record::<LocalRecordDetail>::new(cur_ts, signed_value_descriptor, local_record_detail)?;
let dht_key = Self::get_key(vcrypto.clone(), &record);
local_record_store.new_record(dht_key, record).await?;
// Open the record
self.open_record_common(inner, dht_key, Some(owner), safety_selection)
.await
} }
async fn open_record_common( /// Open an existing local record if it exists,
/// and if it doesnt exist locally, try to pull it from the network and
/// open it and return the opened descriptor
pub async fn open_record(
&self, &self,
mut inner: AsyncMutexGuardArc<StorageManagerInner>,
key: TypedKey, key: TypedKey,
writer: Option<KeyPair>, writer: Option<KeyPair>,
safety_selection: SafetySelection, safety_selection: SafetySelection,
) -> Result<DHTRecordDescriptor, VeilidAPIError> { ) -> Result<DHTRecordDescriptor, VeilidAPIError> {
// Ensure the record is closed let mut inner = self.lock().await?;
if inner.opened_records.contains_key(&key) {
apibail_generic!("record is already open and should be closed first");
}
// See if we have a local record already or not // See if we have a local record already or not
if let Some(res) = inner.open_record_check_existing(key, writer, safety_selection) { if let Some(res) = inner.open_existing_record(key, writer, safety_selection)? {
return res; return Ok(res);
} }
// No record yet, try to get it from the network // No record yet, try to get it from the network
@ -280,86 +239,38 @@ impl StorageManager {
// No last descriptor, no last value // No last descriptor, no last value
// Use the safety selection we opened the record with // Use the safety selection we opened the record with
let subkey: ValueSubkey = 0; let subkey: ValueSubkey = 0;
let result = self let subkey_result = self
.do_get_value(rpc_processor, key, subkey, None, None, safety_selection) .do_get_value(
rpc_processor,
key,
subkey,
safety_selection,
SubkeyResult::default(),
)
.await?; .await?;
// If we got nothing back, the key wasn't found // If we got nothing back, the key wasn't found
if result.value.is_none() && result.descriptor.is_none() { if subkey_result.value.is_none() && subkey_result.descriptor.is_none() {
// No result // No result
apibail_key_not_found!(key); apibail_key_not_found!(key);
}; };
// Must have descriptor
let Some(signed_value_descriptor) = result.descriptor else {
// No descriptor for new record, can't store this
apibail_generic!("no descriptor");
};
let owner = signed_value_descriptor.owner().clone();
// If the writer we chose is also the owner, we have the owner secret
// Otherwise this is just another subkey writer
let owner_secret = if let Some(writer) = writer {
if writer.key == owner {
Some(writer.secret)
} else {
None
}
} else {
None
};
let schema = signed_value_descriptor.schema()?;
// Reopen inner to store value we just got // Reopen inner to store value we just got
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
// Get local record store // Open the new record
let Some(local_record_store) = inner.local_record_store.as_mut() else {
apibail_not_initialized!();
};
// Make and store a new record for this descriptor
let record = Record::<LocalRecordDetail>::new(
get_aligned_timestamp(),
signed_value_descriptor,
LocalRecordDetail { safety_selection },
)?;
local_record_store.new_record(key, record).await?;
// If we got a subkey with the getvalue, it has already been validated against the schema, so store it
if let Some(signed_value_data) = result.value {
// Write subkey to local store
local_record_store
.set_subkey(key, subkey, signed_value_data)
.await?;
}
// Write open record
inner inner
.opened_records .open_new_record(key, writer, subkey, subkey_result, safety_selection)
.insert(key, OpenedRecord::new(writer, safety_selection));
// Make DHT Record Descriptor to return
let descriptor = DHTRecordDescriptor::new(key, owner, owner_secret, schema);
Ok(descriptor)
}
pub async fn open_record(
&self,
key: TypedKey,
writer: Option<KeyPair>,
safety_selection: SafetySelection,
) -> Result<DHTRecordDescriptor, VeilidAPIError> {
let inner = self.lock().await?;
self.open_record_common(inner, key, writer, safety_selection)
.await .await
} }
/// Close an opened local record
pub async fn close_record(&self, key: TypedKey) -> Result<(), VeilidAPIError> { pub async fn close_record(&self, key: TypedKey) -> Result<(), VeilidAPIError> {
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
inner.close_record(key) inner.close_record(key)
} }
/// Delete a local record
pub async fn delete_record(&self, key: TypedKey) -> Result<(), VeilidAPIError> { pub async fn delete_record(&self, key: TypedKey) -> Result<(), VeilidAPIError> {
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
@ -376,6 +287,8 @@ impl StorageManager {
local_record_store.delete_record(key).await local_record_store.delete_record(key).await
} }
/// Get the value of a subkey from an opened local record
/// may refresh the record, and will if it is forced to or the subkey is not available locally yet
pub async fn get_value( pub async fn get_value(
&self, &self,
key: TypedKey, key: TypedKey,
@ -383,23 +296,23 @@ impl StorageManager {
force_refresh: bool, force_refresh: bool,
) -> Result<Option<ValueData>, VeilidAPIError> { ) -> Result<Option<ValueData>, VeilidAPIError> {
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
// Get rpc processor and drop mutex so we don't block while getting the value from the network
let Some(opened_record) = inner.opened_records.remove(&key) else { let Some(opened_record) = inner.opened_records.remove(&key) else {
apibail_generic!("record not open"); apibail_generic!("record not open");
}; };
// See if the requested subkey is our local record store // See if the requested subkey is our local record store
let SubkeyResult { value, descriptor } = inner.handle_get_local_value(key, subkey, true)?; let last_subkey_result = inner.handle_get_local_value(key, subkey, true)?;
// Return the existing value if we have one unless we are forcing a refresh // Return the existing value if we have one unless we are forcing a refresh
if !force_refresh { if !force_refresh {
if let Some(value) = value { if let Some(last_subkey_result_value) = last_subkey_result.value {
return Ok(Some(value.into_value_data())); return Ok(Some(last_subkey_result_value.into_value_data()));
} }
} }
// Refresh if we can // Refresh if we can
// Get rpc processor and drop mutex so we don't block while getting the value from the network
let Some(rpc_processor) = inner.rpc_processor.clone() else { let Some(rpc_processor) = inner.rpc_processor.clone() else {
// Offline, try again later // Offline, try again later
apibail_try_again!(); apibail_try_again!();
@ -410,32 +323,34 @@ impl StorageManager {
// May have last descriptor / value // May have last descriptor / value
// Use the safety selection we opened the record with // Use the safety selection we opened the record with
let opt_last_seq = value.as_ref().map(|v| v.value_data().seq()); let opt_last_seq = last_subkey_result
let result = self .value
.as_ref()
.map(|v| v.value_data().seq());
let subkey_result = self
.do_get_value( .do_get_value(
rpc_processor, rpc_processor,
key, key,
subkey, subkey,
value,
descriptor,
opened_record.safety_selection(), opened_record.safety_selection(),
last_subkey_result,
) )
.await?; .await?;
// See if we got a value back // See if we got a value back
let Some(result_value) = result.value else { let Some(subkey_result_value) = subkey_result.value else {
// If we got nothing back then we also had nothing beforehand, return nothing // If we got nothing back then we also had nothing beforehand, return nothing
return Ok(None); return Ok(None);
}; };
// If we got a new value back then write it to the opened record // If we got a new value back then write it to the opened record
if Some(result_value.value_data().seq()) != opt_last_seq { if Some(subkey_result_value.value_data().seq()) != opt_last_seq {
let mut inner = self.lock().await?; let mut inner = self.lock().await?;
inner inner
.handle_set_local_value(key, subkey, result_value.clone()) .handle_set_local_value(key, subkey, subkey_result_value.clone())
.await?; .await?;
} }
Ok(Some(result_value.into_value_data())) Ok(Some(subkey_result_value.into_value_data()))
} }
pub async fn set_value( pub async fn set_value(

View File

@ -31,6 +31,7 @@ where
} }
/// The result of the do_get_value_operation /// The result of the do_get_value_operation
#[derive(Default, Debug)]
pub struct SubkeyResult { pub struct SubkeyResult {
/// The subkey value if we got one /// The subkey value if we got one
pub value: Option<SignedValueData>, pub value: Option<SignedValueData>,

View File

@ -1,8 +1,8 @@
use super::*; use super::*;
/// Locked structure for storage manager /// Locked structure for storage manager
#[derive(Default)]
pub(super) struct StorageManagerInner { pub(super) struct StorageManagerInner {
unlocked_inner: Arc<StorageManagerUnlockedInner>,
/// If we are started up /// If we are started up
pub initialized: bool, pub initialized: bool,
/// Records that have been 'opened' and are not yet closed /// Records that have been 'opened' and are not yet closed
@ -18,15 +18,74 @@ pub(super) struct StorageManagerInner {
} }
impl StorageManagerInner { impl StorageManagerInner {
pub fn open_record_check_existing( pub fn new(unlocked_inner: Arc<StorageManagerUnlockedInner>) -> Self {
Self {
unlocked_inner,
initialized: false,
opened_records: Default::default(),
local_record_store: Default::default(),
remote_record_store: Default::default(),
rpc_processor: Default::default(),
tick_future: Default::default(),
}
}
pub async fn create_new_owned_local_record(
&mut self,
kind: CryptoKind,
schema: DHTSchema,
safety_selection: SafetySelection,
) -> Result<(TypedKey, KeyPair), VeilidAPIError> {
// Get cryptosystem
let Some(vcrypto) = self.unlocked_inner.crypto.get(kind) else {
apibail_generic!("unsupported cryptosystem");
};
// Get local record store
let Some(local_record_store) = self.local_record_store.as_mut() else {
apibail_not_initialized!();
};
// Compile the dht schema
let schema_data = schema.compile();
// New values require a new owner key
let owner = vcrypto.generate_keypair();
// Make a signed value descriptor for this dht value
let signed_value_descriptor = SignedValueDescriptor::make_signature(
owner.key,
schema_data,
vcrypto.clone(),
owner.secret,
)?;
// Add new local value record
let cur_ts = get_aligned_timestamp();
let local_record_detail = LocalRecordDetail { safety_selection };
let record =
Record::<LocalRecordDetail>::new(cur_ts, signed_value_descriptor, local_record_detail)?;
let dht_key = Self::get_key(vcrypto.clone(), &record);
local_record_store.new_record(dht_key, record).await?;
Ok((dht_key, owner))
}
pub fn open_existing_record(
&mut self, &mut self,
key: TypedKey, key: TypedKey,
writer: Option<KeyPair>, writer: Option<KeyPair>,
safety_selection: SafetySelection, safety_selection: SafetySelection,
) -> Option<Result<DHTRecordDescriptor, VeilidAPIError>> { ) -> Result<Option<DHTRecordDescriptor>, VeilidAPIError> {
// Ensure the record is closed
if self.opened_records.contains_key(&key) {
apibail_generic!("record is already open and should be closed first");
}
// Get local record store // Get local record store
let Some(local_record_store) = self.local_record_store.as_mut() else { let Some(local_record_store) = self.local_record_store.as_mut() else {
return Some(Err(VeilidAPIError::not_initialized())); apibail_not_initialized!();
}; };
// See if we have a local record already or not // See if we have a local record already or not
@ -40,7 +99,7 @@ impl StorageManagerInner {
(r.owner().clone(), r.schema()) (r.owner().clone(), r.schema())
}; };
let Some((owner, schema)) = local_record_store.with_record_mut(key, cb) else { let Some((owner, schema)) = local_record_store.with_record_mut(key, cb) else {
return None; return Ok(None);
}; };
// Had local record // Had local record
@ -62,17 +121,43 @@ impl StorageManagerInner {
// Make DHT Record Descriptor to return // Make DHT Record Descriptor to return
let descriptor = DHTRecordDescriptor::new(key, owner, owner_secret, schema); let descriptor = DHTRecordDescriptor::new(key, owner, owner_secret, schema);
Some(Ok(descriptor)) Ok(Some(descriptor))
} }
pub async fn new_local_record( pub async fn open_new_record(
&mut self, &mut self,
key: TypedKey, key: TypedKey,
writer: Option<KeyPair>,
subkey: ValueSubkey, subkey: ValueSubkey,
signed_value_descriptor: SignedValueDescriptor, subkey_result: SubkeyResult,
signed_value_data: Option<SignedValueData>,
safety_selection: SafetySelection, safety_selection: SafetySelection,
) -> Result<(), VeilidAPIError> { ) -> Result<DHTRecordDescriptor, VeilidAPIError> {
// Ensure the record is closed
if self.opened_records.contains_key(&key) {
panic!("new record should never be opened at this point");
}
// Must have descriptor
let Some(signed_value_descriptor) = subkey_result.descriptor else {
// No descriptor for new record, can't store this
apibail_generic!("no descriptor");
};
// Get owner
let owner = signed_value_descriptor.owner().clone();
// If the writer we chose is also the owner, we have the owner secret
// Otherwise this is just another subkey writer
let owner_secret = if let Some(writer) = writer {
if writer.key == owner {
Some(writer.secret)
} else {
None
}
} else {
None
};
let schema = signed_value_descriptor.schema()?;
// Get local record store // Get local record store
let Some(local_record_store) = self.local_record_store.as_mut() else { let Some(local_record_store) = self.local_record_store.as_mut() else {
apibail_not_initialized!(); apibail_not_initialized!();
@ -87,17 +172,20 @@ impl StorageManagerInner {
local_record_store.new_record(key, record).await?; local_record_store.new_record(key, record).await?;
// If we got a subkey with the getvalue, it has already been validated against the schema, so store it // If we got a subkey with the getvalue, it has already been validated against the schema, so store it
if let Some(signed_value_data) = signed_value_data { if let Some(signed_value_data) = subkey_result.value {
// Write subkey to local store // Write subkey to local store
local_record_store local_record_store
.set_subkey(key, subkey, signed_value_data) .set_subkey(key, subkey, signed_value_data)
.await?; .await?;
} }
// Write open record // Write open record
self.opened_records self.opened_records
.insert(key, OpenedRecord::new(writer, safety_selection)); .insert(key, OpenedRecord::new(writer, safety_selection));
Ok(()) // Make DHT Record Descriptor to return
let descriptor = DHTRecordDescriptor::new(key, owner, owner_secret, schema);
Ok(descriptor)
} }
pub fn close_record(&mut self, key: TypedKey) -> Result<(), VeilidAPIError> { pub fn close_record(&mut self, key: TypedKey) -> Result<(), VeilidAPIError> {
@ -184,4 +272,20 @@ impl StorageManagerInner {
Ok(()) Ok(())
} }
/// # DHT Key = Hash(ownerKeyKind) of: [ ownerKeyValue, schema ]
fn get_key<D>(vcrypto: CryptoSystemVersion, record: &Record<D>) -> TypedKey
where
D: Clone + RkyvArchive + RkyvSerialize<RkyvSerializer>,
for<'t> <D as RkyvArchive>::Archived: CheckBytes<RkyvDefaultValidator<'t>>,
<D as RkyvArchive>::Archived: RkyvDeserialize<D, SharedDeserializeMap>,
{
let compiled = record.descriptor().schema_data();
let mut hash_data = Vec::<u8>::with_capacity(PUBLIC_KEY_LENGTH + 4 + compiled.len());
hash_data.extend_from_slice(&vcrypto.kind().0);
hash_data.extend_from_slice(&record.owner().bytes);
hash_data.extend_from_slice(compiled);
let hash = vcrypto.generate_hash(&hash_data);
TypedKey::new(vcrypto.kind(), hash)
}
} }