Skip to content

Commit 73ae835

Browse files
committed
Basic app skeleton and block parsing
0 parents  commit 73ae835

File tree

7 files changed

+186
-0
lines changed

7 files changed

+186
-0
lines changed

.gitignore

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
.idea
2+
3+
# Compiled files
4+
*.o
5+
*.so
6+
*.rlib
7+
*.dll
8+
9+
# Executables
10+
*.exe
11+
12+
# Generated by Cargo
13+
/target/
14+
15+
/target
16+
**/*.rs.bk
17+
Cargo.lock
18+
19+
# Rust code coverage-specific
20+
/kcov-build/

Cargo.toml

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
[package]
2+
name = "bpdump"
3+
description = "Bitcoin protocol Swiss army knife - like `tcpdump`, but for LNP/BP stack"
4+
version = "0.1.0"
5+
license = "MIT"
6+
authors = ["Dr Maxim Orlovsky <[email protected]>"]
7+
keywords = [ "bitcoin", "lightning", "blockchain" ]
8+
readme = "README.md"
9+
edition = "2018"
10+
11+
[dependencies]
12+
yaml = "0.3.0"
13+
bitcoin = "~0.19.0"
14+
bech32 = "~0.6.0"
15+
bitcoin-bech32 = "~0.9.0"
16+
bitcoin_hashes = "~0.3.0"
17+
18+
[patch.crates-io]
19+
bitcoin = { git = "https://github.com/pandoracore/rust-bitcoin.git", branch = "generic-reader" }
20+
21+
[dependencies.secp256k1]
22+
version = "~0.12.0" # We require this version because of compatibility issues with bitcoin 0.18.0
23+
features = [ "rand" ]
24+
25+
[dependencies.clap]
26+
version = "~2.33"
27+
default-features = true
28+
features = [ "yaml" ]

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2019 Storm
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

README.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
# Bitcoin protocol Swiss army knife - like `tcpdump`, but for LNP/BP stack
2+
3+
WIP
4+
5+
```sh
6+
$ cargo run --package bpdump --features yaml --bin bpdump block sample_data/blk00000.dat
7+
```

cli.yml

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
name: bpdump
2+
version: "0.1.0"
3+
author: Dr Maxim Orlovsky <[email protected]>
4+
about: Bitcoin protocol dumping tool - like TCP dump, but for LNP/BP stack
5+
args:
6+
- COMMAND:
7+
help: command to perform
8+
index: 1
9+
required: true
10+
possible_values:
11+
- block
12+
- tx
13+
- INPUT:
14+
help: Sets the input file to use (otherwise uses STDIN)
15+
required: false
16+
index: 2
17+
- OUTPUT:
18+
help: Sets output file to dump the information (otherwise uses STDOUT)
19+
required: false
20+
index: 3
21+
requires: [ 'INPUT' ]
22+
- verbose:
23+
short: v
24+
multiple: true
25+
help: Sets the level of verbosity
26+
- format:
27+
short: f
28+
multiple: false
29+
takes_value: true
30+
help: Defines output format
31+
value_names:
32+
- csv
33+
- yaml
34+
- json
35+
- text

sample_data/blk00000.dat

45.2 KB
Binary file not shown.

src/main.rs

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
#[macro_use]
2+
extern crate clap;
3+
extern crate bitcoin;
4+
5+
use std::fs::File;
6+
use std::io;
7+
use clap::{App, ArgMatches};
8+
use std::io::BufRead;
9+
use bitcoin::network::stream_reader::StreamReader;
10+
use bitcoin::{Block, BitcoinHash};
11+
12+
13+
#[cfg(feature = "yaml")]
14+
fn main() -> std::io::Result<()> {
15+
eprintln!("\nBitcoin protocol dumping tool\n");
16+
17+
let yaml = load_yaml!("../cli.yml");
18+
let matches = App::from_yaml(yaml).get_matches();
19+
20+
match matches.value_of("COMMAND").unwrap() {
21+
"block" => parse_block(matches),
22+
_ => unreachable!(),
23+
}
24+
}
25+
26+
#[cfg(not(feature = "yaml"))]
27+
fn main() {
28+
// As stated above, if clap is not compiled with the YAML feature, it is disabled.
29+
eprintln!("YAML feature is disabled.");
30+
eprintln!("Pass --features yaml to cargo when trying this example.");
31+
}
32+
33+
fn parse_block(matches: ArgMatches) -> std::io::Result<()> {
34+
let filename = matches.value_of("INPUT");
35+
eprintln!("Parsing block file {}", filename.unwrap_or("STDIN"));
36+
37+
let mut stdin;
38+
let buf_reader: Box<BufRead> = match filename {
39+
Some(name) => Box::new(io::BufReader::new(File::open(name)?)),
40+
None => {
41+
stdin = io::stdin();
42+
Box::new(stdin.lock())
43+
},
44+
};
45+
let mut stream_reader = StreamReader::new(Box::new(buf_reader), None);
46+
47+
let mut blkno = 0;
48+
loop {
49+
// Reading magick number
50+
match stream_reader.read_next::<u32>() {
51+
Ok(0xD9B4BEF9) => (),
52+
_ => {
53+
eprintln!("No magick number found");
54+
break;
55+
}
56+
}
57+
// Skipping block length
58+
eprintln!("Magick number ok");
59+
let _ = stream_reader.read_next::<u32>();
60+
// Reading block
61+
match stream_reader.read_next::<Block>() {
62+
Err(err) => {
63+
eprintln!("{}", err);
64+
break;
65+
},
66+
Ok(block) => {
67+
eprintln!("* read block no {}, id {}", blkno, block.bitcoin_hash());
68+
println!("{:#?}", block.header);
69+
println!("{:#?}", block.txdata[0]);
70+
blkno += 1;
71+
},
72+
}
73+
}
74+
Ok(())
75+
}

0 commit comments

Comments
 (0)