Skip to content

Commit

Permalink
add neuron certificate handling
Browse files Browse the repository at this point in the history
  • Loading branch information
andreea-popescu-reef committed Apr 4, 2024
1 parent 09e0270 commit 472d178
Show file tree
Hide file tree
Showing 7 changed files with 230 additions and 2 deletions.
27 changes: 26 additions & 1 deletion pallets/subtensor/rpc/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,13 @@ pub trait SubtensorCustomApi<BlockHash> {
fn get_neurons(&self, netuid: u16, at: Option<BlockHash>) -> RpcResult<Vec<u8>>;
#[method(name = "neuronInfo_getNeuron")]
fn get_neuron(&self, netuid: u16, uid: u16, at: Option<BlockHash>) -> RpcResult<Vec<u8>>;

#[method(name = "neuronInfo_getNeuronCertificate")]
fn get_neuron_certificate(
&self,
netuid: u16,
uid: u16,
at: Option<BlockHash>,
) -> RpcResult<Vec<u8>>;
#[method(name = "subnetInfo_getSubnetInfo")]
fn get_subnet_info(&self, netuid: u16, at: Option<BlockHash>) -> RpcResult<Vec<u8>>;
#[method(name = "subnetInfo_getSubnetsInfo")]
Expand Down Expand Up @@ -212,6 +218,25 @@ where
})
}

fn get_neuron_certificate(
&self,
netuid: u16,
uid: u16,
at: Option<<Block as BlockT>::Hash>,
) -> RpcResult<Vec<u8>> {
let api = self.client.runtime_api();
let at = at.unwrap_or_else(|| self.client.info().best_hash);

api.get_neuron_certificate(at, netuid, uid).map_err(|e| {
CallError::Custom(ErrorObject::owned(
Error::RuntimeError.into(),
"Unable to get neuron certificate.",
Some(e.to_string()),
))
.into()
})
}

fn get_subnet_info(
&self,
netuid: u16,
Expand Down
1 change: 1 addition & 0 deletions pallets/subtensor/runtime-api/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ sp_api::decl_runtime_apis! {
pub trait NeuronInfoRuntimeApi {
fn get_neurons(netuid: u16) -> Vec<u8>;
fn get_neuron(netuid: u16, uid: u16) -> Vec<u8>;
fn get_neuron_certificate(netuid: u16, uid: u16) -> Vec<u8>;
fn get_neurons_lite(netuid: u16) -> Vec<u8>;
fn get_neuron_lite(netuid: u16, uid: u16) -> Vec<u8>;
}
Expand Down
109 changes: 109 additions & 0 deletions pallets/subtensor/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ mod benchmarks;
// =========================
mod block_step;

mod certificate;
mod epoch;
mod math;
mod registration;
Expand All @@ -57,6 +58,7 @@ pub mod migration;
#[frame_support::pallet]
pub mod pallet {

use crate::certificate::Certificate;
use frame_support::{
dispatch::GetDispatchInfo,
inherent::Vec,
Expand Down Expand Up @@ -522,6 +524,14 @@ pub mod pallet {
pub placeholder2: u8, // --- Axon proto placeholder 1.
}

// --- Struct for NeuronCertificate.
pub type NeuronCertificateOf = NeuronCertificate;

#[derive(Decode, Encode, Default, TypeInfo, PartialEq, Eq, Clone, Debug)]
pub struct NeuronCertificate {
pub certificate: Certificate,
}

// --- Struct for Prometheus.
pub type PrometheusInfoOf = PrometheusInfo;
#[derive(Encode, Decode, Default, TypeInfo, Clone, PartialEq, Eq, Debug)]
Expand Down Expand Up @@ -570,6 +580,16 @@ pub mod pallet {
PrometheusInfoOf,
OptionQuery,
>;
#[pallet::storage] // --- MAP ( netuid, hotkey ) --> certificate
pub(super) type NeuronCertificates<T: Config> = StorageDoubleMap<
_,
Identity,
u16,
Blake2_128Concat,
T::AccountId,
NeuronCertificateOf,
OptionQuery,
>;

// =======================================
// ==== Subnetwork Hyperparam storage ====
Expand Down Expand Up @@ -1446,6 +1466,91 @@ pub mod pallet {
protocol,
placeholder1,
placeholder2,
None,
)
}

// ---- Serves or updates axon /promethteus information for the neuron associated with the caller. If the caller is
// already registered the metadata is updated. If the caller is not registered this call throws NotRegistered.
//
// # Args:
// * 'origin': (<T as frame_system::Config>Origin):
// - The signature of the caller.
//
// * 'netuid' (u16):
// - The u16 network identifier.
//
// * 'version' (u64):
// - The bittensor version identifier.
//
// * 'ip' (u64):
// - The endpoint ip information as a u128 encoded integer.
//
// * 'port' (u16):
// - The endpoint port information as a u16 encoded integer.
//
// * 'ip_type' (u8):
// - The endpoint ip version as a u8, 4 or 6.
//
// * 'protocol' (u8):
// - UDP:1 or TCP:0
//
// * 'placeholder1' (u8):
// - Placeholder for further extra params.
//
// * 'placeholder2' (u8):
// - Placeholder for further extra params.
//
// * 'certificate' (Vec<u8>):
// - TLS certificate
//
// # Event:
// * AxonServed;
// - On successfully serving the axon info.
//
// # Raises:
// * 'NetworkDoesNotExist':
// - Attempting to set weights on a non-existent network.
//
// * 'NotRegistered':
// - Attempting to set weights from a non registered account.
//
// * 'InvalidIpType':
// - The ip type is not 4 or 6.
//
// * 'InvalidIpAddress':
// - The numerically encoded ip address does not resolve to a proper ip.
//
// * 'ServingRateLimitExceeded':
// - Attempting to set prometheus information withing the rate limit min.
//
#[pallet::call_index(40)]
#[pallet::weight((Weight::from_ref_time(19_000_000)
.saturating_add(T::DbWeight::get().reads(2))
.saturating_add(T::DbWeight::get().writes(1)), DispatchClass::Normal, Pays::No))]
pub fn serve_axon_tls(
origin: OriginFor<T>,
netuid: u16,
version: u32,
ip: u128,
port: u16,
ip_type: u8,
protocol: u8,
placeholder1: u8,
placeholder2: u8,
certificate: Certificate,
) -> DispatchResult {
Self::do_serve_axon(
origin,
netuid,
version,
ip,
port,
ip_type,
protocol,
placeholder1,
placeholder2,
Some(certificate),
)
}

Expand Down Expand Up @@ -1869,6 +1974,10 @@ where
let transaction_fee = 0;
Ok((CallType::Serve, transaction_fee, who.clone()))
}
Some(Call::serve_axon_tls { .. }) => {
let transaction_fee = 0;
Ok((CallType::Serve, transaction_fee, who.clone()))
}
Some(Call::register_network { .. }) => {
let transaction_fee = 0;
Ok((CallType::RegisterNetwork, transaction_fee, who.clone()))
Expand Down
22 changes: 22 additions & 0 deletions pallets/subtensor/src/neuron_info.rs
Original file line number Diff line number Diff line change
Expand Up @@ -170,6 +170,28 @@ impl<T: Config> Pallet<T> {
neuron
}

pub fn get_neuron_certificate(netuid: u16, uid: u16) -> Option<NeuronCertificate> {
if !Self::if_subnet_exist(netuid) {
return None;
}

let _hotkey = Self::get_hotkey_for_net_and_uid(netuid, uid);
let hotkey;
if _hotkey.is_err() {
return None;
} else {
// No error, hotkey was registered
hotkey = _hotkey.expect("Hotkey should exist");
}

if Self::has_neuron_certificate(netuid, &hotkey) {
let certificate = NeuronCertificates::<T>::get(netuid, hotkey).unwrap();
Some(certificate)
} else {
None
}
}

fn get_neuron_lite_subnet_exists(netuid: u16, uid: u16) -> Option<NeuronInfoLite<T>> {
let _hotkey = Self::get_hotkey_for_net_and_uid(netuid, uid);
let hotkey;
Expand Down
11 changes: 11 additions & 0 deletions pallets/subtensor/src/serving.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
use super::*;
use crate::certificate::Certificate;
use frame_support::inherent::Vec;
use frame_support::sp_std::vec;

Expand Down Expand Up @@ -63,6 +64,7 @@ impl<T: Config> Pallet<T> {
protocol: u8,
placeholder1: u8,
placeholder2: u8,
certificate: Option<Certificate>,
) -> dispatch::DispatchResult {
// --- 1. We check the callers (hotkey) signature.
let hotkey_id = ensure_signed(origin)?;
Expand All @@ -88,6 +90,11 @@ impl<T: Config> Pallet<T> {
Error::<T>::ServingRateLimitExceeded
);

// --- 5. Check certificate
if let Some(certificate) = certificate {
NeuronCertificates::<T>::insert(netuid, hotkey_id.clone(), NeuronCertificate{certificate})
}

// --- 6. We insert the axon meta.
prev_axon.block = Self::get_current_block_as_u64();
prev_axon.version = version;
Expand Down Expand Up @@ -241,6 +248,10 @@ impl<T: Config> Pallet<T> {
return Axons::<T>::contains_key(netuid, hotkey);
}

pub fn has_neuron_certificate(netuid: u16, hotkey: &T::AccountId) -> bool {
return NeuronCertificates::<T>::contains_key(netuid, hotkey);
}

pub fn has_prometheus_info(netuid: u16, hotkey: &T::AccountId) -> bool {
return Prometheus::<T>::contains_key(netuid, hotkey);
}
Expand Down
50 changes: 50 additions & 0 deletions pallets/subtensor/tests/serving.rs
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,56 @@ fn test_serving_ok() {
});
}

#[test]
fn test_serving_tls_ok() {
new_test_ext().execute_with(|| {
let hotkey_account_id = U256::from(1);
let uid: u16 = 0;
let netuid: u16 = 1;
let tempo: u16 = 13;
let version: u32 = 2;
let ip: u128 = 1676056785;
let port: u16 = 128;
let ip_type: u8 = 4;
let modality: u16 = 0;
let protocol: u8 = 0;
let placeholder1: u8 = 0;
let placeholder2: u8 = 0;
let certificate: Vec<u8> = "CERT".as_bytes().to_vec();
add_network(netuid, tempo, modality);
register_ok_neuron(netuid, hotkey_account_id, U256::from(66), 0);
assert_ok!(SubtensorModule::serve_axon_tls(
<<Test as Config>::RuntimeOrigin>::signed(hotkey_account_id),
netuid,
version,
ip,
port,
ip_type,
protocol,
placeholder1,
placeholder2,
certificate.clone()
));
let stored_certificate = SubtensorModule::get_neuron_certificate(netuid, uid);
assert_eq!(stored_certificate.unwrap().certificate, certificate);
let new_certificate = "UPDATED_CERT".as_bytes().to_vec();
assert_ok!(SubtensorModule::serve_axon_tls(
<<Test as Config>::RuntimeOrigin>::signed(hotkey_account_id),
netuid,
version,
ip,
port,
ip_type,
protocol,
placeholder1,
placeholder2,
new_certificate.clone()
));
let stored_certificate = SubtensorModule::get_neuron_certificate(netuid, uid);
assert_eq!(stored_certificate.unwrap().certificate, new_certificate)
});
}

#[test]
fn test_serving_set_metadata_update() {
new_test_ext().execute_with(|| {
Expand Down
12 changes: 11 additions & 1 deletion runtime/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,7 @@ pub const VERSION: RuntimeVersion = RuntimeVersion {
// `spec_version`, and `authoring_version` are the same between Wasm and native.
// This value is set to 100 to notify Polkadot-JS App (https://polkadot.js.org/apps) to use
// the compatible custom types.
spec_version: 144,
spec_version: 145,
impl_version: 1,
apis: RUNTIME_API_VERSIONS,
transaction_version: 1,
Expand Down Expand Up @@ -1328,6 +1328,16 @@ impl_runtime_apis! {
vec![]
}
}

fn get_neuron_certificate(netuid: u16, uid: u16) -> Vec<u8> {
let _result = SubtensorModule::get_neuron_certificate(netuid, uid);
if _result.is_some() {
let result = _result.expect("Could not get Certificate");
result.encode()
} else {
vec![]
}
}
}

impl subtensor_custom_rpc_runtime_api::SubnetInfoRuntimeApi<Block> for Runtime {
Expand Down

0 comments on commit 472d178

Please sign in to comment.