@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
@@ -0,0 +1,205 @@
|
|
1
|
+
<style>
|
2
|
+
.disabled{
|
3
|
+
opacity: .5;
|
4
|
+
pointer-events: none;
|
5
|
+
}
|
6
|
+
</style>
|
7
|
+
<script type="text/javascript">
|
8
|
+
(() => {
|
9
|
+
const TOOL_CHOICE = [
|
10
|
+
{ value: "none", label: "None" },
|
11
|
+
{ value: "auto", label: "Automatic" },
|
12
|
+
{ value: "required", label: "Required" }
|
13
|
+
]
|
14
|
+
|
15
|
+
const removeDuplicates = (data) => {
|
16
|
+
const intents = {}
|
17
|
+
const tools = []
|
18
|
+
|
19
|
+
data
|
20
|
+
.forEach((intent => {
|
21
|
+
if (intent.type === "OpenAI Tool") {
|
22
|
+
if (!intents[intent.name]) {
|
23
|
+
intents[intent.name] = true
|
24
|
+
tools.push(intent)
|
25
|
+
}
|
26
|
+
} else {
|
27
|
+
tools.push(intent)
|
28
|
+
}
|
29
|
+
}))
|
30
|
+
|
31
|
+
return tools
|
32
|
+
}
|
33
|
+
|
34
|
+
$.getJSON('registered-intents', function (data = RED.settings.callIntentRegistry) {
|
35
|
+
const tools = removeDuplicates(data)
|
36
|
+
window.__tools = getToolOptions(tools)
|
37
|
+
initialize()
|
38
|
+
});
|
39
|
+
|
40
|
+
const getToolOptions = (intents = []) => {
|
41
|
+
return intents
|
42
|
+
.filter(option => {
|
43
|
+
// Remove excluded tool selection
|
44
|
+
// TODO: Deprecated
|
45
|
+
console.log(option.excludeFromOpenAi)
|
46
|
+
return !option.excludeFromOpenAi
|
47
|
+
})
|
48
|
+
.map(intent => {
|
49
|
+
|
50
|
+
if(intent.type === "Register Intent"){
|
51
|
+
return { value: intent.name, label: `${intent.name} (Registered Intent)`}
|
52
|
+
}else {
|
53
|
+
const tool = JSON.parse(intent.tool)
|
54
|
+
return { value: tool.function.name, label: `${intent.name} (Deprecated)`}
|
55
|
+
}
|
56
|
+
})
|
57
|
+
}
|
58
|
+
|
59
|
+
RED.nodes.registerType("LLM Chat", {
|
60
|
+
category: 'AI Intent',
|
61
|
+
color: '#1abc9c',
|
62
|
+
icon:"bot-message-square.svg",
|
63
|
+
defaults: {
|
64
|
+
name: { value: "" },
|
65
|
+
platform: { value: "", type: "platform-configuration", required: false },
|
66
|
+
tools: {value: ""},
|
67
|
+
tool_choice: {value: "none"},
|
68
|
+
conversation_id: {value: ""}
|
69
|
+
|
70
|
+
},
|
71
|
+
inputs: 1,
|
72
|
+
outputs: 1,
|
73
|
+
paletteLabel: "LLM Chat",
|
74
|
+
label: function () {
|
75
|
+
return this.name || "Chat";
|
76
|
+
},
|
77
|
+
|
78
|
+
oneditprepare: function(){
|
79
|
+
$.getJSON('registered-intents', function (data = RED.settings.callIntentRegistry) {
|
80
|
+
const tools = removeDuplicates(data)
|
81
|
+
window.__tools = getToolOptions(tools)
|
82
|
+
|
83
|
+
$("#node-input-tools").typedInput({
|
84
|
+
types: [
|
85
|
+
{
|
86
|
+
value: "",
|
87
|
+
multiple: true,
|
88
|
+
options: window.__tools
|
89
|
+
}
|
90
|
+
]
|
91
|
+
})
|
92
|
+
});
|
93
|
+
$("#node-input-tool_choice").typedInput({
|
94
|
+
types: [
|
95
|
+
{
|
96
|
+
value: "",
|
97
|
+
options: TOOL_CHOICE
|
98
|
+
}
|
99
|
+
]
|
100
|
+
}).on("change", function() {
|
101
|
+
if ($(this).val() === "none") {
|
102
|
+
$("#tools").addClass("disabled");
|
103
|
+
} else {
|
104
|
+
$("#tools").removeClass("disabled");
|
105
|
+
}
|
106
|
+
})
|
107
|
+
}
|
108
|
+
});
|
109
|
+
})()
|
110
|
+
|
111
|
+
</script>
|
112
|
+
|
113
|
+
<script type="text/html" data-template-name="LLM Chat">
|
114
|
+
<div style="display: flex; justify-content: center; margin-bottom: 25px;">
|
115
|
+
<a href="https://youtu.be/2Efb1X6F5UY" target="_blank" referrerpolicy="no-referrer"
|
116
|
+
style="color: #f53b57"><i class="fa fa-youtube"></i><span style="padding-left: 10px;">Watch
|
117
|
+
LLM Chat Node Tutorial</span></a>
|
118
|
+
</div>
|
119
|
+
<div class="form-row">
|
120
|
+
<label for="node-input-name"> Name</label>
|
121
|
+
<input type="text" id="node-input-name" placeholder="Name">
|
122
|
+
</div>
|
123
|
+
|
124
|
+
<div class="form-row">
|
125
|
+
<label for="node-input-platform"> Configuration</label>
|
126
|
+
<input type="text" id="node-input-platform" placeholder="0a1b2c3b4d5e6f">
|
127
|
+
</div>
|
128
|
+
|
129
|
+
|
130
|
+
<div class="form-row">
|
131
|
+
<label for="node-input-conversation_id"> Conversation Id</label>
|
132
|
+
|
133
|
+
<input type="text" id="node-input-conversation_id">
|
134
|
+
</div>
|
135
|
+
|
136
|
+
<div class="form-row">
|
137
|
+
<label for="node-input-tool_choice">Tool Choice</label>
|
138
|
+
|
139
|
+
<input type="text" id="node-input-tool_choice">
|
140
|
+
</div>
|
141
|
+
|
142
|
+
<div class="form-row" id="tools">
|
143
|
+
<label for="node-input-tools"> Tools</label>
|
144
|
+
<input type="text" id="node-input-tools">
|
145
|
+
</div>
|
146
|
+
|
147
|
+
</script>
|
148
|
+
|
149
|
+
<script type="text/html" data-help-name="LLM Chat">
|
150
|
+
<p>Calls an LLM and returns the response.</p>
|
151
|
+
|
152
|
+
<h3>Important</h3>
|
153
|
+
<p>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.</p>
|
154
|
+
|
155
|
+
<h3>Inputs</h3>
|
156
|
+
<dl class="message-properties">
|
157
|
+
<dt><code>msg.payload</code> <span class="property-type">object</span></dt>
|
158
|
+
<dd>
|
159
|
+
<p>The main input to the LLM. Should contain <code>user</code> and optionally <code>system</code> messages.</p>
|
160
|
+
<p>Example: <code>msg.payload = { system: "You are a helpful assistant.", user: "What is the capital of Alaska?" }</code></p>
|
161
|
+
</dd>
|
162
|
+
<dt><code>msg.payload.options</code> <span class="property-type">object</span></dt>
|
163
|
+
<dd>Optional. Allows passing additional parameters to the LLM. See the LLM's documentation for available options (e.g., <code>temperature</code>, <code>max_tokens</code>, <code>format</code> for your platform).</dd>
|
164
|
+
<dt><code>msg.clearChatHistory</code> <span class="property-type">boolean</span></dt>
|
165
|
+
<dd>Optional. If set to <code>true</code>, the conversation history will be cleared for the specified <code>Conversation Id</code>.</dd>
|
166
|
+
</dl>
|
167
|
+
|
168
|
+
|
169
|
+
<h3>Node Configuration</h3>
|
170
|
+
<dl>
|
171
|
+
<dt>Conversation Id <span class="property-type">string</span></dt>
|
172
|
+
<dd>The value provided will be used as a key to store the conversation. The node will keep track of all the messages. If a different node uses the same key, the conversation will be connected and will use the same conversation history. If omitted, each call to the LLM will be isolated.</dd>
|
173
|
+
|
174
|
+
<dt>Tool Choice <span class="property-type">dropdown</span></dt>
|
175
|
+
<dd>This property lets you decide if the LLM should use function calling. If you are using Ollama, not every model supports function calling.
|
176
|
+
<ul>
|
177
|
+
<li><code>None</code>: Disables the Tools setting and the LLM will not use any functions.</li>
|
178
|
+
<li><code>Automatic</code>: Lets the LLM decide if it should choose one of the selected tools.</li>
|
179
|
+
<li><code>Required</code>: Forces the LLM to use at least one of the selected functions.</li>
|
180
|
+
</ul>
|
181
|
+
</dd>
|
182
|
+
|
183
|
+
<dt>Tools <span class="property-type">string</span></dt>
|
184
|
+
<dd>This field is populated by Register Intent nodes. You can choose zero or more functions. Any selected function will be sent to the LLM, which will determine whether or not to call the function depending on the Tool Choice setting.</dd>
|
185
|
+
</dl>
|
186
|
+
|
187
|
+
<h3>System and User Messages</h3>
|
188
|
+
<p>You can pass messages to the LLM by setting <code>msg.payload = { system, user }</code> where <code>system</code> and <code>user</code> are string values. You don't always have to pass <code>system</code>, but the <code>user</code> property is required. For example:</p>
|
189
|
+
<pre>
|
190
|
+
msg.payload = {
|
191
|
+
system: "You are a helpful assistant.",
|
192
|
+
user: "What is the capital of Alaska?"
|
193
|
+
};
|
194
|
+
</pre>
|
195
|
+
<p>To pass additional options (like temperature or max tokens), use <code>msg.payload.options</code>:</p>
|
196
|
+
<pre>
|
197
|
+
msg.payload = {
|
198
|
+
user: "What is capital of alaska. Respond using JSON.",
|
199
|
+
options: {
|
200
|
+
format: "json" // Example for Ollama
|
201
|
+
}
|
202
|
+
};
|
203
|
+
</pre>
|
204
|
+
|
205
|
+
</script>
|
@@ -0,0 +1,73 @@
|
|
1
|
+
const { end } = require("../globalUtils");
|
2
|
+
const { chatGPTHelper } = require("./ChatGPTHelper");
|
3
|
+
const { ollamaHelper } = require("./OllamaHelper");
|
4
|
+
const { geminiHelper } = require("./GeminiHelper");
|
5
|
+
const { azureOpenAIHelper } = require("./AzureOpenAIHelper");
|
6
|
+
const { TYPES } = require("../constants");
|
7
|
+
|
8
|
+
const PLATFORM = [
|
9
|
+
{ value: "gpt", label: "ChatGPT" },
|
10
|
+
{ value: "gemini", label: "Gemini" },
|
11
|
+
{ value: "ollama", label: "Ollama" },
|
12
|
+
{ value: "azureopenai", label: "Azure OpenAI" },
|
13
|
+
];
|
14
|
+
|
15
|
+
module.exports = function (RED) {
|
16
|
+
function LLMChatNode(config) {
|
17
|
+
RED.nodes.createNode(this, config);
|
18
|
+
this.platform = RED.nodes.getNode(config.platform);
|
19
|
+
const node = this;
|
20
|
+
this.on("input", function (msg, send, done = () => { }) {
|
21
|
+
try {
|
22
|
+
node.status({ fill: "green", shape: "ring", text: "Working..." });
|
23
|
+
switch (node.platform.platform) {
|
24
|
+
case "gpt":
|
25
|
+
chatGPTHelper({ node, RED, config, msg }, finish)
|
26
|
+
break
|
27
|
+
case "ollama":
|
28
|
+
ollamaHelper({ node, RED, config, msg }, finish)
|
29
|
+
break
|
30
|
+
case "gemini":
|
31
|
+
geminiHelper({ node, RED, config, msg }, finish)
|
32
|
+
break
|
33
|
+
case "azureopenai":
|
34
|
+
azureOpenAIHelper({ node, RED, config, msg }, finish)
|
35
|
+
break
|
36
|
+
default:
|
37
|
+
node.status({ fill: "red", shape: "ring", text: "Invalid configuration platform" });
|
38
|
+
finish(`The configuration of this node is invalid platform ${node.platform.platform}`)
|
39
|
+
}
|
40
|
+
} catch (e) {
|
41
|
+
finish(e)
|
42
|
+
}
|
43
|
+
|
44
|
+
/**
|
45
|
+
* Callback function to handle errors and completed flows
|
46
|
+
* @param error
|
47
|
+
* @param payload
|
48
|
+
*/
|
49
|
+
function finish(error, payload) {
|
50
|
+
if (error) {
|
51
|
+
node.status({ fill: "red", shape: "dot", text: "Error" });
|
52
|
+
return end(done, error)
|
53
|
+
} else if (send && payload) {
|
54
|
+
send(payload)
|
55
|
+
} else if (payload) {
|
56
|
+
config.send.apply(node, arguments);
|
57
|
+
end(done)
|
58
|
+
}
|
59
|
+
node.status({ fill: "grey", shape: "dot", text: "Done" });
|
60
|
+
|
61
|
+
}
|
62
|
+
});
|
63
|
+
}
|
64
|
+
|
65
|
+
RED.nodes.registerType(TYPES.LLMChat, LLMChatNode, {
|
66
|
+
settings: {
|
67
|
+
lLMChatPlatforms: {
|
68
|
+
value: PLATFORM,
|
69
|
+
exportable: true,
|
70
|
+
},
|
71
|
+
},
|
72
|
+
});
|
73
|
+
};
|
@@ -0,0 +1,136 @@
|
|
1
|
+
<style>
|
2
|
+
.hidden {
|
3
|
+
display: none;
|
4
|
+
}
|
5
|
+
</style>
|
6
|
+
|
7
|
+
<script type="text/javascript">
|
8
|
+
RED.nodes.registerType('platform-configuration', {
|
9
|
+
category: 'config',
|
10
|
+
defaults: {
|
11
|
+
name: { value: "", required: true },
|
12
|
+
platform: { value: "gpt", required: true },
|
13
|
+
url: { value: "" },
|
14
|
+
model: { value: "gpt-4o", required: true },
|
15
|
+
},
|
16
|
+
credentials: {
|
17
|
+
api: { type: "text" },
|
18
|
+
},
|
19
|
+
label: function () {
|
20
|
+
return this.name
|
21
|
+
},
|
22
|
+
oneditprepare: function () {
|
23
|
+
$("#node-config-input-platform").typedInput({
|
24
|
+
types: [
|
25
|
+
{
|
26
|
+
value: "gpt",
|
27
|
+
options: [
|
28
|
+
{ value: "gpt", label: "ChatGPT" },
|
29
|
+
{ value: "gemini", label: "Gemini" },
|
30
|
+
{ value: "ollama", label: "Ollama" },
|
31
|
+
{ value: "azureopenai", label: "Azure OpenAI" },
|
32
|
+
]
|
33
|
+
}
|
34
|
+
]
|
35
|
+
}).on("change", function () {
|
36
|
+
if ($(this).val() === "ollama") {
|
37
|
+
$("#api").addClass("hidden");
|
38
|
+
$("#url").removeClass("hidden");
|
39
|
+
$("#endpoint").addClass("hidden");
|
40
|
+
} else if ($(this).val() === "azureopenai") {
|
41
|
+
$("#api").removeClass("hidden");
|
42
|
+
$("#url").addClass("hidden");
|
43
|
+
$("#endpoint").removeClass("hidden");
|
44
|
+
} else {
|
45
|
+
$("#api").removeClass("hidden");
|
46
|
+
$("#url").addClass("hidden");
|
47
|
+
$("#endpoint").addClass("hidden");
|
48
|
+
}
|
49
|
+
})
|
50
|
+
}
|
51
|
+
});
|
52
|
+
</script>
|
53
|
+
|
54
|
+
<script type="text/html" data-template-name="platform-configuration">
|
55
|
+
<div class="form-row" id="name">
|
56
|
+
<label for="node-config-input-name">Name</label>
|
57
|
+
<input type="text" id="node-config-input-name">
|
58
|
+
</div>
|
59
|
+
|
60
|
+
<div class="form-row" id="platform">
|
61
|
+
<label for="node-config-input-platform">Platform</label>
|
62
|
+
<input type="text" id="node-config-input-platform">
|
63
|
+
</div>
|
64
|
+
|
65
|
+
<div class="form-row" id="model">
|
66
|
+
<label for="node-config-input-model">Model</label>
|
67
|
+
<input type="text" id="node-config-input-model">
|
68
|
+
</div>
|
69
|
+
|
70
|
+
|
71
|
+
<div class="form-row" id="api">
|
72
|
+
<label for="node-config-input-api"><i class="fa fa-bookmark"></i> API Token</label>
|
73
|
+
<input type="text" id="node-config-input-api">
|
74
|
+
</div>
|
75
|
+
|
76
|
+
<div class="form-row hidden" id="url">
|
77
|
+
<label for="node-config-input-url"><i class="fa fa-internet-explorer"></i>Ollama URL</label>
|
78
|
+
<input type="text" id="node-config-input-url">
|
79
|
+
</div>
|
80
|
+
|
81
|
+
<div class="form-row hidden" id="endpoint">
|
82
|
+
<label for="node-config-input-endpoint"><i class="fa fa-internet-explorer"></i>Azure OpenAI Endpoint URL</label>
|
83
|
+
<input type="text" id="node-config-input-endpoint">
|
84
|
+
</div>
|
85
|
+
</script>
|
86
|
+
|
87
|
+
<script type="text/html" data-help-name="platform-configuration">
|
88
|
+
<p>This configuration node allows you to set up an AI platform for use with other AI-related nodes in Node-RED.</p>
|
89
|
+
|
90
|
+
<h3>Configuration Options</h3>
|
91
|
+
<dl class="message-properties">
|
92
|
+
<dt class="required">Name
|
93
|
+
<span class="property-type">string</span>
|
94
|
+
</dt>
|
95
|
+
<dd>A custom name for this configuration.</dd>
|
96
|
+
|
97
|
+
<dt class="required">Platform
|
98
|
+
<span class="property-type">dropdown</span>
|
99
|
+
</dt>
|
100
|
+
<dd>Select the AI platform to use:
|
101
|
+
<ul>
|
102
|
+
<li><b>ChatGPT</b> – OpenAI models (e.g., GPT-4o)</li>
|
103
|
+
<li><b>Gemini</b> – Google AI models</li>
|
104
|
+
<li><b>Ollama</b> – Local AI models</li>
|
105
|
+
<li><b>Azure OpenAI</b> – Azure OpenAI models (e.g., GPT-4o)</li>
|
106
|
+
</ul>
|
107
|
+
</dd>
|
108
|
+
|
109
|
+
<dt class="required">Model
|
110
|
+
<span class="property-type">string</span>
|
111
|
+
</dt>
|
112
|
+
<dd>The specific AI model to use. Defaults to "gpt-4o" for OpenAI or Azure OpenAI.</dd>
|
113
|
+
|
114
|
+
<dt class="optional">API Token
|
115
|
+
<span class="property-type">string</span>
|
116
|
+
</dt>
|
117
|
+
<dd>Required for cloud-based models (OpenAI, Gemini). Enter your API key to authenticate requests.</dd>
|
118
|
+
|
119
|
+
<dt class="optional">Ollama URL
|
120
|
+
<span class="property-type">string</span>
|
121
|
+
</dt>
|
122
|
+
<dd>Required only when using Ollama. Enter the local Ollama server URL.</dd>
|
123
|
+
|
124
|
+
<dt class="optional">Azure OpenAI Endpoint
|
125
|
+
<span class="property-type">string</span>
|
126
|
+
</dt>
|
127
|
+
<dd>Required only when using Azure OpenAI. Enter the given Azure OpenAI Endpoint URL.</dd>
|
128
|
+
</dl>
|
129
|
+
|
130
|
+
<h3>Behavior</h3>
|
131
|
+
<p>If "Ollama" is selected as the platform, the API Token field is hidden, and the Ollama URL field is shown instead.</p>
|
132
|
+
|
133
|
+
<h3>Usage</h3>
|
134
|
+
<p>This configuration node is referenced by AI-related nodes to send requests to the selected AI platform. Ensure it is set up correctly before use.</p>
|
135
|
+
|
136
|
+
</script>
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module.exports = function (RED) {
|
2
|
+
function PlatformConfigurationNode(prop) {
|
3
|
+
RED.nodes.createNode(this, prop);
|
4
|
+
this.url = prop.url;
|
5
|
+
this.platform = prop.platform;
|
6
|
+
this.name = prop.name;
|
7
|
+
this.model = prop.model;
|
8
|
+
}
|
9
|
+
|
10
|
+
RED.nodes.registerType("platform-configuration", PlatformConfigurationNode, {
|
11
|
+
credentials: {
|
12
|
+
api: { type: "text" },
|
13
|
+
endpoint: { type: "text" }
|
14
|
+
}
|
15
|
+
});
|
16
|
+
};
|
@@ -0,0 +1 @@
|
|
1
|
+
<svg fill="none" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"><path d="M16 8.016A8.522 8.522 0 008.016 16h-.032A8.521 8.521 0 000 8.016v-.032A8.521 8.521 0 007.984 0h.032A8.522 8.522 0 0016 7.984v.032z" fill="url(#prefix__paint0_radial_980_20147)"/><defs><radialGradient id="prefix__paint0_radial_980_20147" cx="0" cy="0" r="1" gradientUnits="userSpaceOnUse" gradientTransform="matrix(16.1326 5.4553 -43.70045 129.2322 1.588 6.503)"><stop offset=".067" stop-color="#9168C0"/><stop offset=".343" stop-color="#5684D1"/><stop offset=".672" stop-color="#1BA1E3"/></radialGradient></defs></svg>
|
@@ -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 683 683" 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-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;">
|
4
|
+
<use id="g2523" xlink:href="#_Image1" x="106.724" y="59.301" width="469.463px" height="564.226px" transform="matrix(0.998857,0,0,0.99863,0,0)"/>
|
5
|
+
<defs>
|
6
|
+
<image id="_Image1" width="470px" height="565px" xlink:href=""/>
|
7
|
+
</defs>
|
8
|
+
</svg>
|