Allow Git commits to have zero or one tree (#422)

Summary:
Submodule commits need not have associated tree objects, in case the
repository to which they belong does not exist in our graph. We’d like
to represent submodule commits as actual commits, which necessitates
this change. See #417 for context.

Test Plan:
Existing unit tests suffice.

wchargin-branch: git-affine-trees
This commit is contained in:
William Chargin 2018-06-27 11:47:39 -07:00 committed by GitHub
parent dd83d7b4ab
commit 38c364c916
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 20 additions and 32 deletions

View File

@ -205,22 +205,16 @@ export class GraphView {
} }
} }
// Each commits must have a unique and properly named HAS_TREE edge. // Any HAS_TREE edge to a commit must be properly named. This
// implies that a commit has at most one such edge. (Normal commits
// should have trees, but submodule commits might not.)
for (const rawNode of this._graph.nodes({prefix: GN._Prefix.commit})) { for (const rawNode of this._graph.nodes({prefix: GN._Prefix.commit})) {
const treeNeighbors = Array.from( for (const neighbor of this._graph.neighbors(rawNode, {
this._graph.neighbors(rawNode, {
direction: Direction.OUT, direction: Direction.OUT,
nodePrefix: NodeAddress.empty, nodePrefix: NodeAddress.empty,
edgePrefix: GE._Prefix.hasTree, edgePrefix: GE._Prefix.hasTree,
}) })) {
); const rawEdge = neighbor.edge;
if (treeNeighbors.length !== 1) {
throw new Error(
"invariant violation: commit should have 1 tree, " +
`but has ${treeNeighbors.length}: ${NodeAddress.toString(rawNode)}`
);
}
const rawEdge = treeNeighbors[0].edge;
const edge: GE.HasTreeAddress = (GE.fromRaw( const edge: GE.HasTreeAddress = (GE.fromRaw(
(((rawEdge.address: EdgeAddressT): any): GE.RawAddress) (((rawEdge.address: EdgeAddressT): any): GE.RawAddress)
): any); ): any);
@ -233,6 +227,7 @@ export class GraphView {
); );
} }
} }
}
// All HAS_PARENT edges must map between between the correct commits. // All HAS_PARENT edges must map between between the correct commits.
for (const edge of this._graph.edges({ for (const edge of this._graph.edges({

View File

@ -198,13 +198,6 @@ describe("plugins/git/graphView", () => {
"invariant violation: bad hom: " + edgeToString(badEdge) "invariant violation: bad hom: " + edgeToString(badEdge)
); );
}); });
it("for existence", () => {
const g = baseGraph();
expect(() => new GraphView(g)).toThrow(
"invariant violation: commit should have 1 tree, but has 0: " +
NodeAddress.toString(GN.toRaw(commit))
);
});
it("for correctness", () => { it("for correctness", () => {
const badEdge = {...otherEdge, src: edge.src}; const badEdge = {...otherEdge, src: edge.src};
const g = baseGraph().addEdge(badEdge); const g = baseGraph().addEdge(badEdge);