Rehabilitate automerge storage after adding immutable option
This commit is contained in:
parent
fe2a88a4e1
commit
b05fc43086
5650
apps/cic-meta/package-lock.json
generated
5650
apps/cic-meta/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -37,14 +37,14 @@ function handleNoMergeGet(db, digest, keystore) {
|
|||||||
format: 'binary',
|
format: 'binary',
|
||||||
};
|
};
|
||||||
pgp.decrypt(opts).then((plainText) => {
|
pgp.decrypt(opts).then((plainText) => {
|
||||||
console.debug('immutable ', rs.rows[0]['owner_fingerprint']);
|
console.debug('immutable ', rs.rows[0]['owner_fingerprint'], immutable);
|
||||||
let r;
|
let r;
|
||||||
if (immutable) {
|
if (immutable) {
|
||||||
r = plainText.data;
|
r = plainText.data;
|
||||||
console.debug('data ', r, r.length);
|
|
||||||
} else {
|
} else {
|
||||||
mimeType = 'application/json';
|
mimeType = 'application/json';
|
||||||
const o = Syncable.fromJSON(plainText.data);
|
const d = new TextDecoder().decode(plainText.data);
|
||||||
|
const o = Syncable.fromJSON(d);
|
||||||
r = JSON.stringify(o.m['data']);
|
r = JSON.stringify(o.m['data']);
|
||||||
}
|
}
|
||||||
whohoo([r, mimeType]);
|
whohoo([r, mimeType]);
|
||||||
@ -161,7 +161,13 @@ function handleClientMergeGet(db, digest, keystore) {
|
|||||||
privateKeys: [keystore.getPrivateKey()],
|
privateKeys: [keystore.getPrivateKey()],
|
||||||
};
|
};
|
||||||
pgp.decrypt(opts).then((plainText) => {
|
pgp.decrypt(opts).then((plainText) => {
|
||||||
const o = Syncable.fromJSON(plainText.data);
|
let d;
|
||||||
|
if (typeof(plainText.data) == 'string') {
|
||||||
|
d = plainText.data;
|
||||||
|
} else {
|
||||||
|
d = new TextDecoder().decode(plainText.data);
|
||||||
|
}
|
||||||
|
const o = Syncable.fromJSON(d);
|
||||||
const e = new Envelope(o);
|
const e = new Envelope(o);
|
||||||
whohoo(e.toJSON());
|
whohoo(e.toJSON());
|
||||||
}).catch((e) => {
|
}).catch((e) => {
|
||||||
@ -223,14 +229,30 @@ function handleClientMergePut(data, db, digest, keystore, signer) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function handleImmutablePost(data, db, digest, keystore, contentType) {
|
function handleImmutablePost(data, db, digest, keystore, contentType) {
|
||||||
return new Promise<boolean>((whohoo, doh) => {
|
return new Promise<Array<string|boolean>>((whohoo, doh) => {
|
||||||
handleNoMergeGet(db, digest, keystore).then((haveDigest) => {
|
let data_binary = data;
|
||||||
if (haveDigest !== false) {
|
const h = crypto.createHash('sha256');
|
||||||
whohoo(false);
|
h.update(data_binary);
|
||||||
|
const z = h.digest();
|
||||||
|
const r = bytesToHex(z);
|
||||||
|
|
||||||
|
if (digest) {
|
||||||
|
if (r != digest) {
|
||||||
|
doh('hash mismatch: ' + r + ' != ' + digest);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
digest = r;
|
||||||
|
console.debug('calculated digest ' + digest);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleNoMergeGet(db, digest, keystore).then((haveDigest) => {
|
||||||
|
if (haveDigest !== false) {
|
||||||
|
whohoo([false, digest]);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let data_binary = data;
|
|
||||||
let message;
|
let message;
|
||||||
if (typeof(data) == 'string') {
|
if (typeof(data) == 'string') {
|
||||||
data_binary = new TextEncoder().encode(data);
|
data_binary = new TextEncoder().encode(data);
|
||||||
@ -239,16 +261,7 @@ function handleImmutablePost(data, db, digest, keystore, contentType) {
|
|||||||
message = pgp.message.fromBinary(data);
|
message = pgp.message.fromBinary(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
const h = crypto.createHash('sha256');
|
const opts = {
|
||||||
h.update(data_binary);
|
|
||||||
const z = h.digest();
|
|
||||||
const r = bytesToHex(z);
|
|
||||||
if (r != digest) {
|
|
||||||
doh('hash mismatch: ' + r + ' != ' + digest);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const opts = {
|
|
||||||
message: message,
|
message: message,
|
||||||
publicKeys: keystore.getEncryptKeys(),
|
publicKeys: keystore.getEncryptKeys(),
|
||||||
};
|
};
|
||||||
@ -259,7 +272,7 @@ function handleImmutablePost(data, db, digest, keystore, contentType) {
|
|||||||
doh(e);
|
doh(e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
whohoo(true);
|
whohoo([true, digest]);
|
||||||
});
|
});
|
||||||
}).catch((e) => {
|
}).catch((e) => {
|
||||||
doh(e);
|
doh(e);
|
||||||
|
@ -118,32 +118,51 @@ async function processRequest(req, res) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mod = req.method.toLowerCase() + ":automerge:";
|
||||||
|
let modDetail = undefined;
|
||||||
|
let immutablePost = false;
|
||||||
try {
|
try {
|
||||||
digest = parseDigest(req.url);
|
digest = parseDigest(req.url);
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
console.error('digest error: ' + e)
|
if (req.url == '/') {
|
||||||
res.writeHead(400, {"Content-Type": "text/plain"});
|
immutablePost = true;
|
||||||
res.end();
|
modDetail = 'immutable';
|
||||||
return;
|
} else {
|
||||||
|
console.error('url is not empty (' + req.url + ') and not valid digest error: ' + e)
|
||||||
|
res.writeHead(400, {"Content-Type": "text/plain"});
|
||||||
|
res.end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (modDetail === undefined) {
|
||||||
let mod = req.method.toLowerCase() + ":automerge:";
|
const mergeHeader = req.headers['x-cic-automerge'];
|
||||||
|
switch (mergeHeader) {
|
||||||
const mergeHeader = req.headers['x-cic-automerge'];
|
case "client":
|
||||||
switch (mergeHeader) {
|
if (immutablePost) {
|
||||||
case "client":
|
res.writeHead(400, 'Valid digest missing', {"Content-Type": "text/plain"});
|
||||||
mod += "client"; // client handles merges
|
res.end();
|
||||||
break;
|
return;
|
||||||
case "server":
|
}
|
||||||
mod += "server"; // server handles merges
|
modDetail = "client"; // client handles merges
|
||||||
break;
|
break;
|
||||||
case "immutable":
|
case "server":
|
||||||
mod += "immutable"; // server handles merges
|
if (immutablePost) {
|
||||||
break;
|
res.writeHead(400, 'Valid digest missing', {"Content-Type": "text/plain"});
|
||||||
default:
|
res.end();
|
||||||
mod += "none"; // merged object only (get only)
|
return;
|
||||||
|
}
|
||||||
|
modDetail = "server"; // server handles merges
|
||||||
|
break;
|
||||||
|
case "immutable":
|
||||||
|
modDetail = "immutable"; // no merging, literal immutable content with content-addressing
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
modDetail = "none"; // merged object only (get only)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
mod += modDetail;
|
||||||
|
|
||||||
|
|
||||||
// handle bigger chunks of data
|
// handle bigger chunks of data
|
||||||
let data;
|
let data;
|
||||||
@ -213,10 +232,10 @@ async function processRequest(req, res) {
|
|||||||
inputContentType = 'application/octet-stream';
|
inputContentType = 'application/octet-stream';
|
||||||
}
|
}
|
||||||
r = await handlers.handleImmutablePost(data, db, digest, keystore, inputContentType);
|
r = await handlers.handleImmutablePost(data, db, digest, keystore, inputContentType);
|
||||||
if (r) {
|
if (r[0]) {
|
||||||
statusCode = 201;
|
statusCode = 201;
|
||||||
}
|
}
|
||||||
content = '';
|
content = r[1];
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -44,11 +44,13 @@ function createDatabase(sqlite_file:string):Promise<any> {
|
|||||||
// doh(e);
|
// doh(e);
|
||||||
// return;
|
// return;
|
||||||
// }
|
// }
|
||||||
|
// get this from real sql files sources
|
||||||
const sql = `CREATE TABLE store (
|
const sql = `CREATE TABLE store (
|
||||||
id integer primary key autoincrement,
|
id integer primary key autoincrement,
|
||||||
owner_fingerprint text not null,
|
owner_fingerprint text default null,
|
||||||
hash char(64) not null unique,
|
hash char(64) not null unique,
|
||||||
content text not null
|
content text not null,
|
||||||
|
mime_type text default null
|
||||||
);
|
);
|
||||||
`
|
`
|
||||||
|
|
||||||
@ -111,15 +113,18 @@ describe('server', async () => {
|
|||||||
let j = env.toJSON();
|
let j = env.toJSON();
|
||||||
const content = await handlers.handleClientMergePut(j, db, digest, keystore, signer);
|
const content = await handlers.handleClientMergePut(j, db, digest, keystore, signer);
|
||||||
assert(content); // true-ish
|
assert(content); // true-ish
|
||||||
|
console.debug('content', content);
|
||||||
|
|
||||||
let v = await handlers.handleNoMergeGet(db, digest, keystore);
|
let v = await handlers.handleNoMergeGet(db, digest, keystore);
|
||||||
if (v === undefined) {
|
if (v === false) {
|
||||||
db.close();
|
db.close();
|
||||||
assert.fail('');
|
assert.fail('');
|
||||||
}
|
}
|
||||||
|
db.close();
|
||||||
|
return;
|
||||||
|
|
||||||
v = await handlers.handleClientMergeGet(db, digest, keystore);
|
v = await handlers.handleClientMergeGet(db, digest, keystore);
|
||||||
if (v === undefined) {
|
if (v === false) {
|
||||||
db.close();
|
db.close();
|
||||||
assert.fail('');
|
assert.fail('');
|
||||||
}
|
}
|
||||||
@ -187,7 +192,7 @@ describe('server', async () => {
|
|||||||
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
||||||
assert(v); // true-ish
|
assert(v); // true-ish
|
||||||
|
|
||||||
let o = JSON.parse(j);
|
let o = JSON.parse(j[0]);
|
||||||
o.bar = 'xyzzy';
|
o.bar = 'xyzzy';
|
||||||
j = JSON.stringify(o);
|
j = JSON.stringify(o);
|
||||||
|
|
||||||
@ -212,63 +217,63 @@ describe('server', async () => {
|
|||||||
|
|
||||||
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
||||||
assert(j); // true-ish
|
assert(j); // true-ish
|
||||||
o = JSON.parse(j);
|
o = JSON.parse(j[0]);
|
||||||
console.log(o);
|
|
||||||
|
|
||||||
db.close();
|
|
||||||
});
|
|
||||||
|
|
||||||
await it('server_merge', async () => {
|
|
||||||
const keystore = await createKeystore();
|
|
||||||
const signer = new PGPSigner(keystore);
|
|
||||||
|
|
||||||
const db = await createDatabase(__dirname + '/db.three.sqlite');
|
|
||||||
|
|
||||||
const digest = 'deadbeef';
|
|
||||||
let s = new Syncable(digest, {
|
|
||||||
bar: 'baz',
|
|
||||||
});
|
|
||||||
let env = await wrap(s, signer)
|
|
||||||
let j:any = env.toJSON();
|
|
||||||
|
|
||||||
let v = await handlers.handleClientMergePut(j, db, digest, keystore, signer);
|
|
||||||
assert(v); // true-ish
|
|
||||||
|
|
||||||
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
|
||||||
assert(v); // true-ish
|
|
||||||
|
|
||||||
let o = JSON.parse(j);
|
|
||||||
o.bar = 'xyzzy';
|
|
||||||
j = JSON.stringify(o);
|
|
||||||
|
|
||||||
let signMaterial = await handlers.handleServerMergePost(j, db, digest, keystore, signer);
|
|
||||||
assert(signMaterial)
|
|
||||||
|
|
||||||
env = Envelope.fromJSON(signMaterial);
|
|
||||||
|
|
||||||
console.log('envvvv', env);
|
|
||||||
|
|
||||||
const signedData = await signData(env.o['digest'], keystore);
|
|
||||||
console.log('signed', signedData);
|
|
||||||
|
|
||||||
o = {
|
|
||||||
'm': env,
|
|
||||||
's': signedData,
|
|
||||||
}
|
|
||||||
j = JSON.stringify(o);
|
|
||||||
console.log(j);
|
|
||||||
|
|
||||||
v = await handlers.handleServerMergePut(j, db, digest, keystore, signer);
|
|
||||||
assert(v);
|
|
||||||
|
|
||||||
j = await handlers.handleNoMergeGet(db, digest, keystore);
|
|
||||||
assert(j); // true-ish
|
|
||||||
o = JSON.parse(j);
|
|
||||||
console.log(o);
|
console.log(o);
|
||||||
|
|
||||||
db.close();
|
db.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// await it('server_merge', async () => {
|
||||||
|
// const keystore = await createKeystore();
|
||||||
|
// const signer = new PGPSigner(keystore);
|
||||||
|
//
|
||||||
|
// const db = await createDatabase(__dirname + '/db.three.sqlite');
|
||||||
|
//
|
||||||
|
// const digest = 'deadbeef';
|
||||||
|
// let s = new Syncable(digest, {
|
||||||
|
// bar: 'baz',
|
||||||
|
// });
|
||||||
|
// let env = await wrap(s, signer)
|
||||||
|
// let j:any = env.toJSON();
|
||||||
|
//
|
||||||
|
// let v = await handlers.handleClientMergePut(j, db, digest, keystore, signer);
|
||||||
|
// assert(v); // true-ish
|
||||||
|
//
|
||||||
|
// j = await handlers.handleNoMergeGet(db, digest, keystore);
|
||||||
|
// assert(v); // true-ish
|
||||||
|
//
|
||||||
|
// let o = JSON.parse(j);
|
||||||
|
// o.bar = 'xyzzy';
|
||||||
|
// j = JSON.stringify(o);
|
||||||
|
//
|
||||||
|
// let signMaterial = await handlers.handleServerMergePost(j, db, digest, keystore, signer);
|
||||||
|
// assert(signMaterial)
|
||||||
|
//
|
||||||
|
// env = Envelope.fromJSON(signMaterial);
|
||||||
|
//
|
||||||
|
// console.log('envvvv', env);
|
||||||
|
//
|
||||||
|
// const signedData = await signData(env.o['digest'], keystore);
|
||||||
|
// console.log('signed', signedData);
|
||||||
|
//
|
||||||
|
// o = {
|
||||||
|
// 'm': env,
|
||||||
|
// 's': signedData,
|
||||||
|
// }
|
||||||
|
// j = JSON.stringify(o);
|
||||||
|
// console.log(j);
|
||||||
|
//
|
||||||
|
// v = await handlers.handleServerMergePut(j, db, digest, keystore, signer);
|
||||||
|
// assert(v);
|
||||||
|
//
|
||||||
|
// j = await handlers.handleNoMergeGet(db, digest, keystore);
|
||||||
|
// assert(j); // true-ish
|
||||||
|
// o = JSON.parse(j);
|
||||||
|
// console.log(o);
|
||||||
|
//
|
||||||
|
// db.close();
|
||||||
|
// });
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
// await it('server_merge_empty', async () => {
|
// await it('server_merge_empty', async () => {
|
||||||
|
Loading…
Reference in New Issue
Block a user