Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Add JSON format to import blocks and set it as default #5816

Merged
merged 67 commits into from
May 22, 2020
Merged
Show file tree
Hide file tree
Changes from 66 commits
Commits
Show all changes
67 commits
Select commit Hold shift + click to select a range
0bb1eb7
Add BlockStream Enum and utility fn
Apr 24, 2020
b291ff1
WIP: Modify import closure to work with BlockStream
Apr 24, 2020
49b0144
Fix trait bounds
Apr 24, 2020
cb6aded
Working prototype
Apr 24, 2020
155695d
Revamp block importing
Apr 28, 2020
e59e1ad
Add export_import_flow tests
Apr 28, 2020
7b729fe
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
Apr 28, 2020
69a8bff
Add comments and clean code
Apr 28, 2020
ab173a5
Add more comments in the import fn
Apr 28, 2020
e22ed9b
Add link code to import function
Apr 28, 2020
00fbd6c
Add condition when returning Ready(Ok(()) to make sure we've imported…
Apr 28, 2020
e7a3c4d
Add check for imported blocks in JSON case
Apr 28, 2020
56d49fa
Use rest pattern
Apr 29, 2020
ff75523
Fix compilation error for undeclared variable
Apr 29, 2020
7c66af2
Add polling and waker before pending
Apr 29, 2020
419b0ab
Print read_block_count instead of count
Apr 29, 2020
be1d626
Simplify binary cli option with structopt
Apr 29, 2020
a67a75d
Update test to reflect changes in CLI api
Apr 29, 2020
e1ce1aa
Change Stream to take SignedBlock<B> instead of B
Apr 30, 2020
30f60de
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
Apr 30, 2020
ecf8729
Add comments to BlockStream
Apr 30, 2020
7b8f34c
Move out logic to smaller functions for clearer code
Apr 30, 2020
1a9e910
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
Apr 30, 2020
61260b5
Remove result over import_blocks return type
Apr 30, 2020
4ca8fe6
Check for error in command output rather than simply checking command…
May 1, 2020
c986b44
Revamp export/import/revert testing
May 1, 2020
8b3a17a
Fix minor typos and formatting errors
pscott May 6, 2020
c57415b
Remove unnecessary if condition in terminating condition
pscott May 6, 2020
32eb4af
Explicit error instead of returning it as a string
pscott May 6, 2020
412e942
Pass BlockStream to log_importing_status_updates and simplify matchin…
May 6, 2020
2266d67
Use .contains() instead of regex match
May 6, 2020
f62d5e0
Line break in match block; return future::ready instead of poll_fn
May 6, 2020
ad44445
Update Cargo.lock
May 6, 2020
17f4c74
Add check so that queue doesn't grow too big
May 6, 2020
994157c
Use Iterator instead of Stream
May 7, 2020
4d1234e
Remove allow dead_code
May 7, 2020
37ae67b
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 7, 2020
757693d
Remove outdated comments
pscott May 7, 2020
9865fae
Return Errors instead of logging them
May 7, 2020
1150f12
Simplify match arms
pscott May 8, 2020
377823c
Remove check before terminating block import
May 8, 2020
576606c
Apply suggestions from code review
bkchr May 11, 2020
9b4658a
Check that queue is not full BEFORE calling
May 12, 2020
76a5954
Revert "Remove check before terminating block import"
May 12, 2020
f848a05
Improve unit tests to make sure we actually import blocks
May 13, 2020
d924935
Remove Unpin implementation for BlockIter
May 13, 2020
3d00933
Add prototype of enum for ImportStates
May 18, 2020
fb040fc
Add working prototype for StateMachine
May 18, 2020
4f9bcf8
Add comments for clearer code
May 18, 2020
9a134eb
Add sleep before calling Waker when waiting for import queue
May 19, 2020
fdd3683
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 19, 2020
d98c92d
Add Speedometer
May 19, 2020
344286e
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 20, 2020
5bdc4fd
add dbg!(&log) for test debugging
May 21, 2020
5d1f1e3
Fix lines with more than 100 cols
May 21, 2020
768c309
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 21, 2020
a17cc1e
Fix regex capture for test
May 21, 2020
9d7d7a1
Update regexes to take to capture the whole number
May 21, 2020
03a5cc8
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 21, 2020
ef02ec3
Rename Cmd to Command
pscott May 21, 2020
fb91c53
Actually rename Cmd to Command
May 21, 2020
4f067ef
Apply suggestions from code review
gnunicorn May 22, 2020
0c47a45
Merge branch 'master' of github.com:paritytech/substrate into scott_d…
May 22, 2020
196de60
Fix compilation errors for tests
May 22, 2020
731f1b7
Merge branch 'scott_default_json_import_blocks' of github.com:parityt…
May 22, 2020
f43f3de
Fix compilation errors from code review suggestion
May 22, 2020
cdc1290
Update bin/node/cli/tests/export_import_flow.rs
gavofyork May 22, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion bin/node/cli/tests/check_block_works.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ use assert_cmd::cargo::cargo_bin;
use std::process::Command;
use tempfile::tempdir;

mod common;
pub mod common;

#[test]
fn check_block_works() {
Expand Down
1 change: 0 additions & 1 deletion bin/node/cli/tests/common.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@
// along with this program. If not, see <https://www.gnu.org/licenses/>.

#![cfg(unix)]
#![allow(dead_code)]

use std::{process::{Child, ExitStatus}, thread, time::Duration, path::Path};
use assert_cmd::cargo::cargo_bin;
Expand Down
214 changes: 214 additions & 0 deletions bin/node/cli/tests/export_import_flow.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,214 @@
// This file is part of Substrate.

// Copyright (C) 2020 Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0

// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.

#![cfg(unix)]

use assert_cmd::cargo::cargo_bin;
use std::{process::Command, fs, path::PathBuf};
use tempfile::{tempdir, TempDir};
use regex::Regex;

pub mod common;

fn contains_error(logged_output: &str) -> bool {
logged_output.contains("Error")
}

/// Helper struct to execute the export/import/revert tests.
/// The fields are paths to a temporary directory
struct ExportImportRevertExecutor<'a> {
base_path: &'a TempDir,
exported_blocks_file: &'a PathBuf,
db_path: &'a PathBuf,
num_exported_blocks: Option<u64>,
}

/// Format options for export / import commands.
enum FormatOpt {
Json,
Binary,
}

/// Command corresponding to the different commands we would like to run.
enum SubCommand {
ExportBlocks,
ImportBlocks,
}

impl ToString for SubCommand {
fn to_string(&self) -> String {
match self {
SubCommand::ExportBlocks => String::from("export-blocks"),
SubCommand::ImportBlocks => String::from("import-blocks"),
}
}
}

impl<'a> ExportImportRevertExecutor<'a> {
fn new(
base_path: &'a TempDir,
exported_blocks_file: &'a PathBuf,
db_path: &'a PathBuf
) -> Self {
Self {
base_path,
exported_blocks_file,
db_path,
num_exported_blocks: None,
}
}

/// Helper method to run a command. Returns a string corresponding to what has been logged.
fn run_block_command(&self,
sub_command: SubCommand,
format_opt: FormatOpt,
expected_to_fail: bool
) -> String {
let sub_command_str = sub_command.to_string();
// Adding "--binary" if need be.
let arguments: Vec<&str> = match format_opt {
FormatOpt::Binary => vec![&sub_command_str, "--dev", "--pruning", "archive", "--binary", "-d"],
FormatOpt::Json => vec![&sub_command_str, "--dev", "--pruning", "archive", "-d"],
};

let tmp: TempDir;
// Setting base_path to be a temporary folder if we are importing blocks.
// This allows us to make sure we are importing from scratch.
let base_path = match sub_command {
SubCommand::ExportBlocks => &self.base_path.path(),
SubCommand::ImportBlocks => {
tmp = tempdir().unwrap();
tmp.path()
}
};

// Running the command and capturing the output.
let output = Command::new(cargo_bin("substrate"))
.args(&arguments)
.arg(&base_path)
.arg(&self.exported_blocks_file)
.output()
.unwrap();

let logged_output = String::from_utf8_lossy(&output.stderr).to_string();

dbg!(&output.stderr);
dbg!(&output.status);
gavofyork marked this conversation as resolved.
Show resolved Hide resolved
if expected_to_fail {
// Checking that we did indeed find an error.
assert!(contains_error(&logged_output), "expected to error but did not error!");
assert!(!output.status.success());
} else {
// Making sure no error were logged.
assert!(!contains_error(&logged_output), "expected not to error but error'd!");
assert!(output.status.success());
}

logged_output
}

/// Runs the `export-blocks` command.
fn run_export(&mut self, fmt_opt: FormatOpt) {
let log = self.run_block_command(SubCommand::ExportBlocks, fmt_opt, false);

// Using regex to find out how many block we exported.
let re = Regex::new(r"Exporting blocks from #\d* to #(?P<exported_blocks>\d*)").unwrap();
let caps = re.captures(&log).unwrap();
// Saving the number of blocks we've exported for further use.
self.num_exported_blocks = Some(caps["exported_blocks"].parse::<u64>().unwrap());

let metadata = fs::metadata(&self.exported_blocks_file).unwrap();
assert!(metadata.len() > 0, "file exported_blocks should not be empty");

let _ = fs::remove_dir_all(&self.db_path);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why delete files manually if TempDir is supposed to handle that?

Copy link
Contributor Author

@pscott pscott May 6, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So this was part of the original test and I believe the main idea is to run the node, export the blocks, and then delete the DB to make sure that import-blocks is actually importing blocks, and not "doing nothing".

Maybe what we should add is a test to see that import blocks actually imported blocks? Not sure how to do that, I wouldn't really know what we could query...

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@arkpar Yes that was my code actually. First create the temp directory, export, delete the directory, import and then only let the destructor clean-up. It was done on purpose but you could also create another tempdir instead of reusing one, it might make the code more understandable.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So keep in mind that now that we've changed the test flow, we need to keep the same directory active for the whole testing period, in order to keep the db :)
I believe the remove_dir_all only happens between the export-blocks command and the import-blocks which makes sense, so that we make sure that we are indeed importing stuff! :)

}

/// Runs the `import-blocks` command, asserting that an error was found or
/// not depending on `expected_to_fail`.
fn run_import(&mut self, fmt_opt: FormatOpt, expected_to_fail: bool) {
let log = self.run_block_command(SubCommand::ImportBlocks, fmt_opt, expected_to_fail);

if !expected_to_fail {
// Using regex to find out how much block we imported,
// and what's the best current block.
let re = Regex::new(r"Imported (?P<imported>\d*) blocks. Best: #(?P<best>\d*)").unwrap();
let caps = re.captures(&log).expect("capture should have succeeded");
let imported = caps["imported"].parse::<u64>().unwrap();
let best = caps["best"].parse::<u64>().unwrap();

assert_eq!(
imported,
best,
"numbers of blocks imported and best number differs"
);
assert_eq!(
best,
self.num_exported_blocks.expect("number of exported blocks cannot be None; qed"),
"best block number and number of expected blocks should not differ"
);
}
self.num_exported_blocks = None;
}

/// Runs the `revert` command.
fn run_revert(&self) {
let output = Command::new(cargo_bin("substrate"))
.args(&["revert", "--dev", "--pruning", "archive", "-d"])
.arg(&self.base_path.path())
.output()
.unwrap();

let logged_output = String::from_utf8_lossy(&output.stderr).to_string();

// Reverting should not log any error.
assert!(!contains_error(&logged_output));
// Command should never fail.
assert!(output.status.success());
}

/// Helper function that runs the whole export / import / revert flow and checks for errors.
fn run(&mut self, export_fmt: FormatOpt, import_fmt: FormatOpt, expected_to_fail: bool) {
self.run_export(export_fmt);
self.run_import(import_fmt, expected_to_fail);
self.run_revert();
}
}

#[test]
fn export_import_revert() {
let base_path = tempdir().expect("could not create a temp dir");
let exported_blocks_file = base_path.path().join("exported_blocks");
let db_path = base_path.path().join("db");

common::run_dev_node_for_a_while(base_path.path());

let mut executor = ExportImportRevertExecutor::new(
&base_path,
&exported_blocks_file,
&db_path,
);

// Binary and binary should work.
executor.run(FormatOpt::Binary, FormatOpt::Binary, false);
// Binary and JSON should fail.
executor.run(FormatOpt::Binary, FormatOpt::Json, true);
// JSON and JSON should work.
executor.run(FormatOpt::Json, FormatOpt::Json, false);
// JSON and binary should fail.
executor.run(FormatOpt::Json, FormatOpt::Binary, true);
}
61 changes: 0 additions & 61 deletions bin/node/cli/tests/import_export_and_revert_work.rs

This file was deleted.

2 changes: 1 addition & 1 deletion bin/node/cli/tests/inspect_works.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ use assert_cmd::cargo::cargo_bin;
use std::process::Command;
use tempfile::tempdir;

mod common;
pub mod common;

#[test]
fn inspect_works() {
Expand Down
2 changes: 1 addition & 1 deletion bin/node/cli/tests/purge_chain_works.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ use assert_cmd::cargo::cargo_bin;
use std::process::Command;
use tempfile::tempdir;

mod common;
pub mod common;

#[test]
#[cfg(unix)]
Expand Down
2 changes: 1 addition & 1 deletion bin/node/cli/tests/running_the_node_and_interrupt.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ use assert_cmd::cargo::cargo_bin;
use std::{convert::TryInto, process::Command, thread, time::Duration};
use tempfile::tempdir;

mod common;
pub mod common;

#[test]
#[cfg(unix)]
Expand Down
6 changes: 5 additions & 1 deletion client/cli/src/commands/import_blocks_cmd.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,10 @@ pub struct ImportBlocksCmd {
#[structopt(long = "default-heap-pages", value_name = "COUNT")]
pub default_heap_pages: Option<u32>,

/// Try importing blocks from binary format rather than JSON.
#[structopt(long)]
pub binary: bool,

#[allow(missing_docs)]
#[structopt(flatten)]
pub shared_params: SharedParams,
Expand Down Expand Up @@ -79,7 +83,7 @@ impl ImportBlocksCmd {
};

builder(config)?
.import_blocks(file, false)
.import_blocks(file, false, self.binary)
.await
.map_err(Into::into)
}
Expand Down
1 change: 1 addition & 0 deletions client/service/src/builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -831,6 +831,7 @@ pub trait ServiceBuilderCommand {
self,
input: impl Read + Seek + Send + 'static,
force: bool,
binary: bool,
) -> Pin<Box<dyn Future<Output = Result<(), Error>> + Send>>;

/// Performs the blocks export.
Expand Down
Loading