fcoin/migrate/walletdb5to6.js
2018-08-10 16:23:46 -07:00

265 lines
5.9 KiB
JavaScript

'use strict';
const assert = require('assert');
const bdb = require('bdb');
const bio = require('bufio');
assert(process.argv.length > 2, 'Please pass in a database path.');
let batch;
const db = bdb.create({
location: process.argv[2],
compression: true,
cacheSize: 32 << 20,
createIfMissing: false
});
async function updateVersion() {
const bak = `${process.env.HOME}/wallet-bak-${Date.now()}`;
console.log('Checking version.');
const raw = await db.get('V');
assert(raw, 'No version.');
const version = raw.readUInt32LE(0, true);
if (version !== 5)
throw Error(`DB is version ${version}.`);
console.log('Backing up DB to: %s.', bak);
await db.backup(bak);
const data = Buffer.allocUnsafe(4);
data.writeUInt32LE(6, 0, true);
batch.put('V', data);
}
async function wipeTXDB() {
let total = 0;
const keys = await db.keys();
for (let i = 0; i < keys.length; i++) {
const key = keys[i];
switch (key[0]) {
case 0x62: // b
case 0x63: // c
case 0x65: // e
case 0x74: // t
case 0x6f: // o
case 0x68: // h
batch.del(key);
total += 1;
break;
}
}
batch.del(Buffer.from([0x52])); // R
console.log('Wiped %d txdb records.', total);
}
async function patchAccounts() {
const items = await db.range({
gt: Buffer.from([0x61]), // a
lt: Buffer.from([0x62])
});
for (let i = 0; i < items.length; i++) {
const item = items[i];
const wid = item.key.readUInt32BE(1, true);
const index = item.key.readUInt32BE(5, true);
const account = accountFromRaw(item.value);
console.log('a[%d][%d] -> lookahead=%d', wid, index, account.lookahead);
batch.put(item.key, accountToRaw(account));
console.log('n[%d][%d] -> %s', wid, index, account.name);
batch.put(n(wid, index), Buffer.from(account.name, 'ascii'));
}
}
async function indexPaths() {
const items = await db.range({
gt: Buffer.from([0x50]), // P
lt: Buffer.from([0x51])
});
for (let i = 0; i < items.length; i++) {
const item = items[i];
const wid = item.key.readUInt32BE(1, true);
const hash = item.key.slice(5);
const index = item.value.readUInt32LE(0, true);
console.log('r[%d][%d][%s] -> NUL', wid, index, hash);
batch.put(r(wid, index, hash), Buffer.from([0]));
}
}
async function patchPathMaps() {
const items = await db.range({
gt: Buffer.from([0x70]), // p
lt: Buffer.from([0x71])
});
for (let i = 0; i < items.length; i++) {
const item = items[i];
const hash = item.key.slice(1);
const wids = parseWallets(item.value);
console.log('p[%s] -> u32(%d)', hash, wids.length);
batch.put(item.key, serializeWallets(wids));
}
}
function parseWallets(data) {
const p = bio.read(data);
const wids = [];
while (p.left())
wids.push(p.readU32());
return wids;
}
function serializeWallets(wids) {
const p = bio.write();
p.writeU32(wids.length);
for (let i = 0; i < wids.length; i++) {
const wid = wids[i];
p.writeU32(wid);
}
return p.render();
}
function accountToRaw(account) {
const p = bio.write();
p.writeVarString(account.name, 'ascii');
p.writeU8(account.initialized ? 1 : 0);
p.writeU8(account.witness ? 1 : 0);
p.writeU8(account.type);
p.writeU8(account.m);
p.writeU8(account.n);
p.writeU32(account.accountIndex);
p.writeU32(account.receiveDepth);
p.writeU32(account.changeDepth);
p.writeU32(account.nestedDepth);
p.writeU8(account.lookahead);
p.writeBytes(account.accountKey);
p.writeU8(account.keys.length);
for (let i = 0; i < account.keys.length; i++) {
const key = account.keys[i];
p.writeBytes(key);
}
return p.render();
};
function accountFromRaw(data) {
const account = {};
const p = bio.read(data);
account.name = p.readVarString('ascii');
account.initialized = p.readU8() === 1;
account.witness = p.readU8() === 1;
account.type = p.readU8();
account.m = p.readU8();
account.n = p.readU8();
account.accountIndex = p.readU32();
account.receiveDepth = p.readU32();
account.changeDepth = p.readU32();
account.nestedDepth = p.readU32();
account.lookahead = 10;
account.accountKey = p.readBytes(82);
account.keys = [];
const count = p.readU8();
for (let i = 0; i < count; i++) {
const key = p.readBytes(82);
account.keys.push(key);
}
return account;
}
function n(wid, index) {
const key = Buffer.allocUnsafe(9);
key[0] = 0x6e;
key.writeUInt32BE(wid, 1, true);
key.writeUInt32BE(index, 5, true);
return key;
}
function r(wid, index, hash) {
const key = Buffer.allocUnsafe(1 + 4 + 4 + (hash.length / 2));
key[0] = 0x72;
key.writeUInt32BE(wid, 1, true);
key.writeUInt32BE(index, 5, true);
hash.copy(key, 9);
return key;
}
async function updateLookahead() {
const WalletDB = require('../lib/wallet/walletdb');
const db = new WalletDB({
network: process.argv[3],
db: 'leveldb',
location: process.argv[2],
witness: false,
useCheckpoints: false,
maxFiles: 64,
resolution: false,
verify: false
});
await db.open();
for (let i = 1; i < db.depth; i++) {
const wallet = await db.get(i);
assert(wallet);
console.log('Updating wallet lookahead: %s', wallet.id);
for (let j = 0; j < wallet.accountDepth; j++)
await wallet.setLookahead(j, 20);
}
await db.close();
}
updateLookahead;
async function unstate() {
await db.open();
batch = db.batch();
await wipeTXDB();
await batch.write();
await db.close();
}
(async () => {
await db.open();
batch = db.batch();
console.log('Opened %s.', process.argv[2]);
await updateVersion();
await wipeTXDB();
await patchAccounts();
await indexPaths();
await patchPathMaps();
await batch.write();
await db.close();
// Do not use:
// await updateLookahead();
await unstate();
})().then(() => {
console.log('Migration complete.');
console.log('Rescan is required...');
console.log('Start bcoin with `--start-height=[wallet-creation-height]`.');
process.exit(0);
});