n8n-nodes-memory 0.2.0 → 0.4.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.
@@ -1,5 +1,5 @@
1
- import type { IExecuteFunctions, INodeExecutionData, INodeType, INodeTypeDescription } from 'n8n-workflow';
1
+ import type { INodeType, INodeTypeDescription, ISupplyDataFunctions, SupplyData } from 'n8n-workflow';
2
2
  export declare class Memory implements INodeType {
3
3
  description: INodeTypeDescription;
4
- execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]>;
4
+ supplyData(this: ISupplyDataFunctions, itemIndex: number): Promise<SupplyData>;
5
5
  }
@@ -1,12 +1,16 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.Memory = void 0;
4
- class ApiMemoryAdapter {
5
- constructor(http, apiUrl, sessionId, apiKey) {
6
- this.http = http;
4
+ const memory_1 = require("langchain/memory");
5
+ const chat_history_1 = require("@langchain/core/chat_history");
6
+ const messages_1 = require("@langchain/core/messages");
7
+ class ApiChatHistory extends chat_history_1.BaseListChatMessageHistory {
8
+ constructor(apiUrl, sessionId, apiKey) {
9
+ super();
7
10
  this.apiUrl = apiUrl;
8
11
  this.sessionId = sessionId;
9
12
  this.apiKey = apiKey;
13
+ this.lc_namespace = ['langchain', 'stores', 'message', 'api'];
10
14
  }
11
15
  getHeaders() {
12
16
  const headers = {
@@ -18,66 +22,78 @@ class ApiMemoryAdapter {
18
22
  return headers;
19
23
  }
20
24
  async getMessages() {
25
+ console.log('[ApiChatHistory] getMessages called for session:', this.sessionId);
21
26
  try {
22
- const res = await this.http.helpers.httpRequest({
23
- url: this.apiUrl,
27
+ const url = new URL(this.apiUrl);
28
+ url.searchParams.set('action', 'get');
29
+ url.searchParams.set('sessionId', this.sessionId);
30
+ console.log('[ApiChatHistory] Fetching from:', url.toString());
31
+ const response = await fetch(url.toString(), {
24
32
  method: 'GET',
25
33
  headers: this.getHeaders(),
26
- qs: {
27
- action: 'get',
28
- sessionId: this.sessionId,
29
- },
30
34
  });
31
- const messages = res?.messages ?? [];
32
- return messages.map((m) => ({
33
- role: m.type === 'human' ? 'user' : m.type === 'ai' ? 'assistant' : m.type,
34
- content: m.content,
35
- }));
35
+ if (!response.ok) {
36
+ console.error(`[ApiChatHistory] API error: ${response.status}`);
37
+ return [];
38
+ }
39
+ const data = await response.json();
40
+ const messages = data?.messages ?? [];
41
+ console.log('[ApiChatHistory] Got messages:', messages.length);
42
+ return messages.map((m) => {
43
+ if (m.type === 'human') {
44
+ return new messages_1.HumanMessage(m.content);
45
+ }
46
+ else {
47
+ return new messages_1.AIMessage(m.content);
48
+ }
49
+ });
36
50
  }
37
51
  catch (error) {
38
- console.error('Memory API getMessages error:', error);
52
+ console.error('[ApiChatHistory] getMessages error:', error);
39
53
  return [];
40
54
  }
41
55
  }
42
- async addMessages(messages) {
56
+ async addMessage(message) {
57
+ console.log('[ApiChatHistory] addMessage called for session:', this.sessionId);
43
58
  try {
44
- for (const msg of messages) {
45
- const type = msg.role === 'user' ? 'human' : msg.role === 'assistant' ? 'ai' : msg.role;
46
- await this.http.helpers.httpRequest({
47
- url: this.apiUrl,
48
- method: 'POST',
49
- headers: this.getHeaders(),
50
- body: {
51
- action: 'add',
52
- sessionId: this.sessionId,
53
- message: {
54
- type,
55
- content: msg.content,
56
- },
59
+ const type = message._getType() === 'human' ? 'human' : 'ai';
60
+ await fetch(this.apiUrl, {
61
+ method: 'POST',
62
+ headers: this.getHeaders(),
63
+ body: JSON.stringify({
64
+ action: 'add',
65
+ sessionId: this.sessionId,
66
+ message: {
67
+ type,
68
+ content: message.content,
57
69
  },
58
- });
59
- }
70
+ }),
71
+ });
72
+ console.log('[ApiChatHistory] Message added successfully');
60
73
  }
61
74
  catch (error) {
62
- console.error('Memory API addMessages error:', error);
75
+ console.error('[ApiChatHistory] addMessage error:', error);
76
+ }
77
+ }
78
+ async addMessages(messages) {
79
+ for (const message of messages) {
80
+ await this.addMessage(message);
63
81
  }
64
82
  }
65
83
  async clear() {
84
+ console.log('[ApiChatHistory] clear called for session:', this.sessionId);
66
85
  try {
67
- await this.http.helpers.httpRequest({
68
- url: this.apiUrl,
86
+ await fetch(this.apiUrl, {
69
87
  method: 'POST',
70
88
  headers: this.getHeaders(),
71
- body: {
89
+ body: JSON.stringify({
72
90
  action: 'clear',
73
91
  sessionId: this.sessionId,
74
- },
92
+ }),
75
93
  });
76
- return true;
77
94
  }
78
95
  catch (error) {
79
- console.error('Memory API clear error:', error);
80
- return false;
96
+ console.error('[ApiChatHistory] clear error:', error);
81
97
  }
82
98
  }
83
99
  }
@@ -138,24 +154,35 @@ class Memory {
138
154
  default: '',
139
155
  description: 'Optional API key for authentication',
140
156
  },
157
+ {
158
+ displayName: 'Context Window Length',
159
+ name: 'contextWindowLength',
160
+ type: 'number',
161
+ default: 10,
162
+ description: 'Number of previous messages to include as context',
163
+ },
141
164
  ],
142
165
  };
143
166
  }
144
- async execute() {
145
- const apiUrl = this.getNodeParameter('apiUrl', 0);
146
- const sessionId = this.getNodeParameter('sessionId', 0);
147
- const apiKey = this.getNodeParameter('apiKey', 0) || undefined;
148
- const adapter = new ApiMemoryAdapter(this, apiUrl, sessionId, apiKey);
149
- const item = {
150
- json: {},
151
- context: {
152
- ai: {
153
- memory: adapter,
154
- },
155
- },
167
+ async supplyData(itemIndex) {
168
+ const apiUrl = this.getNodeParameter('apiUrl', itemIndex);
169
+ const sessionId = this.getNodeParameter('sessionId', itemIndex);
170
+ const apiKey = this.getNodeParameter('apiKey', itemIndex, '') || undefined;
171
+ const contextWindowLength = this.getNodeParameter('contextWindowLength', itemIndex, 10);
172
+ console.log('[Memory API] Creating memory for session:', sessionId);
173
+ const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);
174
+ const memory = new memory_1.BufferWindowMemory({
175
+ memoryKey: 'chat_history',
176
+ chatHistory,
177
+ returnMessages: true,
178
+ inputKey: 'input',
179
+ outputKey: 'output',
180
+ k: contextWindowLength,
181
+ });
182
+ return {
183
+ response: memory,
156
184
  };
157
- return [[item]];
158
185
  }
159
186
  }
160
187
  exports.Memory = Memory;
161
- //# 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  IExecuteFunctions,\n  INodeExecutionData,\n  INodeType,\n  INodeTypeDescription,\n} from 'n8n-workflow';\n\ninterface ApiMessage {\n  role: string;\n  content: string;\n}\n\nclass ApiMemoryAdapter {\n  constructor(\n    private http: IExecuteFunctions,\n    private apiUrl: string,\n    private sessionId: string,\n    private apiKey?: string,\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<ApiMessage[]> {\n    try {\n      const res = await this.http.helpers.httpRequest({\n        url: this.apiUrl,\n        method: 'GET',\n        headers: this.getHeaders(),\n        qs: {\n          action: 'get',\n          sessionId: this.sessionId,\n        },\n      });\n\n      const messages = res?.messages ?? [];\n      return messages.map((m: { type: string; content: string }) => ({\n        role: m.type === 'human' ? 'user' : m.type === 'ai' ? 'assistant' : m.type,\n        content: m.content,\n      }));\n    } catch (error) {\n      console.error('Memory API getMessages error:', error);\n      return [];\n    }\n  }\n\n  async addMessages(messages: Array<{ role: string; content: string }>): Promise<void> {\n    try {\n      for (const msg of messages) {\n        const type = msg.role === 'user' ? 'human' : msg.role === 'assistant' ? 'ai' : msg.role;\n        await this.http.helpers.httpRequest({\n          url: this.apiUrl,\n          method: 'POST',\n          headers: this.getHeaders(),\n          body: {\n            action: 'add',\n            sessionId: this.sessionId,\n            message: {\n              type,\n              content: msg.content,\n            },\n          },\n        });\n      }\n    } catch (error) {\n      console.error('Memory API addMessages error:', error);\n    }\n  }\n\n  async clear(): Promise<boolean> {\n    try {\n      await this.http.helpers.httpRequest({\n        url: this.apiUrl,\n        method: 'POST',\n        headers: this.getHeaders(),\n        body: {\n          action: 'clear',\n          sessionId: this.sessionId,\n        },\n      });\n      return true;\n    } catch (error) {\n      console.error('Memory API clear error:', error);\n      return false;\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'],\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 execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {\n    const apiUrl = this.getNodeParameter('apiUrl', 0) as string;\n    const sessionId = this.getNodeParameter('sessionId', 0) as string;\n    const apiKey = (this.getNodeParameter('apiKey', 0) as string) || undefined;\n\n    const adapter = new ApiMemoryAdapter(this, apiUrl, sessionId, apiKey);\n\n    const item: INodeExecutionData = {\n      json: {},\n      context: {\n        ai: {\n          memory: adapter,\n        },\n      } as any,\n    };\n\n    return [[item]];\n  }\n}\n"]}
188
+ //# 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      const url = new URL(this.apiUrl);\n      url.searchParams.set('action', 'get');\n      url.searchParams.set('sessionId', this.sessionId);\n\n      console.log('[ApiChatHistory] Fetching from:', url.toString());\n      const response = await fetch(url.toString(), {\n        method: 'GET',\n        headers: this.getHeaders(),\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.2.0",
3
+ "version": "0.4.0",
4
4
  "description": "Custom memory node for n8n AI Agent with external API support",
5
5
  "keywords": [
6
6
  "n8n-community-node-package",
@@ -38,5 +38,9 @@
38
38
  },
39
39
  "peerDependencies": {
40
40
  "n8n-workflow": "^2.2.1"
41
+ },
42
+ "dependencies": {
43
+ "@langchain/core": "^0.3.0",
44
+ "langchain": "^0.3.0"
41
45
  }
42
46
  }