Spaces:
Running
Running
| // ===== KIMI INTELLIGENT MEMORY SYSTEM ===== | |
| class KimiMemorySystem { | |
| constructor(database) { | |
| this.db = database; | |
| this.memoryEnabled = true; | |
| this.maxMemoryEntries = 100; | |
| this.memoryCategories = { | |
| personal: "Personal Information", | |
| preferences: "Likes & Dislikes", | |
| relationships: "Relationships & People", | |
| activities: "Activities & Hobbies", | |
| goals: "Goals & Aspirations", | |
| experiences: "Shared Experiences", | |
| important: "Important Events" | |
| }; | |
| // Patterns for automatic memory extraction (multilingual) | |
| this.extractionPatterns = { | |
| personal: [ | |
| // English patterns | |
| /(?:my name is|i'm called|call me|i am) (\w+)/i, | |
| /(?:i am|i'm) (\d+) years? old/i, | |
| /(?:i live in|i'm from|from) ([^,.!?]+)/i, | |
| /(?:i work as|my job is|i'm a) ([^,.!?]+)/i, | |
| // French patterns | |
| /(?:je m'appelle|mon nom est|je suis|je me prénomme|je me nomme) ([^,.!?]+)/i, | |
| /(?:j'ai) (\d+) ans?/i, | |
| /(?:j'habite à|je vis à|je viens de) ([^,.!?]+)/i, | |
| /(?:je travaille comme|mon travail est|je suis) ([^,.!?]+)/i, | |
| // Spanish patterns | |
| /(?:me llamo|mi nombre es|soy) ([^,.!?]+)/i, | |
| /(?:tengo) (\d+) años?/i, | |
| /(?:vivo en|soy de) ([^,.!?]+)/i, | |
| /(?:trabajo como|mi trabajo es|soy) ([^,.!?]+)/i, | |
| // Italian patterns | |
| /(?:mi chiamo|il mio nome è|sono) ([^,.!?]+)/i, | |
| /(?:ho) (\d+) anni?/i, | |
| /(?:abito a|vivo a|sono di) ([^,.!?]+)/i, | |
| /(?:lavoro come|il mio lavoro è|sono) ([^,.!?]+)/i, | |
| // German patterns | |
| /(?:ich heiße|mein name ist|ich bin) ([^,.!?]+)/i, | |
| /(?:ich bin) (\d+) jahre? alt/i, | |
| /(?:ich wohne in|ich lebe in|ich komme aus) ([^,.!?]+)/i, | |
| /(?:ich arbeite als|mein beruf ist|ich bin) ([^,.!?]+)/i, | |
| // Japanese patterns | |
| /私の名前は([^。!?!?、,.]+)[ですだ]?/i, | |
| /私は([^。!?!?、,.]+)です/i, | |
| /([^、。!?!?,.]+)と申します/i, | |
| /([^、。!?!?,.]+)といいます/i, | |
| // Chinese patterns | |
| /我叫([^,。!?!?,.]+)/i, | |
| /我的名字是([^,。!?!?,.]+)/i, | |
| /叫我([^,。!?!?,.]+)/i | |
| ], | |
| preferences: [ | |
| // English patterns | |
| /(?:i love|i like|i enjoy|i prefer) ([^,.!?]+)/i, | |
| /(?:i hate|i dislike|i don't like) ([^,.!?]+)/i, | |
| /(?:my favorite|i really like) ([^,.!?]+)/i, | |
| // French patterns | |
| /(?:j'aime|j'adore|je préfère) ([^,.!?]+)/i, | |
| /(?:je déteste|je n'aime pas) ([^,.!?]+)/i, | |
| /(?:mon préféré|ma préférée) (?:est|sont) ([^,.!?]+)/i, | |
| // Explicit memory requests | |
| /(?:ajoute? (?:au|à la) (?:système? )?(?:de )?mémoire|retiens?|mémorise?) (?:que )?(.+)/i, | |
| /(?:add to memory|remember|memorize) (?:that )?(.+)/i | |
| ], | |
| relationships: [ | |
| // English patterns | |
| /(?:my (?:wife|husband|girlfriend|boyfriend|partner)) (?:is|named?) ([^,.!?]+)/i, | |
| /(?:my (?:mother|father|sister|brother|friend)) ([^,.!?]+)/i, | |
| // French patterns | |
| /(?:ma (?:femme|copine|partenaire)|mon (?:mari|copain|partenaire)) (?:s'appelle|est) ([^,.!?]+)/i, | |
| /(?:ma (?:mère|sœur)|mon (?:père|frère|ami)) (?:s'appelle|est) ([^,.!?]+)/i, | |
| // Spanish patterns | |
| /(?:mi (?:esposa|esposo|novia|novio|pareja)) (?:es|se llama) ([^,.!?]+)/i, | |
| /(?:mi (?:madre|padre|hermana|hermano|amigo|amiga)) (?:es|se llama) ([^,.!?]+)/i, | |
| // Italian patterns | |
| /(?:la mia (?:moglie|fidanzata|compagna)|il mio (?:marito|fidanzato|compagno)) (?:è|si chiama) ([^,.!?]+)/i, | |
| /(?:mia (?:madre|sorella)|mio (?:padre|fratello|amico)) (?:è|si chiama) ([^,.!?]+)/i, | |
| // German patterns | |
| /(?:meine (?:frau|freundin|partnerin)|mein (?:mann|freund|partner)) (?:ist|heißt) ([^,.!?]+)/i, | |
| /(?:meine (?:mutter|schwester)|mein (?:vater|bruder|freund)) (?:ist|heißt) ([^,.!?]+)/i, | |
| // Japanese patterns | |
| /(?:私の(?:妻|夫|彼女|彼氏|パートナー))は([^。!?!?、,.]+)(?:です|といいます)/i, | |
| /(?:私の(?:母|父|姉|妹|兄|弟|友達))は([^。!?!?、,.]+)(?:です|といいます)/i, | |
| // Chinese patterns | |
| /(?:我的(?:妻子|丈夫|女朋友|男朋友|伴侣))叫([^,。!?!?,.]+)/i, | |
| /(?:我的(?:妈妈|父亲|姐姐|妹妹|哥哥|弟弟|朋友))叫([^,。!?!?,.]+)/i | |
| ], | |
| activities: [ | |
| // English patterns | |
| /(?:i play|i do|i practice) ([^,.!?]+)/i, | |
| /(?:my hobby is|i hobby) ([^,.!?]+)/i, | |
| // French patterns | |
| /(?:je joue|je fais|je pratique) ([^,.!?]+)/i, | |
| /(?:mon passe-temps|mon hobby) (?:est|c'est) ([^,.!?]+)/i, | |
| // Spanish patterns | |
| /(?:juego|hago|practico) ([^,.!?]+)/i, | |
| /(?:mi pasatiempo|mi hobby) (?:es) ([^,.!?]+)/i, | |
| // Italian patterns | |
| /(?:gioco|faccio|pratico) ([^,.!?]+)/i, | |
| /(?:il mio passatempo|il mio hobby) (?:è) ([^,.!?]+)/i, | |
| // German patterns | |
| /(?:ich spiele|ich mache|ich übe) ([^,.!?]+)/i, | |
| /(?:mein hobby ist) ([^,.!?]+)/i, | |
| // Japanese patterns | |
| /(?:私は)?(?:[^、。!?!?,.]+)が趣味です/i, | |
| /趣味は([^。!?!?、,.]+)です/i, | |
| // Chinese patterns | |
| /(?:我玩|我做|我练习)([^,。!?!?,.]+)/i, | |
| /(?:我的爱好是)([^,。!?!?,.]+)/i | |
| ], | |
| goals: [ | |
| // English patterns | |
| /(?:i want to|i plan to|my goal is) ([^,.!?]+)/i, | |
| /(?:i'm learning|i study) ([^,.!?]+)/i, | |
| // French patterns | |
| /(?:je veux|je vais|mon objectif est) ([^,.!?]+)/i, | |
| /(?:j'apprends|j'étudie) ([^,.!?]+)/i, | |
| // Spanish patterns | |
| /(?:quiero|voy a|mi objetivo es) ([^,.!?]+)/i, | |
| /(?:estoy aprendiendo|estudio) ([^,.!?]+)/i, | |
| // Italian patterns | |
| /(?:voglio|andrò a|il mio obiettivo è) ([^,.!?]+)/i, | |
| /(?:sto imparando|studio) ([^,.!?]+)/i, | |
| // German patterns | |
| /(?:ich möchte|ich will|mein ziel ist) ([^,.!?]+)/i, | |
| /(?:ich lerne|ich studiere) ([^,.!?]+)/i, | |
| // Japanese patterns | |
| /(?:私は)?(?:[^、。!?!?,.]+)したい/i, | |
| /(?:学んでいる|勉強している) ([^。!?!?、,.]+)/i, | |
| // Chinese patterns | |
| /(?:我想|我要|我的目标是)([^,。!?!?,.]+)/i, | |
| /(?:我在学习|我学习)([^,。!?!?,.]+)/i | |
| ], | |
| experiences: [ | |
| // English patterns | |
| /we went to ([^,.!?]+)/i, | |
| /we met (?:at|on|in) ([^,.!?]+)/i, | |
| /our (?:first date|first kiss|trip|vacation) (?:was|was at|was on|was in|was to) ([^,.!?]+)/i, | |
| /our anniversary (?:is|falls on|will be) ([^,.!?]+)/i, | |
| /we moved in (?:together )?(?:on|in)?\s*([^,.!?]+)/i, | |
| // French patterns | |
| /on s'est rencontr[ée]s? (?:à|au|en|le) ([^,.!?]+)/i, | |
| /on est all[ée]s? à ([^,.!?]+)/i, | |
| /notre (?:premier rendez-vous|première sortie) (?:était|c'était) ([^,.!?]+)/i, | |
| /notre anniversaire (?:est|c'est) ([^,.!?]+)/i, | |
| /on a emménagé (?:ensemble\s*)?(?:le|en|à)\s*([^,.!?]+)/i, | |
| // Spanish patterns | |
| /nos conocimos (?:en|el|la) ([^,.!?]+)/i, | |
| /fuimos a ([^,.!?]+)/i, | |
| /nuestra (?:primera cita|primera salida) (?:fue|era) ([^,.!?]+)/i, | |
| /nuestro aniversario (?:es|cae en|será) ([^,.!?]+)/i, | |
| /nos mudamos (?:juntos\s*)?(?:el|en|a)\s*([^,.!?]+)/i, | |
| // Italian patterns | |
| /ci siamo conosciuti (?:a|al|in|il) ([^,.!?]+)/i, | |
| /siamo andati a ([^,.!?]+)/i, | |
| /il nostro (?:primo appuntamento|primo bacio|viaggio) (?:era|è stato) ([^,.!?]+)/i, | |
| /il nostro anniversario (?:è|cade il|sarà) ([^,.!?]+)/i, | |
| /ci siamo trasferiti (?:insieme\s*)?(?:il|in|a)\s*([^,.!?]+)/i, | |
| // German patterns | |
| /wir haben uns (?:in|am) ([^,.!?]+) kennengelernt/i, | |
| /wir sind (?:nach|zu) ([^,.!?]+) (?:gegangen|gefahren)/i, | |
| /unser (?:erstes date|erster kuss|urlaub) (?:war|fand statt) ([^,.!?]+)/i, | |
| /unser jahrestag (?:ist|fällt auf|wird sein) ([^,.!?]+)/i, | |
| /wir sind (?:zusammen )?eingezogen (?:am|im|in)\s*([^,.!?]+)/i, | |
| // Japanese patterns | |
| /私たちは([^、。!?!?,.]+)で出会った/i, | |
| /一緒に([^、。!?!?,.]+)へ行った/i, | |
| /私たちの記念日(?:は)?([^、。!?!?,.]+)/i, | |
| /一緒に引っ越した(?:のは)?([^、。!?!?,.]+)/i, | |
| // Chinese patterns | |
| /我们在([^,。!?!?,.]+)认识/i, | |
| /我们去了([^,。!?!?,.]+)/i, | |
| /我们的纪念日是([^,。!?!?,.]+)/i, | |
| /我们一起搬家(?:是在)?([^,。!?!?,.]+)/i | |
| ], | |
| important: [ | |
| // English patterns | |
| /it's important (?:to remember|that) (.+)/i, | |
| /please remember (.+)/i, | |
| // French patterns | |
| /c'est important (?:de se souvenir|que) (.+)/i, | |
| /merci de te souvenir (.+)/i, | |
| // Spanish patterns | |
| /es importante (?:recordar|que) (.+)/i, | |
| /por favor recuerda (.+)/i, | |
| // Italian patterns | |
| /è importante (?:ricordare|che) (.+)/i, | |
| /per favore ricorda (.+)/i, | |
| // German patterns | |
| /es ist wichtig (?:zu erinnern|dass) (.+)/i, | |
| /bitte erinnere dich an (.+)/i, | |
| // Japanese patterns | |
| /重要なのは(.+)です/i, | |
| /覚えておいてほしいのは(.+)です/i, | |
| // Chinese patterns | |
| /重要的是(.+)/i, | |
| /请记住(.+)/i | |
| ] | |
| }; | |
| } | |
| async init() { | |
| if (!this.db) { | |
| console.warn("Database not available for memory system"); | |
| return; | |
| } | |
| try { | |
| this.memoryEnabled = await this.db.getPreference("memorySystemEnabled", true); | |
| this.selectedCharacter = await this.db.getSelectedCharacter(); | |
| await this.createMemoryTables(); | |
| // Migrer les IDs incompatibles si nécessaire | |
| await this.migrateIncompatibleIDs(); | |
| } catch (error) { | |
| console.error("Memory system initialization error:", error); | |
| } | |
| } | |
| async createMemoryTables() { | |
| // Ensure memory tables exist in database | |
| if (!this.db.db.memories) { | |
| console.warn("Memory table not found in database schema"); | |
| return; | |
| } | |
| } | |
| // MEMORY EXTRACTION from conversation | |
| async extractMemoryFromText(userText, kimiResponse = null) { | |
| if (!this.memoryEnabled || !userText) return []; | |
| const extractedMemories = []; | |
| const text = userText.toLowerCase(); | |
| console.log("🔍 Memory extraction - Processing text:", userText); | |
| // Enhanced extraction with context awareness | |
| const existingMemories = await this.getAllMemories(); | |
| // First, check for explicit memory requests | |
| const explicitRequests = this.detectExplicitMemoryRequests(userText); | |
| if (explicitRequests.length > 0) { | |
| console.log("🎯 Explicit memory requests detected:", explicitRequests); | |
| extractedMemories.push(...explicitRequests); | |
| } | |
| // Extract using patterns | |
| for (const [category, patterns] of Object.entries(this.extractionPatterns)) { | |
| for (const pattern of patterns) { | |
| const match = text.match(pattern); | |
| if (match && match[1]) { | |
| const content = match[1].trim(); | |
| // Skip very short or generic content | |
| if (content.length < 2 || this.isGenericContent(content)) { | |
| continue; | |
| } | |
| // Check if this is a meaningful update to existing memory | |
| const isUpdate = await this.isMemoryUpdate(category, content, existingMemories); | |
| const memory = { | |
| category: category, | |
| type: "auto_extracted", | |
| content: content, | |
| sourceText: userText, | |
| confidence: this.calculateExtractionConfidence(match, userText), | |
| timestamp: new Date(), | |
| character: this.selectedCharacter, | |
| isUpdate: isUpdate | |
| }; | |
| console.log(`💡 Pattern match for ${category}:`, content); | |
| extractedMemories.push(memory); | |
| } | |
| } | |
| } | |
| // Enhanced pattern detection for more natural expressions | |
| const enhancedMemories = await this.detectNaturalExpressions(userText, existingMemories); | |
| extractedMemories.push(...enhancedMemories); | |
| // Save extracted memories with intelligent deduplication | |
| const savedMemories = []; | |
| for (const memory of extractedMemories) { | |
| console.log("💾 Saving memory:", memory.content); | |
| const saved = await this.addMemory(memory); | |
| if (saved) savedMemories.push(saved); | |
| } | |
| if (savedMemories.length > 0) { | |
| console.log(`✅ Successfully extracted and saved ${savedMemories.length} memories`); | |
| } else { | |
| console.log("📝 No memories extracted from this text"); | |
| } | |
| return savedMemories; | |
| } | |
| // Detect explicit memory requests like "ajoute en mémoire que..." | |
| detectExplicitMemoryRequests(text) { | |
| const memories = []; | |
| const lowerText = text.toLowerCase(); | |
| // French patterns for explicit memory requests | |
| const frenchPatterns = [ | |
| /(?:ajoute?s?(?:r)?|retiens?|mémorise?s?|enregistre?s?|sauvegarde?s?)\s+(?:au|à|en|dans)\s+(?:la\s+|le\s+)?(?:système?\s+(?:de\s+)?)?mémoire\s+(?:que\s+)?(.+)/i, | |
| /(?:peux-tu|pourrais-tu|veux-tu)?\s*(?:ajouter|retenir|mémoriser|enregistrer|sauvegarder)\s+(?:que\s+)?(.+)\s+(?:en|dans)\s+(?:la\s+|le\s+)?mémoire/i, | |
| /(?:je\s+veux\s+que\s+tu\s+)?(?:retienne?s|mémorise?s|ajoute?s)\s+(?:que\s+)?(.+)/i | |
| ]; | |
| // English patterns for explicit memory requests | |
| const englishPatterns = [ | |
| /(?:add\s+to\s+memory|remember|memorize|save\s+(?:to\s+)?memory)\s+(?:that\s+)?(.+)/i, | |
| /(?:can\s+you|could\s+you)?\s*(?:add|remember|memorize|save)\s+(?:that\s+)?(.+)\s+(?:to\s+|in\s+)?memory/i, | |
| /(?:i\s+want\s+you\s+to\s+)?(?:remember|memorize|add)\s+(?:that\s+)?(.+)/i | |
| ]; | |
| // Spanish explicit memory requests | |
| const spanishPatterns = [ | |
| /(?:añade|agrega|recuerda|memoriza|guarda)\s+(?:en|a)\s+(?:la\s+)?memoria\s+(?:que\s+)?(.+)/i, | |
| /(?:puedes|podrías)?\s*(?:añadir|agregar|recordar|memorizar|guardar)\s+(?:que\s+)?(.+)\s+(?:en|a)\s+(?:la\s+)?memoria/i, | |
| /(?:quiero\s+que\s+)?(?:recuerdes|memorices|añadas)\s+(?:que\s+)?(.+)/i | |
| ]; | |
| // Italian explicit memory requests | |
| const italianPatterns = [ | |
| /(?:aggiungi|ricorda|memorizza|salva)\s+(?:nella|in)\s+memoria\s+(?:che\s+)?(.+)/i, | |
| /(?:puoi|potresti)?\s*(?:aggiungere|ricordare|memorizzare|salvare)\s+(?:che\s+)?(.+)\s+(?:nella|in)\s+memoria/i, | |
| /(?:voglio\s+che\s+)?(?:ricordi|memorizzi|aggiunga)\s+(?:che\s+)?(.+)/i | |
| ]; | |
| // German explicit memory requests | |
| const germanPatterns = [ | |
| /(?:füge|merke|speichere)\s+(?:es\s+)?(?:in|zur)\s+?gedächtnis|speicher\s+(?:dass\s+)?(.+)/i, | |
| /(?:kannst\s+du|könntest\s+du)?\s*(?:hinzufügen|merken|speichern)\s+(?:dass\s+)?(.+)\s+(?:in|zum)\s+(?:gedächtnis|speicher)/i, | |
| /(?:ich\s+möchte\s+dass\s+du)\s*(?:merkst|speicherst|hinzufügst)\s+(?:dass\s+)?(.+)/i | |
| ]; | |
| // Japanese explicit memory requests | |
| const japanesePatterns = [ | |
| /記憶に(?:追加|保存|覚えて)(?:して)?(?:ほしい|ください)?(?:、)?(.+)/i, | |
| /(?:覚えて|記憶して)(?:ほしい|ください)?(?:、)?(.+)/i | |
| ]; | |
| // Chinese explicit memory requests | |
| const chinesePatterns = [ | |
| /把(.+)记在(?:记忆|内存|记忆库)里/i, | |
| /(?:请)?记住(?:这件事|这个|以下)?(.+)/i, | |
| /保存到记忆(?:里|中)(?:的是)?(.+)/i | |
| ]; | |
| const allPatterns = [ | |
| ...frenchPatterns, | |
| ...englishPatterns, | |
| ...spanishPatterns, | |
| ...italianPatterns, | |
| ...germanPatterns, | |
| ...japanesePatterns, | |
| ...chinesePatterns | |
| ]; | |
| for (const pattern of allPatterns) { | |
| const match = lowerText.match(pattern); | |
| if (match && match[1]) { | |
| const content = match[1].trim(); | |
| // Determine category based on content | |
| const category = this.categorizeExplicitMemory(content); | |
| memories.push({ | |
| category: category, | |
| type: "explicit_request", | |
| content: content, | |
| sourceText: text, | |
| confidence: 1.0, // High confidence for explicit requests | |
| timestamp: new Date(), | |
| character: this.selectedCharacter, | |
| isUpdate: false | |
| }); | |
| break; // Only take the first match to avoid duplicates | |
| } | |
| } | |
| return memories; | |
| } | |
| // Categorize explicit memory based on content analysis | |
| categorizeExplicitMemory(content) { | |
| const lowerContent = content.toLowerCase(); | |
| // Preference indicators | |
| if ( | |
| lowerContent.includes("j'aime") || | |
| lowerContent.includes("i like") || | |
| lowerContent.includes("j'adore") || | |
| lowerContent.includes("i love") || | |
| lowerContent.includes("je préfère") || | |
| lowerContent.includes("i prefer") || | |
| lowerContent.includes("je déteste") || | |
| lowerContent.includes("i hate") | |
| ) { | |
| return "preferences"; | |
| } | |
| // Personal information indicators | |
| if ( | |
| lowerContent.includes("je m'appelle") || | |
| lowerContent.includes("my name is") || | |
| (lowerContent.includes("j'ai") && lowerContent.includes("ans")) || | |
| lowerContent.includes("years old") || | |
| lowerContent.includes("j'habite") || | |
| lowerContent.includes("i live") | |
| ) { | |
| return "personal"; | |
| } | |
| // Relationship indicators | |
| if ( | |
| lowerContent.includes("ma femme") || | |
| lowerContent.includes("my wife") || | |
| lowerContent.includes("mon mari") || | |
| lowerContent.includes("my husband") || | |
| lowerContent.includes("mon ami") || | |
| lowerContent.includes("my friend") || | |
| lowerContent.includes("ma famille") || | |
| lowerContent.includes("my family") | |
| ) { | |
| return "relationships"; | |
| } | |
| // Activity indicators | |
| if ( | |
| lowerContent.includes("je joue") || | |
| lowerContent.includes("i play") || | |
| lowerContent.includes("je pratique") || | |
| lowerContent.includes("i practice") || | |
| lowerContent.includes("mon hobby") || | |
| lowerContent.includes("my hobby") | |
| ) { | |
| return "activities"; | |
| } | |
| // Goal indicators | |
| if ( | |
| lowerContent.includes("je veux") || | |
| lowerContent.includes("i want") || | |
| lowerContent.includes("mon objectif") || | |
| lowerContent.includes("my goal") || | |
| lowerContent.includes("j'apprends") || | |
| lowerContent.includes("i'm learning") | |
| ) { | |
| return "goals"; | |
| } | |
| // Default to preferences for most explicit requests | |
| return "preferences"; | |
| } | |
| // Check if content is too generic to be useful | |
| isGenericContent(content) { | |
| const genericWords = ["yes", "no", "ok", "okay", "sure", "thanks", "hello", "hi", "bye"]; | |
| return genericWords.includes(content.toLowerCase()) || content.length < 2; | |
| } | |
| // Calculate confidence based on context and pattern strength | |
| calculateExtractionConfidence(match, fullText) { | |
| let confidence = 0.6; // Base confidence | |
| // Boost confidence for explicit statements | |
| const lower = fullText.toLowerCase(); | |
| if ( | |
| lower.includes("my name is") || | |
| lower.includes("i am called") || | |
| lower.includes("je m'appelle") || | |
| lower.includes("mon nom est") || | |
| lower.includes("je me prénomme") || | |
| lower.includes("je me nomme") || | |
| lower.includes("me llamo") || | |
| lower.includes("mi nombre es") || | |
| lower.includes("mi chiamo") || | |
| lower.includes("il mio nome è") || | |
| lower.includes("ich heiße") || | |
| lower.includes("mein name ist") || | |
| lower.includes("と申します") || | |
| lower.includes("私の名前は") || | |
| lower.includes("我叫") || | |
| lower.includes("我的名字是") | |
| ) { | |
| confidence += 0.3; | |
| } | |
| // Boost for longer, more specific content | |
| if (match[1] && match[1].trim().length > 10) { | |
| confidence += 0.1; | |
| } | |
| // Reduce confidence for uncertain language | |
| if (fullText.includes("maybe") || fullText.includes("perhaps") || fullText.includes("might")) { | |
| confidence -= 0.2; | |
| } | |
| return Math.min(1.0, Math.max(0.1, confidence)); | |
| } | |
| // Check if this is an update to existing memory rather than new info | |
| async isMemoryUpdate(category, content, existingMemories) { | |
| const categoryMemories = existingMemories.filter(m => m.category === category); | |
| for (const memory of categoryMemories) { | |
| const similarity = this.calculateSimilarity(memory.content, content); | |
| if (similarity > 0.3) { | |
| // Lower threshold for updates | |
| return true; | |
| } | |
| } | |
| return false; | |
| } | |
| // Detect natural expressions that patterns might miss | |
| async detectNaturalExpressions(text, existingMemories) { | |
| const naturalMemories = []; | |
| const lowerText = text.toLowerCase(); | |
| // Detect name mentions in natural context (multilingual) | |
| const namePatterns = [ | |
| // English | |
| /call me (\w+)/i, | |
| /(\w+) here[,.]?/i, | |
| /this is (\w+)/i, | |
| /(\w+) speaking/i, | |
| // French | |
| /appelle-?moi (\w+)/i, | |
| /on m'appelle (\w+)/i, | |
| /c'est (\w+)/i, | |
| // Spanish | |
| /llámame (\w+)/i, | |
| /me llaman (\w+)/i, | |
| /soy (\w+)/i, | |
| // Italian | |
| /chiamami (\w+)/i, | |
| /mi chiamano (\w+)/i, | |
| /sono (\w+)/i, | |
| // German | |
| /nenn mich (\w+)/i, | |
| /man nennt mich (\w+)/i, | |
| /ich bin (\w+)/i, | |
| // Japanese | |
| /(?:私は)?(\w+)です/i, | |
| // Chinese | |
| /我是(\w+)/i, | |
| /叫我(\w+)/i | |
| ]; | |
| for (const pattern of namePatterns) { | |
| const match = lowerText.match(pattern); | |
| if (match && match[1] && match[1].length > 1) { | |
| const name = match[1].trim(); | |
| // Skip if too generic | |
| if (!this.isGenericContent(name) && !this.isCommonWord(name)) { | |
| naturalMemories.push({ | |
| category: "personal", | |
| type: "auto_extracted", | |
| content: name, | |
| sourceText: text, | |
| confidence: 0.7, | |
| timestamp: new Date(), | |
| character: this.selectedCharacter | |
| }); | |
| } | |
| } | |
| } | |
| return naturalMemories; | |
| } | |
| // Check if word is too common to be a name | |
| isCommonWord(word, language = "en") { | |
| // Use existing constants if available | |
| if (window.KIMI_COMMON_WORDS && window.KIMI_COMMON_WORDS[language]) { | |
| return window.KIMI_COMMON_WORDS[language].includes(word.toLowerCase()); | |
| } | |
| // Fallback to original English list | |
| const commonWords = [ | |
| "the", | |
| "and", | |
| "for", | |
| "are", | |
| "but", | |
| "not", | |
| "you", | |
| "all", | |
| "can", | |
| "had", | |
| "her", | |
| "was", | |
| "one", | |
| "our", | |
| "out", | |
| "day", | |
| "get", | |
| "has", | |
| "him", | |
| "his", | |
| "how", | |
| "man", | |
| "new", | |
| "now", | |
| "old", | |
| "see", | |
| "two", | |
| "way", | |
| "who", | |
| "boy", | |
| "did", | |
| "its", | |
| "let", | |
| "put", | |
| "say", | |
| "she", | |
| "too", | |
| "use" | |
| ]; | |
| return commonWords.includes(word.toLowerCase()); | |
| } | |
| // MANUAL MEMORY MANAGEMENT | |
| async addMemory(memoryData) { | |
| if (!this.db || !this.memoryEnabled) return; | |
| try { | |
| // Check for duplicates with intelligent merging | |
| const existing = await this.findSimilarMemory(memoryData); | |
| if (existing) { | |
| // Intelligent merge strategy | |
| return await this.mergeMemories(existing, memoryData); | |
| } | |
| // Add memory with metadata (let DB auto-generate ID) | |
| const now = new Date(); | |
| const memory = { | |
| category: memoryData.category || "personal", | |
| type: memoryData.type || "manual", | |
| content: memoryData.content, | |
| sourceText: memoryData.sourceText || "", | |
| confidence: memoryData.confidence || 1.0, | |
| timestamp: memoryData.timestamp || now, | |
| character: memoryData.character || this.selectedCharacter, | |
| isActive: true, | |
| tags: [...new Set([...(memoryData.tags || []), ...this.deriveMemoryTags(memoryData)])], | |
| lastModified: now, | |
| createdAt: now, | |
| lastAccess: now, | |
| accessCount: 0, | |
| importance: this.calculateImportance(memoryData) | |
| }; | |
| if (this.db.db.memories) { | |
| const id = await this.db.db.memories.add(memory); | |
| memory.id = id; // Store the auto-generated ID | |
| console.log(`Memory added with ID: ${id}`); | |
| } | |
| // Cleanup old memories if we exceed limit | |
| await this.cleanupOldMemories(); | |
| // Notify LLM system to refresh context | |
| this.notifyLLMContextUpdate(); | |
| return memory; | |
| } catch (error) { | |
| console.error("Error adding memory:", error); | |
| } | |
| } | |
| // Intelligent memory merging | |
| async mergeMemories(existingMemory, newMemoryData) { | |
| try { | |
| // Determine merge strategy based on content and confidence | |
| const strategy = this.determineMergeStrategy(existingMemory, newMemoryData); | |
| let mergedContent = existingMemory.content; | |
| let mergedConfidence = existingMemory.confidence; | |
| let mergedTags = [...(existingMemory.tags || [])]; | |
| switch (strategy) { | |
| case "update_content": | |
| // New information is more confident/recent | |
| mergedContent = newMemoryData.content; | |
| mergedConfidence = Math.max(existingMemory.confidence, newMemoryData.confidence || 0.8); | |
| break; | |
| case "merge_content": | |
| // Combine information intelligently | |
| if ( | |
| existingMemory.category === "personal" && | |
| this.areRelatedNames(existingMemory.content, newMemoryData.content) | |
| ) { | |
| // Handle name variants | |
| mergedContent = this.mergeNames(existingMemory.content, newMemoryData.content); | |
| } else { | |
| // General merge - keep most specific | |
| mergedContent = | |
| newMemoryData.content.length > existingMemory.content.length | |
| ? newMemoryData.content | |
| : existingMemory.content; | |
| } | |
| mergedConfidence = (existingMemory.confidence + (newMemoryData.confidence || 0.8)) / 2; | |
| break; | |
| case "add_variant": | |
| // Store as variant/alias | |
| mergedTags.push(`alias:${newMemoryData.content}`); | |
| break; | |
| case "boost_confidence": | |
| // Same content, boost confidence | |
| mergedConfidence = Math.min(1.0, existingMemory.confidence + 0.1); | |
| break; | |
| } | |
| // Update existing memory | |
| const updatedMemory = { | |
| ...existingMemory, | |
| content: mergedContent, | |
| confidence: mergedConfidence, | |
| tags: [...new Set([...mergedTags, ...this.deriveMemoryTags(newMemoryData)])], // Remove duplicates | |
| lastModified: new Date(), | |
| accessCount: (existingMemory.accessCount || 0) + 1, | |
| importance: Math.max(existingMemory.importance || 0.5, this.calculateImportance(newMemoryData)) | |
| }; | |
| await this.updateMemory(existingMemory.id, updatedMemory); | |
| return updatedMemory; | |
| } catch (error) { | |
| console.error("Error merging memories:", error); | |
| return existingMemory; | |
| } | |
| } | |
| // Determine how to merge two related memories | |
| determineMergeStrategy(existing, newData) { | |
| const similarity = this.calculateSimilarity(existing.content, newData.content); | |
| const newConfidence = newData.confidence || 0.8; | |
| // If very similar content but new has higher confidence | |
| if (similarity > 0.9 && newConfidence > existing.confidence) { | |
| return "boost_confidence"; | |
| } | |
| // If moderately similar, decide based on specificity and recency | |
| if (similarity > 0.7) { | |
| if (newData.content.length > existing.content.length * 1.5) { | |
| return "update_content"; // New is more detailed | |
| } else { | |
| return "merge_content"; | |
| } | |
| } | |
| // For names, handle as variants | |
| if (existing.category === "personal" && this.areRelatedNames(existing.content, newData.content)) { | |
| return "add_variant"; | |
| } | |
| // Default to merging | |
| return "merge_content"; | |
| } | |
| // Merge name variants intelligently | |
| mergeNames(name1, name2) { | |
| // Keep the longest/most formal version as primary | |
| if (name1.length > name2.length) { | |
| return name1; | |
| } else if (name2.length > name1.length) { | |
| return name2; | |
| } | |
| // If same length, keep the first one | |
| return name1; | |
| } | |
| // Calculate importance of memory for prioritization | |
| calculateImportance(memoryData) { | |
| let importance = 0.5; // Base importance | |
| // Category base weights | |
| const categoryWeights = { | |
| important: 1.0, | |
| personal: 0.9, | |
| relationships: 0.85, | |
| goals: 0.75, | |
| experiences: 0.65, | |
| preferences: 0.6, | |
| activities: 0.5 | |
| }; | |
| importance = categoryWeights[memoryData.category] || 0.5; | |
| const content = (memoryData.content || "").toLowerCase(); | |
| const tags = new Set([...(memoryData.tags || []), ...this.deriveMemoryTags(memoryData)]); | |
| // Heuristic boosts for meaningful relationship milestones and commitments | |
| const milestoneTags = [ | |
| "relationship:first_meet", | |
| "relationship:first_date", | |
| "relationship:first_kiss", | |
| "relationship:anniversary", | |
| "relationship:moved_in", | |
| "relationship:engaged", | |
| "relationship:married", | |
| "relationship:breakup" | |
| ]; | |
| if ([...tags].some(t => milestoneTags.includes(t))) importance += 0.15; | |
| // Boundaries and consent are high priority to remember | |
| if ([...tags].some(t => t.startsWith("boundary:"))) importance += 0.15; | |
| // Preferences tied to strong like/dislike | |
| if ( | |
| content.includes("i love") || | |
| content.includes("j'adore") || | |
| content.includes("i hate") || | |
| content.includes("je déteste") | |
| ) { | |
| importance += 0.05; | |
| } | |
| // Temporal cues: future commitments or dates | |
| if (/(\bnext\b|\btomorrow\b|\bce soir\b|\bdemain\b|\bmañana\b|\bdomani\b|\bmorgen\b)/i.test(content)) { | |
| importance += 0.05; | |
| } | |
| // Longer details and high confidence | |
| if (memoryData.content && memoryData.content.length > 24) importance += 0.05; | |
| if (memoryData.confidence && memoryData.confidence > 0.9) importance += 0.05; | |
| // Round to two decimals to avoid floating point artifacts | |
| return Math.min(1.0, Math.round(importance * 100) / 100); | |
| } | |
| // Derive semantic tags from memory content to assist prioritization and merging | |
| deriveMemoryTags(memoryData) { | |
| const tags = []; | |
| const text = (memoryData.content || "").toLowerCase(); | |
| const category = memoryData.category || ""; | |
| // Relationship status and milestones | |
| if (/(single|célibataire|soltero|single|ledig)/i.test(text)) tags.push("relationship:status_single"); | |
| if (/(in a relationship|en couple|together|ensemble|pareja|coppia|beziehung)/i.test(text)) | |
| tags.push("relationship:status_in_relationship"); | |
| if (/(engaged|fiancé|fiancée|promis|promised|verlobt)/i.test(text)) tags.push("relationship:status_engaged"); | |
| if (/(married|marié|mariée|casado|sposato|verheiratet)/i.test(text)) tags.push("relationship:status_married"); | |
| if (/(broke up|rupture|separated|separado|separati|getrennt)/i.test(text)) tags.push("relationship:breakup"); | |
| if (/(first date|premier rendez-vous|primera cita|primo appuntamento)/i.test(text)) tags.push("relationship:first_date"); | |
| if (/(first kiss|premier baiser|primer beso|primo bacio)/i.test(text)) tags.push("relationship:first_kiss"); | |
| if (/(anniversary|anniversaire|aniversario|anniversario|jahrestag)/i.test(text)) tags.push("relationship:anniversary"); | |
| if (/(moved in together|emménagé ensemble|mudamos juntos|trasferiti insieme|zusammen eingezogen)/i.test(text)) | |
| tags.push("relationship:moved_in"); | |
| if (/(met at|rencontré à|conocimos en|conosciuti a|kennengelernt)/i.test(text)) tags.push("relationship:first_meet"); | |
| // Boundaries and consent (keep generic and non-graphic) | |
| if (/(i don't like|je n'aime pas|no me gusta|non mi piace|ich mag nicht)\s+[^,.!?]+/i.test(text)) | |
| tags.push("boundary:dislike"); | |
| if (/(i prefer|je préfère|prefiero|preferisco|ich bevorzuge)\s+[^,.!?]+/i.test(text)) tags.push("boundary:preference"); | |
| if (/(no|pas)\s+(?:kissing|baiser|beso|bacio|küssen)/i.test(text)) tags.push("boundary:limit"); | |
| if (/(consent|consentement|consentimiento|consenso|einwilligung)/i.test(text)) tags.push("boundary:consent"); | |
| // Time-related tags | |
| if (/(today|ce jour|hoy|oggi|heute|今日)/i.test(text)) tags.push("time:today"); | |
| if (/(tomorrow|demain|mañana|domani|morgen|明日)/i.test(text)) tags.push("time:tomorrow"); | |
| if (/(next week|semaine prochaine|la próxima semana|la prossima settimana|nächste woche)/i.test(text)) | |
| tags.push("time:next_week"); | |
| // Category-specific hints | |
| if (category === "preferences") tags.push("type:preference"); | |
| if (category === "personal") tags.push("type:personal"); | |
| if (category === "relationships") tags.push("type:relationship"); | |
| if (category === "experiences") tags.push("type:experience"); | |
| if (category === "goals") tags.push("type:goal"); | |
| if (category === "important") tags.push("type:important"); | |
| return tags; | |
| } | |
| async updateMemory(memoryId, updateData) { | |
| if (!this.db) return false; | |
| try { | |
| // Ensure memoryId is the correct type | |
| const numericId = typeof memoryId === "string" ? parseInt(memoryId) : memoryId; | |
| // Vérifier d'abord que la mémoire existe | |
| const existingMemory = await this.db.db.memories.get(numericId); | |
| if (!existingMemory) { | |
| console.error(`❌ Memory with ID ${numericId} not found in database`); | |
| return false; | |
| } | |
| console.log(`🔄 Updating memory ${numericId}:`, { existing: existingMemory, update: updateData }); | |
| const update = { | |
| ...updateData, | |
| lastModified: new Date() | |
| }; | |
| if (this.db.db.memories) { | |
| const result = await this.db.db.memories.update(numericId, update); | |
| console.log(`Memory update result for ID ${numericId}:`, result); | |
| if (result > 0) { | |
| console.log("✅ Memory updated successfully"); | |
| // Notify LLM system to refresh context | |
| this.notifyLLMContextUpdate(); | |
| return true; | |
| } else { | |
| console.error("❌ Memory update failed - no rows affected"); | |
| return false; | |
| } | |
| } | |
| } catch (error) { | |
| console.error("Error updating memory:", error, { memoryId, updateData }); | |
| return false; | |
| } | |
| } | |
| async deleteMemory(memoryId) { | |
| if (!this.db) return false; | |
| try { | |
| // Ensure memoryId is the correct type | |
| const numericId = typeof memoryId === "string" ? parseInt(memoryId) : memoryId; | |
| if (this.db.db.memories) { | |
| const result = await this.db.db.memories.delete(numericId); | |
| console.log(`Memory delete result for ID ${numericId}:`, result); | |
| // Notify LLM system to refresh context | |
| if (result) { | |
| this.notifyLLMContextUpdate(); | |
| } | |
| return result; | |
| } | |
| } catch (error) { | |
| console.error("Error deleting memory:", error, { memoryId }); | |
| return false; | |
| } | |
| } | |
| notifyLLMContextUpdate() { | |
| // Debounce context updates to avoid excessive calls | |
| if (this.contextUpdateTimeout) { | |
| clearTimeout(this.contextUpdateTimeout); | |
| } | |
| this.contextUpdateTimeout = setTimeout(() => { | |
| if (window.kimiLLM && typeof window.kimiLLM.refreshMemoryContext === "function") { | |
| window.kimiLLM.refreshMemoryContext(); | |
| } | |
| }, 500); | |
| } | |
| async getMemoriesByCategory(category, character = null) { | |
| if (!this.db) return []; | |
| try { | |
| character = character || this.selectedCharacter; | |
| if (this.db.db.memories) { | |
| return await this.db.db.memories | |
| .where("[character+category]") | |
| .equals([character, category]) | |
| .and(m => m.isActive) | |
| .reverse() | |
| .sortBy("timestamp"); | |
| } | |
| } catch (error) { | |
| console.error("Error getting memories by category:", error); | |
| return []; | |
| } | |
| } | |
| async getAllMemories(character = null) { | |
| if (!this.db) return []; | |
| try { | |
| character = character || this.selectedCharacter; | |
| if (this.db.db.memories) { | |
| const memories = await this.db.db.memories | |
| .where("character") | |
| .equals(character) | |
| .and(m => m.isActive) | |
| .reverse() | |
| .sortBy("timestamp"); | |
| console.log(`Retrieved ${memories.length} memories for character: ${character}`); | |
| return memories; | |
| } | |
| } catch (error) { | |
| console.error("Error getting all memories:", error); | |
| return []; | |
| } | |
| } | |
| async findSimilarMemory(memoryData) { | |
| if (!this.db) return null; | |
| try { | |
| const memories = await this.getMemoriesByCategory(memoryData.category); | |
| // Enhanced similarity check with multiple criteria | |
| for (const memory of memories) { | |
| const contentSimilarity = this.calculateSimilarity(memory.content, memoryData.content); | |
| // Different thresholds based on category | |
| let threshold = 0.8; | |
| if (memoryData.category === "personal") { | |
| threshold = 0.6; // Names and personal info can vary more | |
| } else if (memoryData.category === "preferences") { | |
| threshold = 0.7; // Preferences can be expressed differently | |
| } | |
| if (contentSimilarity > threshold) { | |
| return memory; | |
| } | |
| // Special handling for names (check if one is contained in the other) | |
| if (memoryData.category === "personal" && this.areRelatedNames(memory.content, memoryData.content)) { | |
| return memory; | |
| } | |
| } | |
| } catch (error) { | |
| console.error("Error finding similar memory:", error); | |
| } | |
| return null; | |
| } | |
| // Check if two names are related (nicknames, variants, etc.) | |
| areRelatedNames(name1, name2) { | |
| const n1 = name1.toLowerCase().trim(); | |
| const n2 = name2.toLowerCase().trim(); | |
| // Exact match | |
| if (n1 === n2) return true; | |
| // One contains the other (Jean-Pierre vs Jean) | |
| if (n1.includes(n2) || n2.includes(n1)) return true; | |
| // Common nickname patterns | |
| const nicknames = { | |
| jean: ["jp", "jeannot"], | |
| pierre: ["pete", "pietro"], | |
| marie: ["mary", "maria"], | |
| michael: ["mike", "mick"], | |
| william: ["bill", "will", "willy"], | |
| robert: ["bob", "rob", "bobby"], | |
| richard: ["rick", "dick", "richie"], | |
| thomas: ["tom", "tommy"], | |
| christopher: ["chris", "kit"], | |
| anthony: ["tony", "ant"] | |
| }; | |
| for (const [full, nicks] of Object.entries(nicknames)) { | |
| if ((n1 === full && nicks.includes(n2)) || (n2 === full && nicks.includes(n1))) { | |
| return true; | |
| } | |
| } | |
| return false; | |
| } | |
| calculateSimilarity(text1, text2) { | |
| // Enhanced similarity calculation | |
| const words1 = text1 | |
| .toLowerCase() | |
| .split(/\s+/) | |
| .filter(w => w.length > 2); | |
| const words2 = text2 | |
| .toLowerCase() | |
| .split(/\s+/) | |
| .filter(w => w.length > 2); | |
| if (words1.length === 0 || words2.length === 0) { | |
| return text1.toLowerCase() === text2.toLowerCase() ? 1 : 0; | |
| } | |
| const intersection = words1.filter(word => words2.includes(word)); | |
| const union = [...new Set([...words1, ...words2])]; | |
| let similarity = intersection.length / union.length; | |
| // Boost similarity for exact substring matches | |
| if (text1.toLowerCase().includes(text2.toLowerCase()) || text2.toLowerCase().includes(text1.toLowerCase())) { | |
| similarity += 0.2; | |
| } | |
| return Math.min(1.0, similarity); | |
| } | |
| async cleanupOldMemories() { | |
| if (!this.db) return; | |
| try { | |
| // Retrieve all active memories for the current character | |
| const memories = await this.getAllMemories(); | |
| // If the number of memories exceeds the limit (this.maxMemoryEntries), | |
| // delete the least important/oldest ones to keep only the most relevant. | |
| if (memories.length > this.maxMemoryEntries) { | |
| // Sort by importance (confidence) and recency (timestamp) | |
| memories.sort((a, b) => { | |
| // Score = confidence * age (the higher the score, the less priority the memory has) | |
| const scoreA = a.confidence * (Date.now() - new Date(a.timestamp).getTime()); | |
| const scoreB = b.confidence * (Date.now() - new Date(b.timestamp).getTime()); | |
| return scoreB - scoreA; | |
| }); | |
| // Delete all memories beyond the limit | |
| const toDelete = memories.slice(this.maxMemoryEntries); | |
| for (const memory of toDelete) { | |
| await this.deleteMemory(memory.id); | |
| } | |
| } | |
| } catch (error) { | |
| console.error("Error cleaning up old memories:", error); | |
| } | |
| } | |
| // MEMORY RETRIEVAL FOR LLM | |
| async getRelevantMemories(context = "", limit = 10) { | |
| if (!this.memoryEnabled) return []; | |
| try { | |
| const allMemories = await this.getAllMemories(); | |
| if (allMemories.length === 0) return []; | |
| if (!context) { | |
| // Return most important and recent memories | |
| return this.selectMostImportantMemories(allMemories, limit); | |
| } | |
| // Score memories based on relevance to context | |
| const scoredMemories = allMemories.map(memory => ({ | |
| ...memory, | |
| relevanceScore: this.calculateRelevance(memory, context) | |
| })); | |
| // Sort by relevance and return top results | |
| scoredMemories.sort((a, b) => b.relevanceScore - a.relevanceScore); | |
| // Filter out very low relevance memories | |
| const relevantMemories = scoredMemories.filter(m => m.relevanceScore > 0.1); | |
| return relevantMemories.slice(0, limit); | |
| } catch (error) { | |
| console.error("Error getting relevant memories:", error); | |
| return []; | |
| } | |
| } | |
| // Select most important memories when no context is provided | |
| selectMostImportantMemories(memories, limit) { | |
| // Score by importance, recency, and access count | |
| const scoredMemories = memories.map(memory => { | |
| let score = memory.importance || 0.5; | |
| // Boost recent memories | |
| const daysSinceCreation = (Date.now() - new Date(memory.timestamp)) / (1000 * 60 * 60 * 24); | |
| score += Math.max(0, (7 - daysSinceCreation) / 7) * 0.2; // Recent boost | |
| // Boost frequently accessed memories | |
| const accessCount = memory.accessCount || 0; | |
| score += Math.min(accessCount / 10, 0.2); // Access boost | |
| // Boost high confidence memories | |
| score += (memory.confidence || 0.5) * 0.1; | |
| return { ...memory, importanceScore: score }; | |
| }); | |
| scoredMemories.sort((a, b) => b.importanceScore - a.importanceScore); | |
| return scoredMemories.slice(0, limit); | |
| } | |
| calculateRelevance(memory, context) { | |
| const contextWords = context | |
| .toLowerCase() | |
| .split(/\s+/) | |
| .filter(w => w.length > 2); | |
| const memoryWords = memory.content | |
| .toLowerCase() | |
| .split(/\s+/) | |
| .filter(w => w.length > 2); | |
| let score = 0; | |
| // Enhanced content similarity with keyword matching | |
| score += this.calculateSimilarity(memory.content, context) * 0.4; | |
| // Keyword matching bonus | |
| let keywordMatches = 0; | |
| for (const word of contextWords) { | |
| if (memoryWords.includes(word)) { | |
| keywordMatches++; | |
| } | |
| } | |
| if (contextWords.length > 0) { | |
| score += (keywordMatches / contextWords.length) * 0.3; | |
| } | |
| // Category relevance bonus based on context | |
| score += this.getCategoryRelevance(memory.category, context) * 0.1; | |
| // Recent memories get bonus for current conversation | |
| const daysSinceCreation = (Date.now() - new Date(memory.timestamp)) / (1000 * 60 * 60 * 24); | |
| score += Math.max(0, (30 - daysSinceCreation) / 30) * 0.1; | |
| // Confidence and importance boost | |
| score += (memory.confidence || 0.5) * 0.05; | |
| score += (memory.importance || 0.5) * 0.05; | |
| return Math.min(1.0, score); | |
| } | |
| // Determine if memory category is relevant to current context | |
| getCategoryRelevance(category, context) { | |
| const contextLower = context.toLowerCase(); | |
| const categoryKeywords = { | |
| personal: [ | |
| "name", | |
| "age", | |
| "live", | |
| "work", | |
| "job", | |
| "who", | |
| "am", | |
| "myself", | |
| "appelle", | |
| "nombre", | |
| "chiamo", | |
| "heiße", | |
| "名前", | |
| "名字", | |
| "我叫" | |
| ], | |
| preferences: [ | |
| "like", | |
| "love", | |
| "hate", | |
| "prefer", | |
| "enjoy", | |
| "favorite", | |
| "dislike", | |
| "j'aime", | |
| "j'adore", | |
| "je préfère", | |
| "je déteste", | |
| "me gusta", | |
| "prefiero", | |
| "odio", | |
| "mi piace", | |
| "preferisco", | |
| "ich mag", | |
| "ich bevorzuge", | |
| "hasse" | |
| ], | |
| relationships: [ | |
| "family", | |
| "friend", | |
| "wife", | |
| "husband", | |
| "partner", | |
| "mother", | |
| "father", | |
| "girlfriend", | |
| "boyfriend", | |
| "anniversary", | |
| "date", | |
| "kiss", | |
| "move in", | |
| "famille", | |
| "ami", | |
| "copine", | |
| "copain", | |
| "anniversaire", | |
| "rendez-vous", | |
| "baiser", | |
| "emménagé", | |
| "pareja", | |
| "cita", | |
| "beso", | |
| "aniversario", | |
| "mudarnos", | |
| "fidanzata", | |
| "fidanzato", | |
| "anniversario", | |
| "bacio", | |
| "trasferiti", | |
| "freundin", | |
| "freund", | |
| "jahrestag", | |
| "kuss", | |
| "eingezogen" | |
| ], | |
| activities: [ | |
| "play", | |
| "hobby", | |
| "sport", | |
| "activity", | |
| "practice", | |
| "do", | |
| "joue", | |
| "passe-temps", | |
| "hobby", | |
| "juego", | |
| "pasatiempo", | |
| "gioco", | |
| "passatempo", | |
| "spiele", | |
| "hobby" | |
| ], | |
| goals: [ | |
| "want", | |
| "plan", | |
| "goal", | |
| "dream", | |
| "hope", | |
| "wish", | |
| "future", | |
| "veux", | |
| "objectif", | |
| "apprends", | |
| "aprendo", | |
| "voglio", | |
| "obiettivo", | |
| "lerne", | |
| "ziel" | |
| ], | |
| experiences: [ | |
| "remember", | |
| "happened", | |
| "story", | |
| "experience", | |
| "time", | |
| "we met", | |
| "first date", | |
| "first kiss", | |
| "anniversary", | |
| "rencontré", | |
| "premier rendez-vous", | |
| "premier baiser", | |
| "anniversaire", | |
| "conocimos", | |
| "primera cita", | |
| "primer beso", | |
| "aniversario", | |
| "conosciuti", | |
| "primo appuntamento", | |
| "primo bacio", | |
| "anniversario", | |
| "kennengelernt", | |
| "erstes date", | |
| "erster kuss", | |
| "jahrestag" | |
| ], | |
| important: [ | |
| "important", | |
| "remember", | |
| "special", | |
| "never forget", | |
| "important", | |
| "souvenir", | |
| "spécial", | |
| "importante", | |
| "recuerda", | |
| "importante", | |
| "ricorda", | |
| "wichtig", | |
| "erinnere" | |
| ] | |
| }; | |
| const keywords = categoryKeywords[category] || []; | |
| let relevance = 0; | |
| for (const keyword of keywords) { | |
| if (contextLower.includes(keyword)) { | |
| relevance += 0.2; | |
| } | |
| } | |
| return Math.min(1.0, relevance); | |
| } | |
| // Update access count when memory is used | |
| async recordMemoryAccess(memoryId) { | |
| try { | |
| const memory = await this.db.db.memories.get(memoryId); | |
| if (memory) { | |
| memory.accessCount = (memory.accessCount || 0) + 1; | |
| memory.lastAccess = new Date(); | |
| await this.db.db.memories.put(memory); | |
| } | |
| } catch (error) { | |
| console.error("Error recording memory access:", error); | |
| } | |
| } | |
| // ===== MEMORY SCORING & RANKING ===== | |
| scoreMemory(memory) { | |
| // Factors: importance (0-1), recency, frequency, confidence | |
| const now = Date.now(); | |
| const created = memory.createdAt | |
| ? new Date(memory.createdAt).getTime() | |
| : memory.timestamp | |
| ? new Date(memory.timestamp).getTime() | |
| : now; | |
| const lastAccess = memory.lastAccess ? new Date(memory.lastAccess).getTime() : created; | |
| const ageMs = Math.max(1, now - created); | |
| const sinceLastAccessMs = Math.max(1, now - lastAccess); | |
| // Recency: exponential decay | |
| const recency = Math.exp(-sinceLastAccessMs / (1000 * 60 * 60 * 24 * 14)); // 14-day half-life approx | |
| const freshness = Math.exp(-ageMs / (1000 * 60 * 60 * 24 * 60)); // 60-day aging | |
| const freq = Math.log10((memory.accessCount || 0) + 1) / Math.log10(50); // normalized frequency (cap ~50) | |
| const importance = typeof memory.importance === "number" ? memory.importance : 0.5; | |
| const confidence = typeof memory.confidence === "number" ? memory.confidence : 0.5; | |
| // Weighted sum | |
| const score = importance * 0.35 + recency * 0.2 + freq * 0.15 + confidence * 0.2 + freshness * 0.1; | |
| return Number(score.toFixed(6)); | |
| } | |
| async getRankedMemories(contextText = "", limit = 7) { | |
| const all = await this.getAllMemories(); | |
| if (!all.length) return []; | |
| // Optional basic context relevance boost | |
| const ctxLower = (contextText || "").toLowerCase(); | |
| return all | |
| .map(m => { | |
| let baseScore = this.scoreMemory(m); | |
| if (ctxLower && m.content && ctxLower.includes(m.content.toLowerCase().split(" ")[0])) { | |
| baseScore += 0.05; // tiny relevance boost | |
| } | |
| return { memory: m, score: baseScore }; | |
| }) | |
| .sort((a, b) => b.score - a.score) | |
| .slice(0, limit) | |
| .map(r => r.memory); | |
| } | |
| // MEMORY STATISTICS | |
| async getMemoryStats() { | |
| try { | |
| const memories = await this.getAllMemories(); | |
| const stats = { | |
| total: memories.length, | |
| byCategory: {}, | |
| averageConfidence: 0, | |
| oldestMemory: null, | |
| newestMemory: null | |
| }; | |
| if (memories.length > 0) { | |
| // Category breakdown | |
| for (const memory of memories) { | |
| stats.byCategory[memory.category] = (stats.byCategory[memory.category] || 0) + 1; | |
| } | |
| // Average confidence | |
| stats.averageConfidence = memories.reduce((sum, m) => sum + m.confidence, 0) / memories.length; | |
| // Oldest and newest | |
| const sortedByDate = [...memories].sort((a, b) => new Date(a.timestamp) - new Date(b.timestamp)); | |
| stats.oldestMemory = sortedByDate[0]; | |
| stats.newestMemory = sortedByDate[sortedByDate.length - 1]; | |
| } | |
| return stats; | |
| } catch (error) { | |
| console.error("Error getting memory stats:", error); | |
| return { total: 0, byCategory: {}, averageConfidence: 0 }; | |
| } | |
| } | |
| // MEMORY TOGGLE | |
| async toggleMemorySystem(enabled) { | |
| this.memoryEnabled = enabled; | |
| if (this.db) { | |
| await this.db.setPreference("memorySystemEnabled", enabled); | |
| } | |
| } | |
| // EXPORT/IMPORT MEMORIES | |
| async exportMemories() { | |
| try { | |
| const memories = await this.getAllMemories(); | |
| return { | |
| exportDate: new Date().toISOString(), | |
| character: this.selectedCharacter, | |
| memories: memories, | |
| version: "1.0" | |
| }; | |
| } catch (error) { | |
| console.error("Error exporting memories:", error); | |
| return null; | |
| } | |
| } | |
| async importMemories(importData) { | |
| if (!importData || !importData.memories) return false; | |
| try { | |
| for (const memory of importData.memories) { | |
| await this.addMemory({ | |
| ...memory, | |
| type: "imported", | |
| character: this.selectedCharacter | |
| }); | |
| } | |
| return true; | |
| } catch (error) { | |
| console.error("Error importing memories:", error); | |
| return false; | |
| } | |
| } | |
| // MIGRATION UTILITIES | |
| async migrateIncompatibleIDs() { | |
| if (!this.db) return false; | |
| try { | |
| console.log("🔧 Début de la migration des IDs incompatibles..."); | |
| // Récupérer toutes les mémoires | |
| const allMemories = await this.db.db.memories.toArray(); | |
| console.log(`📊 ${allMemories.length} mémoires trouvées`); | |
| const incompatibleMemories = allMemories.filter(memory => { | |
| // Les IDs auto-increment sont des entiers séquentiels (1, 2, 3...) | |
| // Les anciens IDs manuels sont des nombres très grands (timestamps) | |
| return memory.id > 10000; // Seuil arbitraire pour détecter les anciens IDs | |
| }); | |
| if (incompatibleMemories.length === 0) { | |
| console.log("✅ Aucune migration nécessaire"); | |
| return true; | |
| } | |
| console.log(`🔄 Migration de ${incompatibleMemories.length} mémoires avec IDs incompatibles`); | |
| // Sauvegarder les données avant suppression | |
| const dataToMigrate = incompatibleMemories.map(memory => { | |
| const { id, ...memoryData } = memory; // Enlever l'ancien ID | |
| return memoryData; | |
| }); | |
| // Supprimer les anciennes entrées | |
| await this.db.db.memories.bulkDelete(incompatibleMemories.map(m => m.id)); | |
| // Réinsérer avec de nouveaux IDs auto-générés | |
| const newIds = await this.db.db.memories.bulkAdd(dataToMigrate); | |
| console.log(`✅ Migration terminée. Nouveaux IDs:`, newIds); | |
| return true; | |
| } catch (error) { | |
| console.error("❌ Erreur lors de la migration:", error); | |
| return false; | |
| } | |
| } | |
| } | |
| window.KimiMemorySystem = KimiMemorySystem; | |
| export default KimiMemorySystem; | |
| window.KimiMemorySystem = KimiMemorySystem; | |