Class: Llmemory::LongTerm::GraphBased::Memory
- Inherits:
-
Object
- Object
- Llmemory::LongTerm::GraphBased::Memory
- Includes:
- MemoryModule
- Defined in:
- lib/llmemory/long_term/graph_based/memory.rb
Instance Attribute Summary collapse
-
#user_id ⇒ Object
readonly
Returns the value of attribute user_id.
Instance Method Summary collapse
-
#forget(ids:, reason: nil) ⇒ Object
Forgetting a knowledge graph is not a simple delete-by-id: edges are soft-archived and nodes can be left orphaned.
-
#initialize(user_id:, storage: nil, vector_store: nil, llm: nil, extractor: nil) ⇒ Memory
constructor
A new instance of Memory.
- #list(user_id: nil, limit: nil) ⇒ Object
- #memorize(conversation_text) ⇒ Object
- #retrieve(query, top_k: 10) ⇒ Object
- #search_candidates(query, user_id: nil, top_k: 20) ⇒ Object
- #stats(user_id: nil) ⇒ Object
- #storage ⇒ Object
-
#write(payload, **_meta) ⇒ Object
— MemoryModule uniform interface —.
Methods included from MemoryModule
Constructor Details
#initialize(user_id:, storage: nil, vector_store: nil, llm: nil, extractor: nil) ⇒ Memory
Returns a new instance of Memory.
17 18 19 20 21 22 23 24 25 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 17 def initialize(user_id:, storage: nil, vector_store: nil, llm: nil, extractor: nil) @user_id = user_id @graph_storage = storage || Storages.build @kg = KnowledgeGraph.new(user_id: user_id, storage: @graph_storage) @conflict_resolver = ConflictResolver.new(@kg) @vector_store = vector_store || build_vector_store @llm = llm || Llmemory::LLM.client @extractor = extractor || Llmemory::Extractors::EntityRelationExtractor.new(llm: @llm) end |
Instance Attribute Details
#user_id ⇒ Object (readonly)
Returns the value of attribute user_id.
103 104 105 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 103 def user_id @user_id end |
Instance Method Details
#forget(ids:, reason: nil) ⇒ Object
Forgetting a knowledge graph is not a simple delete-by-id: edges are soft-archived and nodes can be left orphaned. A dedicated graph edge/node lifecycle (with orphan handling) is a deliberate follow-up.
127 128 129 130 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 127 def forget(ids:, reason: nil) raise NotImplementedError, "Graph forget is not implemented yet; edge/node lifecycle (archival + orphan handling) is a follow-up." end |
#list(user_id: nil, limit: nil) ⇒ Object
115 116 117 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 115 def list(user_id: nil, limit: nil) @graph_storage.list_nodes(user_id || @user_id, limit: limit) end |
#memorize(conversation_text) ⇒ Object
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 27 def memorize(conversation_text) text = Llmemory.configuration.noise_filter_enabled ? NoiseFilter.filter?(conversation_text) : conversation_text.to_s return true if text.strip.empty? data = @extractor.extract(text) rescue { entities: [], relations: [] } data = { entities: [], relations: [] } unless data.is_a?(Hash) entities = Array(data[:entities] || data["entities"]) relations = Array(data[:relations] || data["relations"]) return true if entities.empty? && relations.empty? provenance = Llmemory::Provenance.from_text_fingerprint(text, method: "entity_relation_extraction") name_to_id = {} entities.each do |e| next unless e.is_a?(Hash) entity_type = e[:type] || e["type"] || "concept" name = e[:name] || e["name"] next if name.nil? || name.to_s.strip.empty? id = @kg.add_node(entity_type: entity_type, name: name.to_s.strip, properties: { "provenance" => provenance }) name_to_id[name.to_s.strip] ||= id end relations.each do |r| next unless r.is_a?(Hash) subject = (r[:subject] || r["subject"]).to_s.strip predicate = (r[:predicate] || r["predicate"]).to_s.strip object = (r[:object] || r["object"]).to_s.strip next if subject.empty? || predicate.empty? || object.empty? subject_id = name_to_id[subject] || @kg.add_node(entity_type: "concept", name: subject, properties: { "provenance" => provenance }) object_id = name_to_id[object] || @kg.add_node(entity_type: "concept", name: object, properties: { "provenance" => provenance }) edge = Edge.new( id: nil, user_id: @user_id, subject_id: subject_id, predicate: predicate, target_id: object_id, properties: { "provenance" => provenance }, created_at: Time.now, archived_at: nil ) @conflict_resolver.resolve(edge) edge_id = @kg.add_edge(subject: subject_id, predicate: predicate, object: object_id, properties: { "provenance" => provenance }) text = "#{subject} #{predicate} #{object}" = @vector_store.respond_to?(:embed) ? @vector_store.(text) : nil if && @vector_store.respond_to?(:store) @vector_store.store(id: "edge_#{edge_id}", embedding: , metadata: { text: text, created_at: Time.now }, user_id: @user_id) end end true end |
#retrieve(query, top_k: 10) ⇒ Object
83 84 85 86 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 83 def retrieve(query, top_k: 10) results = hybrid_search(query, top_k: top_k) format_as_context(results) end |
#search_candidates(query, user_id: nil, top_k: 20) ⇒ Object
88 89 90 91 92 93 94 95 96 97 98 99 100 101 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 88 def search_candidates(query, user_id: nil, top_k: 20) uid = user_id || @user_id return [] unless uid == @user_id results = hybrid_search(query, top_k: top_k) results.map do |r| { id: r[:id], text: r[:text], timestamp: r[:created_at] || r[:timestamp], score: r[:score] || 1.0, importance: r[:importance] } end end |
#stats(user_id: nil) ⇒ Object
119 120 121 122 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 119 def stats(user_id: nil) uid = user_id || @user_id { nodes: @graph_storage.count_nodes(uid), edges: @graph_storage.count_edges(uid) } end |
#storage ⇒ Object
105 106 107 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 105 def storage @graph_storage end |
#write(payload, **_meta) ⇒ Object
— MemoryModule uniform interface —
111 112 113 |
# File 'lib/llmemory/long_term/graph_based/memory.rb', line 111 def write(payload, **) memorize(payload) end |