@cpwc/node-red-contrib-ai-intent 3.1.0-alpha
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/.eslintrc +49 -0
- package/.idea/modules.xml +8 -0
- package/.idea/node-red-contrib-ai-intent.iml +12 -0
- package/.idea/vcs.xml +6 -0
- package/LICENSE +21 -0
- package/README.md +233 -0
- package/call-intent/icons/promotion-icon.svg +8 -0
- package/call-intent/index.html +114 -0
- package/call-intent/index.js +110 -0
- package/constants.js +31 -0
- package/database.js +9 -0
- package/examples/home-assistant-automation.json +167 -0
- package/examples/llm-chat-node-example.json +208 -0
- package/examples/openai-call-registered-intent-example.json +174 -0
- package/examples/openai-system-node-example.json +178 -0
- package/examples/openai-tool-node-example.json +120 -0
- package/examples/openai-user-node-exampe.json +234 -0
- package/geminiai-chat/geminiai-configuration/index.html +18 -0
- package/geminiai-chat/geminiai-configuration/index.js +7 -0
- package/geminiai-chat/icons/diamond.svg +8 -0
- package/geminiai-chat/icons/gemini-icon.svg +1 -0
- package/geminiai-chat/icons/gemini.svg +8 -0
- package/geminiai-chat/index.html +189 -0
- package/geminiai-chat/index.js +92 -0
- package/globalUtils.js +39 -0
- package/images/call_register_intent.jpeg +0 -0
- package/images/finally.jpg +0 -0
- package/images/set-config-node.gif +0 -0
- package/llm-chat/AzureOpenAIHelper.js +204 -0
- package/llm-chat/ChatGPTHelper.js +197 -0
- package/llm-chat/GeminiHelper.js +260 -0
- package/llm-chat/OllamaHelper.js +196 -0
- package/llm-chat/icons/bot-message-square.svg +1 -0
- package/llm-chat/icons/brain-circuit.svg +1 -0
- package/llm-chat/icons/chatgpt-icon.svg +7 -0
- package/llm-chat/index.html +205 -0
- package/llm-chat/index.js +73 -0
- package/llm-chat/platform-configuration/index.html +136 -0
- package/llm-chat/platform-configuration/index.js +16 -0
- package/localai-chat/icons/gem-icon.svg +1 -0
- package/localai-chat/icons/llama.svg +8 -0
- package/localai-chat/index.html +244 -0
- package/localai-chat/index.js +108 -0
- package/localai-chat/localai-configuration/index.html +18 -0
- package/localai-chat/localai-configuration/index.js +7 -0
- package/openai-chat/icons/chatgpt-icon.svg +7 -0
- package/openai-chat/index.html +196 -0
- package/openai-chat/index.js +58 -0
- package/openai-chat/openai-configuration/index.html +18 -0
- package/openai-chat/openai-configuration/index.js +7 -0
- package/openai-response/index.html +66 -0
- package/openai-response/index.js +154 -0
- package/openai-system/index.html +68 -0
- package/openai-system/index.js +28 -0
- package/openai-tool/index.html +57 -0
- package/openai-tool/index.js +50 -0
- package/openai-user/index.html +76 -0
- package/openai-user/index.js +26 -0
- package/package.json +49 -0
- package/register-intent/icons/register-icon.svg +8 -0
- package/register-intent/index.html +195 -0
- package/register-intent/index.js +72 -0
- package/register-intent/utils.js +10 -0
- package/utilities/chat-controller.js +249 -0
- package/utilities/chat-ledger.js +122 -0
- package/utilities/conversationHistory.js +68 -0
- package/utilities/format.js +94 -0
- package/utilities/gemini-controller.js +243 -0
- package/utilities/global-context.js +30 -0
- package/utilities/validateSchema.js +74 -0
package/.eslintrc
ADDED
@@ -0,0 +1,49 @@
|
|
1
|
+
{
|
2
|
+
"root": true,
|
3
|
+
"globals": {
|
4
|
+
"monaco": true,
|
5
|
+
"RED": true
|
6
|
+
},
|
7
|
+
"env": {
|
8
|
+
"browser": true,
|
9
|
+
"commonjs": true,
|
10
|
+
"es2021": true,
|
11
|
+
"mocha": true,
|
12
|
+
"jquery": true
|
13
|
+
},
|
14
|
+
"extends": [
|
15
|
+
"eslint:recommended",
|
16
|
+
"standard"
|
17
|
+
],
|
18
|
+
"plugins": [
|
19
|
+
"eslint-plugin-html",
|
20
|
+
"no-only-tests"
|
21
|
+
],
|
22
|
+
"parserOptions": {
|
23
|
+
"ecmaVersion": 12
|
24
|
+
},
|
25
|
+
"rules": {
|
26
|
+
// Built-in rules
|
27
|
+
"indent": [
|
28
|
+
"error",
|
29
|
+
2
|
30
|
+
],
|
31
|
+
"object-shorthand": [
|
32
|
+
"error"
|
33
|
+
],
|
34
|
+
"no-console": [
|
35
|
+
"error",
|
36
|
+
{
|
37
|
+
"allow": [
|
38
|
+
"debug",
|
39
|
+
"info",
|
40
|
+
"warn",
|
41
|
+
"error"
|
42
|
+
]
|
43
|
+
}
|
44
|
+
],
|
45
|
+
// plugin:no-only-tests
|
46
|
+
"no-only-tests/no-only-tests": "error",
|
47
|
+
"n/no-callback-literal": 0
|
48
|
+
}
|
49
|
+
}
|
@@ -0,0 +1,8 @@
|
|
1
|
+
<?xml version="1.0" encoding="UTF-8"?>
|
2
|
+
<project version="4">
|
3
|
+
<component name="ProjectModuleManager">
|
4
|
+
<modules>
|
5
|
+
<module fileurl="file://$PROJECT_DIR$/.idea/node-red-contrib-ai-intent.iml" filepath="$PROJECT_DIR$/.idea/node-red-contrib-ai-intent.iml" />
|
6
|
+
</modules>
|
7
|
+
</component>
|
8
|
+
</project>
|
@@ -0,0 +1,12 @@
|
|
1
|
+
<?xml version="1.0" encoding="UTF-8"?>
|
2
|
+
<module type="WEB_MODULE" version="4">
|
3
|
+
<component name="NewModuleRootManager">
|
4
|
+
<content url="file://$MODULE_DIR$">
|
5
|
+
<excludeFolder url="file://$MODULE_DIR$/.tmp" />
|
6
|
+
<excludeFolder url="file://$MODULE_DIR$/temp" />
|
7
|
+
<excludeFolder url="file://$MODULE_DIR$/tmp" />
|
8
|
+
</content>
|
9
|
+
<orderEntry type="inheritedJdk" />
|
10
|
+
<orderEntry type="sourceFolder" forTests="false" />
|
11
|
+
</component>
|
12
|
+
</module>
|
package/.idea/vcs.xml
ADDED
package/LICENSE
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
MIT License
|
2
|
+
|
3
|
+
Copyright (c) 2023 Michael Montaque
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
13
|
+
copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
@@ -0,0 +1,233 @@
|
|
1
|
+
# AI Intent Version 3
|
2
|
+
|
3
|
+
This is a collection of nodes to help enhance existing automations to be utilized by chatbots and take advantage of LLM's like GPT. There are 4 Nodes in this collection:
|
4
|
+
|
5
|
+
> Note: This uses GPT's Chat Completion API which is now considered Legacy. If it becomes deprecated. I will do my best to update this to leverage something equivalent.
|
6
|
+
### Breaking Changes
|
7
|
+
|
8
|
+
#### v1 - v2
|
9
|
+
There were changes under the hood how Intents are saved. This may cause some of your **Register Intent** and **Call Intent**
|
10
|
+
nodes to become invalid. Usually opening and closing the configuration of the nodes and redeploying should work.
|
11
|
+
If this doesn't work, you may need to redo the nodes from the palette.
|
12
|
+
|
13
|
+
#### v2 - v3
|
14
|
+
A lot of the nodes are being deprecated. The **User node**, **System Node**, **Response Node**, **Tool Node**, **Response Node**
|
15
|
+
**ChatGPT Node**, **Gemini Node**, **LocalAI Node**. Instead, use the **LLM Chat node**. This node replaces all of those nodes.
|
16
|
+
You can now write Function tools in the Register intent node which gives greater versatility.
|
17
|
+
> **PLEASE MIGRATE YOUR AUTOMATIONS TO USE THE LLM CHAT NODE.**
|
18
|
+
---
|
19
|
+
|
20
|
+
## LLM Chat Node
|
21
|
+
|
22
|
+
The **LLM Chat** node enables users to interact with a Large Language Model (LLM) and receive responses. It supports system and user messages, conversation tracking, and additional configuration options.
|
23
|
+
> Watch tutorial: https://youtu.be/2Efb1X6F5UY
|
24
|
+
---
|
25
|
+
|
26
|
+
## **Important**
|
27
|
+
To use this node, you need to configure the connection details. Consult the documentation for your chosen LLM platform (e.g., OpenAI, Ollama, Gemini) for specific instructions.
|
28
|
+
|
29
|
+
---
|
30
|
+
|
31
|
+
## **Inputs**
|
32
|
+
|
33
|
+
### **1. Main Input (`msg.payload`)**
|
34
|
+
The primary input for the LLM. It must include a `user` message and may optionally contain a `system` message.
|
35
|
+
|
36
|
+
#### **Example:**
|
37
|
+
```javascript
|
38
|
+
msg.payload = {
|
39
|
+
system: "You are a helpful assistant.",
|
40
|
+
user: "What is the capital of Alaska?"
|
41
|
+
};
|
42
|
+
```
|
43
|
+
- **`system`** (optional): Provides instructions to guide the model’s behavior.
|
44
|
+
- **`user`** (required): Contains the user's query or command.
|
45
|
+
|
46
|
+
---
|
47
|
+
|
48
|
+
### **2. Additional Options (`msg.payload.options`)**
|
49
|
+
Allows users to send extra parameters to customize LLM responses.
|
50
|
+
|
51
|
+
#### **Example:**
|
52
|
+
```javascript
|
53
|
+
msg.payload = {
|
54
|
+
user: "What is the capital of Alaska? Respond using JSON.",
|
55
|
+
options: {
|
56
|
+
format: "json" // Example for Ollama
|
57
|
+
}
|
58
|
+
};
|
59
|
+
```
|
60
|
+
|
61
|
+
The available options depend on the LLM platform and may include parameters such as:
|
62
|
+
- **`temperature`**: Controls randomness.
|
63
|
+
- **`max_tokens`**: Limits response length.
|
64
|
+
- **`format`**: Defines response structure (e.g., JSON).
|
65
|
+
|
66
|
+
Refer to the LLM's documentation for more details.
|
67
|
+
|
68
|
+
---
|
69
|
+
|
70
|
+
### **3. Clear Chat History (`msg.clearChatHistory`)**
|
71
|
+
- **Type**: `boolean` (optional)
|
72
|
+
- **Behavior**: If set to `true`, the conversation history will be cleared for the specified **Conversation Id**.
|
73
|
+
|
74
|
+
#### **Example:**
|
75
|
+
```javascript
|
76
|
+
msg.clearChatHistory = true;
|
77
|
+
```
|
78
|
+
|
79
|
+
---
|
80
|
+
|
81
|
+
## **Node Configuration**
|
82
|
+
|
83
|
+
### **1. Conversation Id**
|
84
|
+
- **Type**: `string`
|
85
|
+
- **Description**: This value is used as a key to store and track conversation history. If multiple nodes share the same Conversation Id, they will share the same conversation context.
|
86
|
+
- **If omitted**: Each call to the LLM will be independent, without historical context.
|
87
|
+
|
88
|
+
---
|
89
|
+
|
90
|
+
### **2. Tool Choice**
|
91
|
+
Defines whether the LLM should use function calling.
|
92
|
+
|
93
|
+
#### **Options:**
|
94
|
+
- **`None`**: Disables tools; the LLM won’t use functions.
|
95
|
+
- **`Automatic`**: Allows the LLM to decide when to use selected tools.
|
96
|
+
- **`Required`**: Forces the LLM to use at least one selected tool.
|
97
|
+
|
98
|
+
> **Note:** For Ollama, not all models support function calling.
|
99
|
+
|
100
|
+
---
|
101
|
+
|
102
|
+
### **3. Tools**
|
103
|
+
- **Type**: `string`
|
104
|
+
- **Description**: This field is populated by **Register Intent** nodes. It allows selecting one or more functions that the LLM can use.
|
105
|
+
- **Behavior**: The LLM will consider calling the selected functions based on the **Tool Choice** setting.
|
106
|
+
|
107
|
+
---
|
108
|
+
|
109
|
+
## **System and User Messages**
|
110
|
+
Messages must be passed in the `msg.payload` object:
|
111
|
+
```javascript
|
112
|
+
msg.payload = {
|
113
|
+
system: "You are a helpful assistant.",
|
114
|
+
user: "What is the capital of Alaska?"
|
115
|
+
};
|
116
|
+
```
|
117
|
+
|
118
|
+
The **system message** is optional but helps set the LLM’s behavior. The **user message** is required for every interaction.
|
119
|
+
|
120
|
+
To include additional response options:
|
121
|
+
```javascript
|
122
|
+
msg.payload = {
|
123
|
+
user: "What is the capital of Alaska? Respond using JSON.",
|
124
|
+
options: {
|
125
|
+
format: "json" // Example for Ollama
|
126
|
+
}
|
127
|
+
};
|
128
|
+
```
|
129
|
+
|
130
|
+
---
|
131
|
+
|
132
|
+
## **Summary**
|
133
|
+
- Use `msg.payload` with `user` (required) and `system` (optional) messages.
|
134
|
+
- Customize responses with `msg.payload.options`.
|
135
|
+
- Set `msg.clearChatHistory = true` to reset conversation history.
|
136
|
+
- Configure **Conversation Id** to maintain or separate conversation history.
|
137
|
+
- Choose **Tool Choice** and **Tools** to enable function calling when supported.
|
138
|
+
|
139
|
+
For more details, refer to the documentation of your selected LLM platform.
|
140
|
+
|
141
|
+
|
142
|
+
|
143
|
+
---
|
144
|
+
## Register Intent Node
|
145
|
+
|
146
|
+
### Purpose
|
147
|
+
The **Register Intent Node** allows users to define custom actions (intents) that can be triggered by the **Call Intent Node** or AI assistants like OpenAI.
|
148
|
+
> Watch Tutorial: https://youtu.be/FvP04OToeLQ
|
149
|
+
|
150
|
+
### Configuration
|
151
|
+
- **Name (string):** A unique identifier for the intent (alphanumeric, underscores, hyphens, max 64 characters).
|
152
|
+
- **Description (string):** A clear explanation of the intent’s purpose. If using AI assistants, this helps them understand when to trigger the intent.
|
153
|
+
- **Advanced Mode (boolean):** Enables structured AI interactions using a JSON-based schema.
|
154
|
+
- **Tool Schema (JSON, required if Advanced Mode is enabled):** Defines the expected parameters for AI-generated inputs.
|
155
|
+
|
156
|
+
### Usage
|
157
|
+
Place this node at the beginning of a flow to register an intent. Other flows or AI models can then call this intent dynamically. For AI integrations, ensure the description field is precise to improve intent recognition.
|
158
|
+
|
159
|
+
### Example JSON Schema for OpenAI
|
160
|
+
```json
|
161
|
+
{
|
162
|
+
"type": "object",
|
163
|
+
"properties": {
|
164
|
+
"eventName": {
|
165
|
+
"type": "string",
|
166
|
+
"description": "Unique event identifier."
|
167
|
+
},
|
168
|
+
"eventTime": {
|
169
|
+
"type": "string",
|
170
|
+
"format": "date-time",
|
171
|
+
"description": "ISO8601 formatted event time."
|
172
|
+
},
|
173
|
+
"eventPayload": {
|
174
|
+
"type": "string",
|
175
|
+
"description": "Command for smart home actions."
|
176
|
+
}
|
177
|
+
},
|
178
|
+
"required": ["eventName", "eventTime"],
|
179
|
+
"additionalProperties": false
|
180
|
+
}
|
181
|
+
```
|
182
|
+
|
183
|
+
### AI Platform Considerations
|
184
|
+
Different AI platforms may have unique function calling mechanisms. Refer to the official documentation for specifics:
|
185
|
+
- [OpenAI Function Calling](https://platform.openai.com/docs/guides/function-calling)
|
186
|
+
- [Google Gemini API](https://ai.google.dev/gemini-api/docs/function-calling)
|
187
|
+
- [Ollama Tool Support](https://ollama.com/blog/tool-support)
|
188
|
+
|
189
|
+
---
|
190
|
+
## Call Intent Node
|
191
|
+
|
192
|
+
### Purpose
|
193
|
+
The **Call Intent Node** is used to trigger an intent that has been previously registered using the **Register Intent Node**.
|
194
|
+
|
195
|
+
### Configuration
|
196
|
+
- **Name (string):** A label for the node.
|
197
|
+
- **Registered Node Name (string):** Select from a dropdown list of available registered intents or pass the intent dynamically via `msg.payload.nodeName`.
|
198
|
+
|
199
|
+
### Usage
|
200
|
+
This node acts as a trigger for registered intents. It can:
|
201
|
+
- Directly call an intent similar to a link node.
|
202
|
+
- Be placed after an **LLM Chat Node**, allowing AI to determine which intent to execute dynamically.
|
203
|
+
- Trigger multiple intents if the payload contains an array of intent names.
|
204
|
+
|
205
|
+
#### Example Payload for Multiple Calls
|
206
|
+
```json
|
207
|
+
{
|
208
|
+
"payload": [
|
209
|
+
{"nodeName": "turn_on_lights"},
|
210
|
+
{"nodeName": "set_thermostat"}
|
211
|
+
]
|
212
|
+
}
|
213
|
+
```
|
214
|
+
|
215
|
+
---
|
216
|
+
## LLM Chat Node
|
217
|
+
|
218
|
+
### Purpose
|
219
|
+
The **LLM Chat Node** integrates with various AI models to process and respond to natural language input dynamically.
|
220
|
+
|
221
|
+
### Usage
|
222
|
+
Place this node in a flow where dynamic AI-driven responses are required. It can:
|
223
|
+
- Answer user queries.
|
224
|
+
- Process input and pass structured responses to downstream nodes.
|
225
|
+
- Work in conjunction with **Register Intent** and **Call Intent Nodes** to enable complex AI-driven automations.
|
226
|
+
|
227
|
+
### Example Use Case
|
228
|
+
A user asks: *"What’s the weather today?"*
|
229
|
+
1. **LLM Chat Node** processes the request.
|
230
|
+
2. If the AI determines a weather function is needed, it triggers a **Call Intent Node**.
|
231
|
+
3. The **Call Intent Node** invokes a flow that retrieves weather data and returns a response.
|
232
|
+
|
233
|
+
---
|
@@ -0,0 +1,8 @@
|
|
1
|
+
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
2
|
+
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
3
|
+
<svg width="100%" height="100%" viewBox="0 0 513 512" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;">
|
4
|
+
<use xlink:href="#_Image1" x="70.593" y="80.545" width="370.865px" height="350.938px" transform="matrix(0.999637,0,0,0.999824,0,0)"/>
|
5
|
+
<defs>
|
6
|
+
<image id="_Image1" width="371px" height="351px" xlink:href=""/>
|
7
|
+
</defs>
|
8
|
+
</svg>
|
@@ -0,0 +1,114 @@
|
|
1
|
+
<script type="text/javascript">
|
2
|
+
const DYNAMIC_OPTION = {value:"", label:"Use payload.nodeName"}
|
3
|
+
|
4
|
+
|
5
|
+
const getIntentOptions = (intents = []) => {
|
6
|
+
const options = intents
|
7
|
+
.filter(option => {
|
8
|
+
return option.type === "Register Intent" && !option.excludeFromOpenAi
|
9
|
+
}).map(intent => {
|
10
|
+
return { value: intent.id, label: intent.name}
|
11
|
+
})
|
12
|
+
|
13
|
+
options.push(DYNAMIC_OPTION)
|
14
|
+
|
15
|
+
return options
|
16
|
+
}
|
17
|
+
$.getJSON('registered-intents', function (data = RED.settings.callIntentRegistry) {
|
18
|
+
window.__intentOptions = getIntentOptions(data)
|
19
|
+
|
20
|
+
initialize()
|
21
|
+
});
|
22
|
+
|
23
|
+
|
24
|
+
const initialize = () => {
|
25
|
+
RED.nodes.registerType("Call Intent", {
|
26
|
+
category: 'AI Intent',
|
27
|
+
color: '#1abc9c',
|
28
|
+
icon: "promotion-icon.svg",
|
29
|
+
defaults: {
|
30
|
+
name: { value: "" },
|
31
|
+
registeredNodeId: {
|
32
|
+
value: getIntentOptions(RED.settings.callIntentRegistry),
|
33
|
+
validate: (value) => {
|
34
|
+
return window.__intentOptions.some(option => {
|
35
|
+
return option.value === value
|
36
|
+
})
|
37
|
+
}
|
38
|
+
},
|
39
|
+
},
|
40
|
+
inputs: 1,
|
41
|
+
outputs: 0,
|
42
|
+
|
43
|
+
label: function () {
|
44
|
+
return this.name || "Call Intent";
|
45
|
+
},
|
46
|
+
|
47
|
+
oneditprepare: function () {
|
48
|
+
$("#node-input-message").typedInput({
|
49
|
+
type: "str",
|
50
|
+
types: ["str", "num", "bool", "json"],
|
51
|
+
typeField: "#node-input-message-type"
|
52
|
+
})
|
53
|
+
|
54
|
+
$.getJSON('registered-intents', function (data = RED.settings.callIntentRegistry) {
|
55
|
+
window.__intentOptions = getIntentOptions(data)
|
56
|
+
|
57
|
+
$("#node-input-registeredNodeId").typedInput({
|
58
|
+
types: [
|
59
|
+
{
|
60
|
+
value: "",
|
61
|
+
options: window.__intentOptions
|
62
|
+
}
|
63
|
+
]
|
64
|
+
})
|
65
|
+
});
|
66
|
+
|
67
|
+
|
68
|
+
}
|
69
|
+
|
70
|
+
});
|
71
|
+
}
|
72
|
+
|
73
|
+
|
74
|
+
|
75
|
+
</script>
|
76
|
+
|
77
|
+
<script type="text/html" data-template-name="Call Intent">
|
78
|
+
<div style="display: flex; justify-content: center; margin-bottom: 25px;">
|
79
|
+
<a href="https://youtu.be/FvP04OToeLQ" target="_blank" referrerpolicy="no-referrer" style="color: #f53b57"><i
|
80
|
+
class="fa fa-youtube"></i><span style="padding-left: 10px;">Watch
|
81
|
+
Call Intent Node Tutorial</span></a>
|
82
|
+
</div>
|
83
|
+
<div class="form-row">
|
84
|
+
<label for="node-input-name"><i class="fa fa-tag"></i> Name</label>
|
85
|
+
<input type="text" id="node-input-name" placeholder="Name">
|
86
|
+
</div>
|
87
|
+
|
88
|
+
<div class="form-row">
|
89
|
+
<label for="node-input-registeredNodeId"><i class="fa fa-tag"></i> Registered Node Name</label>
|
90
|
+
<input type="text" id="node-input-registeredNodeId" placeholder="Name of registered node">
|
91
|
+
</div>
|
92
|
+
|
93
|
+
</script>
|
94
|
+
|
95
|
+
<script type="text/html" data-help-name="Call Intent">
|
96
|
+
<p>Triggers the Register Intent node that has the corresponding id.</p>
|
97
|
+
|
98
|
+
<h3>Inputs</h3>
|
99
|
+
<dl class="message-properties">
|
100
|
+
<dt>Registered Node Name
|
101
|
+
<span class="property-type">string</span>
|
102
|
+
</dt>
|
103
|
+
<dd>Name of the registed intent node you want to trigger. For convenience you can choose the name from the
|
104
|
+
the dropdown however, you can choose payload.nodeName to allow the node to read the
|
105
|
+
intent name from the msg object - <code>msg.payload.nodeName</code>. This function can also fire multiple times
|
106
|
+
if the payload is an array of the objects with nodeName <code>msg.payload = [{nodeName: "fake_name"}]</code>.
|
107
|
+
</dd>
|
108
|
+
</dl>
|
109
|
+
|
110
|
+
<h3>Details</h3>
|
111
|
+
<p>This node can be used as a standalone way to trigger custom intents (this will act very similar to the link in/out nodes)
|
112
|
+
or this node can be used immediately after the <code>OpenAI Response</code> node. If used after the aforementioned node
|
113
|
+
OpenAI can dynamically trigger any registered intents.</p>
|
114
|
+
</script>
|
@@ -0,0 +1,110 @@
|
|
1
|
+
const PubSub = require("pubsub-js");
|
2
|
+
const { end, ContextDatabase } = require("../globalUtils");
|
3
|
+
const { TYPES} = require("../constants");
|
4
|
+
|
5
|
+
/**
|
6
|
+
* Searches context for an object whose `name` or `id` property matches the given name parameter
|
7
|
+
* and returns the matching object.
|
8
|
+
* @param {string} name
|
9
|
+
* @param {Record<string, object>} context
|
10
|
+
* @returns
|
11
|
+
*/
|
12
|
+
const getMatchingIntentFromContext = (nameOrID = "", context) => {
|
13
|
+
return Object.values(context).find((intent) => {
|
14
|
+
return (
|
15
|
+
(intent.name === nameOrID || intent.id === nameOrID) &&
|
16
|
+
intent.type === TYPES.RegisterIntent
|
17
|
+
);
|
18
|
+
});
|
19
|
+
};
|
20
|
+
|
21
|
+
/**
|
22
|
+
* Fires the callback with either an error string or with the node matching the `nameOrId` value
|
23
|
+
* from the context.
|
24
|
+
* @param {string} nameOrId
|
25
|
+
* @param {object} context
|
26
|
+
* @param {function} callback
|
27
|
+
*/
|
28
|
+
const getNode = (nameOrId, context, callback) => {
|
29
|
+
const node = getMatchingIntentFromContext(nameOrId, context);
|
30
|
+
if (!nameOrId) {
|
31
|
+
callback("payload is missing nodeName property");
|
32
|
+
} else if (!node) {
|
33
|
+
callback(`There is no registered intent with name or id of "${nameOrId}"`);
|
34
|
+
} else {
|
35
|
+
callback(null, node);
|
36
|
+
}
|
37
|
+
};
|
38
|
+
|
39
|
+
const normalizeNames = (intents = []) => {
|
40
|
+
return intents.map((intent) => {
|
41
|
+
if (!intent.name && intent.type === TYPES.OpenAITool) {
|
42
|
+
const tool = JSON.parse(intent.tool);
|
43
|
+
return { ...intent, name: tool.function.name };
|
44
|
+
}
|
45
|
+
return intent;
|
46
|
+
});
|
47
|
+
};
|
48
|
+
|
49
|
+
module.exports = function (RED) {
|
50
|
+
const nodeDB = new ContextDatabase(RED);
|
51
|
+
|
52
|
+
function CallIntentHandlerNode(config) {
|
53
|
+
RED.nodes.createNode(this, config);
|
54
|
+
const node = this;
|
55
|
+
|
56
|
+
this.on("input", function (msg, send, done = () => {}) {
|
57
|
+
const nodeStore = nodeDB.getNodeStore();
|
58
|
+
const { registeredNodeId = "" } = config;
|
59
|
+
|
60
|
+
send =
|
61
|
+
send ||
|
62
|
+
function () {
|
63
|
+
node.send.apply(node, arguments);
|
64
|
+
};
|
65
|
+
|
66
|
+
if (Array.isArray(msg.payload)) {
|
67
|
+
msg.payload.forEach((payload, index) => {
|
68
|
+
const { nodeName } = payload;
|
69
|
+
const body = {...msg, index}
|
70
|
+
getNode(nodeName, nodeStore, (err, registeredNode) => {
|
71
|
+
if (err) {
|
72
|
+
node.warn(err);
|
73
|
+
} else {
|
74
|
+
PubSub.publishSync(registeredNode.id, body);
|
75
|
+
send(body);
|
76
|
+
}
|
77
|
+
});
|
78
|
+
});
|
79
|
+
}
|
80
|
+
else {
|
81
|
+
const nameOrId = msg.payload?.nodeName || registeredNodeId;
|
82
|
+
|
83
|
+
getNode(nameOrId, nodeStore, (err, registeredNode) => {
|
84
|
+
if (err) {
|
85
|
+
node.warn(err);
|
86
|
+
} else {
|
87
|
+
PubSub.publishSync(registeredNode.id, msg);
|
88
|
+
send(msg);
|
89
|
+
}
|
90
|
+
});
|
91
|
+
}
|
92
|
+
end(done);
|
93
|
+
});
|
94
|
+
}
|
95
|
+
|
96
|
+
RED.httpAdmin.get("/registered-intents", function (req, res) {
|
97
|
+
const nodeStore = nodeDB.getNodeStore();
|
98
|
+
const intents = normalizeNames(Object.values(nodeStore))
|
99
|
+
res.json(intents);
|
100
|
+
});
|
101
|
+
|
102
|
+
RED.nodes.registerType(TYPES.CallIntent, CallIntentHandlerNode, {
|
103
|
+
settings: {
|
104
|
+
callIntentRegistry: {
|
105
|
+
value: [],
|
106
|
+
exportable: true,
|
107
|
+
},
|
108
|
+
},
|
109
|
+
});
|
110
|
+
};
|
package/constants.js
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
module.exports = {
|
2
|
+
OPEN_AI_KEY: "openaiAPIKey",
|
3
|
+
GEMINI_AI_KEY: "geminiaiAPIKey",
|
4
|
+
LOCAL_STORAGE_PATH: "localStoragePath",
|
5
|
+
INTENT_STORE: "intent_store",
|
6
|
+
ACTIVE_CONVERSATION: "active_conversation_context",
|
7
|
+
CONVERSATION_CONTEXT: "conversation_context",
|
8
|
+
TYPES: {
|
9
|
+
RegisterIntent: "Register Intent",
|
10
|
+
CallIntent: "Call Intent",
|
11
|
+
OpenAIChat: "OpenAI Chat",
|
12
|
+
LLMChat: "LLM Chat",
|
13
|
+
LocalAIChat: "LocalAI Chat",
|
14
|
+
GeminiaiChat: "GeminiAI Chat",
|
15
|
+
OpenAITool: "OpenAI Tool",
|
16
|
+
OpenAIUser: "OpenAI User",
|
17
|
+
OpenAISystem: "OpenAI System",
|
18
|
+
OpenAIResponse: "OpenAI Response",
|
19
|
+
},
|
20
|
+
ROLES: {
|
21
|
+
User: "user",
|
22
|
+
System: "system",
|
23
|
+
Assistant: "assistant",
|
24
|
+
},
|
25
|
+
TOOL_CHOICE: {
|
26
|
+
Any: "any",
|
27
|
+
None: "none",
|
28
|
+
Auto: "auto",
|
29
|
+
Required: "required"
|
30
|
+
},
|
31
|
+
};
|
package/database.js
ADDED
@@ -0,0 +1,9 @@
|
|
1
|
+
if (typeof localStorage === "undefined" || localStorage === null) {
|
2
|
+
var LocalStorage = require("node-localstorage").LocalStorage;
|
3
|
+
}
|
4
|
+
|
5
|
+
const getStorageAtLocation = (location = "./localstore") => {
|
6
|
+
return new LocalStorage(location);
|
7
|
+
}
|
8
|
+
|
9
|
+
module.exports = { getStorageAtLocation };
|