Skip to content

Commit bbb7e5a

Browse files
committed
Respond to Matt's additional comments.
1 parent 5c785a0 commit bbb7e5a

File tree

3 files changed

+61
-73
lines changed

3 files changed

+61
-73
lines changed

lightning-rapid-gossip-sync/src/error.rs

+2-8
Original file line numberDiff line numberDiff line change
@@ -4,21 +4,17 @@ use lightning::ln::msgs::{DecodeError, LightningError};
44

55
/// All-encompassing standard error type that processing can return
66
pub enum GraphSyncError {
7-
/// IO error wrapper, typically the result of an issue with the file system
8-
IOError(std::io::Error),
97
/// Error trying to read the update data, typically due to an erroneous data length indication
108
/// that is greater than the actual amount of data provided
119
DecodeError(DecodeError),
1210
/// Error applying the patch to the network graph, usually the result of updates that are too
1311
/// old or missing prerequisite data to the application of updates out of order
1412
LightningError(LightningError),
15-
/// Some other error whose nature is indicated in its descriptor string
16-
ProcessingError(String),
1713
}
1814

1915
impl From<std::io::Error> for GraphSyncError {
2016
fn from(error: std::io::Error) -> Self {
21-
Self::IOError(error)
17+
Self::DecodeError(DecodeError::Io(error.kind()))
2218
}
2319
}
2420

@@ -37,10 +33,8 @@ impl From<LightningError> for GraphSyncError {
3733
impl Debug for GraphSyncError {
3834
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
3935
match self {
40-
GraphSyncError::IOError(e) => f.write_fmt(format_args!("IOError: {:?}", e)),
4136
GraphSyncError::DecodeError(e) => f.write_fmt(format_args!("DecodeError: {:?}", e)),
42-
GraphSyncError::LightningError(e) => f.write_fmt(format_args!("LightningError: {:?}", e)),
43-
GraphSyncError::ProcessingError(e) => f.write_fmt(format_args!("ProcessingError: {}", e))
37+
GraphSyncError::LightningError(e) => f.write_fmt(format_args!("LightningError: {:?}", e))
4438
}
4539
}
4640
}

lightning-rapid-gossip-sync/src/lib.rs

+54-12
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,21 @@
1010
//! This crate exposes functionality to rapidly sync gossip data, aimed primarily at mobile
1111
//! devices.
1212
//!
13-
//! The (presumed) server sends a compressed response containing gossip data. The gossip data is
13+
//! The server sends a compressed response containing gossip data. The gossip data is
1414
//! formatted compactly, omitting signatures and opportunistically incremental where previous
1515
//! channel updates are known (a mechanism that is enabled when the timestamp of the last known
1616
//! channel update is communicated). A reference server implementation can be found
17-
//! [here](https://github.com/arik-so/rust-ln-sync).
17+
//! [here](https://github.com/lightningdevkit/rapid-gossip-sync-server).
18+
//!
19+
//! The primary benefit this syncing mechanism provides is that given a trusted server, a
20+
//! low-powered client can offload the validation of gossip signatures. This enables a client to
21+
//! privately calculate routes for payments, and do so much faster and earlier than requiring a full
22+
//! peer-to-peer gossip sync to complete.
23+
//!
24+
//! The reason the rapid sync server requires a modicum of trust is that it could provide bogus
25+
//! data, though at worst, all that would result in is a fake network topology, which wouldn't
26+
//! enable the server to steal or siphon off funds. It could, however, reduce the client's privacy
27+
//! by increasing the likelihood payments will are routed via channels the server controls.
1828
//!
1929
//! The way a server is meant to calculate this rapid gossip sync data is by using a `latest_seen`
2030
//! timestamp provided by the client. It's not included in either channel announcement or update,
@@ -67,12 +77,46 @@ mod tests {
6777
use bitcoin::blockdata::constants::genesis_block;
6878
use bitcoin::Network;
6979

80+
use lightning::ln::msgs::DecodeError;
7081
use lightning::routing::network_graph::NetworkGraph;
7182

7283
use crate::sync_network_graph_with_file_path;
7384

7485
#[test]
7586
fn test_sync_from_file() {
87+
struct FileSyncTest {
88+
directory: String,
89+
}
90+
91+
impl FileSyncTest {
92+
fn new(tmp_directory: &str, valid_response: &[u8]) -> FileSyncTest {
93+
let test = FileSyncTest { directory: tmp_directory.to_owned() };
94+
95+
let graph_sync_test_directory = test.get_test_directory();
96+
fs::create_dir_all(graph_sync_test_directory).unwrap();
97+
98+
let graph_sync_test_file = test.get_test_file_path();
99+
fs::write(&graph_sync_test_file, valid_response).unwrap();
100+
101+
test
102+
}
103+
fn get_test_directory(&self) -> String {
104+
let graph_sync_test_directory = self.directory.clone() + "/graph-sync-tests";
105+
graph_sync_test_directory
106+
}
107+
fn get_test_file_path(&self) -> String {
108+
let graph_sync_test_directory = self.get_test_directory();
109+
let graph_sync_test_file = graph_sync_test_directory.to_owned() + "/test_data.lngossip";
110+
graph_sync_test_file
111+
}
112+
}
113+
114+
impl Drop for FileSyncTest {
115+
fn drop(&mut self) {
116+
fs::remove_dir_all(self.directory.clone()).unwrap();
117+
}
118+
}
119+
76120
// same as incremental_only_update_fails_without_prior_same_direction_updates
77121
let valid_response = vec![
78122
76, 68, 75, 1, 111, 226, 140, 10, 182, 241, 179, 114, 193, 166, 162, 70, 174, 99, 247,
@@ -90,19 +134,16 @@ mod tests {
90134
0, 1, 0, 0, 0, 125, 255, 2, 68, 226, 0, 6, 11, 0, 1, 5, 0, 0, 0, 0, 29, 129, 25, 192,
91135
];
92136

93-
let tmp_directory = "./tmp";
94-
let graph_sync_test_directory = tmp_directory.to_owned() + "/graph-sync-tests";
95-
let graph_sync_test_file = graph_sync_test_directory.to_owned() + "/test_data.lngossip";
96-
fs::create_dir_all(graph_sync_test_directory).unwrap();
97-
fs::write(&graph_sync_test_file, valid_response).unwrap();
137+
let tmp_directory = "./rapid-gossip-sync-tests-tmp";
138+
let sync_test = FileSyncTest::new(tmp_directory, &valid_response);
139+
let graph_sync_test_file = sync_test.get_test_file_path();
98140

99141
let block_hash = genesis_block(Network::Bitcoin).block_hash();
100142
let network_graph = NetworkGraph::new(block_hash);
101143

102144
assert_eq!(network_graph.read_only().channels().len(), 0);
103145

104146
let sync_result = sync_network_graph_with_file_path(&network_graph, &graph_sync_test_file);
105-
fs::remove_dir_all(tmp_directory).unwrap();
106147

107148
if sync_result.is_err() {
108149
panic!("Unexpected sync result: {:?}", sync_result)
@@ -136,8 +177,8 @@ mod tests {
136177
let start = std::time::Instant::now();
137178
let sync_result =
138179
sync_network_graph_with_file_path(&network_graph, "./res/full_graph.lngossip");
139-
if let Err(crate::error::GraphSyncError::IOError(io_error)) = &sync_result {
140-
let error_string = format!("Input file lightning-graph-sync/res/full_graph.lngossip is missing! Download it from https://bitcoin.ninja/ldk-compressed_graph-bc08df7542-2022-05-05.bin\n\n{}", io_error);
180+
if let Err(crate::error::GraphSyncError::DecodeError(DecodeError::Io(io_error))) = &sync_result {
181+
let error_string = format!("Input file lightning-graph-sync/res/full_graph.lngossip is missing! Download it from https://bitcoin.ninja/ldk-compressed_graph-bc08df7542-2022-05-05.bin\n\n{:?}", io_error);
141182
#[cfg(not(require_route_graph_test))]
142183
{
143184
println!("{}", error_string);
@@ -160,6 +201,7 @@ pub mod bench {
160201
use bitcoin::blockdata::constants::genesis_block;
161202
use bitcoin::Network;
162203

204+
use lightning::ln::msgs::DecodeError;
163205
use lightning::routing::network_graph::NetworkGraph;
164206

165207
use crate::sync_network_graph_with_file_path;
@@ -173,8 +215,8 @@ pub mod bench {
173215
&network_graph,
174216
"./res/full_graph.lngossip",
175217
);
176-
if let Err(crate::error::GraphSyncError::IOError(io_error)) = &sync_result {
177-
let error_string = format!("Input file lightning-graph-sync/res/full_graph.lngossip is missing! Download it from https://bitcoin.ninja/ldk-compressed_graph-bc08df7542-2022-05-05.bin\n\n{}", io_error);
218+
if let Err(crate::error::GraphSyncError::DecodeError(DecodeError::Io(io_error))) = &sync_result {
219+
let error_string = format!("Input file lightning-graph-sync/res/full_graph.lngossip is missing! Download it from https://bitcoin.ninja/ldk-compressed_graph-bc08df7542-2022-05-05.bin\n\n{:?}", io_error);
178220
#[cfg(not(require_route_graph_test))]
179221
{
180222
println!("{}", error_string);

lightning-rapid-gossip-sync/src/processing.rs

+5-53
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@ use lightning::routing::network_graph;
1212
use lightning::util::ser::{BigSize, Readable};
1313

1414
use crate::error::GraphSyncError;
15-
use crate::GraphSyncError::ProcessingError;
1615

1716
/// The purpose of this prefix is to identify the serialization format, should other rapid gossip
1817
/// sync formats arise in the future.
@@ -47,9 +46,7 @@ pub(crate) fn update_network_graph_from_byte_stream<R: Read>(
4746
let _deserialize_unsigned = match prefix {
4847
GOSSIP_PREFIX => true,
4948
_ => {
50-
return Err(GraphSyncError::ProcessingError(
51-
"Prefix must equal 'LDK' and the version byte 1".to_string(),
52-
));
49+
return Err(DecodeError::UnknownVersion.into());
5350
}
5451
};
5552

@@ -75,17 +72,13 @@ pub(crate) fn update_network_graph_from_byte_stream<R: Read>(
7572
let scid_delta: BigSize = Readable::read(read_cursor)?;
7673
let short_channel_id = previous_scid
7774
.checked_add(scid_delta.0)
78-
.ok_or(ProcessingError(
79-
"Invalid announcement SCID delta producing u64 overflow.".to_string(),
80-
))?;
75+
.ok_or(DecodeError::InvalidValue)?;
8176
previous_scid = short_channel_id;
8277

8378
let node_id_1_index: BigSize = Readable::read(read_cursor)?;
8479
let node_id_2_index: BigSize = Readable::read(read_cursor)?;
8580
if max(node_id_1_index.0, node_id_2_index.0) >= node_id_count as u64 {
86-
return Err(GraphSyncError::ProcessingError(
87-
"Node id index out of bounds".to_string(),
88-
));
81+
return Err(DecodeError::InvalidValue.into());
8982
};
9083
let node_id_1 = node_ids[node_id_1_index.0 as usize];
9184
let node_id_2 = node_ids[node_id_2_index.0 as usize];
@@ -129,9 +122,7 @@ pub(crate) fn update_network_graph_from_byte_stream<R: Read>(
129122
let scid_delta: BigSize = Readable::read(read_cursor)?;
130123
let short_channel_id = previous_scid
131124
.checked_add(scid_delta.0)
132-
.ok_or(ProcessingError(
133-
"Invalid update SCID delta producing u64 overflow.".to_string(),
134-
))?;
125+
.ok_or(DecodeError::InvalidValue)?;
135126
previous_scid = short_channel_id;
136127

137128
let channel_flags: u8 = Readable::read(read_cursor)?;
@@ -225,14 +216,7 @@ pub(crate) fn update_network_graph_from_byte_stream<R: Read>(
225216
network_graph.update_channel_unsigned(&synthetic_update)?;
226217
}
227218

228-
let input_termination_check: Result<u8, DecodeError> = Readable::read(read_cursor);
229-
if let Err(DecodeError::ShortRead) = input_termination_check {
230-
Ok(())
231-
} else {
232-
Err(GraphSyncError::ProcessingError(
233-
"unexpected data found".to_string(),
234-
))
235-
}
219+
Ok(())
236220
}
237221

238222
#[cfg(test)]
@@ -276,38 +260,6 @@ mod tests {
276260
}
277261
}
278262

279-
#[test]
280-
fn network_graph_fails_to_update_from_extended_input() {
281-
let block_hash = genesis_block(Network::Bitcoin).block_hash();
282-
let network_graph = NetworkGraph::new(block_hash);
283-
284-
let example_input = vec![
285-
76, 68, 75, 1, 111, 226, 140, 10, 182, 241, 179, 114, 193, 166, 162, 70, 174, 99, 247,
286-
79, 147, 30, 131, 101, 225, 90, 8, 156, 104, 214, 25, 0, 0, 0, 0, 0, 97, 227, 98, 218,
287-
0, 0, 0, 4, 2, 22, 7, 207, 206, 25, 164, 197, 231, 230, 231, 56, 102, 61, 250, 251,
288-
187, 172, 38, 46, 79, 247, 108, 44, 155, 48, 219, 238, 252, 53, 192, 6, 67, 2, 36, 125,
289-
157, 176, 223, 175, 234, 116, 94, 248, 201, 225, 97, 235, 50, 47, 115, 172, 63, 136,
290-
88, 216, 115, 11, 111, 217, 114, 84, 116, 124, 231, 107, 2, 158, 1, 242, 121, 152, 106,
291-
204, 131, 186, 35, 93, 70, 216, 10, 237, 224, 183, 89, 95, 65, 3, 83, 185, 58, 138,
292-
181, 64, 187, 103, 127, 68, 50, 2, 201, 19, 17, 138, 136, 149, 185, 226, 156, 137, 175,
293-
110, 32, 237, 0, 217, 90, 31, 100, 228, 149, 46, 219, 175, 168, 77, 4, 143, 38, 128,
294-
76, 97, 0, 0, 0, 2, 0, 0, 255, 8, 153, 192, 0, 2, 27, 0, 0, 0, 1, 0, 0, 255, 2, 68,
295-
226, 0, 6, 11, 0, 1, 2, 3, 0, 0, 0, 2, 0, 40, 0, 0, 0, 0, 0, 0, 3, 232, 0, 0, 0, 100,
296-
0, 0, 2, 224, 0, 0, 0, 0, 29, 129, 25, 192, 255, 8, 153, 192, 0, 2, 27, 0, 0, 36, 0, 0,
297-
0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 58, 85, 116, 216, 255, 2, 68, 226, 0, 6, 11, 0, 1, 24, 0,
298-
0, 3, 232, 0, 0, 0, 1, 10,
299-
];
300-
let update_result = update_network_graph(&network_graph, &example_input[..]);
301-
assert!(update_result.is_err());
302-
303-
let error_string = if let Err(GraphSyncError::ProcessingError(error)) = update_result {
304-
error
305-
} else {
306-
panic!("Unexpected update result: {:?}", update_result)
307-
};
308-
assert!(error_string.contains("unexpected data found"));
309-
}
310-
311263
#[test]
312264
fn incremental_only_update_fails_without_prior_announcements() {
313265
let incremental_update_input = vec![

0 commit comments

Comments
 (0)