Compare commits
10 Commits
mimo/code/
...
fix/1427-t
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f791cca049 | ||
| 324cdb0d26 | |||
| b4473267e0 | |||
| ed733d4eea | |||
| 7c9f4310d0 | |||
| 2016a7e076 | |||
| 15b9a4398c | |||
| 3f7277d920 | |||
|
|
ec2ed3c62f | ||
|
|
11175e72c0 |
3
app.js
3
app.js
@@ -734,6 +734,9 @@ async function init() {
|
|||||||
const response = await fetch('./portals.json');
|
const response = await fetch('./portals.json');
|
||||||
const portalData = await response.json();
|
const portalData = await response.json();
|
||||||
createPortals(portalData);
|
createPortals(portalData);
|
||||||
|
|
||||||
|
// Start portal hot-reload watcher
|
||||||
|
if (window.PortalHotReload) PortalHotReload.start(5000);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('Failed to load portals.json:', e);
|
console.error('Failed to load portals.json:', e);
|
||||||
addChatMessage('error', 'Portal registry offline. Check logs.');
|
addChatMessage('error', 'Portal registry offline. Check logs.');
|
||||||
|
|||||||
@@ -397,6 +397,7 @@
|
|||||||
<script src="./boot.js"></script>
|
<script src="./boot.js"></script>
|
||||||
<script src="./avatar-customization.js"></script>
|
<script src="./avatar-customization.js"></script>
|
||||||
<script src="./lod-system.js"></script>
|
<script src="./lod-system.js"></script>
|
||||||
|
<script src="./portal-hot-reload.js"></script>
|
||||||
<script>
|
<script>
|
||||||
function openMemoryFilter() { renderFilterList(); document.getElementById('memory-filter').style.display = 'flex'; }
|
function openMemoryFilter() { renderFilterList(); document.getElementById('memory-filter').style.display = 'flex'; }
|
||||||
function closeMemoryFilter() { document.getElementById('memory-filter').style.display = 'none'; }
|
function closeMemoryFilter() { document.getElementById('memory-filter').style.display = 'none'; }
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ from typing import Any, Callable, Optional
|
|||||||
|
|
||||||
import websockets
|
import websockets
|
||||||
|
|
||||||
from bannerlord_trace import BannerlordTraceLogger
|
from nexus.bannerlord_trace import BannerlordTraceLogger
|
||||||
|
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
# ═══════════════════════════════════════════════════════════════════════════
|
||||||
# CONFIGURATION
|
# CONFIGURATION
|
||||||
|
|||||||
@@ -304,6 +304,43 @@ async def inject_event(event_type: str, ws_url: str, **kwargs):
|
|||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def clean_lines(text: str) -> str:
|
||||||
|
"""Remove ANSI codes and collapse whitespace from log text."""
|
||||||
|
import re
|
||||||
|
text = strip_ansi(text)
|
||||||
|
text = re.sub(r'\s+', ' ', text).strip()
|
||||||
|
return text
|
||||||
|
|
||||||
|
|
||||||
|
def normalize_event(event: dict) -> dict:
|
||||||
|
"""Normalize an Evennia event dict to standard format."""
|
||||||
|
return {
|
||||||
|
"type": event.get("type", "unknown"),
|
||||||
|
"actor": event.get("actor", event.get("name", "")),
|
||||||
|
"room": event.get("room", event.get("location", "")),
|
||||||
|
"message": event.get("message", event.get("text", "")),
|
||||||
|
"timestamp": event.get("timestamp", ""),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def parse_room_output(text: str) -> dict:
|
||||||
|
"""Parse Evennia room output into structured data."""
|
||||||
|
import re
|
||||||
|
lines = text.strip().split("\n")
|
||||||
|
result = {"name": "", "description": "", "exits": [], "objects": []}
|
||||||
|
if lines:
|
||||||
|
result["name"] = strip_ansi(lines[0]).strip()
|
||||||
|
if len(lines) > 1:
|
||||||
|
result["description"] = strip_ansi(lines[1]).strip()
|
||||||
|
for line in lines[2:]:
|
||||||
|
line = strip_ansi(line).strip()
|
||||||
|
if line.startswith("Exits:"):
|
||||||
|
result["exits"] = [e.strip() for e in line[6:].split(",") if e.strip()]
|
||||||
|
elif line.startswith("You see:"):
|
||||||
|
result["objects"] = [o.strip() for o in line[8:].split(",") if o.strip()]
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
parser = argparse.ArgumentParser(description="Evennia -> Nexus WebSocket Bridge")
|
parser = argparse.ArgumentParser(description="Evennia -> Nexus WebSocket Bridge")
|
||||||
sub = parser.add_subparsers(dest="mode")
|
sub = parser.add_subparsers(dest="mode")
|
||||||
|
|||||||
@@ -44,9 +44,13 @@ class MemPalaceResult:
|
|||||||
|
|
||||||
|
|
||||||
def _get_client(palace_path: Path):
|
def _get_client(palace_path: Path):
|
||||||
"""Return a ChromaDB persistent client, or raise MemPalaceUnavailable."""
|
"""Return a ChromaDB persistent client, or raise MemPalaceUnavailable.
|
||||||
|
|
||||||
|
Telemetry is disabled for sovereignty — no data leaks to Chroma Inc.
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
import chromadb # type: ignore
|
import chromadb # type: ignore
|
||||||
|
from chromadb.config import Settings
|
||||||
except ImportError as exc:
|
except ImportError as exc:
|
||||||
raise MemPalaceUnavailable(
|
raise MemPalaceUnavailable(
|
||||||
"ChromaDB is not installed. "
|
"ChromaDB is not installed. "
|
||||||
@@ -59,7 +63,10 @@ def _get_client(palace_path: Path):
|
|||||||
"Run 'mempalace mine' to initialise the palace."
|
"Run 'mempalace mine' to initialise the palace."
|
||||||
)
|
)
|
||||||
|
|
||||||
return chromadb.PersistentClient(path=str(palace_path))
|
return chromadb.PersistentClient(
|
||||||
|
path=str(palace_path),
|
||||||
|
settings=Settings(anonymized_telemetry=False),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def search_memories(
|
def search_memories(
|
||||||
|
|||||||
105
portal-hot-reload.js
Normal file
105
portal-hot-reload.js
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
/**
|
||||||
|
* Portal Hot-Reload for The Nexus
|
||||||
|
*
|
||||||
|
* Watches portals.json for changes and hot-reloads portal list
|
||||||
|
* without server restart. Existing connections unaffected.
|
||||||
|
*
|
||||||
|
* Usage:
|
||||||
|
* PortalHotReload.start(intervalMs);
|
||||||
|
* PortalHotReload.stop();
|
||||||
|
* PortalHotReload.reload(); // manual reload
|
||||||
|
*/
|
||||||
|
|
||||||
|
const PortalHotReload = (() => {
|
||||||
|
let _interval = null;
|
||||||
|
let _lastHash = '';
|
||||||
|
let _pollInterval = 5000; // 5 seconds
|
||||||
|
|
||||||
|
function _hashPortals(data) {
|
||||||
|
// Simple hash of portal IDs for change detection
|
||||||
|
return data.map(p => p.id || p.name).sort().join(',');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function _checkForChanges() {
|
||||||
|
try {
|
||||||
|
const response = await fetch('./portals.json?t=' + Date.now());
|
||||||
|
if (!response.ok) return;
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const hash = _hashPortals(data);
|
||||||
|
|
||||||
|
if (hash !== _lastHash) {
|
||||||
|
console.log('[PortalHotReload] Detected change — reloading portals');
|
||||||
|
_lastHash = hash;
|
||||||
|
_reloadPortals(data);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Silent fail — file might be mid-write
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function _reloadPortals(data) {
|
||||||
|
// Remove old portals from scene
|
||||||
|
if (typeof portals !== 'undefined' && Array.isArray(portals)) {
|
||||||
|
portals.forEach(p => {
|
||||||
|
if (p.group && typeof scene !== 'undefined' && scene) {
|
||||||
|
scene.remove(p.group);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
portals.length = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create new portals
|
||||||
|
if (typeof createPortals === 'function') {
|
||||||
|
createPortals(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Re-register with spatial search if available
|
||||||
|
if (window.SpatialSearch && typeof portals !== 'undefined') {
|
||||||
|
portals.forEach(p => {
|
||||||
|
if (p.config && p.config.name && p.group) {
|
||||||
|
SpatialSearch.register('portal', p, p.config.name);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notify
|
||||||
|
if (typeof addChatMessage === 'function') {
|
||||||
|
addChatMessage('system', `Portals reloaded: ${data.length} portals active`);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`[PortalHotReload] Reloaded ${data.length} portals`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function start(intervalMs) {
|
||||||
|
if (_interval) return;
|
||||||
|
_pollInterval = intervalMs || _pollInterval;
|
||||||
|
|
||||||
|
// Initial load
|
||||||
|
fetch('./portals.json').then(r => r.json()).then(data => {
|
||||||
|
_lastHash = _hashPortals(data);
|
||||||
|
}).catch(() => {});
|
||||||
|
|
||||||
|
_interval = setInterval(_checkForChanges, _pollInterval);
|
||||||
|
console.log(`[PortalHotReload] Watching portals.json every ${_pollInterval}ms`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function stop() {
|
||||||
|
if (_interval) {
|
||||||
|
clearInterval(_interval);
|
||||||
|
_interval = null;
|
||||||
|
console.log('[PortalHotReload] Stopped');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function reload() {
|
||||||
|
const response = await fetch('./portals.json?t=' + Date.now());
|
||||||
|
const data = await response.json();
|
||||||
|
_lastHash = _hashPortals(data);
|
||||||
|
_reloadPortals(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { start, stop, reload };
|
||||||
|
})();
|
||||||
|
|
||||||
|
window.PortalHotReload = PortalHotReload;
|
||||||
@@ -26,7 +26,7 @@ HERMES_CONTEXT = [
|
|||||||
|
|
||||||
class RelevanceEngine:
|
class RelevanceEngine:
|
||||||
def __init__(self, collection_name: str = "deep_dive"):
|
def __init__(self, collection_name: str = "deep_dive"):
|
||||||
self.client = chromadb.PersistentClient(path="./chroma_db")
|
self.client = chromadb.PersistentClient(path="./chroma_db", settings=chromadb.config.Settings(anonymized_telemetry=False))
|
||||||
self.embedding_fn = embedding_functions.SentenceTransformerEmbeddingFunction(
|
self.embedding_fn = embedding_functions.SentenceTransformerEmbeddingFunction(
|
||||||
model_name="all-MiniLM-L6-v2"
|
model_name="all-MiniLM-L6-v2"
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ VIOLATION_KEYWORDS = [
|
|||||||
|
|
||||||
def audit(palace_path: Path):
|
def audit(palace_path: Path):
|
||||||
violations = []
|
violations = []
|
||||||
client = chromadb.PersistentClient(path=str(palace_path))
|
client = chromadb.PersistentClient(path=str(palace_path), settings=chromadb.config.Settings(anonymized_telemetry=False))
|
||||||
try:
|
try:
|
||||||
col = client.get_collection("mempalace_drawers")
|
col = client.get_collection("mempalace_drawers")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ DOCS_PER_ROOM = 5
|
|||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
client = chromadb.PersistentClient(path=PALACE_PATH)
|
client = chromadb.PersistentClient(path=PALACE_PATH, settings=chromadb.config.Settings(anonymized_telemetry=False))
|
||||||
col = client.get_collection("mempalace_drawers")
|
col = client.get_collection("mempalace_drawers")
|
||||||
|
|
||||||
# Discover rooms in this wing
|
# Discover rooms in this wing
|
||||||
|
|||||||
Reference in New Issue
Block a user