#![cfg_attr(not(feature = "std"), no_std)]
#[cfg(feature = "runtime-benchmarks")]
mod benchmarking;
#[cfg(test)]
mod tests;
pub mod weights;
extern crate alloc;
use crate::weights::WeightInfo;
#[cfg(not(feature = "std"))]
use alloc::collections::BTreeSet;
#[cfg(not(feature = "std"))]
use alloc::vec::Vec;
use codec::{Decode, Encode};
use frame_support::dispatch::{DispatchResult, DispatchResultWithPostInfo};
use frame_support::ensure;
use frame_support::traits::Time;
use frame_support::weights::Weight;
pub use pallet::*;
use scale_info::TypeInfo;
use sp_auto_id::auto_id_runtime_interface::{decode_tbs_certificate, verify_signature};
use sp_auto_id::{DerVec, SignatureVerificationRequest, Validity};
use sp_core::{blake2_256, H256, U256};
#[cfg(feature = "std")]
use std::collections::BTreeSet;
use subspace_runtime_primitives::Moment;
pub type Identifier = H256;
pub type Serial = U256;
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub struct X509Certificate {
pub issuer_id: Option<Identifier>,
pub serial: U256,
pub subject_common_name: Vec<u8>,
pub subject_public_key_info: DerVec,
pub validity: Validity,
pub raw: DerVec,
pub issued_serials: BTreeSet<Serial>,
pub nonce: U256,
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub enum Certificate {
X509(X509Certificate),
}
impl Certificate {
#[cfg(test)]
fn subject_common_name(&self) -> Vec<u8> {
match self {
Certificate::X509(cert) => cert.subject_common_name.clone(),
}
}
fn subject_public_key_info(&self) -> DerVec {
match self {
Certificate::X509(cert) => cert.subject_public_key_info.clone(),
}
}
fn issue_certificate_serial<T: Config>(&mut self, serial: U256) -> DispatchResult {
match self {
Certificate::X509(cert) => {
ensure!(
!cert.issued_serials.contains(&serial),
Error::<T>::CertificateSerialAlreadyIssued
);
cert.issued_serials.insert(serial);
Ok(())
}
}
}
fn issuer_id(&self) -> Option<Identifier> {
match self {
Certificate::X509(cert) => cert.issuer_id,
}
}
fn serial(&self) -> Serial {
match self {
Certificate::X509(cert) => cert.serial,
}
}
fn issued_serials(&self) -> BTreeSet<Serial> {
match self {
Certificate::X509(cert) => cert.issued_serials.clone(),
}
}
pub(crate) fn is_valid_at(&self, time: Moment) -> bool {
match self {
Certificate::X509(cert) => cert.validity.is_valid_at(time),
}
}
fn derive_identifier(&self) -> Identifier {
match &self {
Certificate::X509(cert) => {
if let Some(issuer_id) = cert.issuer_id {
let mut data = issuer_id.to_fixed_bytes().to_vec();
data.extend_from_slice(cert.subject_common_name.as_ref());
blake2_256(&data).into()
} else {
blake2_256(cert.subject_common_name.as_ref()).into()
}
}
}
}
fn nonce(&self) -> U256 {
match self {
Certificate::X509(cert) => cert.nonce,
}
}
fn inc_nonce<T: Config>(&mut self) -> DispatchResult {
match self {
Certificate::X509(cert) => {
cert.nonce = cert
.nonce
.checked_add(U256::one())
.ok_or(Error::<T>::NonceOverflow)?;
Ok(())
}
}
}
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub struct AutoId {
pub certificate: Certificate,
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub enum RegisterAutoIdX509 {
Root {
certificate: DerVec,
signature_algorithm: DerVec,
signature: Vec<u8>,
},
Leaf {
issuer_id: Identifier,
certificate: DerVec,
signature_algorithm: DerVec,
signature: Vec<u8>,
},
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub enum RenewAutoId {
X509(RenewX509Certificate),
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub struct RenewX509Certificate {
issuer_id: Option<Identifier>,
certificate: DerVec,
signature_algorithm: DerVec,
signature: Vec<u8>,
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub struct Signature {
pub signature_algorithm: DerVec,
pub value: Vec<u8>,
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub enum RegisterAutoId {
X509(RegisterAutoIdX509),
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub enum CertificateActionType {
RevokeCertificate,
DeactivateAutoId,
}
#[derive(Debug, Decode, Encode, TypeInfo, PartialEq, Eq, Clone)]
pub struct CertificateAction {
pub id: Identifier,
pub nonce: U256,
pub action_type: CertificateActionType,
}
#[frame_support::pallet]
mod pallet {
use super::*;
use crate::weights::WeightInfo;
use crate::{AutoId, Identifier, RegisterAutoId, Serial, Signature};
use frame_support::pallet_prelude::*;
use frame_support::traits::Time;
use frame_system::pallet_prelude::*;
#[pallet::config]
pub trait Config: frame_system::Config {
type RuntimeEvent: From<Event<Self>> + IsType<<Self as frame_system::Config>::RuntimeEvent>;
type Time: Time<Moment = subspace_runtime_primitives::Moment>;
type Weights: WeightInfo;
}
#[pallet::pallet]
#[pallet::without_storage_info]
pub struct Pallet<T>(_);
#[pallet::storage]
pub(super) type AutoIds<T> = StorageMap<_, Identity, Identifier, AutoId, OptionQuery>;
#[pallet::storage]
pub(super) type CertificateRevocationList<T> =
StorageMap<_, Identity, Identifier, BTreeSet<Serial>, OptionQuery>;
#[pallet::error]
pub enum Error<T> {
UnknownIssuer,
UnknownAutoId,
InvalidCertificate,
InvalidSignature,
CertificateSerialAlreadyIssued,
ExpiredCertificate,
CertificateRevoked,
CertificateAlreadyRevoked,
NonceOverflow,
AutoIdIdentifierAlreadyExists,
AutoIdIdentifierMismatch,
PublicKeyMismatch,
}
#[pallet::event]
#[pallet::generate_deposit(pub (super) fn deposit_event)]
pub enum Event<T: Config> {
NewAutoIdRegistered(Identifier),
CertificateRevoked(Identifier),
AutoIdDeactivated(Identifier),
AutoIdRenewed(Identifier),
}
#[pallet::call]
impl<T: Config> Pallet<T> {
#[pallet::call_index(0)]
#[pallet::weight(Pallet::<T>::max_register_auto_id_weight())]
pub fn register_auto_id(
origin: OriginFor<T>,
req: RegisterAutoId,
) -> DispatchResultWithPostInfo {
ensure_signed(origin)?;
Self::do_register_auto_id(req)
}
#[pallet::call_index(1)]
#[pallet::weight(Pallet::<T>::max_revoke_auto_id_weight())]
pub fn revoke_certificate(
origin: OriginFor<T>,
auto_id_identifier: Identifier,
signature: Signature,
) -> DispatchResultWithPostInfo {
ensure_signed(origin)?;
Self::do_revoke_certificate(auto_id_identifier, signature)
}
#[pallet::call_index(2)]
#[pallet::weight(<T as Config>::Weights::deactivate_auto_id())]
pub fn deactivate_auto_id(
origin: OriginFor<T>,
auto_id_identifier: Identifier,
signature: Signature,
) -> DispatchResult {
ensure_signed(origin)?;
Self::do_deactivate_auto_id(auto_id_identifier, signature)?;
Ok(())
}
#[pallet::call_index(3)]
#[pallet::weight(Pallet::<T>::max_renew_auto_id_weight())]
pub fn renew_auto_id(
origin: OriginFor<T>,
auto_id_identifier: Identifier,
req: RenewAutoId,
) -> DispatchResultWithPostInfo {
ensure_signed(origin)?;
Self::do_renew_auto_id_certificate(auto_id_identifier, req)
}
}
}
impl<T: Config> Pallet<T> {
pub(crate) fn do_register_auto_id(req: RegisterAutoId) -> DispatchResultWithPostInfo {
let current_time = T::Time::now();
let (certificate, weight) = match req {
RegisterAutoId::X509(x509_req) => match x509_req {
RegisterAutoIdX509::Root {
certificate,
signature_algorithm,
signature,
} => {
let tbs_certificate = decode_tbs_certificate(certificate.clone())
.ok_or(Error::<T>::InvalidCertificate)?;
let req = SignatureVerificationRequest {
public_key_info: tbs_certificate.subject_public_key_info.clone(),
signature_algorithm,
data: certificate.0.clone(),
signature,
};
verify_signature(req).ok_or(Error::<T>::InvalidSignature)?;
ensure!(
tbs_certificate.validity.is_valid_at(current_time),
Error::<T>::ExpiredCertificate
);
(
Certificate::X509(X509Certificate {
issuer_id: None,
serial: tbs_certificate.serial,
subject_common_name: tbs_certificate.subject_common_name,
subject_public_key_info: tbs_certificate.subject_public_key_info,
validity: tbs_certificate.validity,
raw: certificate,
issued_serials: BTreeSet::from([tbs_certificate.serial]),
nonce: U256::zero(),
}),
T::Weights::register_issuer_auto_id(),
)
}
RegisterAutoIdX509::Leaf {
issuer_id,
certificate,
signature_algorithm,
signature,
} => {
let mut issuer_auto_id =
AutoIds::<T>::get(issuer_id).ok_or(Error::<T>::UnknownIssuer)?;
let issuer_public_key_info =
issuer_auto_id.certificate.subject_public_key_info();
ensure!(
issuer_auto_id.certificate.is_valid_at(current_time),
Error::<T>::ExpiredCertificate
);
let tbs_certificate = decode_tbs_certificate(certificate.clone())
.ok_or(Error::<T>::InvalidCertificate)?;
let req = SignatureVerificationRequest {
public_key_info: issuer_public_key_info,
signature_algorithm,
data: certificate.0.clone(),
signature,
};
verify_signature(req).ok_or(Error::<T>::InvalidSignature)?;
ensure!(
tbs_certificate.validity.is_valid_at(current_time),
Error::<T>::ExpiredCertificate
);
ensure!(
!CertificateRevocationList::<T>::get(issuer_id).is_some_and(|serials| {
serials.iter().any(|s| {
*s == issuer_auto_id.certificate.serial()
|| *s == tbs_certificate.serial
})
}),
Error::<T>::CertificateRevoked
);
issuer_auto_id
.certificate
.issue_certificate_serial::<T>(tbs_certificate.serial)?;
AutoIds::<T>::insert(issuer_id, issuer_auto_id);
(
Certificate::X509(X509Certificate {
issuer_id: Some(issuer_id),
serial: tbs_certificate.serial,
subject_common_name: tbs_certificate.subject_common_name,
subject_public_key_info: tbs_certificate.subject_public_key_info,
validity: tbs_certificate.validity,
raw: certificate,
issued_serials: BTreeSet::from([tbs_certificate.serial]),
nonce: U256::zero(),
}),
T::Weights::register_leaf_auto_id(),
)
}
},
};
let auto_id_identifier = certificate.derive_identifier();
let auto_id = AutoId { certificate };
ensure!(
!AutoIds::<T>::contains_key(auto_id_identifier),
Error::<T>::AutoIdIdentifierAlreadyExists
);
AutoIds::<T>::insert(auto_id_identifier, auto_id);
Self::deposit_event(Event::<T>::NewAutoIdRegistered(auto_id_identifier));
Ok(Some(weight).into())
}
fn do_renew_auto_id_certificate(
auto_id_identifier: Identifier,
req: RenewAutoId,
) -> DispatchResultWithPostInfo {
let current_time = T::Time::now();
let auto_id = AutoIds::<T>::get(auto_id_identifier).ok_or(Error::<T>::UnknownAutoId)?;
let RenewAutoId::X509(req) = req;
let tbs_certificate = decode_tbs_certificate(req.certificate.clone())
.ok_or(Error::<T>::InvalidCertificate)?;
let (issuer_public_key_info, weight) = match req.issuer_id {
None => {
CertificateRevocationList::<T>::mutate(auto_id_identifier, |serials| {
serials
.get_or_insert_with(BTreeSet::new)
.insert(auto_id.certificate.serial());
});
(
auto_id.certificate.subject_public_key_info(),
T::Weights::renew_issuer_auto_id(),
)
}
Some(issuer_id) => {
let mut issuer_auto_id =
AutoIds::<T>::get(issuer_id).ok_or(Error::<T>::UnknownIssuer)?;
let issuer_public_key_info = issuer_auto_id.certificate.subject_public_key_info();
ensure!(
issuer_auto_id.certificate.is_valid_at(current_time),
Error::<T>::ExpiredCertificate
);
ensure!(
!CertificateRevocationList::<T>::get(issuer_id).is_some_and(|serials| {
serials.iter().any(|s| {
*s == issuer_auto_id.certificate.serial()
|| *s == tbs_certificate.serial
})
}),
Error::<T>::CertificateRevoked
);
issuer_auto_id
.certificate
.issue_certificate_serial::<T>(tbs_certificate.serial)?;
CertificateRevocationList::<T>::mutate(issuer_id, |serials| {
serials
.get_or_insert_with(BTreeSet::new)
.insert(auto_id.certificate.serial());
});
AutoIds::<T>::insert(issuer_id, issuer_auto_id);
(issuer_public_key_info, T::Weights::renew_leaf_auto_id())
}
};
let signature_req = SignatureVerificationRequest {
public_key_info: issuer_public_key_info,
signature_algorithm: req.signature_algorithm,
data: req.certificate.0.clone(),
signature: req.signature,
};
verify_signature(signature_req).ok_or(Error::<T>::InvalidSignature)?;
ensure!(
tbs_certificate.validity.is_valid_at(current_time),
Error::<T>::ExpiredCertificate
);
ensure!(
auto_id.certificate.subject_public_key_info()
== tbs_certificate.subject_public_key_info,
Error::<T>::PublicKeyMismatch
);
let updated_certificate = Certificate::X509(X509Certificate {
issuer_id: req.issuer_id,
serial: tbs_certificate.serial,
subject_common_name: tbs_certificate.subject_common_name,
subject_public_key_info: tbs_certificate.subject_public_key_info,
validity: tbs_certificate.validity,
raw: req.certificate,
issued_serials: auto_id.certificate.issued_serials(),
nonce: auto_id.certificate.nonce(),
});
ensure!(
auto_id_identifier == updated_certificate.derive_identifier(),
Error::<T>::AutoIdIdentifierMismatch
);
let updated_auto_id = AutoId {
certificate: updated_certificate,
};
AutoIds::<T>::insert(auto_id_identifier, updated_auto_id);
Self::deposit_event(Event::<T>::AutoIdRenewed(auto_id_identifier));
Ok(Some(weight).into())
}
fn do_verify_signature(
auto_id: &AutoId,
signing_data: CertificateAction,
signature: Signature,
) -> DispatchResult {
let Signature {
signature_algorithm,
value: signature,
} = signature;
let req = SignatureVerificationRequest {
public_key_info: auto_id.certificate.subject_public_key_info(),
signature_algorithm,
data: signing_data.encode(),
signature,
};
verify_signature(req).ok_or(Error::<T>::InvalidSignature)?;
Ok(())
}
fn do_revoke_certificate(
auto_id_identifier: Identifier,
signature: Signature,
) -> DispatchResultWithPostInfo {
let auto_id = AutoIds::<T>::get(auto_id_identifier).ok_or(Error::<T>::UnknownAutoId)?;
let (issuer_id, mut issuer_auto_id, weight) = match auto_id.certificate.issuer_id() {
Some(issuer_id) => (
issuer_id,
AutoIds::<T>::get(issuer_id).ok_or(Error::<T>::UnknownIssuer)?,
T::Weights::revoke_leaf_auto_id(),
),
None => (
auto_id_identifier,
auto_id.clone(),
T::Weights::revoke_issuer_auto_id(),
),
};
ensure!(
!CertificateRevocationList::<T>::get(issuer_id).is_some_and(|serials| {
serials.iter().any(|s| {
*s == auto_id.certificate.serial() || *s == issuer_auto_id.certificate.serial()
})
}),
Error::<T>::CertificateAlreadyRevoked
);
Self::do_verify_signature(
&issuer_auto_id,
CertificateAction {
id: auto_id_identifier,
nonce: issuer_auto_id.certificate.nonce(),
action_type: CertificateActionType::RevokeCertificate,
},
signature,
)?;
CertificateRevocationList::<T>::mutate(issuer_id, |serials| {
serials
.get_or_insert_with(BTreeSet::new)
.insert(auto_id.certificate.serial());
});
issuer_auto_id.certificate.inc_nonce::<T>()?;
AutoIds::<T>::insert(issuer_id, issuer_auto_id);
Self::deposit_event(Event::<T>::CertificateRevoked(auto_id_identifier));
Ok(Some(weight).into())
}
fn do_deactivate_auto_id(
auto_id_identifier: Identifier,
signature: Signature,
) -> DispatchResult {
let auto_id = AutoIds::<T>::get(auto_id_identifier).ok_or(Error::<T>::UnknownIssuer)?;
Self::do_verify_signature(
&auto_id,
CertificateAction {
id: auto_id_identifier,
nonce: auto_id.certificate.nonce(),
action_type: CertificateActionType::DeactivateAutoId,
},
signature,
)?;
AutoIds::<T>::remove(auto_id_identifier);
Self::deposit_event(Event::<T>::AutoIdDeactivated(auto_id_identifier));
Ok(())
}
fn max_register_auto_id_weight() -> Weight {
T::Weights::register_issuer_auto_id().max(T::Weights::register_leaf_auto_id())
}
fn max_revoke_auto_id_weight() -> Weight {
T::Weights::revoke_issuer_auto_id().max(T::Weights::revoke_leaf_auto_id())
}
fn max_renew_auto_id_weight() -> Weight {
T::Weights::renew_issuer_auto_id().max(T::Weights::renew_leaf_auto_id())
}
}