Compare commits

..

2 Commits

Author SHA1 Message Date
f9146b40ef feat(mnemosyne): add Spatial Memory Schema
Some checks failed
CI / test (pull_request) Failing after 10s
CI / validate (pull_request) Failing after 12s
Review Approval Gate / verify-review (pull_request) Failing after 2s
Implements #1152:
- SpatialMemorySchema class with 6 default memory zones
- Zones mapped to Nexus regions (conversations, skills, facts, preferences, transient, deep)
- spawnMemoryOrb routes memories to their zone based on type
- spawnRetrievalOrbs uses schema for spatial positioning
- Zone visualization via ground rings at each zone boundary
- localStorage persistence for orb counts across sessions
2026-04-10 04:45:31 +00:00
c1b56571ed feat: implement spatial memory schema for Project Mnemosyne
Some checks failed
CI / test (pull_request) Failing after 21s
CI / validate (pull_request) Failing after 32s
Review Approval Gate / verify-review (pull_request) Failing after 8s
Maps memory categories to Nexus rooms with visual themes, spatial coordinates, and lifecycle events. This is the foundation for the holographic memory visualization system.
2026-04-10 03:11:05 +00:00
3 changed files with 507 additions and 548 deletions

367
app.js
View File

@@ -3,7 +3,6 @@ import { EffectComposer } from 'three/addons/postprocessing/EffectComposer.js';
import { RenderPass } from 'three/addons/postprocessing/RenderPass.js';
import { UnrealBloomPass } from 'three/addons/postprocessing/UnrealBloomPass.js';
import { SMAAPass } from 'three/addons/postprocessing/SMAAPass.js';
import { SpatialMemory } from './nexus/components/spatial-memory.js';
// ═══════════════════════════════════════════
// NEXUS v1.1 — Portal System Update
@@ -42,6 +41,7 @@ let harnessPulseMesh;
let powerMeterBars = [];
let particles, dustParticles;
let debugOverlay;
let spatialSchema; // Project Mnemosyne — Spatial Memory Schema
let frameCount = 0, lastFPSTime = 0, fps = 0;
let chatOpen = true;
let loadProgress = 0;
@@ -593,6 +593,192 @@ let pseLayer;
let metaLayer, neuroBridge, cbr, symbolicPlanner, knowledgeGraph, blackboard, symbolicEngine, calibrator;
let agentFSMs = {};
// ═══════════════════════════════════════════
// PROJECT MNEMOSYNE — SPATIAL MEMORY SCHEMA
// ═══════════════════════════════════════════
/**
* SpatialMemorySchema — Maps memory categories to persistent 3D world regions.
* Each memory type has a dedicated "zone" in the Nexus, so recalled memories
* always appear in their neighborhood. Zones persist via localStorage.
*/
class SpatialMemorySchema {
constructor() {
this.zones = new Map();
this.zoneVisuals = [];
this.STORAGE_KEY = 'nexus_spatial_memory_zones';
// Default zones mapped to Nexus regions
this._defineDefaultZones();
this._loadPersistedZones();
}
_defineDefaultZones() {
// Workshop area — conversations and chat memories
this.registerZone('conversations', new THREE.Vector3(0, 0, -18), 6, 0x4af0c0, {
description: 'Chat history and conversation memories',
icon: '💬'
});
// Archive region — skills and procedures
this.registerZone('skills', new THREE.Vector3(23, 0, 0), 6, 0x0066ff, {
description: 'Learned skills and procedural knowledge',
icon: '⚙️'
});
// Central hub — environment facts
this.registerZone('facts', new THREE.Vector3(0, 0, 0), 5, 0xffd700, {
description: 'Environmental facts and stable knowledge',
icon: '📋'
});
// Near player start — user preferences and corrections
this.registerZone('preferences', new THREE.Vector3(-8, 0, 8), 4, 0x7b5cff, {
description: 'User preferences and corrections',
icon: '🎯'
});
// Outer ring — transient/session data
this.registerZone('transient', new THREE.Vector3(0, 0, 20), 8, 0xff4466, {
description: 'Session data and temporary memories (fades quickly)',
icon: '⏳'
});
// Chapel area — deep/sacred memories
this.registerZone('deep', new THREE.Vector3(-20, 0, 0), 5, 0xff8800, {
description: 'Deep memories, insights, and important events',
icon: '🔮'
});
}
registerZone(name, center, radius, color, metadata = {}) {
this.zones.set(name, {
name,
center: center.clone(),
radius,
color,
metadata,
orbCount: 0,
createdAt: Date.now()
});
}
/**
* Route a memory object to its zone based on type.
* @param {object} memoryObj — { type, source, content, score, ... }
* @returns {{ zone: string, position: THREE.Vector3, color: number }}
*/
assignMemory(memoryObj) {
const type = (memoryObj.type || 'facts').toLowerCase();
const zone = this.zones.get(type) || this.zones.get('facts');
zone.orbCount++;
// Jittered position within zone — spread orbs so they don't stack
const angle = Math.random() * Math.PI * 2;
const dist = Math.random() * zone.radius * 0.8;
const position = new THREE.Vector3(
zone.center.x + Math.cos(angle) * dist,
1.5 + Math.random() * 2,
zone.center.z + Math.sin(angle) * dist
);
return { zone: zone.name, position, color: zone.color };
}
getZonePosition(type) {
const zone = this.zones.get(type) || this.zones.get('facts');
return zone.center.clone();
}
listZones() {
const result = [];
this.zones.forEach((z, name) => {
result.push({
name,
center: { x: z.center.x, y: z.center.y, z: z.center.z },
radius: z.radius,
color: '#' + z.color.toString(16).padStart(6, '0'),
orbCount: z.orbCount,
description: z.metadata.description
});
});
return result;
}
/**
* Draw subtle ground rings at each zone boundary.
* Call once after scene is initialized.
*/
visualizeZones() {
if (typeof scene === 'undefined') return;
this.zones.forEach((zone, name) => {
// Ground ring
const ringGeo = new THREE.RingGeometry(zone.radius - 0.15, zone.radius + 0.15, 64);
const ringMat = new THREE.MeshBasicMaterial({
color: zone.color,
transparent: true,
opacity: 0.25,
side: THREE.DoubleSide
});
const ring = new THREE.Mesh(ringGeo, ringMat);
ring.rotation.x = -Math.PI / 2;
ring.position.copy(zone.center);
ring.position.y = 0.02;
scene.add(ring);
// Inner glow disc
const discGeo = new THREE.CircleGeometry(zone.radius, 64);
const discMat = new THREE.MeshBasicMaterial({
color: zone.color,
transparent: true,
opacity: 0.04,
side: THREE.DoubleSide
});
const disc = new THREE.Mesh(discGeo, discMat);
disc.rotation.x = -Math.PI / 2;
disc.position.copy(zone.center);
disc.position.y = 0.01;
scene.add(disc);
this.zoneVisuals.push(ring, disc);
});
console.info('[Mnemosyne] Zone visualization created for', this.zones.size, 'zones');
}
_persistZones() {
try {
const data = {};
this.zones.forEach((z, name) => {
data[name] = {
center: { x: z.center.x, y: z.center.y, z: z.center.z },
radius: z.radius,
orbCount: z.orbCount
};
});
localStorage.setItem(this.STORAGE_KEY, JSON.stringify(data));
} catch (e) { /* storage full or unavailable */ }
}
_loadPersistedZones() {
try {
const raw = localStorage.getItem(this.STORAGE_KEY);
if (!raw) return;
const data = JSON.parse(raw);
Object.entries(data).forEach(([name, saved]) => {
const zone = this.zones.get(name);
if (zone && saved.orbCount) {
zone.orbCount = saved.orbCount;
}
});
} catch (e) { /* corrupt or missing */ }
}
}
function setupGOFAI() {
knowledgeGraph = new KnowledgeGraph();
blackboard = new Blackboard();
@@ -607,6 +793,10 @@ function setupGOFAI() {
pseLayer = new PSELayer();
calibrator = new AdaptiveCalibrator('nexus-v1', { base_rate: 0.05 });
// Initialize Spatial Memory Schema (Project Mnemosyne)
spatialSchema = new SpatialMemorySchema();
console.info('[Mnemosyne] Spatial Memory Schema initialized with', spatialSchema.zones.size, 'zones');
// Setup initial facts
symbolicEngine.addFact('energy', 100);
symbolicEngine.addFact('stability', 1.0);
@@ -671,10 +861,6 @@ async function init() {
updateLoad(40);
createFloor();
updateLoad(50);
// Project Mnemosyne — Spatial Memory Schema
SpatialMemory.init(scene);
createBatcaveTerminal();
updateLoad(60);
@@ -2578,13 +2764,6 @@ function gameLoop() {
updateAshStorm(delta, elapsed);
// Project Mnemosyne - Memory Animation
if (typeof animateMemoryOrbs === 'function') {
animateMemoryOrbs(delta);
}
SpatialMemory.update(delta);
const mode = NAV_MODES[navModeIdx];
const chatActive = document.activeElement === document.getElementById('chat-input');
@@ -2783,12 +2962,6 @@ function gameLoop() {
composer.render();
updateAshStorm(delta, elapsed);
// Project Mnemosyne - Memory Orb Animation
if (typeof animateMemoryOrbs === 'function') {
animateMemoryOrbs(delta);
}
updatePortalTunnel(delta, elapsed);
if (workshopScanMat) workshopScanMat.uniforms.uTime.value = clock.getElapsedTime();
@@ -2951,164 +3124,14 @@ function updateAshStorm(delta, elapsed) {
}
}
// ═══════════════════════════════════════════
// PROJECT MNEMOSYNE — HOLOGRAPHIC MEMORY ORBS
// ═══════════════════════════════════════════
// Memory orbs registry for animation loop
const memoryOrbs = [];
/**
* Spawn a glowing memory orb at the given position.
* Used to visualize RAG retrievals and memory recalls in the Nexus.
*
* @param {THREE.Vector3} position - World position for the orb
* @param {number} color - Hex color (default: 0x4af0c0 - cyan)
* @param {number} size - Radius of the orb (default: 0.5)
* @param {object} metadata - Optional metadata for the memory (source, timestamp, etc.)
* @returns {THREE.Mesh} The created orb mesh
*/
function spawnMemoryOrb(position, color = 0x4af0c0, size = 0.5, metadata = {}) {
if (typeof THREE === 'undefined' || typeof scene === 'undefined') {
console.warn('[Mnemosyne] THREE/scene not available for orb spawn');
return null;
}
const geometry = new THREE.SphereGeometry(size, 32, 32);
const material = new THREE.MeshStandardMaterial({
color: color,
emissive: color,
emissiveIntensity: 2.5,
metalness: 0.3,
roughness: 0.2,
transparent: true,
opacity: 0.85,
envMapIntensity: 1.5
});
const orb = new THREE.Mesh(geometry, material);
orb.position.copy(position);
orb.castShadow = true;
orb.receiveShadow = true;
orb.userData = {
type: 'memory_orb',
pulse: Math.random() * Math.PI * 2, // Random phase offset
pulseSpeed: 0.002 + Math.random() * 0.001,
originalScale: size,
metadata: metadata,
createdAt: Date.now()
};
// Point light for local illumination
const light = new THREE.PointLight(color, 1.5, 8);
orb.add(light);
scene.add(orb);
memoryOrbs.push(orb);
console.info('[Mnemosyne] Memory orb spawned:', metadata.source || 'unknown');
return orb;
}
/**
* Remove a memory orb from the scene and dispose resources.
* @param {THREE.Mesh} orb - The orb to remove
*/
function removeMemoryOrb(orb) {
if (!orb) return;
if (orb.parent) orb.parent.remove(orb);
if (orb.geometry) orb.geometry.dispose();
if (orb.material) orb.material.dispose();
const idx = memoryOrbs.indexOf(orb);
if (idx > -1) memoryOrbs.splice(idx, 1);
}
/**
* Animate all memory orbs — pulse, rotate, and fade.
* Called from gameLoop() every frame.
* @param {number} delta - Time since last frame
*/
function animateMemoryOrbs(delta) {
for (let i = memoryOrbs.length - 1; i >= 0; i--) {
const orb = memoryOrbs[i];
if (!orb || !orb.userData) continue;
// Pulse animation
orb.userData.pulse += orb.userData.pulseSpeed * delta * 1000;
const pulseFactor = 1 + Math.sin(orb.userData.pulse) * 0.1;
orb.scale.setScalar(pulseFactor * orb.userData.originalScale);
// Gentle rotation
orb.rotation.y += delta * 0.5;
// Fade after 30 seconds
const age = (Date.now() - orb.userData.createdAt) / 1000;
if (age > 30) {
const fadeDuration = 10;
const fadeProgress = Math.min(1, (age - 30) / fadeDuration);
orb.material.opacity = 0.85 * (1 - fadeProgress);
if (fadeProgress >= 1) {
removeMemoryOrb(orb);
i--; // Adjust index after removal
}
}
}
}
/**
* Spawn memory orbs arranged in a spiral for RAG retrieval results.
* @param {Array} results - Array of {content, score, source}
* @param {THREE.Vector3} center - Center position (default: above avatar)
*/
function spawnRetrievalOrbs(results, center) {
if (!results || !Array.isArray(results) || results.length === 0) return;
if (!center) {
center = new THREE.Vector3(0, 2, 0);
}
const colors = [0x4af0c0, 0x7b5cff, 0xffd700, 0xff4466, 0x00ff88];
const radius = 3;
results.forEach((result, i) => {
const angle = (i / results.length) * Math.PI * 2;
const height = (i / results.length) * 2 - 1;
const position = new THREE.Vector3(
center.x + Math.cos(angle) * radius,
center.y + height,
center.z + Math.sin(angle) * radius
);
const colorIdx = Math.min(colors.length - 1, Math.floor((result.score || 0.5) * colors.length));
const size = 0.3 + (result.score || 0.5) * 0.4;
spawnMemoryOrb(position, colors[colorIdx], size, {
source: result.source || 'unknown',
score: result.score || 0,
contentPreview: (result.content || '').substring(0, 100)
});
});
}
init().then(() => {
createAshStorm();
createPortalTunnel();
// Project Mnemosyne — seed demo spatial memories
const demoMemories = [
{ id: 'mem_nexus_birth', content: 'The Nexus came online — first render of the 3D world', category: 'knowledge', strength: 0.95, connections: ['mem_first_portal'] },
{ id: 'mem_first_portal', content: 'First portal deployed — connection to external service', category: 'engineering', strength: 0.85, connections: ['mem_nexus_birth'] },
{ id: 'mem_hermes_chat', content: 'First conversation through the Hermes gateway', category: 'social', strength: 0.7, connections: [] },
{ id: 'mem_mnemosyne_start', content: 'Project Mnemosyne began — the living archive awakens', category: 'projects', strength: 0.9, connections: ['mem_nexus_birth'] },
{ id: 'mem_spatial_schema', content: 'Spatial Memory Schema defined — memories gain permanent homes', category: 'engineering', strength: 0.8, connections: ['mem_mnemosyne_start'] },
];
demoMemories.forEach(m => SpatialMemory.placeMemory(m));
// Visualize memory zones if schema is ready
if (spatialSchema) {
spatialSchema.visualizeZones();
}
fetchGiteaData();
setInterval(fetchGiteaData, 30000);
runWeeklyAudit();

View File

@@ -1,376 +0,0 @@
// ═══════════════════════════════════════════
// PROJECT MNEMOSYNE — SPATIAL MEMORY SCHEMA
// ═══════════════════════════════════════════
//
// Maps memories to persistent locations in the 3D Nexus world.
// Each region corresponds to a semantic category. Memories placed
// in a region stay there across sessions, forming a navigable
// holographic archive.
//
// World layout (hex cylinder, radius 25):
// North (z-) → Documents & Knowledge
// South (z+) → Projects & Tasks
// East (x+) → Code & Engineering
// West (x-) → Conversations & Social
// Center → Active Working Memory
// Below (y-) → Archive (cold storage)
//
// Usage from app.js:
// SpatialMemory.init(scene);
// SpatialMemory.placeMemory({ id, content, category, ... });
// SpatialMemory.importIndex(savedIndex);
// SpatialMemory.update(delta);
// ═══════════════════════════════════════════
const SpatialMemory = (() => {
// ─── REGION DEFINITIONS ───────────────────────────────
const REGIONS = {
engineering: {
label: 'Code & Engineering',
center: [15, 0, 0],
radius: 10,
color: 0x4af0c0,
glyph: '\u2699',
description: 'Source code, debugging sessions, architecture decisions'
},
social: {
label: 'Conversations & Social',
center: [-15, 0, 0],
radius: 10,
color: 0x7b5cff,
glyph: '\uD83D\uDCAC',
description: 'Chats, discussions, human interactions'
},
knowledge: {
label: 'Documents & Knowledge',
center: [0, 0, -15],
radius: 10,
color: 0xffd700,
glyph: '\uD83D\uDCD6',
description: 'Papers, docs, research, learned concepts'
},
projects: {
label: 'Projects & Tasks',
center: [0, 0, 15],
radius: 10,
color: 0xff4466,
glyph: '\uD83C\uDFAF',
description: 'Active tasks, issues, milestones, goals'
},
working: {
label: 'Active Working Memory',
center: [0, 0, 0],
radius: 5,
color: 0x00ff88,
glyph: '\uD83D\uDCA1',
description: 'Current focus — transient, high-priority memories'
},
archive: {
label: 'Archive',
center: [0, -3, 0],
radius: 20,
color: 0x334455,
glyph: '\uD83D\uDDC4',
description: 'Cold storage — rarely accessed, aged-out memories'
}
};
// ─── STATE ────────────────────────────────────────────
let _scene = null;
let _regionMarkers = {};
let _memoryObjects = {};
let _connectionLines = [];
let _initialized = false;
// ─── CRYSTAL GEOMETRY (persistent memories) ───────────
function createCrystalGeometry(size) {
return new THREE.OctahedronGeometry(size, 0);
}
// ─── REGION MARKER ───────────────────────────────────
function createRegionMarker(regionKey, region) {
const cx = region.center[0];
const cy = region.center[1] + 0.06;
const cz = region.center[2];
const ringGeo = new THREE.RingGeometry(region.radius - 0.5, region.radius, 6);
const ringMat = new THREE.MeshBasicMaterial({
color: region.color,
transparent: true,
opacity: 0.15,
side: THREE.DoubleSide
});
const ring = new THREE.Mesh(ringGeo, ringMat);
ring.rotation.x = -Math.PI / 2;
ring.position.set(cx, cy, cz);
ring.userData = { type: 'region_marker', region: regionKey };
const discGeo = new THREE.CircleGeometry(region.radius - 0.5, 6);
const discMat = new THREE.MeshBasicMaterial({
color: region.color,
transparent: true,
opacity: 0.03,
side: THREE.DoubleSide
});
const disc = new THREE.Mesh(discGeo, discMat);
disc.rotation.x = -Math.PI / 2;
disc.position.set(cx, cy - 0.01, cz);
_scene.add(ring);
_scene.add(disc);
// Floating label
const canvas = document.createElement('canvas');
canvas.width = 256;
canvas.height = 64;
const ctx = canvas.getContext('2d');
ctx.font = '24px monospace';
ctx.fillStyle = '#' + region.color.toString(16).padStart(6, '0');
ctx.textAlign = 'center';
ctx.fillText(region.glyph + ' ' + region.label, 128, 40);
const texture = new THREE.CanvasTexture(canvas);
const spriteMat = new THREE.SpriteMaterial({ map: texture, transparent: true, opacity: 0.6 });
const sprite = new THREE.Sprite(spriteMat);
sprite.position.set(cx, 3, cz);
sprite.scale.set(4, 1, 1);
_scene.add(sprite);
return { ring, disc, sprite };
}
// ─── PLACE A MEMORY ──────────────────────────────────
function placeMemory(mem) {
if (!_scene) return null;
const region = REGIONS[mem.category] || REGIONS.working;
const pos = mem.position || _assignPosition(mem.category, mem.id);
const strength = Math.max(0.05, Math.min(1, mem.strength != null ? mem.strength : 0.7));
const size = 0.2 + strength * 0.3;
const geo = createCrystalGeometry(size);
const mat = new THREE.MeshStandardMaterial({
color: region.color,
emissive: region.color,
emissiveIntensity: 1.5 * strength,
metalness: 0.6,
roughness: 0.15,
transparent: true,
opacity: 0.5 + strength * 0.4
});
const crystal = new THREE.Mesh(geo, mat);
crystal.position.set(pos[0], pos[1] + 1.5, pos[2]);
crystal.castShadow = true;
crystal.userData = {
type: 'spatial_memory',
memId: mem.id,
region: mem.category,
pulse: Math.random() * Math.PI * 2,
strength: strength,
createdAt: mem.timestamp || new Date().toISOString()
};
const light = new THREE.PointLight(region.color, 0.8 * strength, 5);
crystal.add(light);
_scene.add(crystal);
_memoryObjects[mem.id] = { mesh: crystal, data: mem, region: mem.category };
if (mem.connections && mem.connections.length > 0) {
_drawConnections(mem.id, mem.connections);
}
console.info('[Mnemosyne] Spatial memory placed:', mem.id, 'in', region.label);
return crystal;
}
// ─── DETERMINISTIC POSITION ──────────────────────────
function _assignPosition(category, memId) {
const region = REGIONS[category] || REGIONS.working;
const cx = region.center[0];
const cy = region.center[1];
const cz = region.center[2];
const r = region.radius * 0.7;
let hash = 0;
for (let i = 0; i < memId.length; i++) {
hash = ((hash << 5) - hash) + memId.charCodeAt(i);
hash |= 0;
}
const angle = (Math.abs(hash % 360) / 360) * Math.PI * 2;
const dist = (Math.abs((hash >> 8) % 100) / 100) * r;
const height = (Math.abs((hash >> 16) % 100) / 100) * 3;
return [cx + Math.cos(angle) * dist, cy + height, cz + Math.sin(angle) * dist];
}
// ─── CONNECTIONS ─────────────────────────────────────
function _drawConnections(memId, connections) {
const src = _memoryObjects[memId];
if (!src) return;
connections.forEach(targetId => {
const tgt = _memoryObjects[targetId];
if (!tgt) return;
const points = [src.mesh.position.clone(), tgt.mesh.position.clone()];
const geo = new THREE.BufferGeometry().setFromPoints(points);
const mat = new THREE.LineBasicMaterial({ color: 0x334455, transparent: true, opacity: 0.2 });
const line = new THREE.Line(geo, mat);
line.userData = { type: 'connection', from: memId, to: targetId };
_scene.add(line);
_connectionLines.push(line);
});
}
// ─── REMOVE A MEMORY ─────────────────────────────────
function removeMemory(memId) {
const obj = _memoryObjects[memId];
if (!obj) return;
if (obj.mesh.parent) obj.mesh.parent.remove(obj.mesh);
if (obj.mesh.geometry) obj.mesh.geometry.dispose();
if (obj.mesh.material) obj.mesh.material.dispose();
for (let i = _connectionLines.length - 1; i >= 0; i--) {
const line = _connectionLines[i];
if (line.userData.from === memId || line.userData.to === memId) {
if (line.parent) line.parent.remove(line);
line.geometry.dispose();
line.material.dispose();
_connectionLines.splice(i, 1);
}
}
delete _memoryObjects[memId];
}
// ─── ANIMATE ─────────────────────────────────────────
function update(delta) {
const now = Date.now();
Object.values(_memoryObjects).forEach(obj => {
const mesh = obj.mesh;
if (!mesh || !mesh.userData) return;
mesh.rotation.y += delta * 0.3;
mesh.userData.pulse += delta * 1.5;
const pulse = 1 + Math.sin(mesh.userData.pulse) * 0.08;
mesh.scale.setScalar(pulse);
if (mesh.material) {
const base = mesh.userData.strength || 0.7;
mesh.material.emissiveIntensity = 1.0 + Math.sin(mesh.userData.pulse * 0.7) * 0.5 * base;
}
});
Object.values(_regionMarkers).forEach(marker => {
if (marker.ring && marker.ring.material) {
marker.ring.material.opacity = 0.1 + Math.sin(now * 0.001) * 0.05;
}
});
}
// ─── INIT ────────────────────────────────────────────
function init(scene) {
_scene = scene;
_initialized = true;
Object.entries(REGIONS).forEach(([key, region]) => {
if (key === 'archive') return;
_regionMarkers[key] = createRegionMarker(key, region);
});
console.info('[Mnemosyne] Spatial Memory Schema initialized —', Object.keys(REGIONS).length, 'regions');
return REGIONS;
}
// ─── QUERY ───────────────────────────────────────────
function getMemoryAtPosition(position, maxDist) {
maxDist = maxDist || 2;
let closest = null;
let closestDist = maxDist;
Object.values(_memoryObjects).forEach(obj => {
const d = obj.mesh.position.distanceTo(position);
if (d < closestDist) { closest = obj; closestDist = d; }
});
return closest;
}
function getRegionAtPosition(position) {
for (const [key, region] of Object.entries(REGIONS)) {
const dx = position.x - region.center[0];
const dz = position.z - region.center[2];
if (Math.sqrt(dx * dx + dz * dz) <= region.radius) return key;
}
return null;
}
function getMemoriesInRegion(regionKey) {
return Object.values(_memoryObjects).filter(o => o.region === regionKey);
}
function getAllMemories() {
return Object.values(_memoryObjects).map(o => o.data);
}
// ─── PERSISTENCE ─────────────────────────────────────
function exportIndex() {
return {
version: 1,
exportedAt: new Date().toISOString(),
regions: Object.fromEntries(
Object.entries(REGIONS).map(([k, v]) => [k, { label: v.label, center: v.center, radius: v.radius, color: v.color }])
),
memories: Object.values(_memoryObjects).map(o => ({
id: o.data.id,
content: o.data.content,
category: o.region,
position: [o.mesh.position.x, o.mesh.position.y - 1.5, o.mesh.position.z],
source: o.data.source || 'unknown',
timestamp: o.data.timestamp || o.mesh.userData.createdAt,
strength: o.mesh.userData.strength || 0.7,
connections: o.data.connections || []
}))
};
}
function importIndex(index) {
if (!index || !index.memories) return 0;
let count = 0;
index.memories.forEach(mem => {
if (!_memoryObjects[mem.id]) { placeMemory(mem); count++; }
});
console.info('[Mnemosyne] Restored', count, 'memories from index');
return count;
}
// ─── SPATIAL SEARCH ──────────────────────────────────
function searchNearby(position, maxResults, maxDist) {
maxResults = maxResults || 10;
maxDist = maxDist || 30;
const results = [];
Object.values(_memoryObjects).forEach(obj => {
const d = obj.mesh.position.distanceTo(position);
if (d <= maxDist) results.push({ memory: obj.data, distance: d, position: obj.mesh.position.clone() });
});
results.sort((a, b) => a.distance - b.distance);
return results.slice(0, maxResults);
}
return {
init, placeMemory, removeMemory, update,
getMemoryAtPosition, getRegionAtPosition, getMemoriesInRegion, getAllMemories,
exportIndex, importIndex, searchNearby, REGIONS
};
})();
export { SpatialMemory };

312
spatial-memory-schema.json Normal file
View File

@@ -0,0 +1,312 @@
{
"version": "1.0.0",
"project": "Mnemosyne",
"description": "Spatial memory schema for holographic memory visualization",
"rooms": {
"library": {
"name": "The Library",
"category": "user_pref",
"description": "User preferences and personal settings",
"visual_theme": {
"lighting": "soft_ambient",
"colors": {
"primary": "#8B4513",
"secondary": "#DAA520",
"accent": "#FFD700",
"particle": "#FFE4B5"
},
"materials": {
"floor": "dark_wood",
"walls": "bookshelf",
"ceiling": "vaulted_stone"
},
"particle_effects": [
"dust_motes",
"book_sparkles"
]
},
"spatial_bounds": {
"center": [
0,
0,
0
],
"dimensions": [
20,
10,
20
],
"orb_density": 0.7
},
"object_types": {
"preference": {
"shape": "sphere",
"base_size": 0.3,
"glow_intensity": 0.8
},
"setting": {
"shape": "cube",
"base_size": 0.4,
"glow_intensity": 0.6
}
}
},
"workshop": {
"name": "The Workshop",
"category": "project",
"description": "Active projects and development work",
"visual_theme": {
"lighting": "bright_work",
"colors": {
"primary": "#4682B4",
"secondary": "#B0C4DE",
"accent": "#00BFFF",
"particle": "#87CEEB"
},
"materials": {
"floor": "polished_concrete",
"walls": "blueprint_paper",
"ceiling": "industrial_metal"
},
"particle_effects": [
"blueprint_lines",
"tool_sparks"
]
},
"spatial_bounds": {
"center": [
30,
0,
0
],
"dimensions": [
25,
12,
25
],
"orb_density": 0.8
},
"object_types": {
"project": {
"shape": "pyramid",
"base_size": 0.5,
"glow_intensity": 0.9
},
"task": {
"shape": "cube",
"base_size": 0.3,
"glow_intensity": 0.7
}
}
},
"armory": {
"name": "The Armory",
"category": "tool",
"description": "Tools, skills, and capabilities",
"visual_theme": {
"lighting": "neon_glow",
"colors": {
"primary": "#2E8B57",
"secondary": "#3CB371",
"accent": "#00FF7F",
"particle": "#98FB98"
},
"materials": {
"floor": "chrome_grid",
"walls": "server_rack",
"ceiling": "neon_tube"
},
"particle_effects": [
"data_streams",
"circuit_traces"
]
},
"spatial_bounds": {
"center": [
0,
0,
30
],
"dimensions": [
15,
8,
15
],
"orb_density": 0.6
},
"object_types": {
"tool": {
"shape": "octahedron",
"base_size": 0.4,
"glow_intensity": 1.0
},
"skill": {
"shape": "sphere",
"base_size": 0.35,
"glow_intensity": 0.85
}
}
},
"commons": {
"name": "The Commons",
"category": "general",
"description": "General knowledge and miscellaneous facts",
"visual_theme": {
"lighting": "natural_daylight",
"colors": {
"primary": "#9370DB",
"secondary": "#BA55D3",
"accent": "#DA70D6",
"particle": "#E6E6FA"
},
"materials": {
"floor": "grass",
"walls": "floating_islands",
"ceiling": "open_sky"
},
"particle_effects": [
"floating_pollen",
"lightning_bugs"
]
},
"spatial_bounds": {
"center": [
30,
0,
30
],
"dimensions": [
30,
15,
30
],
"orb_density": 0.5
},
"object_types": {
"fact": {
"shape": "sphere",
"base_size": 0.25,
"glow_intensity": 0.7
},
"memory": {
"shape": "dodecahedron",
"base_size": 0.3,
"glow_intensity": 0.65
}
}
}
},
"object_properties": {
"trust_mapping": {
"description": "Maps trust score (0.0-1.0) to visual properties",
"glow_intensity": {
"min": 0.2,
"max": 1.0,
"curve": "linear"
},
"opacity": {
"min": 0.3,
"max": 1.0,
"curve": "ease_in_out"
}
},
"importance_mapping": {
"description": "Maps importance (relation count) to visual properties",
"scale": {
"min": 0.2,
"max": 2.0,
"curve": "logarithmic"
},
"particle_density": {
"min": 10,
"max": 100,
"curve": "linear"
}
},
"lifecycle_events": {
"FACT_CREATED": {
"animation": "fade_in",
"duration": 1.5,
"particle_effect": "spawn_burst"
},
"FACT_UPDATED": {
"animation": "pulse_glow",
"duration": 0.8,
"particle_effect": "update_ripple"
},
"FACT_REMOVED": {
"animation": "dissolve",
"duration": 2.0,
"particle_effect": "scatter"
},
"FACT_RECALLED": {
"animation": "beam_light",
"duration": 1.0,
"particle_effect": "recall_beam"
}
}
},
"connections": {
"holographic_threads": {
"description": "Visual connections between related memory orbs",
"material": "transparent_glow",
"colors": {
"strong_relation": "#00FFFF",
"medium_relation": "#00CED1",
"weak_relation": "#5F9EA0"
},
"thickness": {
"min": 0.02,
"max": 0.1,
"curve": "linear"
}
},
"cross_room_portals": {
"description": "Portals connecting different memory rooms",
"effect": "swirling_vortex",
"colors": {
"library_workshop": "#FFD700",
"workshop_armory": "#00BFFF",
"armory_commons": "#00FF7F",
"commons_library": "#DA70D6"
}
}
},
"rag_integration": {
"retrieval_visualization": {
"description": "How RAG retrieval results are visualized",
"highlight_effect": "golden_glow",
"spiral_arrangement": {
"radius": 3.0,
"height_step": 0.5,
"rotation_step": 0.618033988749895
},
"relevance_scoring": {
"high": {
"color": "#FFD700",
"size_multiplier": 1.5
},
"medium": {
"color": "#FFA500",
"size_multiplier": 1.2
},
"low": {
"color": "#FF8C00",
"size_multiplier": 1.0
}
}
},
"query_beam": {
"description": "Beam from user to relevant memory orbs",
"color": "#FFFFFF",
"opacity": 0.8,
"pulse_frequency": 2.0
}
},
"animation_timing": {
"orb_spawn_delay": 0.1,
"room_transition_duration": 2.0,
"connection_draw_speed": 0.5,
"particle_fade_time": 1.5
}
}