Skip to content

Commit

Permalink
gossipd: only do (automatic) store compaction at startup.
Browse files Browse the repository at this point in the history
Rewriting the gossip_store is much more trivial when we don't have
any pointers into it, so add some simple offline compaction code
and disable the automatic compaction code.

Signed-off-by: Rusty Russell <[email protected]>
  • Loading branch information
rustyrussell authored and niftynei committed Jun 22, 2019
1 parent c15d9ed commit c303d7d
Show file tree
Hide file tree
Showing 2 changed files with 93 additions and 16 deletions.
98 changes: 83 additions & 15 deletions gossipd/gossip_store.c
Original file line number Diff line number Diff line change
Expand Up @@ -77,12 +77,95 @@ static bool append_msg(int fd, const u8 *msg, u32 timestamp, u64 *len)
return writev(fd, iov, ARRAY_SIZE(iov)) == sizeof(hdr) + msglen;
}

/* Read gossip store entries, copy non-deleted ones. This code is written
* as simply and robustly as possible! */
static void gossip_store_compact_offline(void)
{
size_t count = 0, deleted = 0;
int old_fd, new_fd;
struct gossip_hdr hdr;
u8 version;

old_fd = open(GOSSIP_STORE_FILENAME, O_RDONLY);
if (old_fd == -1)
return;
new_fd = open(GOSSIP_STORE_TEMP_FILENAME, O_RDWR|O_TRUNC|O_CREAT, 0600);
if (new_fd < 0) {
status_broken(
"Could not open file for gossip_store compaction");
goto close_old;
}

if (!read_all(old_fd, &version, sizeof(version))
|| version != GOSSIP_STORE_VERSION) {
status_broken("gossip_store_compact: bad version");
goto close_and_delete;
}

if (!write_all(new_fd, &version, sizeof(version))) {
status_broken("gossip_store_compact_offline: writing version to store: %s",
strerror(errno));
goto close_and_delete;
}

/* Read everything, write non-deleted ones to new_fd */
while (read_all(old_fd, &hdr, sizeof(hdr))) {
size_t msglen;
u8 *msg;

msglen = (be32_to_cpu(hdr.len) & ~GOSSIP_STORE_LEN_DELETED_BIT);
msg = tal_arr(NULL, u8, msglen);
if (!read_all(old_fd, msg, msglen)) {
status_broken("gossip_store_compact_offline: reading msg len %zu from store: %s",
msglen, strerror(errno));
tal_free(msg);
goto close_and_delete;
}

if (be32_to_cpu(hdr.len) & GOSSIP_STORE_LEN_DELETED_BIT) {
deleted++;
tal_free(msg);
continue;
}

if (!write_all(new_fd, &hdr, sizeof(hdr))
|| !write_all(new_fd, msg, msglen)) {
status_broken("gossip_store_compact_offline: writing msg len %zu to new store: %s",
msglen, strerror(errno));
tal_free(msg);
goto close_and_delete;
}
tal_free(msg);
count++;
}
if (close(new_fd) != 0) {
status_broken("gossip_store_compact_offline: closing new store: %s",
strerror(errno));
goto close_old;
}
close(old_fd);
if (rename(GOSSIP_STORE_TEMP_FILENAME, GOSSIP_STORE_FILENAME) != 0) {
status_broken("gossip_store_compact_offline: rename failed: %s",
strerror(errno));
}
status_debug("gossip_store_compact_offline: %zu deleted, %zu copied",
deleted, count);
return;

close_and_delete:
close(new_fd);
close_old:
close(old_fd);
unlink(GOSSIP_STORE_TEMP_FILENAME);
}

struct gossip_store *gossip_store_new(struct routing_state *rstate,
struct list_head *peers)
{
struct gossip_store *gs = tal(rstate, struct gossip_store);
gs->count = gs->deleted = 0;
gs->writable = true;
gossip_store_compact_offline();
gs->fd = open(GOSSIP_STORE_FILENAME, O_RDWR|O_APPEND|O_CREAT, 0600);
gs->rstate = rstate;
gs->disable_compaction = false;
Expand Down Expand Up @@ -352,19 +435,6 @@ bool gossip_store_compact(struct gossip_store *gs)
return false;
}

static void gossip_store_maybe_compact(struct gossip_store *gs)
{
/* Don't compact while loading! */
if (!gs->writable)
return;
if (gs->count < 1000)
return;
if (gs->deleted < gs->count / 4)
return;

gossip_store_compact(gs);
}

u64 gossip_store_add(struct gossip_store *gs, const u8 *gossip_msg,
u32 timestamp,
const u8 *addendum)
Expand Down Expand Up @@ -460,8 +530,6 @@ void gossip_store_delete(struct gossip_store *gs,
if (type == WIRE_CHANNEL_ANNOUNCEMENT)
delete_by_index(gs, next_index,
WIRE_GOSSIP_STORE_CHANNEL_AMOUNT);

gossip_store_maybe_compact(gs);
}

const u8 *gossip_store_get(const tal_t *ctx,
Expand Down
11 changes: 10 additions & 1 deletion tests/test_gossip.py
Original file line number Diff line number Diff line change
Expand Up @@ -922,7 +922,7 @@ def test_gossip_store_load_announce_before_update(node_factory):

l1.start()
# May preceed the Started msg waited for in 'start'.
wait_for(lambda: l1.daemon.is_in_log(r'gossip_store: Read 1/1/1/0 cannounce/cupdate/nannounce/cdelete from store \(1 deleted\) in 912 bytes'))
wait_for(lambda: l1.daemon.is_in_log(r'gossip_store: Read 1/1/1/0 cannounce/cupdate/nannounce/cdelete from store \(0 deleted\) in 770 bytes'))
assert not l1.daemon.is_in_log('gossip_store.*truncating')

# Extra sanity check if we can.
Expand Down Expand Up @@ -1283,3 +1283,12 @@ def test_gossip_store_load_no_channel_update(node_factory):

with open(os.path.join(l1.daemon.lightning_dir, 'gossip_store'), "rb") as f:
assert bytearray(f.read()) == bytearray.fromhex("07")


def test_gossip_store_compact_on_load(node_factory, bitcoind):
l2 = setup_gossip_store_test(node_factory, bitcoind)

l2.restart()

wait_for(lambda: l2.daemon.is_in_log('gossip_store_compact_offline: 9 deleted, 9 copied'))
wait_for(lambda: l2.daemon.is_in_log(r'gossip_store: Read 1/4/2/0 cannounce/cupdate/nannounce/cdelete from store \(0 deleted\) in 1446 bytes'))

0 comments on commit c303d7d

Please sign in to comment.