Initiatives: add EdgeSpec support to Initiative type (#1764)

Because the `Initiative` type now supports `EdgeSpec`, we're no longer
discarding entries when converting between `InitiativeFile` and
`Initiative`.

To make this commit smaller and easier to review, we're not yet adding
support to add `NodeEntry` to the graph though, instead
`createWeightedGraph` ignores the entries for now.
This commit is contained in:
Robin van Boven 2020-04-27 18:31:10 +02:00 committed by GitHub
parent 0a671025d7
commit 9e407b3cac
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 133 additions and 54 deletions

View File

@ -127,20 +127,32 @@ exports[`plugins/initiatives/initiativesDirectory loadDirectory should handle an
\\"http://foo.bar/A/champ\\" \\"http://foo.bar/A/champ\\"
], ],
\\"completed\\": true, \\"completed\\": true,
\\"contributions\\": [ \\"contributions\\": {
\\"http://foo.bar/A/contrib\\" \\"entries\\": [
], ],
\\"dependencies\\": [ \\"urls\\": [
\\"http://foo.bar/A/dep\\" \\"http://foo.bar/A/contrib\\"
], ]
},
\\"dependencies\\": {
\\"entries\\": [
],
\\"urls\\": [
\\"http://foo.bar/A/dep\\"
]
},
\\"id\\": [ \\"id\\": [
\\"INITIATIVE_FILE\\", \\"INITIATIVE_FILE\\",
\\"http://example.com/initiatives\\", \\"http://example.com/initiatives\\",
\\"initiative-A.json\\" \\"initiative-A.json\\"
], ],
\\"references\\": [ \\"references\\": {
\\"http://foo.bar/A/ref\\" \\"entries\\": [
], ],
\\"urls\\": [
\\"http://foo.bar/A/ref\\"
]
},
\\"timestampMs\\": 1578520917711, \\"timestampMs\\": 1578520917711,
\\"title\\": \\"Initiative A\\", \\"title\\": \\"Initiative A\\",
\\"weight\\": { \\"weight\\": {
@ -153,20 +165,32 @@ exports[`plugins/initiatives/initiativesDirectory loadDirectory should handle an
\\"http://foo.bar/B/champ\\" \\"http://foo.bar/B/champ\\"
], ],
\\"completed\\": false, \\"completed\\": false,
\\"contributions\\": [ \\"contributions\\": {
\\"http://foo.bar/B/contrib\\" \\"entries\\": [
], ],
\\"dependencies\\": [ \\"urls\\": [
\\"http://foo.bar/B/dep\\" \\"http://foo.bar/B/contrib\\"
], ]
},
\\"dependencies\\": {
\\"entries\\": [
],
\\"urls\\": [
\\"http://foo.bar/B/dep\\"
]
},
\\"id\\": [ \\"id\\": [
\\"INITIATIVE_FILE\\", \\"INITIATIVE_FILE\\",
\\"http://example.com/initiatives\\", \\"http://example.com/initiatives\\",
\\"initiative-B.json\\" \\"initiative-B.json\\"
], ],
\\"references\\": [ \\"references\\": {
\\"http://foo.bar/B/ref\\" \\"entries\\": [
], ],
\\"urls\\": [
\\"http://foo.bar/B/ref\\"
]
},
\\"timestampMs\\": 1578520917722, \\"timestampMs\\": 1578520917722,
\\"title\\": \\"Initiative B\\", \\"title\\": \\"Initiative B\\",
\\"weight\\": { \\"weight\\": {
@ -179,17 +203,56 @@ exports[`plugins/initiatives/initiativesDirectory loadDirectory should handle an
\\"http://foo.bar/C/champ\\" \\"http://foo.bar/C/champ\\"
], ],
\\"completed\\": false, \\"completed\\": false,
\\"contributions\\": [ \\"contributions\\": {
], \\"entries\\": [
\\"dependencies\\": [ {
], \\"contributors\\": [
\\"https://foo.bar/C/contrib-user\\"
],
\\"key\\": \\"add-test-contrib\\",
\\"timestampMs\\": 1578520917733,
\\"title\\": \\"Add test contrib\\",
\\"weight\\": 10
}
],
\\"urls\\": [
]
},
\\"dependencies\\": {
\\"entries\\": [
{
\\"contributors\\": [
\\"https://foo.bar/C/dep-user\\"
],
\\"key\\": \\"add-test-dependency\\",
\\"timestampMs\\": 1578002517700,
\\"title\\": \\"Add test dependency\\",
\\"weight\\": null
}
],
\\"urls\\": [
]
},
\\"id\\": [ \\"id\\": [
\\"INITIATIVE_FILE\\", \\"INITIATIVE_FILE\\",
\\"http://example.com/initiatives\\", \\"http://example.com/initiatives\\",
\\"initiative-C.json\\" \\"initiative-C.json\\"
], ],
\\"references\\": [ \\"references\\": {
], \\"entries\\": [
{
\\"contributors\\": [
\\"https://foo.bar/C/ref-user\\"
],
\\"key\\": \\"add-test-reference\\",
\\"timestampMs\\": 1578520917733,
\\"title\\": \\"Add test reference\\",
\\"weight\\": null
}
],
\\"urls\\": [
]
},
\\"timestampMs\\": 1578520917733, \\"timestampMs\\": 1578520917733,
\\"title\\": \\"Initiative C\\", \\"title\\": \\"Initiative C\\",
\\"weight\\": { \\"weight\\": {

View File

@ -12,6 +12,7 @@ import {type WeightedGraph as WeightedGraphT} from "../../core/weightedGraph";
import * as WeightedGraph from "../../core/weightedGraph"; import * as WeightedGraph from "../../core/weightedGraph";
import type {NodeWeight} from "../../core/weights"; import type {NodeWeight} from "../../core/weights";
import type {ReferenceDetector, URL} from "../../core/references"; import type {ReferenceDetector, URL} from "../../core/references";
import type {EdgeSpec} from "./edgeSpec";
import type {Initiative, InitiativeRepository} from "./initiative"; import type {Initiative, InitiativeRepository} from "./initiative";
import {addressFromId} from "./initiative"; import {addressFromId} from "./initiative";
import { import {
@ -90,9 +91,14 @@ export function createWeightedGraph(
// Generic approach to adding edges when the reference detector has a hit. // Generic approach to adding edges when the reference detector has a hit.
const edgeHandler = ( const edgeHandler = (
urls: $ReadOnlyArray<URL>, edges: $ReadOnlyArray<URL> | EdgeSpec,
createEdge: EdgeFactoryT createEdge: EdgeFactoryT
) => { ) => {
// TODO: this is a temporary implementation, which takes an EdgeSpec and
// just takes the $ReadOnlyArray<URL>. This is only done to allow support
// for graphing NodeEntries to be added in a separate commit.
const urls = Array.isArray(edges) ? edges : edges.urls;
for (const url of urls) { for (const url of urls) {
const addr = refs.addressFromUrl(url); const addr = refs.addressFromUrl(url);
if (!addr) continue; if (!addr) continue;

View File

@ -26,9 +26,9 @@ function _createInitiative(overrides?: $Shape<Initiative>): Initiative {
title: "Unset test initiative", title: "Unset test initiative",
timestampMs: Timestamp.fromNumber(123), timestampMs: Timestamp.fromNumber(123),
completed: false, completed: false,
dependencies: [], contributions: {urls: [], entries: []},
references: [], dependencies: {urls: [], entries: []},
contributions: [], references: {urls: [], entries: []},
champions: [], champions: [],
...overrides, ...overrides,
}; };
@ -237,7 +237,10 @@ describe("plugins/initiatives/createGraph", () => {
// Given // Given
const {repo, refs} = example(); const {repo, refs} = example();
repo.addInitiative({ repo.addInitiative({
dependencies: ["https://example.com/1"], dependencies: {
urls: ["https://example.com/1"],
entries: [],
},
}); });
// When // When
@ -253,7 +256,10 @@ describe("plugins/initiatives/createGraph", () => {
// Given // Given
const {repo, refs} = example(); const {repo, refs} = example();
repo.addInitiative({ repo.addInitiative({
references: ["https://example.com/2"], references: {
urls: ["https://example.com/2"],
entries: [],
},
}); });
// When // When
@ -269,7 +275,10 @@ describe("plugins/initiatives/createGraph", () => {
// Given // Given
const {repo, refs} = example(); const {repo, refs} = example();
repo.addInitiative({ repo.addInitiative({
contributions: ["https://example.com/3"], contributions: {
urls: ["https://example.com/3"],
entries: [],
},
}); });
// When // When
@ -304,7 +313,10 @@ describe("plugins/initiatives/createGraph", () => {
const {repo, refs} = example(); const {repo, refs} = example();
refs.addReference("https://example.com/1", exampleNodeAddress(1)); refs.addReference("https://example.com/1", exampleNodeAddress(1));
repo.addInitiative({ repo.addInitiative({
dependencies: ["https://example.com/1", "https://example.com/99"], dependencies: {
urls: ["https://example.com/1", "https://example.com/99"],
entries: [],
},
}); });
// When // When
@ -335,7 +347,10 @@ describe("plugins/initiatives/createGraph", () => {
const {repo, refs} = example(); const {repo, refs} = example();
refs.addReference("https://example.com/2", exampleNodeAddress(2)); refs.addReference("https://example.com/2", exampleNodeAddress(2));
repo.addInitiative({ repo.addInitiative({
references: ["https://example.com/2", "https://example.com/99"], references: {
urls: ["https://example.com/2", "https://example.com/99"],
entries: [],
},
}); });
// When // When
@ -366,7 +381,10 @@ describe("plugins/initiatives/createGraph", () => {
const {repo, refs} = example(); const {repo, refs} = example();
refs.addReference("https://example.com/3", exampleNodeAddress(3)); refs.addReference("https://example.com/3", exampleNodeAddress(3));
repo.addInitiative({ repo.addInitiative({
contributions: ["https://example.com/3", "https://example.com/99"], contributions: {
urls: ["https://example.com/3", "https://example.com/99"],
entries: [],
},
}); });
// When // When

View File

@ -4,6 +4,7 @@ import {type URL} from "../../core/references";
import {type NodeAddressT, NodeAddress} from "../../core/graph"; import {type NodeAddressT, NodeAddress} from "../../core/graph";
import {type NodeWeight} from "../../core/weights"; import {type NodeWeight} from "../../core/weights";
import {type TimestampMs} from "../../util/timestamp"; import {type TimestampMs} from "../../util/timestamp";
import {type EdgeSpec} from "./edgeSpec";
import {initiativeNodeType} from "./declaration"; import {initiativeNodeType} from "./declaration";
// Composite ID, used as input for NodeAddressT. // Composite ID, used as input for NodeAddressT.
@ -48,9 +49,9 @@ export type Initiative = {|
+timestampMs: TimestampMs, +timestampMs: TimestampMs,
+weight?: InitiativeWeight, +weight?: InitiativeWeight,
+completed: boolean, +completed: boolean,
+dependencies: $ReadOnlyArray<URL>, +dependencies: EdgeSpec,
+references: $ReadOnlyArray<URL>, +references: EdgeSpec,
+contributions: $ReadOnlyArray<URL>, +contributions: EdgeSpec,
+champions: $ReadOnlyArray<URL>, +champions: $ReadOnlyArray<URL>,
|}; |};

View File

@ -7,11 +7,11 @@ import {type URL} from "../../core/references";
import {type NodeAddressT} from "../../core/graph"; import {type NodeAddressT} from "../../core/graph";
import * as Timestamp from "../../util/timestamp"; import * as Timestamp from "../../util/timestamp";
import {compatReader} from "../../backend/compatIO"; import {compatReader} from "../../backend/compatIO";
import {normalizeEdgeSpec} from "./edgeSpec";
import { import {
type ReferenceDetector, type ReferenceDetector,
MappedReferenceDetector, MappedReferenceDetector,
} from "../../core/references"; } from "../../core/references";
import {type EdgeSpecJson} from "./edgeSpec";
import { import {
type Initiative, type Initiative,
type InitiativeRepository, type InitiativeRepository,
@ -165,9 +165,9 @@ export function _convertToInitiatives(
id: initiativeFileId(directory, fileName), id: initiativeFileId(directory, fileName),
timestampMs, timestampMs,
champions: champions || [], champions: champions || [],
contributions: _lossyURLFromEdgeSpecJson(contributions), contributions: normalizeEdgeSpec(contributions, timestampMs),
dependencies: _lossyURLFromEdgeSpecJson(dependencies), dependencies: normalizeEdgeSpec(dependencies, timestampMs),
references: _lossyURLFromEdgeSpecJson(references), references: normalizeEdgeSpec(references, timestampMs),
}; };
initiatives.push(initiative); initiatives.push(initiative);
@ -175,15 +175,6 @@ export function _convertToInitiatives(
return initiatives; return initiatives;
} }
// TODO: this is a temporary function, which reverts an ?EdgeSpecJson to just
// $ReadOnlyArray<URL>. It only exists to allow the `Initiative` type to add
// support for EdgeSpec in a separate commit.
export function _lossyURLFromEdgeSpecJson(
json?: EdgeSpecJson
): $ReadOnlyArray<URL> {
return (json || {}).urls || [];
}
// Creates a reference map using `initiativeFileURL`. // Creates a reference map using `initiativeFileURL`.
export function _createReferenceMap( export function _createReferenceMap(
initiatives: $ReadOnlyArray<Initiative> initiatives: $ReadOnlyArray<Initiative>

View File

@ -7,6 +7,7 @@ import stringify from "json-stable-stringify";
import {MappedReferenceDetector} from "../../core/references"; import {MappedReferenceDetector} from "../../core/references";
import * as Timestamp from "../../util/timestamp"; import * as Timestamp from "../../util/timestamp";
import {type Initiative, createId, addressFromId} from "./initiative"; import {type Initiative, createId, addressFromId} from "./initiative";
import {normalizeEdgeSpec} from "./edgeSpec";
import { import {
type InitiativesDirectory, type InitiativesDirectory,
loadDirectory, loadDirectory,
@ -16,7 +17,6 @@ import {
_validateUrl, _validateUrl,
_convertToInitiatives, _convertToInitiatives,
_createReferenceMap, _createReferenceMap,
_lossyURLFromEdgeSpecJson,
} from "./initiativesDirectory"; } from "./initiativesDirectory";
import {type InitiativeFile} from "./initiativeFile"; import {type InitiativeFile} from "./initiativeFile";
@ -55,9 +55,9 @@ const exampleInitiative = (remoteUrl: string, fileName: string): Initiative => {
id: createId("INITIATIVE_FILE", remoteUrl, fileName), id: createId("INITIATIVE_FILE", remoteUrl, fileName),
timestampMs, timestampMs,
champions: champions || [], champions: champions || [],
contributions: _lossyURLFromEdgeSpecJson(contributions), contributions: normalizeEdgeSpec(contributions, timestampMs),
dependencies: _lossyURLFromEdgeSpecJson(dependencies), dependencies: normalizeEdgeSpec(dependencies, timestampMs),
references: _lossyURLFromEdgeSpecJson(references), references: normalizeEdgeSpec(references, timestampMs),
}; };
}; };