n8n-nodes-memory 0.3.0 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -22,20 +22,24 @@ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
22
22
  return headers;
23
23
  }
24
24
  async getMessages() {
25
+ console.log('[ApiChatHistory] getMessages called for session:', this.sessionId);
25
26
  try {
26
- const url = new URL(this.apiUrl);
27
- url.searchParams.set('action', 'get');
28
- url.searchParams.set('sessionId', this.sessionId);
29
- const response = await fetch(url.toString(), {
30
- method: 'GET',
27
+ console.log('[ApiChatHistory] Fetching from:', this.apiUrl);
28
+ const response = await fetch(this.apiUrl, {
29
+ method: 'POST',
31
30
  headers: this.getHeaders(),
31
+ body: JSON.stringify({
32
+ action: 'get',
33
+ sessionId: this.sessionId,
34
+ }),
32
35
  });
33
36
  if (!response.ok) {
34
- console.error(`Memory API error: ${response.status}`);
37
+ console.error(`[ApiChatHistory] API error: ${response.status}`);
35
38
  return [];
36
39
  }
37
40
  const data = await response.json();
38
41
  const messages = data?.messages ?? [];
42
+ console.log('[ApiChatHistory] Got messages:', messages.length);
39
43
  return messages.map((m) => {
40
44
  if (m.type === 'human') {
41
45
  return new messages_1.HumanMessage(m.content);
@@ -46,11 +50,12 @@ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
46
50
  });
47
51
  }
48
52
  catch (error) {
49
- console.error('Memory API getMessages error:', error);
53
+ console.error('[ApiChatHistory] getMessages error:', error);
50
54
  return [];
51
55
  }
52
56
  }
53
57
  async addMessage(message) {
58
+ console.log('[ApiChatHistory] addMessage called for session:', this.sessionId);
54
59
  try {
55
60
  const type = message._getType() === 'human' ? 'human' : 'ai';
56
61
  await fetch(this.apiUrl, {
@@ -65,9 +70,10 @@ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
65
70
  },
66
71
  }),
67
72
  });
73
+ console.log('[ApiChatHistory] Message added successfully');
68
74
  }
69
75
  catch (error) {
70
- console.error('Memory API addMessage error:', error);
76
+ console.error('[ApiChatHistory] addMessage error:', error);
71
77
  }
72
78
  }
73
79
  async addMessages(messages) {
@@ -76,6 +82,7 @@ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
76
82
  }
77
83
  }
78
84
  async clear() {
85
+ console.log('[ApiChatHistory] clear called for session:', this.sessionId);
79
86
  try {
80
87
  await fetch(this.apiUrl, {
81
88
  method: 'POST',
@@ -87,7 +94,7 @@ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
87
94
  });
88
95
  }
89
96
  catch (error) {
90
- console.error('Memory API clear error:', error);
97
+ console.error('[ApiChatHistory] clear error:', error);
91
98
  }
92
99
  }
93
100
  }
@@ -148,6 +155,13 @@ class Memory {
148
155
  default: '',
149
156
  description: 'Optional API key for authentication',
150
157
  },
158
+ {
159
+ displayName: 'Context Window Length',
160
+ name: 'contextWindowLength',
161
+ type: 'number',
162
+ default: 10,
163
+ description: 'Number of previous messages to include as context',
164
+ },
151
165
  ],
152
166
  };
153
167
  }
@@ -155,13 +169,16 @@ class Memory {
155
169
  const apiUrl = this.getNodeParameter('apiUrl', itemIndex);
156
170
  const sessionId = this.getNodeParameter('sessionId', itemIndex);
157
171
  const apiKey = this.getNodeParameter('apiKey', itemIndex, '') || undefined;
172
+ const contextWindowLength = this.getNodeParameter('contextWindowLength', itemIndex, 10);
173
+ console.log('[Memory API] Creating memory for session:', sessionId);
158
174
  const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);
159
- const memory = new memory_1.BufferMemory({
175
+ const memory = new memory_1.BufferWindowMemory({
160
176
  memoryKey: 'chat_history',
161
177
  chatHistory,
162
178
  returnMessages: true,
163
179
  inputKey: 'input',
164
180
  outputKey: 'output',
181
+ k: contextWindowLength,
165
182
  });
166
183
  return {
167
184
  response: memory,
@@ -169,4 +186,4 @@ class Memory {
169
186
  }
170
187
  }
171
188
  exports.Memory = Memory;
172
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Memory.node.js","sourceRoot":"","sources":["../../../src/nodes/Memory/Memory.node.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  INodeType,\n  INodeTypeDescription,\n  ISupplyDataFunctions,\n  SupplyData,\n} from 'n8n-workflow';\nimport { BufferMemory } from 'langchain/memory';\nimport { BaseListChatMessageHistory } from '@langchain/core/chat_history';\nimport {\n  BaseMessage,\n  HumanMessage,\n  AIMessage,\n} from '@langchain/core/messages';\n\nclass ApiChatHistory extends BaseListChatMessageHistory {\n  lc_namespace = ['langchain', 'stores', 'message', 'api'];\n\n  constructor(\n    private apiUrl: string,\n    private sessionId: string,\n    private apiKey?: string,\n  ) {\n    super();\n  }\n\n  private getHeaders(): Record<string, string> {\n    const headers: Record<string, string> = {\n      'Content-Type': 'application/json',\n    };\n    if (this.apiKey) {\n      headers['Authorization'] = `Bearer ${this.apiKey}`;\n    }\n    return headers;\n  }\n\n  async getMessages(): Promise<BaseMessage[]> {\n    try {\n      const url = new URL(this.apiUrl);\n      url.searchParams.set('action', 'get');\n      url.searchParams.set('sessionId', this.sessionId);\n\n      const response = await fetch(url.toString(), {\n        method: 'GET',\n        headers: this.getHeaders(),\n      });\n\n      if (!response.ok) {\n        console.error(`Memory API error: ${response.status}`);\n        return [];\n      }\n\n      const data = await response.json() as { messages?: Array<{ type: string; content: string }> };\n      const messages = data?.messages ?? [];\n\n      return messages.map((m: { type: string; content: string }) => {\n        if (m.type === 'human') {\n          return new HumanMessage(m.content);\n        } else {\n          return new AIMessage(m.content);\n        }\n      });\n    } catch (error) {\n      console.error('Memory API getMessages error:', error);\n      return [];\n    }\n  }\n\n  async addMessage(message: BaseMessage): Promise<void> {\n    try {\n      const type = message._getType() === 'human' ? 'human' : 'ai';\n\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: JSON.stringify({\n          action: 'add',\n          sessionId: this.sessionId,\n          message: {\n            type,\n            content: message.content,\n          },\n        }),\n      });\n    } catch (error) {\n      console.error('Memory API addMessage error:', error);\n    }\n  }\n\n  async addMessages(messages: BaseMessage[]): Promise<void> {\n    for (const message of messages) {\n      await this.addMessage(message);\n    }\n  }\n\n  async clear(): Promise<void> {\n    try {\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: JSON.stringify({\n          action: 'clear',\n          sessionId: this.sessionId,\n        }),\n      });\n    } catch (error) {\n      console.error('Memory API clear error:', error);\n    }\n  }\n}\n\nexport class Memory implements INodeType {\n  description: INodeTypeDescription = {\n    displayName: 'Memory API',\n    name: 'memoryApi',\n    icon: 'fa:brain',\n    group: ['transform'],\n    version: 1,\n    subtitle: 'External API Memory',\n    description: 'Use an external API as chat memory storage',\n    defaults: {\n      name: 'Memory API',\n    },\n    codex: {\n      categories: ['AI'],\n      subcategories: {\n        AI: ['Memory'],\n      },\n      resources: {\n        primaryDocumentation: [\n          {\n            url: 'https://github.com/filipexyz/n8n-nodes-memory',\n          },\n        ],\n      },\n    },\n    inputs: [],\n    outputs: ['ai_memory' as const],\n    outputNames: ['Memory'],\n    properties: [\n      {\n        displayName: 'API URL',\n        name: 'apiUrl',\n        type: 'string',\n        default: '',\n        required: true,\n        description: 'URL of the memory API (e.g., n8n webhook URL)',\n        placeholder: 'https://your-n8n.com/webhook/memory',\n      },\n      {\n        displayName: 'Session ID',\n        name: 'sessionId',\n        type: 'string',\n        default: '={{ $json.sessionId }}',\n        required: true,\n        description: 'Unique identifier for the conversation session',\n      },\n      {\n        displayName: 'API Key',\n        name: 'apiKey',\n        type: 'string',\n        typeOptions: {\n          password: true,\n        },\n        default: '',\n        description: 'Optional API key for authentication',\n      },\n    ],\n  };\n\n  async supplyData(\n    this: ISupplyDataFunctions,\n    itemIndex: number\n  ): Promise<SupplyData> {\n    const apiUrl = this.getNodeParameter('apiUrl', itemIndex) as string;\n    const sessionId = this.getNodeParameter('sessionId', itemIndex) as string;\n    const apiKey = (this.getNodeParameter('apiKey', itemIndex, '') as string) || undefined;\n\n    const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);\n\n    const memory = new BufferMemory({\n      memoryKey: 'chat_history',\n      chatHistory,\n      returnMessages: true,\n      inputKey: 'input',\n      outputKey: 'output',\n    });\n\n    return {\n      response: memory,\n    };\n  }\n}\n"]}
189
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Memory.node.js","sourceRoot":"","sources":["../../../src/nodes/Memory/Memory.node.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  INodeType,\n  INodeTypeDescription,\n  ISupplyDataFunctions,\n  SupplyData,\n} from 'n8n-workflow';\nimport { BufferWindowMemory } from 'langchain/memory';\nimport { BaseListChatMessageHistory } from '@langchain/core/chat_history';\nimport {\n  BaseMessage,\n  HumanMessage,\n  AIMessage,\n} from '@langchain/core/messages';\n\nclass ApiChatHistory extends BaseListChatMessageHistory {\n  lc_namespace = ['langchain', 'stores', 'message', 'api'];\n\n  constructor(\n    private apiUrl: string,\n    private sessionId: string,\n    private apiKey?: string,\n  ) {\n    super();\n  }\n\n  private getHeaders(): Record<string, string> {\n    const headers: Record<string, string> = {\n      'Content-Type': 'application/json',\n    };\n    if (this.apiKey) {\n      headers['Authorization'] = `Bearer ${this.apiKey}`;\n    }\n    return headers;\n  }\n\n  async getMessages(): Promise<BaseMessage[]> {\n    console.log('[ApiChatHistory] getMessages called for session:', this.sessionId);\n    try {\n      console.log('[ApiChatHistory] Fetching from:', this.apiUrl);\n      const response = await fetch(this.apiUrl, {\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: JSON.stringify({\n          action: 'get',\n          sessionId: this.sessionId,\n        }),\n      });\n\n      if (!response.ok) {\n        console.error(`[ApiChatHistory] API error: ${response.status}`);\n        return [];\n      }\n\n      const data = await response.json() as { messages?: Array<{ type: string; content: string }> };\n      const messages = data?.messages ?? [];\n      console.log('[ApiChatHistory] Got messages:', messages.length);\n\n      return messages.map((m: { type: string; content: string }) => {\n        if (m.type === 'human') {\n          return new HumanMessage(m.content);\n        } else {\n          return new AIMessage(m.content);\n        }\n      });\n    } catch (error) {\n      console.error('[ApiChatHistory] getMessages error:', error);\n      return [];\n    }\n  }\n\n  async addMessage(message: BaseMessage): Promise<void> {\n    console.log('[ApiChatHistory] addMessage called for session:', this.sessionId);\n    try {\n      const type = message._getType() === 'human' ? 'human' : 'ai';\n\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: JSON.stringify({\n          action: 'add',\n          sessionId: this.sessionId,\n          message: {\n            type,\n            content: message.content,\n          },\n        }),\n      });\n      console.log('[ApiChatHistory] Message added successfully');\n    } catch (error) {\n      console.error('[ApiChatHistory] addMessage error:', error);\n    }\n  }\n\n  async addMessages(messages: BaseMessage[]): Promise<void> {\n    for (const message of messages) {\n      await this.addMessage(message);\n    }\n  }\n\n  async clear(): Promise<void> {\n    console.log('[ApiChatHistory] clear called for session:', this.sessionId);\n    try {\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: JSON.stringify({\n          action: 'clear',\n          sessionId: this.sessionId,\n        }),\n      });\n    } catch (error) {\n      console.error('[ApiChatHistory] clear error:', error);\n    }\n  }\n}\n\nexport class Memory implements INodeType {\n  description: INodeTypeDescription = {\n    displayName: 'Memory API',\n    name: 'memoryApi',\n    icon: 'fa:brain',\n    group: ['transform'],\n    version: 1,\n    subtitle: 'External API Memory',\n    description: 'Use an external API as chat memory storage',\n    defaults: {\n      name: 'Memory API',\n    },\n    codex: {\n      categories: ['AI'],\n      subcategories: {\n        AI: ['Memory'],\n      },\n      resources: {\n        primaryDocumentation: [\n          {\n            url: 'https://github.com/filipexyz/n8n-nodes-memory',\n          },\n        ],\n      },\n    },\n    inputs: [],\n    outputs: ['ai_memory' as const],\n    outputNames: ['Memory'],\n    properties: [\n      {\n        displayName: 'API URL',\n        name: 'apiUrl',\n        type: 'string',\n        default: '',\n        required: true,\n        description: 'URL of the memory API (e.g., n8n webhook URL)',\n        placeholder: 'https://your-n8n.com/webhook/memory',\n      },\n      {\n        displayName: 'Session ID',\n        name: 'sessionId',\n        type: 'string',\n        default: '={{ $json.sessionId }}',\n        required: true,\n        description: 'Unique identifier for the conversation session',\n      },\n      {\n        displayName: 'API Key',\n        name: 'apiKey',\n        type: 'string',\n        typeOptions: {\n          password: true,\n        },\n        default: '',\n        description: 'Optional API key for authentication',\n      },\n      {\n        displayName: 'Context Window Length',\n        name: 'contextWindowLength',\n        type: 'number',\n        default: 10,\n        description: 'Number of previous messages to include as context',\n      },\n    ],\n  };\n\n  async supplyData(\n    this: ISupplyDataFunctions,\n    itemIndex: number\n  ): Promise<SupplyData> {\n    const apiUrl = this.getNodeParameter('apiUrl', itemIndex) as string;\n    const sessionId = this.getNodeParameter('sessionId', itemIndex) as string;\n    const apiKey = (this.getNodeParameter('apiKey', itemIndex, '') as string) || undefined;\n    const contextWindowLength = this.getNodeParameter('contextWindowLength', itemIndex, 10) as number;\n\n    console.log('[Memory API] Creating memory for session:', sessionId);\n\n    const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);\n\n    const memory = new BufferWindowMemory({\n      memoryKey: 'chat_history',\n      chatHistory,\n      returnMessages: true,\n      inputKey: 'input',\n      outputKey: 'output',\n      k: contextWindowLength,\n    });\n\n    return {\n      response: memory,\n    };\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "n8n-nodes-memory",
3
- "version": "0.3.0",
3
+ "version": "0.5.0",
4
4
  "description": "Custom memory node for n8n AI Agent with external API support",
5
5
  "keywords": [
6
6
  "n8n-community-node-package",