fcoin/migrate/walletdb2to3.js
2016-10-02 23:24:13 -07:00

263 lines
5.6 KiB
JavaScript

var bcoin = require('../');
var walletdb = require('../lib/wallet/walletdb');
var Path = require('../lib/wallet/path');
var Account = require('../lib/wallet/account');
var layout = walletdb.layout;
var co = bcoin.co;
var assert = require('assert');
var file = process.argv[2];
var BufferReader = require('../lib/utils/reader');
var BufferWriter = require('../lib/utils/writer');
var db, batch;
assert(typeof file === 'string', 'Please pass in a database path.');
file = file.replace(/\.ldb$/, '');
db = bcoin.ldb({
location: file,
db: 'leveldb',
compression: true,
cacheSize: 16 << 20,
writeBufferSize: 8 << 20,
createIfMissing: false,
bufferKeys: true
});
var updateVersion = co(function* updateVersion() {
var data, ver;
console.log('Checking version.');
data = yield db.get('V');
if (!data)
return;
ver = data.readUInt32LE(0, true);
if (ver !== 2)
throw Error('DB is version ' + ver + '.');
console.log('Backing up DB.');
yield db.backup(process.env.HOME + '/walletdb-bak-' + Date.now() + '.ldb');
ver = new Buffer(4);
ver.writeUInt32LE(3, 0, true);
batch.put('R', ver);
});
var updatePathMap = co(function* updatePathMap() {
var i, iter, item, oldPaths, oldPath, hash, path, keys, key, ring;
var total = 0;
iter = db.iterator({
gte: layout.p(constants.NULL_HASH),
lte: layout.p(constants.HIGH_HASH),
values: true
});
console.log('Migrating path map.');
for (;;) {
item = yield iter.next();
if (!item)
break;
total++;
hash = layout.pp(item.key);
oldPaths = parsePaths(data, hash);
keys = Object.keys(oldPaths);
for (i = 0; i < keys.length; i++) {
keys[i] = +keys[i];
key = keys[i];
oldPath = oldPaths[key];
path = new Path(oldPath);
if (path.data) {
if (path.encrypted) {
console.log(
'Cannot migrate encrypted import: %s (%s)',
path.data.toString('hex'),
path.toAddress().toBase58());
continue;
}
ring = keyFromRaw(path.data);
path.data = new bcoin.keyring(ring).toRaw();
}
batch.put(layout.P(key, hash), path.toRaw());
}
batch.put(layout.p(hash), serializeWallets(keys));
}
console.log('Migrated %d paths.', total);
});
var updateAccounts = co(function* updateAccounts() {
var total = 0;
var i, iter, item, account;
iter = db.iterator({
gte: layout.a(0, 0),
lte: layout.a(0xffffffff, 0xffffffff),
values: true
});
console.log('Migrating accounts.');
for (;;) {
item = yield iter.next();
if (!item)
break;
total++;
account = accountFromRaw(item.value, item.key);
account = new Account({ network: account.network, options: {} }, account);
batch.put(layout.a(account.wid, account.accountIndex), account.toRaw());
}
console.log('Migrated %d accounts.', total);
});
function pathFromRaw(data) {
var path = {};
var p = new BufferReader(data);
path.wid = p.readU32();
path.name = p.readVarString('utf8');
path.account = p.readU32();
switch (p.readU8()) {
case 0:
path.keyType = 0;
path.branch = p.readU32();
path.index = p.readU32();
if (p.readU8() === 1)
assert(false, 'Cannot migrate custom redeem script.');
break;
case 1:
path.keyType = 1;
path.encrypted = p.readU8() === 1;
path.data = p.readVarBytes();
path.branch = -1;
path.index = -1;
break;
default:
assert(false);
break;
}
path.version = p.read8();
path.type = p.readU8();
return path;
}
function parsePaths(data, hash) {
var p = new BufferReader(data);
var out = {};
var path;
while (p.left()) {
path = pathFromRaw(p);
out[path.wid] = path;
if (hash)
path.hash = hash;
}
return out;
}
function serializeWallets(wallets) {
var p = new BufferWriter();
var i, wid;
for (i = 0; i < wallets.length; i++) {
wid = wallets[i];
p.writeU32(wid);
}
return p.render();
}
function readAccountKey(key) {
return {
wid: key.readUInt32BE(1, true),
index: key.readUInt32E(5, true)
};
}
function accountFromRaw(data, dbkey) {
var account = {};
var p = new BufferReader(data);
var i, count, key;
dbkey = readAccountKey(dbkey);
account.wid = dbkey.wid;
account.network = bcoin.network.fromMagic(p.readU32());
account.name = p.readVarString('utf8');
account.initialized = p.readU8() === 1;
account.type = p.readU8();
account.m = p.readU8();
account.n = p.readU8();
account.witness = p.readU8() === 1;
account.accountIndex = p.readU32();
account.receiveDepth = p.readU32();
account.changeDepth = p.readU32();
account.accountKey = bcoin.hd.fromRaw(p.readBytes(82));
account.keys = [];
account.watchOnly = false;
account.nestedDepth = 0;
count = p.readU8();
for (i = 0; i < count; i++) {
key = bcoin.hd.fromRaw(p.readBytes(82));
account.keys.push(key);
}
return account;
}
function keyFromRaw(data, network) {
var ring = {};
var p = new BufferReader(data);
var key, script;
ring.network = bcoin.network.get(network);
ring.witness = p.readU8() === 1;
key = p.readVarBytes();
if (key.length === 32) {
ring.privateKey = key;
ring.publicKey = bcoin.ec.publicKeyCreate(key, true);
} else {
ring.publicKey = key;
}
script = p.readVarBytes();
if (script.length > 0)
ring.script = bcoin.script.fromRaw(script);
return ring;
}
co.spawn(function *() {
yield db.open();
batch = db.batch();
console.log('Opened %s.', file);
yield updateVersion();
yield updatePathMap();
yield updateAccounts();
yield batch.write();
}).then(function() {
console.log('Migration complete.');
process.exit(0);
});