diff --git a/packages/ui/certd-client/index.html b/packages/ui/certd-client/index.html
index 1740a304..f6398595 100644
--- a/packages/ui/certd-client/index.html
+++ b/packages/ui/certd-client/index.html
@@ -7,6 +7,17 @@
Loading
+
diff --git a/packages/ui/certd-client/public/iframe.js b/packages/ui/certd-client/public/iframe.js
new file mode 100644
index 00000000..22bfaf74
--- /dev/null
+++ b/packages/ui/certd-client/public/iframe.js
@@ -0,0 +1,151 @@
+function embedChatbot() {
+ const chatBtnId = "fastgpt-chatbot-button";
+ const chatWindowId = "fastgpt-chatbot-window";
+ const script = document.getElementById("chatbot-iframe");
+ const botSrc = script?.getAttribute("data-bot-src");
+ const defaultOpen = script?.getAttribute("data-default-open") === "true";
+ const canDrag = script?.getAttribute("data-drag") === "true";
+ const MessageIcon =
+ script?.getAttribute("data-open-icon") ||
+ ``;
+ const CloseIcon =
+ script?.getAttribute("data-close-icon") ||
+ "";
+
+ if (!botSrc) {
+ console.error(`Can't find appid`);
+ return;
+ }
+ if (document.getElementById(chatBtnId)) {
+ return;
+ }
+
+ const ChatBtn = document.createElement("div");
+ ChatBtn.id = chatBtnId;
+ ChatBtn.style.cssText = "position: fixed; bottom: 30px; right: 60px; width: 40px; height: 40px; cursor: pointer; z-index: 2147483647; transition: 0;";
+
+ const ChatBtnDiv = document.createElement("img");
+ ChatBtnDiv.src = defaultOpen ? CloseIcon : MessageIcon;
+ ChatBtnDiv.setAttribute("width", "100%");
+ ChatBtnDiv.setAttribute("height", "100%");
+ ChatBtnDiv.draggable = false;
+
+ const parent = document.createElement("div");
+ const iframe = document.createElement("iframe");
+ iframe.allow = "*";
+ iframe.referrerPolicy = "no-referrer";
+ iframe.title = "FastGPT Chat Window";
+ parent.id = chatWindowId;
+ iframe.src = botSrc;
+ parent.style.cssText =
+ "border: none; position: fixed; flex-direction: column; justify-content: space-between; box-shadow: rgba(150, 150, 150, 0.2) 0px 10px 30px 0px, rgba(150, 150, 150, 0.2) 0px 0px 0px 1px; width: 375px; height: 667px; max-width: 90vw; max-height: 85vh; border-radius: 0.75rem; display: flex; z-index: 2147483647; overflow: hidden; left: unset; background-color: #F3F4F6;";
+ parent.style.visibility = defaultOpen ? "unset" : "hidden";
+ iframe.style.cssText = "border: none;width:100%;height:100%";
+
+ document.body.appendChild(parent);
+ parent.appendChild(iframe);
+
+ const tips = document.createElement("div");
+ tips.innerText = "内容由AI生成,仅供参考";
+ tips.style.cssText = "padding:5px;font-size:10px;display:flex; justify-content:center;border-top: 1px solid #ddd;";
+ parent.appendChild(tips);
+
+ const observer = new MutationObserver(mutations => {
+ mutations.forEach(mutation => {
+ if (mutation.type === "attributes" && mutation.attributeName === "data-bot-src") {
+ const newBotSrc = script.getAttribute("data-bot-src");
+ if (newBotSrc) {
+ iframe.src = newBotSrc;
+ }
+ }
+ });
+ });
+ observer.observe(script, {
+ attributes: true,
+ attributeFilter: ["data-bot-src"],
+ });
+
+ let chatBtnDragged = false;
+ let chatBtnDown = false;
+ let chatBtnMouseX;
+ let chatBtnMouseY;
+
+ const updateChatWindowPosition = () => {
+ const chatWindow = document.getElementById(chatWindowId);
+ const btn = ChatBtn.getBoundingClientRect();
+ const [vw, vh, ww, wh] = [window.innerWidth, window.innerHeight, chatWindow.offsetWidth, chatWindow.offsetHeight];
+
+ let right = 0;
+ if (btn.left >= ww) {
+ right = vw - btn.left + 10; // 左侧有空间则放在左侧,间距 10
+ } else if (vw - btn.right >= ww) {
+ right = vw - btn.right - ww - 10; // 右侧有空间则放在右侧
+ }
+
+ let bottom = Math.max(30, vh - btn.bottom); // 聊天窗口底部和按钮对齐,最少 30
+ if (btn.top < wh) {
+ bottom = Math.min(bottom, vh - wh - 30); // 确保聊天窗口不超出顶部
+ }
+
+ chatWindow.style.right = `${right}px`;
+ chatWindow.style.bottom = `${bottom}px`;
+ };
+
+ ChatBtn.addEventListener("click", function () {
+ if (chatBtnDragged) {
+ chatBtnDragged = false;
+ return;
+ }
+ const chatWindow = document.getElementById(chatWindowId);
+
+ if (!chatWindow) return;
+ const visibilityVal = chatWindow.style.visibility;
+ if (visibilityVal === "hidden") {
+ chatWindow.style.visibility = "unset";
+ ChatBtnDiv.src = CloseIcon;
+ } else {
+ chatWindow.style.visibility = "hidden";
+ ChatBtnDiv.src = MessageIcon;
+ }
+ });
+
+ ChatBtn.addEventListener("mousedown", e => {
+ e.stopPropagation();
+ chatBtnMouseX = e.clientX;
+ chatBtnMouseY = e.clientY;
+ chatBtnDown = true;
+
+ ChatBtn.initialRight = parseInt(ChatBtn.style.right) || 60;
+ ChatBtn.initialBottom = parseInt(ChatBtn.style.bottom) || 30;
+ });
+
+ window.addEventListener("mousemove", e => {
+ e.stopPropagation();
+ if (!canDrag || !chatBtnDown) return;
+
+ chatBtnDragged = true;
+
+ const deltaX = e.clientX - chatBtnMouseX;
+ const deltaY = e.clientY - chatBtnMouseY;
+
+ let newRight = ChatBtn.initialRight - deltaX;
+ let newBottom = ChatBtn.initialBottom - deltaY;
+
+ newRight = Math.max(20, Math.min(window.innerWidth - 60, newRight));
+ newBottom = Math.max(30, Math.min(window.innerHeight - 70, newBottom));
+
+ ChatBtn.style.right = `${newRight}px`;
+ ChatBtn.style.bottom = `${newBottom}px`;
+
+ updateChatWindowPosition();
+ });
+
+ window.addEventListener("mouseup", e => {
+ chatBtnDown = false;
+ });
+
+ ChatBtn.appendChild(ChatBtnDiv);
+ document.body.appendChild(ChatBtn);
+ updateChatWindowPosition();
+}
+window.addEventListener("load", embedChatbot);
diff --git a/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts b/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts
index 4def0157..9a0c9f93 100644
--- a/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts
+++ b/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts
@@ -64,7 +64,7 @@ export class UploadCertToHostPlugin extends AbstractTaskPlugin {
crtPath!: string;
@TaskInput({
title: '私钥保存路径',
- helper: '需要有写入权限,路径要包含私钥文件名,例如:/tmp/cert.key',
+ helper: '原本的私钥保存路径,需要有写入权限,路径要包含私钥文件名,例如:/tmp/cert.key',
component: {
placeholder: '/root/deploy/nginx/cert.key',
},
@@ -212,8 +212,9 @@ export class UploadCertToHostPlugin extends AbstractTaskPlugin {
name: 'a-textarea',
vModel: 'value',
rows: 6,
+ placeholder: 'systemctl restart nginx ',
},
- helper: '上传后执行脚本命令,不填则不执行\n注意:如果目标主机是windows,且终端是cmd,系统会自动将多行命令通过“&&”连接成一行',
+ helper: '上传后执行脚本命令,让证书生效(比如重启nginx),不填则不执行\n注意:如果目标主机是windows,且终端是cmd,系统会自动将多行命令通过“&&”连接成一行',
required: false,
})
script!: string;