mirror of
https://github.com/status-im/sourcecred.git
synced 2025-01-11 21:24:35 +00:00
mirror: update EAV primitives (#1342)
Summary: This commit modifies `_updateOwnData` to write to both the old type-specific primitives tables as well as the new EAV table. This establishes the invariant that a node with non-null `last_update` will always have primitive data (if its object type has primitive fields). Test Plan: Existing tests expanded. Commenting out each of the `updateEavPrimitive` calls (independently) causes a test to fail. Note that every test that queries an internal `primitives_*` table to inspect the database state has been expanded to make an equivalent query against the `primitives` table as well. wchargin-branch: mirror-eav-update
This commit is contained in:
parent
463f3a073a
commit
3cb22565e5
@ -196,7 +196,7 @@ export class Mirror {
|
|||||||
// it requires bumping the version, bump it: requiring some extra
|
// it requires bumping the version, bump it: requiring some extra
|
||||||
// one-time cache resets is okay; doing the wrong thing is not.
|
// one-time cache resets is okay; doing the wrong thing is not.
|
||||||
const blob = stringify({
|
const blob = stringify({
|
||||||
version: "MIRROR_v5",
|
version: "MIRROR_v6",
|
||||||
schema: this._schema,
|
schema: this._schema,
|
||||||
options: {
|
options: {
|
||||||
blacklistedIds: this._blacklistedIds,
|
blacklistedIds: this._blacklistedIds,
|
||||||
@ -1356,7 +1356,7 @@ export class Mirror {
|
|||||||
].join("_"): string): any);
|
].join("_"): string): any);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
const updatePrimitives: ({|
|
const updateTypeSpecificPrimitives: ({|
|
||||||
+id: Schema.ObjectId,
|
+id: Schema.ObjectId,
|
||||||
// These keys can be top-level primitive fields or the primitive
|
// These keys can be top-level primitive fields or the primitive
|
||||||
// children of a nested field. The values are the JSON encodings
|
// children of a nested field. The values are the JSON encodings
|
||||||
@ -1387,6 +1387,19 @@ export class Mirror {
|
|||||||
);
|
);
|
||||||
return _makeSingleUpdateFunction(stmt);
|
return _makeSingleUpdateFunction(stmt);
|
||||||
})();
|
})();
|
||||||
|
const updateEavPrimitive: ({|
|
||||||
|
+id: Schema.ObjectId,
|
||||||
|
+fieldname: string,
|
||||||
|
+value: string | 0 | 1,
|
||||||
|
|}) => void = _makeSingleUpdateFunction(
|
||||||
|
db.prepare(
|
||||||
|
dedent`\
|
||||||
|
UPDATE primitives
|
||||||
|
SET value = :value
|
||||||
|
WHERE object_id = :id AND fieldname = :fieldname
|
||||||
|
`
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
for (const entry of queryResult) {
|
for (const entry of queryResult) {
|
||||||
const primitives: {|
|
const primitives: {|
|
||||||
@ -1406,9 +1419,9 @@ export class Mirror {
|
|||||||
`of type ${s(typename)} (got ${(primitive: empty)})`
|
`of type ${s(typename)} (got ${(primitive: empty)})`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
primitives[
|
const jsonValue = JSON.stringify(primitive);
|
||||||
parameterNameFor.topLevelField(fieldname)
|
primitives[parameterNameFor.topLevelField(fieldname)] = jsonValue;
|
||||||
] = JSON.stringify(primitive);
|
updateEavPrimitive({id: entry.id, fieldname, value: jsonValue});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add nested primitives.
|
// Add nested primitives.
|
||||||
@ -1428,6 +1441,11 @@ export class Mirror {
|
|||||||
}
|
}
|
||||||
primitives[parameterNameFor.topLevelField(nestFieldname)] =
|
primitives[parameterNameFor.topLevelField(nestFieldname)] =
|
||||||
topLevelNested == null ? 0 : 1;
|
topLevelNested == null ? 0 : 1;
|
||||||
|
updateEavPrimitive({
|
||||||
|
id: entry.id,
|
||||||
|
fieldname: nestFieldname,
|
||||||
|
value: topLevelNested == null ? 0 : 1,
|
||||||
|
});
|
||||||
const eggFields = objectType.nestedFields[nestFieldname].primitives;
|
const eggFields = objectType.nestedFields[nestFieldname].primitives;
|
||||||
for (const eggFieldname of Object.keys(eggFields)) {
|
for (const eggFieldname of Object.keys(eggFields)) {
|
||||||
const eggValue: PrimitiveResult | NodeFieldResult =
|
const eggValue: PrimitiveResult | NodeFieldResult =
|
||||||
@ -1442,13 +1460,19 @@ export class Mirror {
|
|||||||
`of type ${s(typename)} (got ${(primitive: empty)})`
|
`of type ${s(typename)} (got ${(primitive: empty)})`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
const jsonValue = JSON.stringify(primitive);
|
||||||
primitives[
|
primitives[
|
||||||
parameterNameFor.nestedField(nestFieldname, eggFieldname)
|
parameterNameFor.nestedField(nestFieldname, eggFieldname)
|
||||||
] = JSON.stringify(primitive);
|
] = jsonValue;
|
||||||
|
updateEavPrimitive({
|
||||||
|
id: entry.id,
|
||||||
|
fieldname: `${nestFieldname}.${eggFieldname}`,
|
||||||
|
value: jsonValue,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
updatePrimitives(primitives);
|
updateTypeSpecificPrimitives(primitives);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -932,6 +932,30 @@ describe("graphql/mirror", () => {
|
|||||||
|
|
||||||
// Check that some objects have the right primitives.
|
// Check that some objects have the right primitives.
|
||||||
// (These poke at the internals of the storage format a bit.)
|
// (These poke at the internals of the storage format a bit.)
|
||||||
|
|
||||||
|
// Check primitives (EAV format).
|
||||||
|
expect(
|
||||||
|
db
|
||||||
|
.prepare(
|
||||||
|
"SELECT object_id AS id, fieldname, value " +
|
||||||
|
"FROM objects JOIN primitives ON objects.id = object_id " +
|
||||||
|
"WHERE " +
|
||||||
|
"typename IN ('Repository', 'Issue', 'User', 'ClosedEvent') " +
|
||||||
|
"ORDER BY id, fieldname ASC"
|
||||||
|
)
|
||||||
|
.all()
|
||||||
|
).toEqual([
|
||||||
|
{id: "issue:#1", fieldname: "title", value: '"something wicked"'},
|
||||||
|
{id: "issue:#1", fieldname: "url", value: '"url://foo/bar/issue/1"'},
|
||||||
|
{id: "issue:#2", fieldname: "title", value: '"this way comes"'},
|
||||||
|
{id: "issue:#2", fieldname: "url", value: '"url://foo/bar/issue/2"'},
|
||||||
|
{id: "repo:foo/bar", fieldname: "url", value: '"url://foo/bar"'},
|
||||||
|
{id: "user:alice", fieldname: "login", value: null},
|
||||||
|
{id: "user:alice", fieldname: "url", value: null},
|
||||||
|
// nothing for `ClosedEvent` (it has no primitive fields)
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Check primitives (legacy format).
|
||||||
expect(
|
expect(
|
||||||
db
|
db
|
||||||
.prepare("SELECT * FROM primitives_Repository ORDER BY id ASC")
|
.prepare("SELECT * FROM primitives_Repository ORDER BY id ASC")
|
||||||
@ -942,13 +966,13 @@ describe("graphql/mirror", () => {
|
|||||||
).toEqual([
|
).toEqual([
|
||||||
{
|
{
|
||||||
id: "issue:#1",
|
id: "issue:#1",
|
||||||
url: JSON.stringify("url://foo/bar/issue/1"),
|
|
||||||
title: JSON.stringify("something wicked"),
|
title: JSON.stringify("something wicked"),
|
||||||
|
url: JSON.stringify("url://foo/bar/issue/1"),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "issue:#2",
|
id: "issue:#2",
|
||||||
url: JSON.stringify("url://foo/bar/issue/2"),
|
|
||||||
title: JSON.stringify("this way comes"),
|
title: JSON.stringify("this way comes"),
|
||||||
|
url: JSON.stringify("url://foo/bar/issue/2"),
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
expect(
|
expect(
|
||||||
@ -2142,9 +2166,26 @@ describe("graphql/mirror", () => {
|
|||||||
expect(
|
expect(
|
||||||
db.prepare("SELECT * FROM primitives_Issue ORDER BY id ASC").all()
|
db.prepare("SELECT * FROM primitives_Issue ORDER BY id ASC").all()
|
||||||
).toEqual([
|
).toEqual([
|
||||||
{id: "issue:#1", url: '"url://issue/1"', title: "13.75"},
|
{id: "issue:#1", title: "13.75", url: '"url://issue/1"'},
|
||||||
{id: "issue:#2", url: "null", title: "false"},
|
{id: "issue:#2", title: "false", url: "null"},
|
||||||
{id: "issue:#3", url: null, title: null},
|
{id: "issue:#3", title: null, url: null},
|
||||||
|
]);
|
||||||
|
expect(
|
||||||
|
db
|
||||||
|
.prepare(
|
||||||
|
"SELECT object_id AS id, fieldname, value " +
|
||||||
|
"FROM objects JOIN primitives ON objects.id = object_id " +
|
||||||
|
"WHERE typename = 'Issue' " +
|
||||||
|
"ORDER BY id, fieldname ASC"
|
||||||
|
)
|
||||||
|
.all()
|
||||||
|
).toEqual([
|
||||||
|
{id: "issue:#1", fieldname: "title", value: "13.75"},
|
||||||
|
{id: "issue:#1", fieldname: "url", value: '"url://issue/1"'},
|
||||||
|
{id: "issue:#2", fieldname: "title", value: "false"},
|
||||||
|
{id: "issue:#2", fieldname: "url", value: "null"},
|
||||||
|
{id: "issue:#3", fieldname: "title", value: null},
|
||||||
|
{id: "issue:#3", fieldname: "url", value: null},
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
it("stores data with non-`null` nested fields", () => {
|
it("stores data with non-`null` nested fields", () => {
|
||||||
@ -2178,9 +2219,9 @@ describe("graphql/mirror", () => {
|
|||||||
).toEqual([
|
).toEqual([
|
||||||
{
|
{
|
||||||
id: "commit:oid",
|
id: "commit:oid",
|
||||||
oid: '"yes"',
|
|
||||||
author: +true,
|
author: +true,
|
||||||
"author.date": '"today"',
|
"author.date": '"today"',
|
||||||
|
oid: '"yes"',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "commit:zzz",
|
id: "commit:zzz",
|
||||||
@ -2189,6 +2230,23 @@ describe("graphql/mirror", () => {
|
|||||||
"author.date": "null",
|
"author.date": "null",
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
expect(
|
||||||
|
db
|
||||||
|
.prepare(
|
||||||
|
"SELECT object_id AS id, fieldname, value " +
|
||||||
|
"FROM objects JOIN primitives ON objects.id = object_id " +
|
||||||
|
"WHERE typename = 'Commit' " +
|
||||||
|
"ORDER BY id, fieldname ASC"
|
||||||
|
)
|
||||||
|
.all()
|
||||||
|
).toEqual([
|
||||||
|
{id: "commit:oid", fieldname: "author", value: +true},
|
||||||
|
{id: "commit:oid", fieldname: "author.date", value: '"today"'},
|
||||||
|
{id: "commit:oid", fieldname: "oid", value: '"yes"'},
|
||||||
|
{id: "commit:zzz", fieldname: "author", value: +true},
|
||||||
|
{id: "commit:zzz", fieldname: "author.date", value: "null"},
|
||||||
|
{id: "commit:zzz", fieldname: "oid", value: '"zzz"'},
|
||||||
|
]);
|
||||||
expect(
|
expect(
|
||||||
db.prepare("SELECT * FROM links ORDER BY parent_id ASC").all()
|
db.prepare("SELECT * FROM links ORDER BY parent_id ASC").all()
|
||||||
).toEqual([
|
).toEqual([
|
||||||
@ -2247,6 +2305,20 @@ describe("graphql/mirror", () => {
|
|||||||
"author.date": "null",
|
"author.date": "null",
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
expect(
|
||||||
|
db
|
||||||
|
.prepare(
|
||||||
|
"SELECT object_id AS id, fieldname, value " +
|
||||||
|
"FROM objects JOIN primitives ON objects.id = object_id " +
|
||||||
|
"WHERE typename = 'Commit' " +
|
||||||
|
"ORDER BY id, fieldname ASC"
|
||||||
|
)
|
||||||
|
.all()
|
||||||
|
).toEqual([
|
||||||
|
{id: "commit:oid", fieldname: "author", value: +false},
|
||||||
|
{id: "commit:oid", fieldname: "author.date", value: "null"},
|
||||||
|
{id: "commit:oid", fieldname: "oid", value: '"mmm"'},
|
||||||
|
]);
|
||||||
expect(
|
expect(
|
||||||
db.prepare("SELECT * FROM links ORDER BY parent_id ASC").all()
|
db.prepare("SELECT * FROM links ORDER BY parent_id ASC").all()
|
||||||
).toEqual([
|
).toEqual([
|
||||||
@ -2289,6 +2361,17 @@ describe("graphql/mirror", () => {
|
|||||||
.prepare("SELECT * FROM primitives_LockedEvent ORDER BY id ASC")
|
.prepare("SELECT * FROM primitives_LockedEvent ORDER BY id ASC")
|
||||||
.all()
|
.all()
|
||||||
).toEqual([{id: "dos"}, {id: "uno"}]);
|
).toEqual([{id: "dos"}, {id: "uno"}]);
|
||||||
|
expect(
|
||||||
|
db
|
||||||
|
.prepare(
|
||||||
|
"SELECT objects.id AS o_id, primitives.rowid AS p_rowid " +
|
||||||
|
"FROM objects LEFT OUTER JOIN primitives " +
|
||||||
|
"ON objects.id = object_id " +
|
||||||
|
"WHERE typename = 'LockedEvent' " +
|
||||||
|
"ORDER BY o_id ASC"
|
||||||
|
)
|
||||||
|
.all()
|
||||||
|
).toEqual([{o_id: "dos", p_rowid: null}, {o_id: "uno", p_rowid: null}]);
|
||||||
expect(
|
expect(
|
||||||
db
|
db
|
||||||
.prepare("SELECT * FROM links ORDER BY parent_id ASC")
|
.prepare("SELECT * FROM links ORDER BY parent_id ASC")
|
||||||
|
Loading…
x
Reference in New Issue
Block a user