n8n-nodes-memory 0.5.1 → 0.6.1

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.
package/README.md CHANGED
@@ -1,6 +1,13 @@
1
1
  # n8n-nodes-memory
2
2
 
3
- Custom memory node for n8n AI Agent with external API support.
3
+ Custom memory nodes for n8n AI Agent with workflow-based storage.
4
+
5
+ ## Two Nodes
6
+
7
+ | Node | Use Case |
8
+ |------|----------|
9
+ | **Memory API** | External webhook/API as memory backend |
10
+ | **Memory Workflow** | Sub-workflow within n8n as memory backend |
4
11
 
5
12
  ## The Power: Memory as Workflows
6
13
 
@@ -81,6 +88,8 @@ Message types: `human` or `ai`
81
88
 
82
89
  ## Parameters
83
90
 
91
+ ### Memory API
92
+
84
93
  | Parameter | Description |
85
94
  |-----------|-------------|
86
95
  | API URL | Your memory webhook URL |
@@ -88,6 +97,14 @@ Message types: `human` or `ai`
88
97
  | API Key | Optional Bearer token |
89
98
  | Context Window Length | Messages to include (default: 10) |
90
99
 
100
+ ### Memory Workflow
101
+
102
+ | Parameter | Description |
103
+ |-----------|-------------|
104
+ | Workflow ID | ID of the sub-workflow to execute |
105
+ | Session ID | Unique conversation identifier |
106
+ | Context Window Length | Messages to include (default: 10) |
107
+
91
108
  ## How It Works
92
109
 
93
110
  Implements LangChain's `BaseListChatMessageHistory` with `BufferWindowMemory`. The AI Agent calls your workflows automatically during conversation.
package/dist/index.d.ts CHANGED
@@ -1 +1,2 @@
1
- export { Memory } from './nodes/Memory/Memory.node';
1
+ export { MemoryApi } from './nodes/Memory/MemoryApi.node';
2
+ export { MemoryWorkflow } from './nodes/Memory/MemoryWorkflow.node';
package/dist/index.js CHANGED
@@ -1,6 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Memory = void 0;
4
- var Memory_node_1 = require("./nodes/Memory/Memory.node");
5
- Object.defineProperty(exports, "Memory", { enumerable: true, get: function () { return Memory_node_1.Memory; } });
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMERBQW9EO0FBQTNDLHFHQUFBLE1BQU0sT0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IE1lbW9yeSB9IGZyb20gJy4vbm9kZXMvTWVtb3J5L01lbW9yeS5ub2RlJztcbiJdfQ==
3
+ exports.MemoryWorkflow = exports.MemoryApi = void 0;
4
+ var MemoryApi_node_1 = require("./nodes/Memory/MemoryApi.node");
5
+ Object.defineProperty(exports, "MemoryApi", { enumerable: true, get: function () { return MemoryApi_node_1.MemoryApi; } });
6
+ var MemoryWorkflow_node_1 = require("./nodes/Memory/MemoryWorkflow.node");
7
+ Object.defineProperty(exports, "MemoryWorkflow", { enumerable: true, get: function () { return MemoryWorkflow_node_1.MemoryWorkflow; } });
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsZ0VBQTBEO0FBQWpELDJHQUFBLFNBQVMsT0FBQTtBQUNsQiwwRUFBb0U7QUFBM0QscUhBQUEsY0FBYyxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgTWVtb3J5QXBpIH0gZnJvbSAnLi9ub2Rlcy9NZW1vcnkvTWVtb3J5QXBpLm5vZGUnO1xuZXhwb3J0IHsgTWVtb3J5V29ya2Zsb3cgfSBmcm9tICcuL25vZGVzL01lbW9yeS9NZW1vcnlXb3JrZmxvdy5ub2RlJztcbiJdfQ==
@@ -1,5 +1,5 @@
1
1
  import type { INodeType, INodeTypeDescription, ISupplyDataFunctions, SupplyData } from 'n8n-workflow';
2
- export declare class Memory implements INodeType {
2
+ export declare class MemoryApi implements INodeType {
3
3
  description: INodeTypeDescription;
4
4
  supplyData(this: ISupplyDataFunctions, itemIndex: number): Promise<SupplyData>;
5
5
  }
@@ -0,0 +1,189 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MemoryApi = void 0;
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();
10
+ this.apiUrl = apiUrl;
11
+ this.sessionId = sessionId;
12
+ this.apiKey = apiKey;
13
+ this.lc_namespace = ['langchain', 'stores', 'message', 'api'];
14
+ }
15
+ getHeaders() {
16
+ const headers = {
17
+ 'Content-Type': 'application/json',
18
+ };
19
+ if (this.apiKey) {
20
+ headers['Authorization'] = `Bearer ${this.apiKey}`;
21
+ }
22
+ return headers;
23
+ }
24
+ async getMessages() {
25
+ console.log('[ApiChatHistory] getMessages called for session:', this.sessionId);
26
+ try {
27
+ console.log('[ApiChatHistory] Fetching from:', this.apiUrl);
28
+ const response = await fetch(this.apiUrl, {
29
+ method: 'POST',
30
+ headers: this.getHeaders(),
31
+ body: JSON.stringify({
32
+ action: 'get',
33
+ sessionId: this.sessionId,
34
+ }),
35
+ });
36
+ if (!response.ok) {
37
+ console.error(`[ApiChatHistory] API error: ${response.status}`);
38
+ return [];
39
+ }
40
+ const data = await response.json();
41
+ const messages = data?.messages ?? [];
42
+ console.log('[ApiChatHistory] Got messages:', messages.length);
43
+ return messages.map((m) => {
44
+ if (m.type === 'human') {
45
+ return new messages_1.HumanMessage(m.content);
46
+ }
47
+ else {
48
+ return new messages_1.AIMessage(m.content);
49
+ }
50
+ });
51
+ }
52
+ catch (error) {
53
+ console.error('[ApiChatHistory] getMessages error:', error);
54
+ return [];
55
+ }
56
+ }
57
+ async addMessage(message) {
58
+ console.log('[ApiChatHistory] addMessage called for session:', this.sessionId);
59
+ try {
60
+ const type = message._getType() === 'human' ? 'human' : 'ai';
61
+ await fetch(this.apiUrl, {
62
+ method: 'POST',
63
+ headers: this.getHeaders(),
64
+ body: JSON.stringify({
65
+ action: 'add',
66
+ sessionId: this.sessionId,
67
+ message: {
68
+ type,
69
+ content: message.content,
70
+ },
71
+ }),
72
+ });
73
+ console.log('[ApiChatHistory] Message added successfully');
74
+ }
75
+ catch (error) {
76
+ console.error('[ApiChatHistory] addMessage error:', error);
77
+ }
78
+ }
79
+ async addMessages(messages) {
80
+ for (const message of messages) {
81
+ await this.addMessage(message);
82
+ }
83
+ }
84
+ async clear() {
85
+ console.log('[ApiChatHistory] clear called for session:', this.sessionId);
86
+ try {
87
+ await fetch(this.apiUrl, {
88
+ method: 'POST',
89
+ headers: this.getHeaders(),
90
+ body: JSON.stringify({
91
+ action: 'clear',
92
+ sessionId: this.sessionId,
93
+ }),
94
+ });
95
+ }
96
+ catch (error) {
97
+ console.error('[ApiChatHistory] clear error:', error);
98
+ }
99
+ }
100
+ }
101
+ class MemoryApi {
102
+ constructor() {
103
+ this.description = {
104
+ displayName: 'Memory API',
105
+ name: 'memoryApi',
106
+ icon: 'fa:brain',
107
+ group: ['transform'],
108
+ version: 1,
109
+ subtitle: 'External API Memory',
110
+ description: 'Use an external API as chat memory storage',
111
+ defaults: {
112
+ name: 'Memory API',
113
+ },
114
+ codex: {
115
+ categories: ['AI'],
116
+ subcategories: {
117
+ AI: ['Memory'],
118
+ },
119
+ resources: {
120
+ primaryDocumentation: [
121
+ {
122
+ url: 'https://github.com/filipexyz/n8n-nodes-memory',
123
+ },
124
+ ],
125
+ },
126
+ },
127
+ inputs: [],
128
+ outputs: ['ai_memory'],
129
+ outputNames: ['Memory'],
130
+ properties: [
131
+ {
132
+ displayName: 'API URL',
133
+ name: 'apiUrl',
134
+ type: 'string',
135
+ default: '',
136
+ required: true,
137
+ description: 'URL of the memory API (e.g., n8n webhook URL)',
138
+ placeholder: 'https://your-n8n.com/webhook/memory',
139
+ },
140
+ {
141
+ displayName: 'Session ID',
142
+ name: 'sessionId',
143
+ type: 'string',
144
+ default: '={{ $json.sessionId }}',
145
+ required: true,
146
+ description: 'Unique identifier for the conversation session',
147
+ },
148
+ {
149
+ displayName: 'API Key',
150
+ name: 'apiKey',
151
+ type: 'string',
152
+ typeOptions: {
153
+ password: true,
154
+ },
155
+ default: '',
156
+ description: 'Optional API key for authentication',
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
+ },
165
+ ],
166
+ };
167
+ }
168
+ async supplyData(itemIndex) {
169
+ const apiUrl = this.getNodeParameter('apiUrl', itemIndex);
170
+ const sessionId = this.getNodeParameter('sessionId', itemIndex);
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);
174
+ const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);
175
+ const memory = new memory_1.BufferWindowMemory({
176
+ memoryKey: 'chat_history',
177
+ chatHistory,
178
+ returnMessages: true,
179
+ inputKey: 'input',
180
+ outputKey: 'output',
181
+ k: contextWindowLength,
182
+ });
183
+ return {
184
+ response: memory,
185
+ };
186
+ }
187
+ }
188
+ exports.MemoryApi = MemoryApi;
189
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MemoryApi.node.js","sourceRoot":"","sources":["../../../src/nodes/Memory/MemoryApi.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 MemoryApi 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"]}
@@ -0,0 +1,5 @@
1
+ import type { INodeType, INodeTypeDescription, ISupplyDataFunctions, SupplyData } from 'n8n-workflow';
2
+ export declare class MemoryWorkflow implements INodeType {
3
+ description: INodeTypeDescription;
4
+ supplyData(this: ISupplyDataFunctions, itemIndex: number): Promise<SupplyData>;
5
+ }
@@ -0,0 +1,161 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MemoryWorkflow = void 0;
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 WorkflowChatHistory extends chat_history_1.BaseListChatMessageHistory {
8
+ constructor(executeWorkflow, sessionId) {
9
+ super();
10
+ this.executeWorkflow = executeWorkflow;
11
+ this.sessionId = sessionId;
12
+ this.lc_namespace = ['langchain', 'stores', 'message', 'workflow'];
13
+ }
14
+ async getMessages() {
15
+ console.log('[WorkflowChatHistory] getMessages called for session:', this.sessionId);
16
+ try {
17
+ const result = await this.executeWorkflow({
18
+ action: 'get',
19
+ sessionId: this.sessionId,
20
+ });
21
+ const messages = result?.messages ?? [];
22
+ console.log('[WorkflowChatHistory] Got messages:', messages.length);
23
+ return messages.map((m) => {
24
+ if (m.type === 'human') {
25
+ return new messages_1.HumanMessage(m.content);
26
+ }
27
+ else {
28
+ return new messages_1.AIMessage(m.content);
29
+ }
30
+ });
31
+ }
32
+ catch (error) {
33
+ console.error('[WorkflowChatHistory] getMessages error:', error);
34
+ return [];
35
+ }
36
+ }
37
+ async addMessage(message) {
38
+ console.log('[WorkflowChatHistory] addMessage called for session:', this.sessionId);
39
+ try {
40
+ const type = message._getType() === 'human' ? 'human' : 'ai';
41
+ await this.executeWorkflow({
42
+ action: 'add',
43
+ sessionId: this.sessionId,
44
+ message: {
45
+ type,
46
+ content: message.content,
47
+ },
48
+ });
49
+ console.log('[WorkflowChatHistory] Message added successfully');
50
+ }
51
+ catch (error) {
52
+ console.error('[WorkflowChatHistory] addMessage error:', error);
53
+ }
54
+ }
55
+ async addMessages(messages) {
56
+ for (const message of messages) {
57
+ await this.addMessage(message);
58
+ }
59
+ }
60
+ async clear() {
61
+ console.log('[WorkflowChatHistory] clear called for session:', this.sessionId);
62
+ try {
63
+ await this.executeWorkflow({
64
+ action: 'clear',
65
+ sessionId: this.sessionId,
66
+ });
67
+ }
68
+ catch (error) {
69
+ console.error('[WorkflowChatHistory] clear error:', error);
70
+ }
71
+ }
72
+ }
73
+ class MemoryWorkflow {
74
+ constructor() {
75
+ this.description = {
76
+ displayName: 'Memory Workflow',
77
+ name: 'memoryWorkflow',
78
+ icon: 'fa:project-diagram',
79
+ group: ['transform'],
80
+ version: 1,
81
+ subtitle: 'Sub-Workflow Memory',
82
+ description: 'Use a sub-workflow as chat memory storage',
83
+ defaults: {
84
+ name: 'Memory Workflow',
85
+ },
86
+ codex: {
87
+ categories: ['AI'],
88
+ subcategories: {
89
+ AI: ['Memory'],
90
+ },
91
+ resources: {
92
+ primaryDocumentation: [
93
+ {
94
+ url: 'https://github.com/filipexyz/n8n-nodes-memory',
95
+ },
96
+ ],
97
+ },
98
+ },
99
+ inputs: [],
100
+ outputs: ['ai_memory'],
101
+ outputNames: ['Memory'],
102
+ properties: [
103
+ {
104
+ displayName: 'Workflow',
105
+ name: 'workflowId',
106
+ type: 'workflowSelector',
107
+ default: '',
108
+ required: true,
109
+ description: 'The workflow to execute for memory operations',
110
+ },
111
+ {
112
+ displayName: 'Session ID',
113
+ name: 'sessionId',
114
+ type: 'string',
115
+ default: '={{ $json.sessionId }}',
116
+ required: true,
117
+ description: 'Unique identifier for the conversation session',
118
+ },
119
+ {
120
+ displayName: 'Context Window Length',
121
+ name: 'contextWindowLength',
122
+ type: 'number',
123
+ default: 10,
124
+ description: 'Number of previous messages to include as context',
125
+ },
126
+ ],
127
+ };
128
+ }
129
+ async supplyData(itemIndex) {
130
+ const workflowIdParam = this.getNodeParameter('workflowId', itemIndex);
131
+ const workflowId = typeof workflowIdParam === 'string' ? workflowIdParam : workflowIdParam.value;
132
+ const sessionId = this.getNodeParameter('sessionId', itemIndex);
133
+ const contextWindowLength = this.getNodeParameter('contextWindowLength', itemIndex, 10);
134
+ console.log('[Memory Workflow] Creating memory for session:', sessionId);
135
+ const nodeContext = this;
136
+ const executeWorkflow = async (data) => {
137
+ const workflowInfo = {
138
+ id: workflowId,
139
+ };
140
+ const items = [{ json: data }];
141
+ const result = await nodeContext.executeWorkflow(workflowInfo, items);
142
+ // Get the output from the last node
143
+ const lastNodeData = result.data?.[0]?.[0]?.json;
144
+ return lastNodeData ?? {};
145
+ };
146
+ const chatHistory = new WorkflowChatHistory(executeWorkflow, sessionId);
147
+ const memory = new memory_1.BufferWindowMemory({
148
+ memoryKey: 'chat_history',
149
+ chatHistory,
150
+ returnMessages: true,
151
+ inputKey: 'input',
152
+ outputKey: 'output',
153
+ k: contextWindowLength,
154
+ });
155
+ return {
156
+ response: memory,
157
+ };
158
+ }
159
+ }
160
+ exports.MemoryWorkflow = MemoryWorkflow;
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MemoryWorkflow.node.js","sourceRoot":"","sources":["../../../src/nodes/Memory/MemoryWorkflow.node.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  INodeType,\n  INodeTypeDescription,\n  ISupplyDataFunctions,\n  SupplyData,\n  IExecuteWorkflowInfo,\n  INodeExecutionData,\n  IDataObject,\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\ntype WorkflowExecutor = (data: Record<string, unknown>) => Promise<Record<string, unknown>>;\n\nclass WorkflowChatHistory extends BaseListChatMessageHistory {\n  lc_namespace = ['langchain', 'stores', 'message', 'workflow'];\n\n  constructor(\n    private executeWorkflow: WorkflowExecutor,\n    private sessionId: string,\n  ) {\n    super();\n  }\n\n  async getMessages(): Promise<BaseMessage[]> {\n    console.log('[WorkflowChatHistory] getMessages called for session:', this.sessionId);\n    try {\n      const result = await this.executeWorkflow({\n        action: 'get',\n        sessionId: this.sessionId,\n      });\n\n      const messages = (result?.messages as Array<{ type: string; content: string }>) ?? [];\n      console.log('[WorkflowChatHistory] Got messages:', messages.length);\n\n      return messages.map((m) => {\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('[WorkflowChatHistory] getMessages error:', error);\n      return [];\n    }\n  }\n\n  async addMessage(message: BaseMessage): Promise<void> {\n    console.log('[WorkflowChatHistory] addMessage called for session:', this.sessionId);\n    try {\n      const type = message._getType() === 'human' ? 'human' : 'ai';\n\n      await this.executeWorkflow({\n        action: 'add',\n        sessionId: this.sessionId,\n        message: {\n          type,\n          content: message.content,\n        },\n      });\n      console.log('[WorkflowChatHistory] Message added successfully');\n    } catch (error) {\n      console.error('[WorkflowChatHistory] 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('[WorkflowChatHistory] clear called for session:', this.sessionId);\n    try {\n      await this.executeWorkflow({\n        action: 'clear',\n        sessionId: this.sessionId,\n      });\n    } catch (error) {\n      console.error('[WorkflowChatHistory] clear error:', error);\n    }\n  }\n}\n\nexport class MemoryWorkflow implements INodeType {\n  description: INodeTypeDescription = {\n    displayName: 'Memory Workflow',\n    name: 'memoryWorkflow',\n    icon: 'fa:project-diagram',\n    group: ['transform'],\n    version: 1,\n    subtitle: 'Sub-Workflow Memory',\n    description: 'Use a sub-workflow as chat memory storage',\n    defaults: {\n      name: 'Memory Workflow',\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: 'Workflow',\n        name: 'workflowId',\n        type: 'workflowSelector',\n        default: '',\n        required: true,\n        description: 'The workflow to execute for memory operations',\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: '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 workflowIdParam = this.getNodeParameter('workflowId', itemIndex) as string | { value: string };\n    const workflowId = typeof workflowIdParam === 'string' ? workflowIdParam : workflowIdParam.value;\n    const sessionId = this.getNodeParameter('sessionId', itemIndex) as string;\n    const contextWindowLength = this.getNodeParameter('contextWindowLength', itemIndex, 10) as number;\n\n    console.log('[Memory Workflow] Creating memory for session:', sessionId);\n\n    const nodeContext = this;\n\n    const executeWorkflow: WorkflowExecutor = async (data) => {\n      const workflowInfo: IExecuteWorkflowInfo = {\n        id: workflowId,\n      };\n\n      const items: INodeExecutionData[] = [{ json: data as IDataObject }];\n      const result = await nodeContext.executeWorkflow(workflowInfo, items);\n\n      // Get the output from the last node\n      const lastNodeData = result.data?.[0]?.[0]?.json;\n      return lastNodeData as Record<string, unknown> ?? {};\n    };\n\n    const chatHistory = new WorkflowChatHistory(executeWorkflow, sessionId);\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.5.1",
3
+ "version": "0.6.1",
4
4
  "description": "Custom memory node for n8n AI Agent with external API support",
5
5
  "keywords": [
6
6
  "n8n-community-node-package",
@@ -28,7 +28,8 @@
28
28
  "n8n": {
29
29
  "n8nNodesApiVersion": 1,
30
30
  "nodes": [
31
- "dist/nodes/Memory/Memory.node.js"
31
+ "dist/nodes/Memory/MemoryApi.node.js",
32
+ "dist/nodes/Memory/MemoryWorkflow.node.js"
32
33
  ],
33
34
  "credentials": []
34
35
  },
@@ -1,18 +0,0 @@
1
- import { BaseListChatMessageHistory } from '@langchain/core/chat_history';
2
- import { BaseMessage } from '@langchain/core/messages';
3
- export interface ApiMemoryHistoryConfig {
4
- apiUrl: string;
5
- sessionId: string;
6
- apiKey?: string;
7
- }
8
- export declare class ApiMemoryHistory extends BaseListChatMessageHistory {
9
- lc_namespace: string[];
10
- private apiUrl;
11
- private sessionId;
12
- private apiKey?;
13
- constructor(config: ApiMemoryHistoryConfig);
14
- getMessages(): Promise<BaseMessage[]>;
15
- addMessage(message: BaseMessage): Promise<void>;
16
- addMessages(messages: BaseMessage[]): Promise<void>;
17
- clear(): Promise<void>;
18
- }
@@ -1,113 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ApiMemoryHistory = void 0;
4
- const chat_history_1 = require("@langchain/core/chat_history");
5
- const messages_1 = require("@langchain/core/messages");
6
- class ApiMemoryHistory extends chat_history_1.BaseListChatMessageHistory {
7
- constructor(config) {
8
- super();
9
- this.lc_namespace = ['langchain', 'stores', 'message', 'api'];
10
- this.apiUrl = config.apiUrl;
11
- this.sessionId = config.sessionId;
12
- this.apiKey = config.apiKey;
13
- }
14
- async getMessages() {
15
- try {
16
- const url = new URL(this.apiUrl);
17
- url.searchParams.set('sessionId', this.sessionId);
18
- url.searchParams.set('action', 'get');
19
- const headers = {
20
- 'Content-Type': 'application/json',
21
- };
22
- if (this.apiKey) {
23
- headers['Authorization'] = `Bearer ${this.apiKey}`;
24
- }
25
- const response = await fetch(url.toString(), {
26
- method: 'GET',
27
- headers,
28
- });
29
- if (!response.ok) {
30
- console.error(`Memory API error: ${response.status}`);
31
- return [];
32
- }
33
- const data = await response.json();
34
- const messages = data.messages || [];
35
- return messages.map((msg) => {
36
- switch (msg.type) {
37
- case 'human':
38
- return new messages_1.HumanMessage(msg.content);
39
- case 'ai':
40
- return new messages_1.AIMessage(msg.content);
41
- case 'system':
42
- return new messages_1.SystemMessage(msg.content);
43
- default:
44
- return new messages_1.HumanMessage(msg.content);
45
- }
46
- });
47
- }
48
- catch (error) {
49
- console.error('Failed to get messages from API:', error);
50
- return [];
51
- }
52
- }
53
- async addMessage(message) {
54
- try {
55
- const headers = {
56
- 'Content-Type': 'application/json',
57
- };
58
- if (this.apiKey) {
59
- headers['Authorization'] = `Bearer ${this.apiKey}`;
60
- }
61
- let type = 'human';
62
- if (message._getType() === 'ai') {
63
- type = 'ai';
64
- }
65
- else if (message._getType() === 'system') {
66
- type = 'system';
67
- }
68
- await fetch(this.apiUrl, {
69
- method: 'POST',
70
- headers,
71
- body: JSON.stringify({
72
- action: 'add',
73
- sessionId: this.sessionId,
74
- message: {
75
- type,
76
- content: message.content,
77
- },
78
- }),
79
- });
80
- }
81
- catch (error) {
82
- console.error('Failed to add message to API:', error);
83
- }
84
- }
85
- async addMessages(messages) {
86
- for (const message of messages) {
87
- await this.addMessage(message);
88
- }
89
- }
90
- async clear() {
91
- try {
92
- const headers = {
93
- 'Content-Type': 'application/json',
94
- };
95
- if (this.apiKey) {
96
- headers['Authorization'] = `Bearer ${this.apiKey}`;
97
- }
98
- await fetch(this.apiUrl, {
99
- method: 'POST',
100
- headers,
101
- body: JSON.stringify({
102
- action: 'clear',
103
- sessionId: this.sessionId,
104
- }),
105
- });
106
- }
107
- catch (error) {
108
- console.error('Failed to clear messages from API:', error);
109
- }
110
- }
111
- }
112
- exports.ApiMemoryHistory = ApiMemoryHistory;
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ApiMemoryHistory.js","sourceRoot":"","sources":["../../../src/nodes/Memory/ApiMemoryHistory.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseListChatMessageHistory,\n} from '@langchain/core/chat_history';\nimport {\n  BaseMessage,\n  HumanMessage,\n  AIMessage,\n  SystemMessage,\n} from '@langchain/core/messages';\n\nexport interface ApiMemoryHistoryConfig {\n  apiUrl: string;\n  sessionId: string;\n  apiKey?: string;\n}\n\ninterface ApiMessage {\n  type: 'human' | 'ai' | 'system';\n  content: string;\n}\n\ninterface ApiResponse {\n  messages?: ApiMessage[];\n}\n\nexport class ApiMemoryHistory extends BaseListChatMessageHistory {\n  lc_namespace = ['langchain', 'stores', 'message', 'api'];\n\n  private apiUrl: string;\n  private sessionId: string;\n  private apiKey?: string;\n\n  constructor(config: ApiMemoryHistoryConfig) {\n    super();\n    this.apiUrl = config.apiUrl;\n    this.sessionId = config.sessionId;\n    this.apiKey = config.apiKey;\n  }\n\n  async getMessages(): Promise<BaseMessage[]> {\n    try {\n      const url = new URL(this.apiUrl);\n      url.searchParams.set('sessionId', this.sessionId);\n      url.searchParams.set('action', 'get');\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n      };\n      if (this.apiKey) {\n        headers['Authorization'] = `Bearer ${this.apiKey}`;\n      }\n\n      const response = await fetch(url.toString(), {\n        method: 'GET',\n        headers,\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 ApiResponse;\n      const messages: ApiMessage[] = data.messages || [];\n\n      return messages.map((msg) => {\n        switch (msg.type) {\n          case 'human':\n            return new HumanMessage(msg.content);\n          case 'ai':\n            return new AIMessage(msg.content);\n          case 'system':\n            return new SystemMessage(msg.content);\n          default:\n            return new HumanMessage(msg.content);\n        }\n      });\n    } catch (error) {\n      console.error('Failed to get messages from API:', error);\n      return [];\n    }\n  }\n\n  async addMessage(message: BaseMessage): Promise<void> {\n    try {\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n      };\n      if (this.apiKey) {\n        headers['Authorization'] = `Bearer ${this.apiKey}`;\n      }\n\n      let type: 'human' | 'ai' | 'system' = 'human';\n      if (message._getType() === 'ai') {\n        type = 'ai';\n      } else if (message._getType() === 'system') {\n        type = 'system';\n      }\n\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers,\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('Failed to add message to API:', 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      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n      };\n      if (this.apiKey) {\n        headers['Authorization'] = `Bearer ${this.apiKey}`;\n      }\n\n      await fetch(this.apiUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          action: 'clear',\n          sessionId: this.sessionId,\n        }),\n      });\n    } catch (error) {\n      console.error('Failed to clear messages from API:', error);\n    }\n  }\n}\n"]}
@@ -1,189 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Memory = void 0;
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();
10
- this.apiUrl = apiUrl;
11
- this.sessionId = sessionId;
12
- this.apiKey = apiKey;
13
- this.lc_namespace = ['langchain', 'stores', 'message', 'api'];
14
- }
15
- getHeaders() {
16
- const headers = {
17
- 'Content-Type': 'application/json',
18
- };
19
- if (this.apiKey) {
20
- headers['Authorization'] = `Bearer ${this.apiKey}`;
21
- }
22
- return headers;
23
- }
24
- async getMessages() {
25
- console.log('[ApiChatHistory] getMessages called for session:', this.sessionId);
26
- try {
27
- console.log('[ApiChatHistory] Fetching from:', this.apiUrl);
28
- const response = await fetch(this.apiUrl, {
29
- method: 'POST',
30
- headers: this.getHeaders(),
31
- body: JSON.stringify({
32
- action: 'get',
33
- sessionId: this.sessionId,
34
- }),
35
- });
36
- if (!response.ok) {
37
- console.error(`[ApiChatHistory] API error: ${response.status}`);
38
- return [];
39
- }
40
- const data = await response.json();
41
- const messages = data?.messages ?? [];
42
- console.log('[ApiChatHistory] Got messages:', messages.length);
43
- return messages.map((m) => {
44
- if (m.type === 'human') {
45
- return new messages_1.HumanMessage(m.content);
46
- }
47
- else {
48
- return new messages_1.AIMessage(m.content);
49
- }
50
- });
51
- }
52
- catch (error) {
53
- console.error('[ApiChatHistory] getMessages error:', error);
54
- return [];
55
- }
56
- }
57
- async addMessage(message) {
58
- console.log('[ApiChatHistory] addMessage called for session:', this.sessionId);
59
- try {
60
- const type = message._getType() === 'human' ? 'human' : 'ai';
61
- await fetch(this.apiUrl, {
62
- method: 'POST',
63
- headers: this.getHeaders(),
64
- body: JSON.stringify({
65
- action: 'add',
66
- sessionId: this.sessionId,
67
- message: {
68
- type,
69
- content: message.content,
70
- },
71
- }),
72
- });
73
- console.log('[ApiChatHistory] Message added successfully');
74
- }
75
- catch (error) {
76
- console.error('[ApiChatHistory] addMessage error:', error);
77
- }
78
- }
79
- async addMessages(messages) {
80
- for (const message of messages) {
81
- await this.addMessage(message);
82
- }
83
- }
84
- async clear() {
85
- console.log('[ApiChatHistory] clear called for session:', this.sessionId);
86
- try {
87
- await fetch(this.apiUrl, {
88
- method: 'POST',
89
- headers: this.getHeaders(),
90
- body: JSON.stringify({
91
- action: 'clear',
92
- sessionId: this.sessionId,
93
- }),
94
- });
95
- }
96
- catch (error) {
97
- console.error('[ApiChatHistory] clear error:', error);
98
- }
99
- }
100
- }
101
- class Memory {
102
- constructor() {
103
- this.description = {
104
- displayName: 'Memory API',
105
- name: 'memoryApi',
106
- icon: 'fa:brain',
107
- group: ['transform'],
108
- version: 1,
109
- subtitle: 'External API Memory',
110
- description: 'Use an external API as chat memory storage',
111
- defaults: {
112
- name: 'Memory API',
113
- },
114
- codex: {
115
- categories: ['AI'],
116
- subcategories: {
117
- AI: ['Memory'],
118
- },
119
- resources: {
120
- primaryDocumentation: [
121
- {
122
- url: 'https://github.com/filipexyz/n8n-nodes-memory',
123
- },
124
- ],
125
- },
126
- },
127
- inputs: [],
128
- outputs: ['ai_memory'],
129
- outputNames: ['Memory'],
130
- properties: [
131
- {
132
- displayName: 'API URL',
133
- name: 'apiUrl',
134
- type: 'string',
135
- default: '',
136
- required: true,
137
- description: 'URL of the memory API (e.g., n8n webhook URL)',
138
- placeholder: 'https://your-n8n.com/webhook/memory',
139
- },
140
- {
141
- displayName: 'Session ID',
142
- name: 'sessionId',
143
- type: 'string',
144
- default: '={{ $json.sessionId }}',
145
- required: true,
146
- description: 'Unique identifier for the conversation session',
147
- },
148
- {
149
- displayName: 'API Key',
150
- name: 'apiKey',
151
- type: 'string',
152
- typeOptions: {
153
- password: true,
154
- },
155
- default: '',
156
- description: 'Optional API key for authentication',
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
- },
165
- ],
166
- };
167
- }
168
- async supplyData(itemIndex) {
169
- const apiUrl = this.getNodeParameter('apiUrl', itemIndex);
170
- const sessionId = this.getNodeParameter('sessionId', itemIndex);
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);
174
- const chatHistory = new ApiChatHistory(apiUrl, sessionId, apiKey);
175
- const memory = new memory_1.BufferWindowMemory({
176
- memoryKey: 'chat_history',
177
- chatHistory,
178
- returnMessages: true,
179
- inputKey: 'input',
180
- outputKey: 'output',
181
- k: contextWindowLength,
182
- });
183
- return {
184
- response: memory,
185
- };
186
- }
187
- }
188
- exports.Memory = Memory;
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"]}