fedimint_core/net/
api_announcement.rsuse std::collections::BTreeMap;
use bitcoin::hashes::{sha256, Hash};
use bitcoin::secp256k1::Message;
use fedimint_core::db::DatabaseLookup;
use fedimint_core::encoding::{Decodable, Encodable};
use fedimint_core::task::MaybeSend;
use fedimint_core::PeerId;
use futures::StreamExt;
use jsonrpsee_core::Serialize;
use serde::Deserialize;
use crate::db::{
Database, DatabaseKey, DatabaseKeyPrefix, DatabaseRecord, IDatabaseTransactionOpsCoreTyped,
};
use crate::task::MaybeSync;
use crate::util::SafeUrl;
const API_ANNOUNCEMENT_MESSAGE_TAG: &[u8] = b"fedimint-api-announcement";
#[derive(Debug, Serialize, Deserialize, Clone, Eq, Hash, PartialEq, Encodable, Decodable)]
pub struct ApiAnnouncement {
pub api_url: SafeUrl,
pub nonce: u64,
}
#[derive(Debug, Serialize, Deserialize, Clone, Eq, Hash, PartialEq, Encodable, Decodable)]
pub struct SignedApiAnnouncement {
pub api_announcement: ApiAnnouncement,
pub signature: secp256k1::schnorr::Signature,
}
#[derive(Debug, Serialize, Deserialize, Clone, Eq, Hash, PartialEq, Encodable, Decodable)]
pub struct SignedApiAnnouncementSubmission {
#[serde(flatten)]
pub signed_api_announcement: SignedApiAnnouncement,
pub peer_id: PeerId,
}
impl ApiAnnouncement {
pub fn new(api_url: SafeUrl, nonce: u64) -> Self {
Self { api_url, nonce }
}
pub fn tagged_hash(&self) -> sha256::Hash {
let mut msg = API_ANNOUNCEMENT_MESSAGE_TAG.to_vec();
self.consensus_encode(&mut msg)
.expect("writing to vec is infallible");
sha256::Hash::hash(&msg)
}
pub fn sign<C: secp256k1::Signing>(
&self,
ctx: &secp256k1::Secp256k1<C>,
key: &secp256k1::Keypair,
) -> SignedApiAnnouncement {
let msg = Message::from_digest(*self.tagged_hash().as_ref());
let signature = ctx.sign_schnorr(&msg, key);
SignedApiAnnouncement {
api_announcement: self.clone(),
signature,
}
}
}
impl SignedApiAnnouncement {
pub fn verify<C: secp256k1::Verification>(
&self,
ctx: &secp256k1::Secp256k1<C>,
pk: &secp256k1::PublicKey,
) -> bool {
let msg = Message::from_digest(*self.api_announcement.tagged_hash().as_ref());
ctx.verify_schnorr(&self.signature, &msg, &pk.x_only_public_key().0)
.is_ok()
}
}
pub async fn override_api_urls<P>(
db: &Database,
cfg_api_urls: impl IntoIterator<Item = (PeerId, SafeUrl)>,
db_key_prefix: &P,
key_to_peer_id: impl Fn(&P::Record) -> PeerId,
) -> BTreeMap<PeerId, SafeUrl>
where
P: DatabaseLookup + DatabaseKeyPrefix + MaybeSend + MaybeSync,
P::Record: DatabaseRecord<Value = SignedApiAnnouncement> + DatabaseKey + MaybeSend + MaybeSync,
{
let mut db_api_urls = db
.begin_transaction_nc()
.await
.find_by_prefix(db_key_prefix)
.await
.map(|(key, announcement)| (key_to_peer_id(&key), announcement.api_announcement.api_url))
.collect::<BTreeMap<_, _>>()
.await;
cfg_api_urls
.into_iter()
.map(|(peer_id, cfg_api_url)| {
(peer_id, db_api_urls.remove(&peer_id).unwrap_or(cfg_api_url))
})
.collect::<BTreeMap<_, _>>()
}