1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
|
// Analogous implementation for withReverseReferenceList, for contributions.
// This is all duplicate code and both should be ported to the same underlying
// data form later on.
//
// This implementation uses a global cache (via WeakMap) to attempt to speed
// up subsequent similar accesses.
//
// This has absolutely not been rigorously tested with altering properties of
// data objects in a wiki data array which is reused. If a new wiki data array
// is used, a fresh cache will always be created.
import {input, templateCompositeFrom} from '#composite';
import {exitWithoutDependency} from '#composite/control-flow';
import inputWikiData from './inputWikiData.js';
// Mapping of reference list property to WeakMap.
// Each WeakMap maps a wiki data array to another weak map,
// which in turn maps each referenced thing to an array of
// things referencing it.
const caches = new Map();
export default templateCompositeFrom({
annotation: `withReverseContributionList`,
inputs: {
data: inputWikiData({allowMixedTypes: false}),
list: input({type: 'string'}),
},
outputs: ['#reverseContributionList'],
steps: () => [
exitWithoutDependency({
dependency: input('data'),
value: input.value([]),
mode: input.value('empty'),
}),
{
dependencies: [input.myself(), input('data'), input('list')],
compute: (continuation, {
[input.myself()]: myself,
[input('data')]: data,
[input('list')]: list,
}) => {
if (!caches.has(list)) {
caches.set(list, new WeakMap());
}
const cache = caches.get(list);
if (!cache.has(data)) {
const cacheRecord = new WeakMap();
for (const referencingThing of data) {
const referenceList = referencingThing[list];
// Destructuring {who} is the only unique part of the
// withReverseContributionList implementation, compared to
// withReverseReferneceList.
for (const {who: referencedThing} of referenceList) {
if (cacheRecord.has(referencedThing)) {
cacheRecord.get(referencedThing).push(referencingThing);
} else {
cacheRecord.set(referencedThing, [referencingThing]);
}
}
}
cache.set(data, cacheRecord);
}
return continuation({
['#reverseContributionList']:
cache.get(data).get(myself) ?? [],
});
},
},
],
});
|