-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathref_merging.ts
46 lines (42 loc) · 1.54 KB
/
ref_merging.ts
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
import {refSorter} from "./js/google_drive/ref_sorter";
import {ListMultimap} from "./multimap";
function refSuffixIndex(ref: string): number {
if (ref.includes("-")) {
return -1;
}
return Math.max(ref.lastIndexOf(":"), ref.lastIndexOf(" "));
}
function splitOnRefSuffix(ref: string): string[] {
const splitPoint = refSuffixIndex(ref);
if (splitPoint === -1) {
return [ref];
}
return [ref.substring(0, splitPoint), ref.substring(splitPoint + 1)];
}
export function mergeRefs(refs: string[]): ListMultimap<string, string> {
refs = Array.from(new Set(refs));
const unmerged: string[] = [];
const bySharedPrefix = new ListMultimap<string, string>();
for (const ref of refs) {
const pieces = splitOnRefSuffix(ref);
if (pieces.length === 1) {
unmerged.push(ref);
} else {
bySharedPrefix.put(pieces[0], ref);
}
}
const mergedRefs = new ListMultimap<string, string>();
for (const mergeableValues of Array.from(bySharedPrefix.asMap().values())) {
if (mergeableValues.length === 1) {
mergedRefs.put(mergeableValues[0], mergeableValues[0]);
} else {
mergeableValues.sort(refSorter);
const [prefix, firstSuffix] = splitOnRefSuffix(mergeableValues[0]);
const [, lastSuffix] = splitOnRefSuffix(mergeableValues.at(-1)!);
const splitChar = mergeableValues[0].charAt(refSuffixIndex(mergeableValues[0]));
mergedRefs.putAll(`${prefix}${splitChar}${firstSuffix}-${lastSuffix}`, mergeableValues);
}
}
unmerged.forEach(x => mergedRefs.put(x, x));
return mergedRefs.sortedCopy();
}