From 6c58ef8e22fda25f37a6eb5890f34a48b25da43f Mon Sep 17 00:00:00 2001
From: "(quasar) nebula" <qznebula@protonmail.com>
Date: Sat, 11 Jan 2025 17:06:04 -0400
Subject: reverse: fill out reverseHelper

---
 src/reverse.js | 59 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++-
 1 file changed, 58 insertions(+), 1 deletion(-)

(limited to 'src')

diff --git a/src/reverse.js b/src/reverse.js
index 324cbabe..9cab5ef8 100644
--- a/src/reverse.js
+++ b/src/reverse.js
@@ -1,10 +1,67 @@
 import * as fr from './find-reverse.js';
 
+import {sortByDate} from '#sort';
+import {stitchArrays} from '#sugar';
+
 function reverseHelper(spec) {
   const cache = new WeakMap();
 
   return (thing, data) => {
-    return ({spec, from: thing, data: data.length});
+    // Check for an existing cache record which corresponds to this data.
+    // If it exists, query it for the requested thing, and return that;
+    // if it doesn't, create it and put it where it needs to be.
+
+    if (cache.has(data)) {
+      return cache.get(data).get(thing) ?? [];
+    }
+
+    const cacheRecord = new WeakMap();
+    cache.set(data, cacheRecord);
+
+    // Get the referencing and referenced things. This is the meat of how
+    // one reverse spec is different from another.
+
+    const referencingThings =
+      data.flatMap(thing => spec.referencing(thing));
+
+    const referencedThings =
+      referencingThings.map(thing => spec.referenced(thing));
+
+    // Actually fill in the cache record. Since we're building up a *reverse*
+    // reference list, track connections in terms of the referenced thing.
+    // Also gather all referenced things into a set, for sorting purposes.
+
+    const allReferencedThings = new Set();
+
+    stitchArrays({
+      referencingThing: referencingThings,
+      referencedThings: referencedThings,
+    }).forEach(({referencingThing, referencedThings}) => {
+        for (const referencedThing of referencedThings) {
+          if (cacheRecord.has(referencedThing)) {
+            cacheRecord.get(referencedThing).push(referencingThing);
+          } else {
+            cacheRecord.set(referencedThing, [referencingThing]);
+            allReferencedThings.add(referencedThing);
+          }
+        }
+      });
+
+    // Sort the entries in the cache records, too, just by date. The rest of
+    // sorting should be handled externally - either preceding the reverse
+    // call (changing the data input) or following (sorting the output).
+
+    for (const referencedThing of allReferencedThings) {
+      if (cacheRecord.has(referencedThing)) {
+        const referencingThings = cacheRecord.get(referencedThing);
+        sortByDate(referencingThings);
+      }
+    }
+
+    // Then just pluck out the requested thing from the now-filled
+    // cache record!
+
+    return cacheRecord.get(thing) ?? [];
   };
 }
 
-- 
cgit 1.3.0-6-gf8a5