Merge branch 'master' into grim
Continuous Integration / Linux Tests (api) (push) Has been cancelled
Continuous Integration / Linux Tests (chain) (push) Has been cancelled
Continuous Integration / Linux Tests (core) (push) Has been cancelled
Continuous Integration / Linux Tests (keychain) (push) Has been cancelled
Continuous Integration / Linux Tests (p2p) (push) Has been cancelled
Continuous Integration / Linux Tests (pool) (push) Has been cancelled
Continuous Integration / Linux Tests (servers) (push) Has been cancelled
Continuous Integration / Linux Tests (src) (push) Has been cancelled
Continuous Integration / Linux Tests (store) (push) Has been cancelled
Continuous Integration / Linux Tests (util) (push) Has been cancelled
Continuous Integration / macOS Tests (push) Has been cancelled
Continuous Integration / Windows Tests (push) Has been cancelled
Continuous Integration / Linux Tests (api) (push) Has been cancelled
Continuous Integration / Linux Tests (chain) (push) Has been cancelled
Continuous Integration / Linux Tests (core) (push) Has been cancelled
Continuous Integration / Linux Tests (keychain) (push) Has been cancelled
Continuous Integration / Linux Tests (p2p) (push) Has been cancelled
Continuous Integration / Linux Tests (pool) (push) Has been cancelled
Continuous Integration / Linux Tests (servers) (push) Has been cancelled
Continuous Integration / Linux Tests (src) (push) Has been cancelled
Continuous Integration / Linux Tests (store) (push) Has been cancelled
Continuous Integration / Linux Tests (util) (push) Has been cancelled
Continuous Integration / macOS Tests (push) Has been cancelled
Continuous Integration / Windows Tests (push) Has been cancelled
# Conflicts: # p2p/Cargo.toml
This commit is contained in:
Generated
+11
-11
@@ -957,7 +957,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin"
|
name = "grin"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"blake2-rfc",
|
"blake2-rfc",
|
||||||
"built",
|
"built",
|
||||||
@@ -987,7 +987,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_api"
|
name = "grin_api"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"async-stream",
|
"async-stream",
|
||||||
"bytes 1.7.1",
|
"bytes 1.7.1",
|
||||||
@@ -1020,7 +1020,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_chain"
|
name = "grin_chain"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bit-vec",
|
"bit-vec",
|
||||||
"bitflags 1.3.2",
|
"bitflags 1.3.2",
|
||||||
@@ -1044,7 +1044,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_config"
|
name = "grin_config"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"dirs",
|
"dirs",
|
||||||
"grin_core",
|
"grin_core",
|
||||||
@@ -1060,7 +1060,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_core"
|
name = "grin_core"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"blake2-rfc",
|
"blake2-rfc",
|
||||||
"byteorder",
|
"byteorder",
|
||||||
@@ -1086,7 +1086,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_keychain"
|
name = "grin_keychain"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"blake2-rfc",
|
"blake2-rfc",
|
||||||
"byteorder",
|
"byteorder",
|
||||||
@@ -1107,7 +1107,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_p2p"
|
name = "grin_p2p"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitflags 1.3.2",
|
"bitflags 1.3.2",
|
||||||
"built",
|
"built",
|
||||||
@@ -1130,7 +1130,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_pool"
|
name = "grin_pool"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"blake2-rfc",
|
"blake2-rfc",
|
||||||
"chrono",
|
"chrono",
|
||||||
@@ -1162,7 +1162,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_servers"
|
name = "grin_servers"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"async-stream",
|
"async-stream",
|
||||||
"chrono",
|
"chrono",
|
||||||
@@ -1192,7 +1192,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_store"
|
name = "grin_store"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"chrono",
|
"chrono",
|
||||||
@@ -1214,7 +1214,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "grin_util"
|
name = "grin_util"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"backtrace",
|
"backtrace",
|
||||||
|
|||||||
+12
-12
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin"
|
name = "grin"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -34,15 +34,15 @@ serde_json = "1"
|
|||||||
log = "0.4"
|
log = "0.4"
|
||||||
term = "0.6"
|
term = "0.6"
|
||||||
|
|
||||||
grin_api = { path = "./api", version = "5.4.0" }
|
grin_api = { path = "./api", version = "5.4.1" }
|
||||||
grin_config = { path = "./config", version = "5.4.0" }
|
grin_config = { path = "./config", version = "5.4.1" }
|
||||||
grin_chain = { path = "./chain", version = "5.4.0" }
|
grin_chain = { path = "./chain", version = "5.4.1" }
|
||||||
grin_core = { path = "./core", version = "5.4.0" }
|
grin_core = { path = "./core", version = "5.4.1" }
|
||||||
grin_keychain = { path = "./keychain", version = "5.4.0" }
|
grin_keychain = { path = "./keychain", version = "5.4.1" }
|
||||||
grin_p2p = { path = "./p2p", version = "5.4.0" }
|
grin_p2p = { path = "./p2p", version = "5.4.1" }
|
||||||
grin_servers = { path = "./servers", version = "5.4.0" }
|
grin_servers = { path = "./servers", version = "5.4.1" }
|
||||||
grin_util = { path = "./util", version = "5.4.0" }
|
grin_util = { path = "./util", version = "5.4.1" }
|
||||||
grin_store = { path = "./store", version = "5.4.0" }
|
grin_store = { path = "./store", version = "5.4.1" }
|
||||||
|
|
||||||
[dependencies.cursive]
|
[dependencies.cursive]
|
||||||
version = "0.21"
|
version = "0.21"
|
||||||
@@ -53,5 +53,5 @@ features = ["pancurses-backend"]
|
|||||||
built = { version = "0.8.0", features = ["git2"]}
|
built = { version = "0.8.0", features = ["git2"]}
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
grin_chain = { path = "./chain", version = "5.4.0" }
|
grin_chain = { path = "./chain", version = "5.4.1" }
|
||||||
grin_store = { path = "./store", version = "5.4.0" }
|
grin_store = { path = "./store", version = "5.4.1" }
|
||||||
|
|||||||
+7
-7
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_api"
|
name = "grin_api"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "APIs for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "APIs for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -32,9 +32,9 @@ async-stream = "0.3"
|
|||||||
url = "2.1"
|
url = "2.1"
|
||||||
bytes = "1"
|
bytes = "1"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_chain = { path = "../chain", version = "5.4.0" }
|
grin_chain = { path = "../chain", version = "5.4.1" }
|
||||||
grin_p2p = { path = "../p2p", version = "5.4.0" }
|
grin_p2p = { path = "../p2p", version = "5.4.1" }
|
||||||
grin_pool = { path = "../pool", version = "5.4.0" }
|
grin_pool = { path = "../pool", version = "5.4.1" }
|
||||||
grin_store = { path = "../store", version = "5.4.0" }
|
grin_store = { path = "../store", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|||||||
+5
-5
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_chain"
|
name = "grin_chain"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -23,10 +23,10 @@ chrono = "0.4.11"
|
|||||||
lru-cache = "0.1"
|
lru-cache = "0.1"
|
||||||
lazy_static = "1"
|
lazy_static = "1"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_keychain = { path = "../keychain", version = "5.4.0" }
|
grin_keychain = { path = "../keychain", version = "5.4.1" }
|
||||||
grin_store = { path = "../store", version = "5.4.0" }
|
grin_store = { path = "../store", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
env_logger = "0.7"
|
env_logger = "0.7"
|
||||||
|
|||||||
@@ -280,6 +280,102 @@ pub struct BitmapSegment {
|
|||||||
proof: SegmentProof,
|
proof: SegmentProof,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl BitmapSegment {
|
||||||
|
// Matches the upper end of the currently served PIBD bitmap segment range.
|
||||||
|
const MAX_SEGMENT_HEIGHT: u8 = 13;
|
||||||
|
|
||||||
|
fn max_chunks(identifier: &SegmentIdentifier) -> Result<usize, ser::Error> {
|
||||||
|
if identifier.height > Self::MAX_SEGMENT_HEIGHT {
|
||||||
|
return Err(ser::Error::TooLargeReadErr);
|
||||||
|
}
|
||||||
|
1usize
|
||||||
|
.checked_shl(identifier.height as u32)
|
||||||
|
.ok_or(ser::Error::TooLargeReadErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn leaf_offset(identifier: &SegmentIdentifier) -> Result<u64, ser::Error> {
|
||||||
|
let segment_capacity = 1u64
|
||||||
|
.checked_shl(identifier.height as u32)
|
||||||
|
.ok_or(ser::Error::TooLargeReadErr)?;
|
||||||
|
segment_capacity
|
||||||
|
.checked_mul(identifier.idx)
|
||||||
|
.ok_or(ser::Error::TooLargeReadErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn n_chunks(blocks: &[BitmapBlock]) -> Result<usize, ser::Error> {
|
||||||
|
let (last, full_blocks) = blocks.split_last().ok_or(ser::Error::CorruptedData)?;
|
||||||
|
for block in full_blocks {
|
||||||
|
if block.try_n_chunks()? != BitmapBlock::NCHUNKS {
|
||||||
|
return Err(ser::Error::CorruptedData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let last_chunks = last.try_n_chunks()?;
|
||||||
|
if last_chunks == 0 {
|
||||||
|
return Err(ser::Error::CorruptedData);
|
||||||
|
}
|
||||||
|
full_blocks
|
||||||
|
.len()
|
||||||
|
.checked_mul(BitmapBlock::NCHUNKS)
|
||||||
|
.and_then(|n| n.checked_add(last_chunks))
|
||||||
|
.ok_or(ser::Error::TooLargeReadErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn validate_blocks(
|
||||||
|
identifier: &SegmentIdentifier,
|
||||||
|
blocks: &[BitmapBlock],
|
||||||
|
) -> Result<usize, ser::Error> {
|
||||||
|
let offset = Self::leaf_offset(identifier)?;
|
||||||
|
let n_chunks = Self::n_chunks(blocks)?;
|
||||||
|
if n_chunks > Self::max_chunks(identifier)? {
|
||||||
|
return Err(ser::Error::TooLargeReadErr);
|
||||||
|
}
|
||||||
|
offset
|
||||||
|
.checked_add((n_chunks - 1) as u64)
|
||||||
|
.ok_or(ser::Error::TooLargeReadErr)?;
|
||||||
|
Ok(n_chunks)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Convert this bitmap segment into a PMMR segment, validating its encoded shape.
|
||||||
|
pub fn into_segment(self) -> Result<Segment<BitmapChunk>, ser::Error> {
|
||||||
|
let BitmapSegment {
|
||||||
|
identifier,
|
||||||
|
blocks,
|
||||||
|
proof,
|
||||||
|
} = self;
|
||||||
|
|
||||||
|
let n_chunks = Self::validate_blocks(&identifier, &blocks)?;
|
||||||
|
let mut leaf_pos = Vec::with_capacity(n_chunks);
|
||||||
|
let mut chunks = Vec::with_capacity(n_chunks);
|
||||||
|
let offset = Self::leaf_offset(&identifier)?;
|
||||||
|
for i in 0..(n_chunks as u64) {
|
||||||
|
let insertion_idx = offset.checked_add(i).ok_or(ser::Error::TooLargeReadErr)?;
|
||||||
|
leaf_pos.push(pmmr::insertion_to_pmmr_index(insertion_idx));
|
||||||
|
chunks.push(BitmapChunk::new());
|
||||||
|
}
|
||||||
|
|
||||||
|
for (block_idx, block) in blocks.into_iter().enumerate() {
|
||||||
|
block.try_n_chunks()?;
|
||||||
|
let offset = block_idx * BitmapBlock::NCHUNKS;
|
||||||
|
for (i, _) in block.inner.iter().enumerate().filter(|&(_, v)| v) {
|
||||||
|
chunks
|
||||||
|
.get_mut(offset + i / BitmapChunk::LEN_BITS)
|
||||||
|
.ok_or(ser::Error::CorruptedData)?
|
||||||
|
.0
|
||||||
|
.set(i % BitmapChunk::LEN_BITS, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Segment::from_parts(
|
||||||
|
identifier,
|
||||||
|
Vec::new(),
|
||||||
|
Vec::new(),
|
||||||
|
leaf_pos,
|
||||||
|
chunks,
|
||||||
|
proof,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Writeable for BitmapSegment {
|
impl Writeable for BitmapSegment {
|
||||||
fn write<W: Writer>(&self, writer: &mut W) -> Result<(), ser::Error> {
|
fn write<W: Writer>(&self, writer: &mut W) -> Result<(), ser::Error> {
|
||||||
Writeable::write(&self.identifier, writer)?;
|
Writeable::write(&self.identifier, writer)?;
|
||||||
@@ -297,10 +393,20 @@ impl Readable for BitmapSegment {
|
|||||||
let identifier: SegmentIdentifier = Readable::read(reader)?;
|
let identifier: SegmentIdentifier = Readable::read(reader)?;
|
||||||
|
|
||||||
let n_blocks = reader.read_u16()? as usize;
|
let n_blocks = reader.read_u16()? as usize;
|
||||||
|
if n_blocks == 0 {
|
||||||
|
return Err(ser::Error::CorruptedData);
|
||||||
|
}
|
||||||
|
let max_blocks = (BitmapSegment::max_chunks(&identifier)? + BitmapBlock::NCHUNKS - 1)
|
||||||
|
/ BitmapBlock::NCHUNKS;
|
||||||
|
if n_blocks > max_blocks {
|
||||||
|
return Err(ser::Error::TooLargeReadErr);
|
||||||
|
}
|
||||||
|
BitmapSegment::leaf_offset(&identifier)?;
|
||||||
let mut blocks = Vec::<BitmapBlock>::with_capacity(n_blocks);
|
let mut blocks = Vec::<BitmapBlock>::with_capacity(n_blocks);
|
||||||
for _ in 0..n_blocks {
|
for _ in 0..n_blocks {
|
||||||
blocks.push(Readable::read(reader)?);
|
blocks.push(Readable::read(reader)?);
|
||||||
}
|
}
|
||||||
|
BitmapSegment::validate_blocks(&identifier, &blocks)?;
|
||||||
let proof = Readable::read(reader)?;
|
let proof = Readable::read(reader)?;
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@@ -348,36 +454,7 @@ impl From<Segment<BitmapChunk>> for BitmapSegment {
|
|||||||
// TODO: this can be sped up with some `unsafe` code
|
// TODO: this can be sped up with some `unsafe` code
|
||||||
impl From<BitmapSegment> for Segment<BitmapChunk> {
|
impl From<BitmapSegment> for Segment<BitmapChunk> {
|
||||||
fn from(segment: BitmapSegment) -> Self {
|
fn from(segment: BitmapSegment) -> Self {
|
||||||
let BitmapSegment {
|
segment.into_segment().expect("valid bitmap segment")
|
||||||
identifier,
|
|
||||||
blocks,
|
|
||||||
proof,
|
|
||||||
} = segment;
|
|
||||||
|
|
||||||
// Count the number of chunks taking into account that the final block might be smaller
|
|
||||||
let n_chunks = (blocks.len() - 1) * BitmapBlock::NCHUNKS
|
|
||||||
+ blocks.last().map(|b| b.n_chunks()).unwrap_or(0);
|
|
||||||
let mut leaf_pos = Vec::with_capacity(n_chunks);
|
|
||||||
let mut chunks = Vec::with_capacity(n_chunks);
|
|
||||||
let offset = (1 << identifier.height) * identifier.idx;
|
|
||||||
for i in 0..(n_chunks as u64) {
|
|
||||||
leaf_pos.push(pmmr::insertion_to_pmmr_index(offset + i));
|
|
||||||
chunks.push(BitmapChunk::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
for (block_idx, block) in blocks.into_iter().enumerate() {
|
|
||||||
assert!(block.inner.len() <= BitmapBlock::NBITS as usize);
|
|
||||||
let offset = block_idx * BitmapBlock::NCHUNKS;
|
|
||||||
for (i, _) in block.inner.iter().enumerate().filter(|&(_, v)| v) {
|
|
||||||
chunks
|
|
||||||
.get_mut(offset + i / BitmapChunk::LEN_BITS)
|
|
||||||
.unwrap()
|
|
||||||
.0
|
|
||||||
.set(i % BitmapChunk::LEN_BITS, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Segment::from_parts(identifier, Vec::new(), Vec::new(), leaf_pos, chunks, proof)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -401,12 +478,16 @@ impl BitmapBlock {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn n_chunks(&self) -> usize {
|
fn try_n_chunks(&self) -> Result<usize, ser::Error> {
|
||||||
let length = self.inner.len();
|
let length = self.inner.len();
|
||||||
assert_eq!(length % BitmapChunk::LEN_BITS, 0);
|
if length % BitmapChunk::LEN_BITS != 0 {
|
||||||
|
return Err(ser::Error::CorruptedData);
|
||||||
|
}
|
||||||
let n_chunks = length / BitmapChunk::LEN_BITS;
|
let n_chunks = length / BitmapChunk::LEN_BITS;
|
||||||
assert!(n_chunks <= BitmapBlock::NCHUNKS);
|
if n_chunks > BitmapBlock::NCHUNKS {
|
||||||
n_chunks
|
return Err(ser::Error::TooLargeReadErr);
|
||||||
|
}
|
||||||
|
Ok(n_chunks)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
use self::chain::txhashset::{BitmapAccumulator, BitmapSegment};
|
use self::chain::txhashset::{BitmapAccumulator, BitmapSegment};
|
||||||
use self::core::core::pmmr::segment::{Segment, SegmentIdentifier};
|
use self::core::core::pmmr::segment::{Segment, SegmentIdentifier};
|
||||||
use self::core::ser::{
|
use self::core::ser::{
|
||||||
BinReader, BinWriter, DeserializationMode, ProtocolVersion, Readable, Writeable,
|
self, BinReader, BinWriter, DeserializationMode, ProtocolVersion, Readable, Writeable,
|
||||||
};
|
};
|
||||||
use croaring::Bitmap;
|
use croaring::Bitmap;
|
||||||
use grin_chain as chain;
|
use grin_chain as chain;
|
||||||
@@ -10,6 +10,29 @@ use grin_util::secp::rand::Rng;
|
|||||||
use rand::thread_rng;
|
use rand::thread_rng;
|
||||||
use std::io::Cursor;
|
use std::io::Cursor;
|
||||||
|
|
||||||
|
fn push_u16(bytes: &mut Vec<u8>, n: u16) {
|
||||||
|
bytes.extend_from_slice(&n.to_be_bytes());
|
||||||
|
}
|
||||||
|
|
||||||
|
fn push_u64(bytes: &mut Vec<u8>, n: u64) {
|
||||||
|
bytes.extend_from_slice(&n.to_be_bytes());
|
||||||
|
}
|
||||||
|
|
||||||
|
fn bitmap_segment_header(height: u8, idx: u64, n_blocks: u16) -> Vec<u8> {
|
||||||
|
let mut bytes = vec![height];
|
||||||
|
push_u64(&mut bytes, idx);
|
||||||
|
push_u16(&mut bytes, n_blocks);
|
||||||
|
bytes
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_bitmap_segment(bytes: &[u8]) -> Result<BitmapSegment, ser::Error> {
|
||||||
|
ser::deserialize(
|
||||||
|
&mut &bytes[..],
|
||||||
|
ProtocolVersion(1),
|
||||||
|
DeserializationMode::default(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
fn test_roundtrip(entries: usize) {
|
fn test_roundtrip(entries: usize) {
|
||||||
let mut rng = thread_rng();
|
let mut rng = thread_rng();
|
||||||
|
|
||||||
@@ -63,7 +86,7 @@ fn test_roundtrip(entries: usize) {
|
|||||||
assert_eq!(bms, bms2);
|
assert_eq!(bms, bms2);
|
||||||
|
|
||||||
// Convert back to `Segment`
|
// Convert back to `Segment`
|
||||||
let segment2 = Segment::from(bms2);
|
let segment2 = bms2.into_segment().unwrap();
|
||||||
assert_eq!(segment, segment2);
|
assert_eq!(segment, segment2);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,3 +106,39 @@ fn abundant_segment_ser_roundtrip() {
|
|||||||
let max = 1 << 16;
|
let max = 1 << 16;
|
||||||
test_roundtrip(thread_rng().gen_range(max - 4096, max - 1024));
|
test_roundtrip(thread_rng().gen_range(max - 4096, max - 1024));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bitmap_segment_read_rejects_empty_blocks() {
|
||||||
|
let bytes = bitmap_segment_header(9, 0, 0);
|
||||||
|
assert_eq!(
|
||||||
|
read_bitmap_segment(&bytes).err(),
|
||||||
|
Some(ser::Error::CorruptedData)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bitmap_segment_read_rejects_too_many_blocks() {
|
||||||
|
let bytes = bitmap_segment_header(9, 0, 9);
|
||||||
|
assert_eq!(
|
||||||
|
read_bitmap_segment(&bytes).err(),
|
||||||
|
Some(ser::Error::TooLargeReadErr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bitmap_segment_read_rejects_too_large_height() {
|
||||||
|
let bytes = bitmap_segment_header(14, 0, 1);
|
||||||
|
assert_eq!(
|
||||||
|
read_bitmap_segment(&bytes).err(),
|
||||||
|
Some(ser::Error::TooLargeReadErr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bitmap_segment_read_rejects_offset_overflow() {
|
||||||
|
let bytes = bitmap_segment_header(13, u64::MAX, 1);
|
||||||
|
assert_eq!(
|
||||||
|
read_bitmap_segment(&bytes).err(),
|
||||||
|
Some(ser::Error::TooLargeReadErr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
+5
-5
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_config"
|
name = "grin_config"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Configuration for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Configuration for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -16,10 +16,10 @@ serde_derive = "1"
|
|||||||
toml = "0.5"
|
toml = "0.5"
|
||||||
dirs = "2.0"
|
dirs = "2.0"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_servers = { path = "../servers", version = "5.4.0" }
|
grin_servers = { path = "../servers", version = "5.4.1" }
|
||||||
grin_p2p = { path = "../p2p", version = "5.4.0" }
|
grin_p2p = { path = "../p2p", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
pretty_assertions = "0.6.1"
|
pretty_assertions = "0.6.1"
|
||||||
|
|||||||
+3
-3
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_core"
|
name = "grin_core"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -28,8 +28,8 @@ chrono = { version = "0.4.11", features = ["serde"] }
|
|||||||
zeroize = { version = "1.1", features =["zeroize_derive"] }
|
zeroize = { version = "1.1", features =["zeroize_derive"] }
|
||||||
bytes = "0.5"
|
bytes = "0.5"
|
||||||
|
|
||||||
keychain = { package = "grin_keychain", path = "../keychain", version = "5.4.0" }
|
keychain = { package = "grin_keychain", path = "../keychain", version = "5.4.1" }
|
||||||
util = { package = "grin_util", path = "../util", version = "5.4.0" }
|
util = { package = "grin_util", path = "../util", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
serde_json = "1"
|
serde_json = "1"
|
||||||
|
|||||||
@@ -21,6 +21,39 @@ use croaring::Bitmap;
|
|||||||
use std::cmp::min;
|
use std::cmp::min;
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
|
|
||||||
|
const MAX_SEGMENT_READ_ITEMS: u64 = 1_000_000;
|
||||||
|
const SEGMENT_READ_PREALLOC_ITEMS: u64 = 1024;
|
||||||
|
|
||||||
|
fn read_segment_item_count<R: Reader>(reader: &mut R) -> Result<u64, Error> {
|
||||||
|
let count = reader.read_u64()?;
|
||||||
|
if count > MAX_SEGMENT_READ_ITEMS {
|
||||||
|
return Err(Error::TooLargeReadErr);
|
||||||
|
}
|
||||||
|
Ok(count)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_segment_positions<R: Reader>(reader: &mut R, count: u64) -> Result<Vec<u64>, Error> {
|
||||||
|
let mut positions = Vec::with_capacity(min(count, SEGMENT_READ_PREALLOC_ITEMS) as usize);
|
||||||
|
let mut last_pos = 0;
|
||||||
|
for _ in 0..count {
|
||||||
|
let pos = reader.read_u64()?;
|
||||||
|
if pos <= last_pos {
|
||||||
|
return Err(Error::SortError);
|
||||||
|
}
|
||||||
|
last_pos = pos;
|
||||||
|
positions.push(pos - 1);
|
||||||
|
}
|
||||||
|
Ok(positions)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_segment_items<T: Readable, R: Reader>(reader: &mut R, count: u64) -> Result<Vec<T>, Error> {
|
||||||
|
let mut items = Vec::with_capacity(min(count, SEGMENT_READ_PREALLOC_ITEMS) as usize);
|
||||||
|
for _ in 0..count {
|
||||||
|
items.push(T::read(reader)?);
|
||||||
|
}
|
||||||
|
Ok(items)
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Eq, PartialEq)]
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
||||||
/// Possible segment types, according to this desegmenter
|
/// Possible segment types, according to this desegmenter
|
||||||
pub enum SegmentType {
|
pub enum SegmentType {
|
||||||
@@ -568,39 +601,13 @@ impl<T: Readable> Readable for Segment<T> {
|
|||||||
fn read<R: Reader>(reader: &mut R) -> Result<Self, Error> {
|
fn read<R: Reader>(reader: &mut R) -> Result<Self, Error> {
|
||||||
let identifier = Readable::read(reader)?;
|
let identifier = Readable::read(reader)?;
|
||||||
|
|
||||||
let n_hashes = reader.read_u64()? as usize;
|
let n_hashes = read_segment_item_count(reader)?;
|
||||||
let mut hash_pos = Vec::with_capacity(n_hashes);
|
let hash_pos = read_segment_positions(reader, n_hashes)?;
|
||||||
let mut last_pos = 0;
|
let hashes = read_segment_items(reader, n_hashes)?;
|
||||||
for _ in 0..n_hashes {
|
|
||||||
let pos = reader.read_u64()?;
|
|
||||||
if pos <= last_pos {
|
|
||||||
return Err(Error::SortError);
|
|
||||||
}
|
|
||||||
last_pos = pos;
|
|
||||||
hash_pos.push(pos - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut hashes = Vec::<Hash>::with_capacity(n_hashes);
|
let n_leaves = read_segment_item_count(reader)?;
|
||||||
for _ in 0..n_hashes {
|
let leaf_pos = read_segment_positions(reader, n_leaves)?;
|
||||||
hashes.push(Readable::read(reader)?);
|
let leaf_data = read_segment_items(reader, n_leaves)?;
|
||||||
}
|
|
||||||
|
|
||||||
let n_leaves = reader.read_u64()? as usize;
|
|
||||||
let mut leaf_pos = Vec::with_capacity(n_leaves);
|
|
||||||
last_pos = 0;
|
|
||||||
for _ in 0..n_leaves {
|
|
||||||
let pos = reader.read_u64()?;
|
|
||||||
if pos <= last_pos {
|
|
||||||
return Err(Error::SortError);
|
|
||||||
}
|
|
||||||
last_pos = pos;
|
|
||||||
leaf_pos.push(pos - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut leaf_data = Vec::<T>::with_capacity(n_leaves);
|
|
||||||
for _ in 0..n_leaves {
|
|
||||||
leaf_data.push(Readable::read(reader)?);
|
|
||||||
}
|
|
||||||
|
|
||||||
let proof = Readable::read(reader)?;
|
let proof = Readable::read(reader)?;
|
||||||
|
|
||||||
@@ -823,12 +830,8 @@ impl SegmentProof {
|
|||||||
|
|
||||||
impl Readable for SegmentProof {
|
impl Readable for SegmentProof {
|
||||||
fn read<R: Reader>(reader: &mut R) -> Result<Self, Error> {
|
fn read<R: Reader>(reader: &mut R) -> Result<Self, Error> {
|
||||||
let n_hashes = reader.read_u64()? as usize;
|
let n_hashes = read_segment_item_count(reader)?;
|
||||||
let mut hashes = Vec::with_capacity(n_hashes);
|
let hashes = read_segment_items(reader, n_hashes)?;
|
||||||
for _ in 0..n_hashes {
|
|
||||||
let hash: Hash = Readable::read(reader)?;
|
|
||||||
hashes.push(hash);
|
|
||||||
}
|
|
||||||
Ok(Self { hashes })
|
Ok(Self { hashes })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,10 +16,15 @@ mod common;
|
|||||||
|
|
||||||
use self::core::core::pmmr;
|
use self::core::core::pmmr;
|
||||||
use self::core::core::{Segment, SegmentIdentifier};
|
use self::core::core::{Segment, SegmentIdentifier};
|
||||||
|
use self::core::ser::{self, DeserializationMode, ProtocolVersion};
|
||||||
use common::TestElem;
|
use common::TestElem;
|
||||||
use grin_core as core;
|
use grin_core as core;
|
||||||
use grin_core::core::pmmr::ReadablePMMR;
|
use grin_core::core::pmmr::ReadablePMMR;
|
||||||
|
|
||||||
|
fn push_u64(bytes: &mut Vec<u8>, n: u64) {
|
||||||
|
bytes.extend_from_slice(&n.to_be_bytes());
|
||||||
|
}
|
||||||
|
|
||||||
fn test_unprunable_size(height: u8, n_leaves: u32) {
|
fn test_unprunable_size(height: u8, n_leaves: u32) {
|
||||||
let size = 1u64 << height;
|
let size = 1u64 << height;
|
||||||
let n_segments = (n_leaves as u64 + size - 1) / size;
|
let n_segments = (n_leaves as u64 + size - 1) / size;
|
||||||
@@ -59,3 +64,30 @@ fn unprunable_mmr() {
|
|||||||
test_unprunable_size(3, i);
|
test_unprunable_size(3, i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn segment_read_rejects_large_hash_count() {
|
||||||
|
let mut bytes = vec![1];
|
||||||
|
push_u64(&mut bytes, 0);
|
||||||
|
push_u64(&mut bytes, 1_000_001);
|
||||||
|
|
||||||
|
let res: Result<Segment<TestElem>, _> = ser::deserialize(
|
||||||
|
&mut &bytes[..],
|
||||||
|
ProtocolVersion(1),
|
||||||
|
DeserializationMode::default(),
|
||||||
|
);
|
||||||
|
assert_eq!(res.err(), Some(ser::Error::TooLargeReadErr));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn segment_proof_read_rejects_large_hash_count() {
|
||||||
|
let mut bytes = vec![];
|
||||||
|
push_u64(&mut bytes, 1_000_001);
|
||||||
|
|
||||||
|
let res: Result<self::core::core::SegmentProof, _> = ser::deserialize(
|
||||||
|
&mut &bytes[..],
|
||||||
|
ProtocolVersion(1),
|
||||||
|
DeserializationMode::default(),
|
||||||
|
);
|
||||||
|
assert_eq!(res.err(), Some(ser::Error::TooLargeReadErr));
|
||||||
|
}
|
||||||
|
|||||||
+2
-2
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_keychain"
|
name = "grin_keychain"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -26,4 +26,4 @@ ripemd160 = "0.9"
|
|||||||
sha2 = "0.9"
|
sha2 = "0.9"
|
||||||
pbkdf2 = "0.8"
|
pbkdf2 = "0.8"
|
||||||
|
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|||||||
+6
-6
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_p2p"
|
name = "grin_p2p"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -23,13 +23,13 @@ log = "0.4"
|
|||||||
chrono = { version = "0.4.11", features = ["serde"] }
|
chrono = { version = "0.4.11", features = ["serde"] }
|
||||||
bytes = "0.5"
|
bytes = "0.5"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_store = { path = "../store", version = "5.4.0" }
|
grin_store = { path = "../store", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
grin_chain = { path = "../chain", version = "5.4.0" }
|
grin_chain = { path = "../chain", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
grin_pool = { path = "../pool", version = "5.4.0" }
|
grin_pool = { path = "../pool", version = "5.4.1" }
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
built = { version = "0.8.0", features = ["git2"]}
|
built = { version = "0.8.0", features = ["git2"]}
|
||||||
+1
-1
@@ -382,7 +382,7 @@ impl MessageHandler for Protocol {
|
|||||||
block_hash,
|
block_hash,
|
||||||
output_root
|
output_root
|
||||||
);
|
);
|
||||||
adapter.receive_bitmap_segment(block_hash, output_root, segment.into())?;
|
adapter.receive_bitmap_segment(block_hash, output_root, segment.into_segment()?)?;
|
||||||
Consumed::None
|
Consumed::None
|
||||||
}
|
}
|
||||||
Message::OutputSegment(req) => {
|
Message::OutputSegment(req) => {
|
||||||
|
|||||||
+5
-5
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_pool"
|
name = "grin_pool"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Chain implementation for grin, a simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -18,9 +18,9 @@ thiserror = "1"
|
|||||||
log = "0.4"
|
log = "0.4"
|
||||||
chrono = "0.4.11"
|
chrono = "0.4.11"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_keychain = { path = "../keychain", version = "5.4.0" }
|
grin_keychain = { path = "../keychain", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
grin_chain = { path = "../chain", version = "5.4.0" }
|
grin_chain = { path = "../chain", version = "5.4.1" }
|
||||||
|
|||||||
+9
-9
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_servers"
|
name = "grin_servers"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -27,11 +27,11 @@ async-stream = "0.3"
|
|||||||
rustls = "0.20"
|
rustls = "0.20"
|
||||||
walkdir = "2.3.1"
|
walkdir = "2.3.1"
|
||||||
|
|
||||||
grin_api = { path = "../api", version = "5.4.0" }
|
grin_api = { path = "../api", version = "5.4.1" }
|
||||||
grin_chain = { path = "../chain", version = "5.4.0" }
|
grin_chain = { path = "../chain", version = "5.4.1" }
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_keychain = { path = "../keychain", version = "5.4.0" }
|
grin_keychain = { path = "../keychain", version = "5.4.1" }
|
||||||
grin_p2p = { path = "../p2p", version = "5.4.0" }
|
grin_p2p = { path = "../p2p", version = "5.4.1" }
|
||||||
grin_pool = { path = "../pool", version = "5.4.0" }
|
grin_pool = { path = "../pool", version = "5.4.1" }
|
||||||
grin_store = { path = "../store", version = "5.4.0" }
|
grin_store = { path = "../store", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|||||||
+3
-3
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_store"
|
name = "grin_store"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
@@ -21,8 +21,8 @@ serde_derive = "1"
|
|||||||
thiserror = "1"
|
thiserror = "1"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
||||||
grin_core = { path = "../core", version = "5.4.0" }
|
grin_core = { path = "../core", version = "5.4.1" }
|
||||||
grin_util = { path = "../util", version = "5.4.0" }
|
grin_util = { path = "../util", version = "5.4.1" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
chrono = "0.4.11"
|
chrono = "0.4.11"
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "grin_util"
|
name = "grin_util"
|
||||||
version = "5.4.0"
|
version = "5.4.1"
|
||||||
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
authors = ["Grin Developers <mimblewimble@lists.launchpad.net>"]
|
||||||
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
description = "Simple, private and scalable cryptocurrency implementation based on the Mimblewimble chain format."
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
|
|||||||
Reference in New Issue
Block a user