| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- const path = require('path')
- const fs = require('fs')
- const ROUTES = [
- { path: '/public/latest', mod: null, description: 'home, latest posts, public timeline, news, recent activity' },
- { path: '/feed', mod: 'feedMod', description: 'feed, microblog, opinions, share thoughts, vote on posts, refeeds' },
- { path: '/forum', mod: 'forumMod', description: 'forum, discussions, threads, debates, conversation by category' },
- { path: '/inhabitants', mod: 'inhabitantsMod', description: 'inhabitants, users, people, profiles, contacts, follow, block' },
- { path: '/tribes', mod: 'tribeMod', description: 'tribes, groups, communities, private rooms, sub-tribes, governance' },
- { path: '/chats', mod: 'chatMod', description: 'chats, messaging, encrypted rooms, group conversations' },
- { path: '/pads', mod: 'padMod', description: 'pads, collaborative editor, shared notes, encrypted documents' },
- { path: '/calendars', mod: 'calendarMod', description: 'calendar, events by date, schedule, reminders, recurring dates' },
- { path: '/maps', mod: 'mapMod', description: 'maps, locations, markers, geography, places' },
- { path: '/events', mod: 'eventMod', description: 'events, agenda, meetups, gatherings, RSVP' },
- { path: '/agenda', mod: 'agendaMod', description: 'agenda, scheduled items, upcoming, my dates' },
- { path: '/tasks', mod: 'taskMod', description: 'tasks, todo, assignments, work items, priorities' },
- { path: '/projects', mod: 'projectMod', description: 'projects, milestones, backers, crowdfunding, bounties' },
- { path: '/jobs', mod: 'jobMod', description: 'jobs, work, hiring, salaries, vacancies, applications' },
- { path: '/market', mod: 'marketMod', description: 'market, marketplace, buy, sell, items, auctions, ECO' },
- { path: '/shops', mod: 'shopMod', description: 'shops, stores, products, ecommerce, vendors' },
- { path: '/banking', mod: 'bankingMod', description: 'banking, wallet, ECO balance, send money, transfers, payments, UBI claim' },
- { path: '/transfers', mod: 'transferMod', description: 'transfers, payments, money movements, ECO transactions, history' },
- { path: '/wallet', mod: 'walletMod', description: 'wallet, ECOin address, send and receive, QR code, balance' },
- { path: '/parliament', mod: 'parliamentMod', description: 'parliament, governance, government, proposals, laws, leaders, voting' },
- { path: '/courts', mod: 'courtsMod', description: 'courts, judges, accusations, mediators, justice, disputes' },
- { path: '/votations', mod: 'votationsMod', description: 'votations, polls, surveys, multi-option votes' },
- { path: '/votes', mod: 'votesMod', description: 'votes, ballots, decisions, polling, voting' },
- { path: '/opinions', mod: 'opinionsMod', description: 'opinions, reactions, ratings, sentiment, expressing views' },
- { path: '/trending', mod: 'trendingMod', description: 'trending, popular, hot, top voted, what is being discussed' },
- { path: '/reports', mod: 'reportsMod', description: 'reports, bug reports, abuse, incidents, severity, confirmations' },
- { path: '/audios', mod: 'audioMod', description: 'audios, music, podcasts, voice recordings, sound files' },
- { path: '/videos', mod: 'videoMod', description: 'videos, films, clips, recordings, watch' },
- { path: '/images', mod: 'imageMod', description: 'images, photos, pictures, gallery, memes' },
- { path: '/documents', mod: 'documentMod', description: 'documents, PDFs, files, papers, references' },
- { path: '/bookmarks', mod: 'bookmarkMod', description: 'bookmarks, links, saved websites, favorites' },
- { path: '/torrents', mod: 'torrentMod', description: 'torrents, magnet links, file sharing, downloads' },
- { path: '/tags', mod: 'tagsMod', description: 'tags, hashtags, topics, categories, labels' },
- { path: '/search', mod: null, description: 'search, find, query, lookup' },
- { path: '/inbox', mod: null, description: 'inbox, notifications, mentions, alerts, messages addressed to me' },
- { path: '/pm', mod: 'privateMessageMod', description: 'private messages, direct messages, DMs, encrypted PM' },
- { path: '/publish', mod: null, description: 'publish, write, create post, new entry, compose' },
- { path: '/games', mod: 'gameMod', description: 'games, play, mini-games, scoring, fun' },
- { path: '/pixelia', mod: 'pixeliaMod', description: 'pixelia, pixel canvas, draw, collaborative pixel art' },
- { path: '/cv', mod: 'cvMod', description: 'cv, curriculum, resume, my profile, skills, experiences' },
- { path: '/legacy', mod: 'legacyMod', description: 'legacy, export data, import, backup, restore identity' },
- { path: '/cipher', mod: 'cipherMod', description: 'cipher, encrypt, decrypt, password, vault' },
- { path: '/stats', mod: 'statsMod', description: 'stats, statistics, KPIs, metrics, dashboard, carbon footprint' },
- { path: '/blockchain', mod: 'blockchainMod', description: 'blockchain, blocks, explorer, ledger, chain' },
- { path: '/peers', mod: 'peersMod', description: 'peers, connections, network, nodes, who am I connected to' },
- { path: '/invites', mod: 'invitesMod', description: 'invites, pub invitations, join code, follow PUB' },
- { path: '/graphos', mod: 'graphosMod', description: 'graphos, network map, visualization, relationship graph' },
- { path: '/modules', mod: null, description: 'modules, features, enable disable plugins, settings' },
- { path: '/settings', mod: null, description: 'settings, preferences, language, theme, configuration' },
- { path: '/favorites', mod: 'favoritesMod', description: 'favorites, starred items, saved content' },
- { path: '/logs', mod: 'logsMod', description: 'logs, life log, personal records, journal, experiences' }
- ]
- const CACHE_FILE = path.join(__dirname, 'embeddings', 'routes_cache.json')
- let cache = null
- const buildCacheKey = () => ROUTES.map(r => r.path + '|' + r.description).join('\n')
- const loadCache = () => {
- try {
- if (!fs.existsSync(CACHE_FILE)) return null
- const data = JSON.parse(fs.readFileSync(CACHE_FILE, 'utf8'))
- if (data.key === buildCacheKey() && Array.isArray(data.entries)) return data.entries
- return null
- } catch (_) {
- return null
- }
- }
- const saveCache = (entries) => {
- try {
- fs.mkdirSync(path.dirname(CACHE_FILE), { recursive: true })
- fs.writeFileSync(CACHE_FILE, JSON.stringify({ key: buildCacheKey(), entries }, null, 2), 'utf8')
- } catch (_) {}
- }
- const ensureIndex = async ({ embed }) => {
- if (cache) return cache
- const cached = loadCache()
- if (cached) { cache = cached; return cache }
- const entries = []
- for (const r of ROUTES) {
- const vec = await embed(r.description)
- if (!vec) return null
- entries.push({ path: r.path, mod: r.mod, vector: vec })
- }
- cache = entries
- saveCache(entries)
- return cache
- }
- const resolveBest = async (queryVector, { isModuleEnabled, threshold = 0.4, embed } = {}) => {
- const idx = await ensureIndex({ embed })
- if (!idx) return null
- let best = null
- for (const entry of idx) {
- if (entry.mod && typeof isModuleEnabled === 'function' && !isModuleEnabled(entry.mod)) continue
- const score = (() => {
- let s = 0
- for (let i = 0; i < queryVector.length; i++) s += queryVector[i] * entry.vector[i]
- return s
- })()
- if (!best || score > best.score) best = { path: entry.path, score }
- }
- if (!best || best.score < threshold) return null
- return best
- }
- module.exports = { ROUTES, ensureIndex, resolveBest }
|