Skip to content

Commit

Permalink
fix(swingset): fix transcript extract/replay tools
Browse files Browse the repository at this point in the history
These had bitrotted:
* transcript now includes whole syscall result, even failures
* handle kerneldb bundleIDs
* new LMDB in swingstore
* gcEveryCrank=false else all GC is disabled (refs #5600)
* fake kernelKeeper/slogger needed new methods

The new code also attempts to track `deliveryNum`, to align it with
the original slog, but I'm not positive it is accurate.

closes #5602
  • Loading branch information
warner authored and mergify[bot] committed Jun 29, 2022
1 parent ec95ec4 commit 56e3d7c
Show file tree
Hide file tree
Showing 3 changed files with 85 additions and 59 deletions.
115 changes: 73 additions & 42 deletions packages/SwingSet/misc-tools/extract-transcript-from-kerneldb.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
// XXX this is wrong; it needs to use the swingstore instead of opening the LMDB
// file directly, then use stream store reads to get the transcript entries.
import lmdb from 'lmdb';
import 'lmdb';
import '@endo/init';
import process from 'process';
import fs from 'fs';
import { isSwingStore, openSwingStore } from '@agoric/swing-store';

const argv = process.argv.splice(2);
const dirPath = argv[0];
Expand All @@ -16,19 +16,12 @@ if (!dirPath) {
process.exit(0);
}

const lmdbEnv = new lmdb.Env();
lmdbEnv.open({
path: dirPath,
mapSize: 2 * 1024 * 1024 * 1024, // XXX need to tune this
});

const dbi = lmdbEnv.openDbi({
name: 'swingset-kernel-state',
create: false,
});
const txn = lmdbEnv.beginTxn();
if (!isSwingStore(dirPath)) {
throw Error(`${dirPath} does not appear to be a swingstore (no ./data.mdb)`);
}
const { kvStore, streamStore } = openSwingStore(dirPath);
function get(key) {
return txn.getString(dbi, key);
return kvStore.get(key);
}

const allVatNames = JSON.parse(get('vat.names'));
Expand All @@ -38,15 +31,27 @@ if (!vatName) {
console.log(`all vats:`);
for (const name of allVatNames) {
const vatID = get(`vat.name.${name}`);
const transcriptLength = Number(get(`${vatID}.t.nextID`));
console.log(`${vatID} : ${name} (${transcriptLength} deliveries)`);
const startPos = JSON.parse(get(`${vatID}.t.startPosition`)).itemCount;
const endPos = JSON.parse(get(`${vatID}.t.endPosition`)).itemCount;
const len = `${endPos - startPos}`;
const status = `(static)`;
console.log(
`${vatID.padEnd(3)} : ${name.padEnd(15)} ${status.padEnd(
20,
)} ${len.padStart(10)} deliveries`,
);
}
for (const vatID of allDynamicVatIDs) {
const transcriptLength = Number(get(`${vatID}.t.nextID`));
const startPos = JSON.parse(get(`${vatID}.t.startPosition`)).itemCount;
const endPos = JSON.parse(get(`${vatID}.t.endPosition`)).itemCount;
const len = `${endPos - startPos}`;
const options = JSON.parse(get(`${vatID}.options`));
const { name, managerType } = options;
const status = `(dynamic, ${managerType})`;
console.log(
`${vatID} : (dynamic)`,
get(`${vatID}.options`),
` (${transcriptLength} deliveries)`,
`${vatID.padEnd(3)} : ${name.padEnd(15)} ${status.padEnd(
20,
)} ${len.padStart(10)} deliveries`,
);
}
} else {
Expand All @@ -64,7 +69,18 @@ if (!vatName) {

const dynamic = allDynamicVatIDs.includes(vatID);
const source = JSON.parse(get(`${vatID}.source`));
const vatSourceBundle = source.bundle || get(`bundle.${source.bundleName}`);
let vatSourceBundle;
if (source.bundleID) {
console.log(`source bundleID: ${source.bundleID}`);
vatSourceBundle = JSON.parse(get(`bundle.${source.bundleID}`));
} else {
// this doesn't actually happen, now that Zoe launches ZCF by bundlecap
vatSourceBundle = JSON.parse(source.bundle);
const { moduleFormat, endoZipBase64 } = vatSourceBundle;
console.log(
`source is bundle, format=${moduleFormat}, endoZipBase64.length=${endoZipBase64.length}`,
);
}
const options = JSON.parse(get(`${vatID}.options`));
console.log(`options:`, options);
const { vatParameters } = options;
Expand All @@ -83,28 +99,43 @@ if (!vatName) {
fs.writeSync(fd, JSON.stringify(first));
fs.writeSync(fd, '\n');

const transcriptLength = Number(get(`${vatID}.t.nextID`));
// The streamStore holds concatenated transcripts from all upgraded
// versions. For each old version, it holds every delivery from
// `startVat` through `stopVat`. For the current version, it holds
// every delivery from `startVat` up through the last delivery
// attempted, which might include one or more deliveries that have
// been rewound (either because a single crank was abandoned, or the
// host application failed to commit the kvStore).
//
// The kvStore `${vatID}.t.startPosition` tells us the index of the
// first entry of the most recent version (the most recent
// `startVat`), while `${vatID}.t.endPosition` tells us the last
// committed entry.
//
// We ignore the heap snapshot ID, because we're deliberately
// replaying everything from `startVat`, and therefore we also
// ignore `snapshot.startPos` (the index of the first delivery
// *after* the snapshot was taken, where normal operation would
// start a replay).

const startPos = JSON.parse(get(`${vatID}.t.startPosition`));
const endPos = JSON.parse(get(`${vatID}.t.endPosition`));
const transcriptLength = endPos.itemCount - startPos.itemCount;
console.log(`${transcriptLength} transcript entries`);
for (let i = 0; i < transcriptLength; i += 1) {
const t = { transcriptNum, ...JSON.parse(get(`${vatID}.t.${i}`)) };
transcriptNum += 1;
// vatstoreGet can lack .response when key was missing
// vatstoreSet has .response: null
// console.log(`t.${i} : ${t}`);

let deliveryNum = 0;
const transcriptStream = `transcript-${vatID}`;
const stream = streamStore.readStream(transcriptStream, startPos, endPos);
for (const entry of stream) {
// entry is JSON.stringify({ d, syscalls }), syscall is { d, response }
const t = { transcriptNum, ...JSON.parse(entry) };
// console.log(`t.${deliveryNum} : ${t}`);
fs.writeSync(fd, `${JSON.stringify(t)}\n`);
// eslint-disable-next-line no-unused-vars
deliveryNum += 1;
transcriptNum += 1;
}
fs.closeSync(fd);

/*
let c = new lmdb.Cursor(txn, dbi);
let key = c.goToFirst();
while(0) {
//console.log(key);
console.log(key, txn.getString(dbi, key));
key = c.goToNext();
if (!key) {
break;
}
}
*/
fs.closeSync(fd);
console.log(`wrote ${transcriptNum} entries for vat ${vatID} into ${fn}`);
}
14 changes: 1 addition & 13 deletions packages/SwingSet/misc-tools/extract-transcript-from-slogfile.js
Original file line number Diff line number Diff line change
Expand Up @@ -66,19 +66,7 @@ async function run() {
}
case 'syscall-result': {
console.log(` -- syscall-result`);
// The vatstoreGet result is null when key was missing. To match the
// transcript extracted from a DB (in which a missing key gets
// 'undefined', causing the .response to be omitted entirely),
// special-case a vatstoreGet with a null result.
//
// vatstoreSet result is always null
const [status, result] = e.vsr; // e.g. ["ok", string]
if (status !== 'ok') {
throw Error(`encountered non-ok syscall result`);
}
if (!(syscall.d[0] === 'vatstoreGet' && result === null)) {
syscall.response = result;
}
syscall.response = e.vsr;
syscalls.push(syscall);
break;
}
Expand Down
15 changes: 11 additions & 4 deletions packages/SwingSet/misc-tools/replay-transcript.js
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ function compareSyscalls(vatID, originalSyscall, newSyscall) {
// 3.8s v8-false, 27.5s v8-gc
// 10.8s xs-no-gc, 15s xs-gc
const worker = 'xs-worker';
const gcEveryCrank = false;
const gcEveryCrank = true; // false means GC is hard-disabled

async function replay(transcriptFile) {
let vatID; // we learn this from the first line of the transcript
Expand All @@ -58,8 +58,13 @@ async function replay(transcriptFile) {
addToTranscript: () => undefined,
getLastSnapshot: () => undefined,
}),
getRelaxDurabilityRules: () => false,
};
const kernelSlog = {
write() {},
delivery: () => () => undefined,
syscall: () => () => undefined,
};
const kernelSlog = { write() {} };
const testLog = undefined;
const meterControl = makeDummyMeterControl();
const gcTools = harden({
Expand Down Expand Up @@ -127,6 +132,7 @@ async function replay(transcriptFile) {
transcriptF = transcriptF.pipe(zlib.createGunzip());
}
const lines = readline.createInterface({ input: transcriptF });
let deliveryNum = 0; // TODO is this aligned?
let lineNumber = 1;
for await (const line of lines) {
if (lineNumber % 1000 === 0) {
Expand Down Expand Up @@ -160,14 +166,15 @@ async function replay(transcriptFile) {
// syscalls = [{ d, response }, ..]
// console.log(`replaying:`);
console.log(
`delivery ${lineNumber}:`,
`delivery ${deliveryNum} (L ${lineNumber}):`,
JSON.stringify(delivery).slice(0, 200),
);
// for (const s of syscalls) {
// s.response = 'nope';
// console.log(` syscall:`, s.d, s.response);
// }
await manager.replayOneDelivery(delivery, syscalls);
await manager.replayOneDelivery(delivery, syscalls, deliveryNum);
deliveryNum += 1;
// console.log(`dr`, dr);
}
}
Expand Down

0 comments on commit 56e3d7c

Please sign in to comment.