// This file is part of Substrate. // Copyright (C) Parity Technologies (UK) Ltd. // SPDX-License-Identifier: Apache-2.0 // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. //! A set of well-known keys used for testing. pub use sp_core::bandersnatch; use sp_core::{ bandersnatch::{Pair, Public, Signature}, crypto::UncheckedFrom, ByteArray, Pair as PairT, }; use lazy_static::lazy_static; use std::{collections::HashMap, ops::Deref, sync::Mutex}; /// Set of test accounts. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, strum::Display, strum::EnumIter)] pub enum Keyring { Alice, Bob, Charlie, Dave, Eve, Ferdie, One, Two, } const PUBLIC_RAW_LEN: usize = ::LEN; impl Keyring { pub fn from_public(who: &Public) -> Option { Self::iter().find(|&k| &Public::from(k) == who) } pub fn from_raw_public(who: [u8; PUBLIC_RAW_LEN]) -> Option { Self::from_public(&Public::unchecked_from(who)) } pub fn to_raw_public(self) -> [u8; PUBLIC_RAW_LEN] { *Public::from(self).as_ref() } pub fn to_raw_public_vec(self) -> Vec { Public::from(self).to_raw_vec() } pub fn sign(self, msg: &[u8]) -> Signature { Pair::from(self).sign(msg) } pub fn pair(self) -> Pair { Pair::from_string(&format!("//{}", <&'static str>::from(self)), None) .expect("static values are known good; qed") } /// Returns an iterator over all test accounts. pub fn iter() -> impl Iterator { ::iter() } pub fn public(self) -> Public { self.pair().public() } pub fn to_seed(self) -> String { format!("//{}", self) } /// Create a crypto `Pair` from a numeric value. pub fn numeric(idx: usize) -> Pair { Pair::from_string(&format!("//{}", idx), None).expect("numeric values are known good; qed") } } impl From for &'static str { fn from(k: Keyring) -> Self { match k { Keyring::Alice => "Alice", Keyring::Bob => "Bob", Keyring::Charlie => "Charlie", Keyring::Dave => "Dave", Keyring::Eve => "Eve", Keyring::Ferdie => "Ferdie", Keyring::One => "One", Keyring::Two => "Two", } } } #[derive(Debug)] pub struct ParseKeyringError; impl std::fmt::Display for ParseKeyringError { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "ParseKeyringError") } } impl std::str::FromStr for Keyring { type Err = ParseKeyringError; fn from_str(s: &str) -> Result::Err> { match s { "Alice" => Ok(Keyring::Alice), "Bob" => Ok(Keyring::Bob), "Charlie" => Ok(Keyring::Charlie), "Dave" => Ok(Keyring::Dave), "Eve" => Ok(Keyring::Eve), "Ferdie" => Ok(Keyring::Ferdie), "One" => Ok(Keyring::One), "Two" => Ok(Keyring::Two), _ => Err(ParseKeyringError), } } } lazy_static! { static ref PRIVATE_KEYS: Mutex> = Mutex::new(Keyring::iter().map(|who| (who, who.pair())).collect()); static ref PUBLIC_KEYS: HashMap = PRIVATE_KEYS .lock() .unwrap() .iter() .map(|(&who, pair)| (who, pair.public())) .collect(); } impl From for Public { fn from(k: Keyring) -> Self { *(*PUBLIC_KEYS).get(&k).unwrap() } } impl From for Pair { fn from(k: Keyring) -> Self { k.pair() } } impl From for [u8; PUBLIC_RAW_LEN] { fn from(k: Keyring) -> Self { *(*PUBLIC_KEYS).get(&k).unwrap().as_ref() } } impl From for &'static [u8; PUBLIC_RAW_LEN] { fn from(k: Keyring) -> Self { PUBLIC_KEYS.get(&k).unwrap().as_ref() } } impl AsRef<[u8; PUBLIC_RAW_LEN]> for Keyring { fn as_ref(&self) -> &[u8; PUBLIC_RAW_LEN] { PUBLIC_KEYS.get(self).unwrap().as_ref() } } impl AsRef for Keyring { fn as_ref(&self) -> &Public { PUBLIC_KEYS.get(self).unwrap() } } impl Deref for Keyring { type Target = [u8; PUBLIC_RAW_LEN]; fn deref(&self) -> &[u8; PUBLIC_RAW_LEN] { PUBLIC_KEYS.get(self).unwrap().as_ref() } } #[cfg(test)] mod tests { use super::*; use sp_core::{bandersnatch::Pair, Pair as PairT}; #[test] fn should_work() { assert!(Pair::verify( &Keyring::Alice.sign(b"I am Alice!"), b"I am Alice!", &Keyring::Alice.public(), )); assert!(!Pair::verify( &Keyring::Alice.sign(b"I am Alice!"), b"I am Bob!", &Keyring::Alice.public(), )); assert!(!Pair::verify( &Keyring::Alice.sign(b"I am Alice!"), b"I am Alice!", &Keyring::Bob.public(), )); } }