2019-05-15 01:41:25 +00:00
|
|
|
use crate::case_result::CaseResult;
|
|
|
|
use crate::cases::*;
|
|
|
|
use crate::doc_header::DocHeader;
|
2019-05-15 05:08:48 +00:00
|
|
|
use crate::eth_specs::{MainnetEthSpec, MinimalEthSpec};
|
2019-05-15 07:14:28 +00:00
|
|
|
use crate::yaml_decode::{extract_yaml_by_key, yaml_split_header_and_cases, YamlDecode};
|
2019-05-15 01:41:25 +00:00
|
|
|
use crate::EfTest;
|
|
|
|
use serde_derive::Deserialize;
|
2019-05-14 01:13:28 +00:00
|
|
|
use std::{fs::File, io::prelude::*, path::PathBuf};
|
2019-05-15 05:08:48 +00:00
|
|
|
use types::EthSpec;
|
2019-05-14 01:13:28 +00:00
|
|
|
|
|
|
|
#[derive(Debug, Deserialize)]
|
2019-05-15 01:15:34 +00:00
|
|
|
pub struct Doc {
|
2019-05-15 07:14:28 +00:00
|
|
|
pub header_yaml: String,
|
|
|
|
pub cases_yaml: String,
|
2019-05-15 02:30:42 +00:00
|
|
|
pub path: PathBuf,
|
2019-05-14 01:13:28 +00:00
|
|
|
}
|
|
|
|
|
2019-05-15 01:15:34 +00:00
|
|
|
impl Doc {
|
2019-05-14 05:08:42 +00:00
|
|
|
fn from_path(path: PathBuf) -> Self {
|
2019-05-15 02:30:42 +00:00
|
|
|
let mut file = File::open(path.clone()).unwrap();
|
2019-05-14 01:13:28 +00:00
|
|
|
|
|
|
|
let mut yaml = String::new();
|
|
|
|
file.read_to_string(&mut yaml).unwrap();
|
|
|
|
|
2019-05-15 07:14:28 +00:00
|
|
|
let (header_yaml, cases_yaml) = yaml_split_header_and_cases(yaml.clone());
|
|
|
|
|
|
|
|
Self {
|
|
|
|
header_yaml,
|
|
|
|
cases_yaml,
|
|
|
|
path,
|
|
|
|
}
|
2019-05-14 01:13:28 +00:00
|
|
|
}
|
|
|
|
|
2019-05-15 02:30:42 +00:00
|
|
|
pub fn test_results(&self) -> Vec<CaseResult> {
|
2019-05-15 07:14:28 +00:00
|
|
|
let header: DocHeader = serde_yaml::from_str(&self.header_yaml.as_str()).unwrap();
|
2019-05-14 01:13:28 +00:00
|
|
|
|
2019-05-14 05:08:42 +00:00
|
|
|
match (
|
|
|
|
header.runner.as_ref(),
|
|
|
|
header.handler.as_ref(),
|
|
|
|
header.config.as_ref(),
|
|
|
|
) {
|
2019-05-22 03:59:54 +00:00
|
|
|
("ssz", "uint", _) => run_test::<SszGeneric, MainnetEthSpec>(self),
|
|
|
|
("ssz", "static", "minimal") => run_test::<SszStatic, MinimalEthSpec>(self),
|
|
|
|
("ssz", "static", "mainnet") => run_test::<SszStatic, MainnetEthSpec>(self),
|
2019-05-21 02:46:22 +00:00
|
|
|
("bls", "aggregate_pubkeys", "mainnet") => {
|
2019-05-22 03:59:54 +00:00
|
|
|
run_test::<BlsAggregatePubkeys, MainnetEthSpec>(self)
|
2019-05-21 02:46:22 +00:00
|
|
|
}
|
|
|
|
("bls", "aggregate_sigs", "mainnet") => {
|
2019-05-22 03:59:54 +00:00
|
|
|
run_test::<BlsAggregateSigs, MainnetEthSpec>(self)
|
2019-05-21 02:46:22 +00:00
|
|
|
}
|
|
|
|
("bls", "msg_hash_compressed", "mainnet") => {
|
2019-05-22 03:59:54 +00:00
|
|
|
run_test::<BlsG2Compressed, MainnetEthSpec>(self)
|
2019-05-21 02:46:22 +00:00
|
|
|
}
|
2019-05-22 04:18:48 +00:00
|
|
|
// Note this test fails due to a difference in our internal representations. It does
|
|
|
|
// not effect verification or external representation.
|
|
|
|
//
|
|
|
|
// It is skipped.
|
|
|
|
("bls", "msg_hash_uncompressed", "mainnet") => vec![],
|
2019-05-22 03:59:54 +00:00
|
|
|
("bls", "priv_to_pub", "mainnet") => run_test::<BlsPrivToPub, MainnetEthSpec>(self),
|
|
|
|
("bls", "sign_msg", "mainnet") => run_test::<BlsSign, MainnetEthSpec>(self),
|
2019-05-14 05:08:42 +00:00
|
|
|
(runner, handler, config) => panic!(
|
|
|
|
"No implementation for runner: \"{}\", handler: \"{}\", config: \"{}\"",
|
|
|
|
runner, handler, config
|
2019-05-14 01:13:28 +00:00
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn assert_tests_pass(path: PathBuf) {
|
2019-05-15 02:30:42 +00:00
|
|
|
let doc = Self::from_path(path);
|
|
|
|
let results = doc.test_results();
|
2019-05-14 01:13:28 +00:00
|
|
|
|
2019-05-15 02:30:42 +00:00
|
|
|
if results.iter().any(|r| r.result.is_err()) {
|
|
|
|
print_failures(&doc, &results);
|
|
|
|
panic!("Tests failed (see above)");
|
2019-05-15 05:08:48 +00:00
|
|
|
} else {
|
|
|
|
println!("Passed {} tests in {:?}", results.len(), doc.path);
|
2019-05-14 01:13:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-15 07:14:28 +00:00
|
|
|
pub fn run_test<T, E: EthSpec>(doc: &Doc) -> Vec<CaseResult>
|
2019-05-14 01:13:28 +00:00
|
|
|
where
|
2019-05-15 01:45:50 +00:00
|
|
|
Cases<T>: EfTest + YamlDecode,
|
2019-05-14 01:13:28 +00:00
|
|
|
{
|
2019-05-15 01:45:50 +00:00
|
|
|
// Extract only the "test_cases" YAML as a stand-alone string.
|
2019-05-15 07:14:28 +00:00
|
|
|
//let test_cases_yaml = extract_yaml_by_key(self., "test_cases");
|
2019-05-14 01:13:28 +00:00
|
|
|
|
2019-05-15 01:45:50 +00:00
|
|
|
// Pass only the "test_cases" YAML string to `yaml_decode`.
|
2019-05-15 07:14:28 +00:00
|
|
|
let test_cases: Cases<T> = Cases::yaml_decode(&doc.cases_yaml).unwrap();
|
2019-05-14 05:08:42 +00:00
|
|
|
|
2019-05-15 01:27:22 +00:00
|
|
|
test_cases.test_results::<E>()
|
2019-05-14 01:13:28 +00:00
|
|
|
}
|
2019-05-15 02:30:42 +00:00
|
|
|
|
|
|
|
pub fn print_failures(doc: &Doc, results: &[CaseResult]) {
|
2019-05-15 07:14:28 +00:00
|
|
|
let header: DocHeader = serde_yaml::from_str(&doc.header_yaml).unwrap();
|
2019-05-15 02:30:42 +00:00
|
|
|
let failures: Vec<&CaseResult> = results.iter().filter(|r| r.result.is_err()).collect();
|
|
|
|
|
|
|
|
println!("--------------------------------------------------");
|
|
|
|
println!("Test Failure");
|
|
|
|
println!("Title: {}", header.title);
|
|
|
|
println!("File: {:?}", doc.path);
|
|
|
|
println!("");
|
|
|
|
println!(
|
|
|
|
"{} tests, {} failures, {} passes.",
|
|
|
|
results.len(),
|
|
|
|
failures.len(),
|
|
|
|
results.len() - failures.len()
|
|
|
|
);
|
|
|
|
println!("");
|
|
|
|
|
|
|
|
for failure in failures {
|
|
|
|
println!("-------");
|
2019-05-15 03:23:52 +00:00
|
|
|
println!("case[{}].result:", failure.case_index);
|
|
|
|
println!("{:#?}", failure.result);
|
2019-05-15 02:30:42 +00:00
|
|
|
}
|
|
|
|
println!("");
|
|
|
|
}
|