190 lines
5.5 KiB
JavaScript
190 lines
5.5 KiB
JavaScript
import { hbs } from "ember-cli-htmlbars";
|
|
import { ajax } from "discourse/lib/ajax";
|
|
import { popupAjaxError } from "discourse/lib/ajax-error";
|
|
import { withPluginApi } from "discourse/lib/plugin-api";
|
|
import { registerWidgetShim } from "discourse/widgets/render-glimmer";
|
|
import ShareModal from "../discourse/components/modal/share-modal";
|
|
import streamText from "../discourse/lib/ai-streamer";
|
|
import copyConversation from "../discourse/lib/copy-conversation";
|
|
const AUTO_COPY_THRESHOLD = 4;
|
|
import AiBotHeaderIcon from "../discourse/components/ai-bot-header-icon";
|
|
import { showShareConversationModal } from "../discourse/lib/ai-bot-helper";
|
|
|
|
let enabledChatBotIds = [];
|
|
function isGPTBot(user) {
|
|
return user && enabledChatBotIds.includes(user.id);
|
|
}
|
|
|
|
function attachHeaderIcon(api) {
|
|
api.headerIcons.add("ai", AiBotHeaderIcon);
|
|
}
|
|
|
|
function initializeAIBotReplies(api) {
|
|
api.addPostMenuButton("cancel-gpt", (post) => {
|
|
if (isGPTBot(post.user)) {
|
|
return {
|
|
icon: "pause",
|
|
action: "cancelStreaming",
|
|
title: "discourse_ai.ai_bot.cancel_streaming",
|
|
className: "btn btn-default cancel-streaming",
|
|
position: "first",
|
|
};
|
|
}
|
|
});
|
|
|
|
api.attachWidgetAction("post", "cancelStreaming", function () {
|
|
ajax(`/discourse-ai/ai-bot/post/${this.model.id}/stop-streaming`, {
|
|
type: "POST",
|
|
})
|
|
.then(() => {
|
|
document
|
|
.querySelector(`#post_${this.model.post_number}`)
|
|
.classList.remove("streaming");
|
|
})
|
|
.catch(popupAjaxError);
|
|
});
|
|
|
|
api.modifyClass("controller:topic", {
|
|
pluginId: "discourse-ai",
|
|
|
|
onAIBotStreamedReply: function (data) {
|
|
streamText(this.model.postStream, data);
|
|
},
|
|
subscribe: function () {
|
|
this._super();
|
|
|
|
if (
|
|
this.model.isPrivateMessage &&
|
|
this.model.details.allowed_users &&
|
|
this.model.details.allowed_users.filter(isGPTBot).length >= 1
|
|
) {
|
|
// we attempt to recover the last message in the bus
|
|
// so we subscribe at -2
|
|
this.messageBus.subscribe(
|
|
`discourse-ai/ai-bot/topic/${this.model.id}`,
|
|
this.onAIBotStreamedReply.bind(this),
|
|
-2
|
|
);
|
|
}
|
|
},
|
|
unsubscribe: function () {
|
|
this.messageBus.unsubscribe("discourse-ai/ai-bot/topic/*");
|
|
this._super();
|
|
},
|
|
});
|
|
}
|
|
|
|
function initializePersonaDecorator(api) {
|
|
let topicController = null;
|
|
api.decorateWidget(`poster-name:after`, (dec) => {
|
|
if (!isGPTBot(dec.attrs.user)) {
|
|
return;
|
|
}
|
|
// this is hacky and will need to change
|
|
// trouble is we need to get the model for the topic
|
|
// and it is not available in the decorator
|
|
// long term this will not be a problem once we remove widgets and
|
|
// have a saner structure for our model
|
|
topicController =
|
|
topicController || api.container.lookup("controller:topic");
|
|
|
|
return dec.widget.attach("persona-flair", {
|
|
topicController,
|
|
});
|
|
});
|
|
|
|
registerWidgetShim(
|
|
"persona-flair",
|
|
"span.persona-flair",
|
|
hbs`{{@data.topicController.model.ai_persona_name}}`
|
|
);
|
|
}
|
|
|
|
const MAX_PERSONA_USER_ID = -1200;
|
|
|
|
function initializeShareButton(api) {
|
|
const currentUser = api.getCurrentUser();
|
|
if (!currentUser || !currentUser.ai_enabled_chat_bots) {
|
|
return;
|
|
}
|
|
|
|
let shareAiResponse = async function ({ post, showFeedback }) {
|
|
if (post.post_number <= AUTO_COPY_THRESHOLD) {
|
|
await copyConversation(post.topic, 1, post.post_number);
|
|
showFeedback("discourse_ai.ai_bot.conversation_shared");
|
|
} else {
|
|
modal.show(ShareModal, { model: post });
|
|
}
|
|
};
|
|
|
|
api.addPostMenuButton("share", (post) => {
|
|
// for backwards compat so we don't break if topic is undefined
|
|
if (post.topic?.archetype !== "private_message") {
|
|
return;
|
|
}
|
|
|
|
if (
|
|
!currentUser.ai_enabled_chat_bots.any(
|
|
(bot) => post.username === bot.username
|
|
)
|
|
) {
|
|
// special handling for personas (persona bot users start at ID -1200 and go down)
|
|
if (post.user_id > MAX_PERSONA_USER_ID) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
return {
|
|
action: shareAiResponse,
|
|
icon: "share",
|
|
className: "post-action-menu__share",
|
|
title: "discourse_ai.ai_bot.share",
|
|
position: "first",
|
|
};
|
|
});
|
|
|
|
const modal = api.container.lookup("service:modal");
|
|
}
|
|
|
|
function initializeShareTopicButton(api) {
|
|
const modal = api.container.lookup("service:modal");
|
|
const currentUser = api.container.lookup("current-user:main");
|
|
|
|
api.registerTopicFooterButton({
|
|
id: "share-ai-conversation",
|
|
icon: "share-alt",
|
|
label: "discourse_ai.ai_bot.share_ai_conversation.name",
|
|
title: "discourse_ai.ai_bot.share_ai_conversation.title",
|
|
action() {
|
|
showShareConversationModal(modal, this.topic.id);
|
|
},
|
|
classNames: ["share-ai-conversation-button"],
|
|
dependentKeys: ["topic.ai_persona_name"],
|
|
displayed() {
|
|
return (
|
|
currentUser?.can_share_ai_bot_conversations &&
|
|
this.topic.ai_persona_name
|
|
);
|
|
},
|
|
});
|
|
}
|
|
|
|
export default {
|
|
name: "discourse-ai-bot-replies",
|
|
|
|
initialize(container) {
|
|
const user = container.lookup("service:current-user");
|
|
|
|
if (user?.ai_enabled_chat_bots) {
|
|
enabledChatBotIds = user.ai_enabled_chat_bots.map((bot) => bot.id);
|
|
withPluginApi("1.6.0", attachHeaderIcon);
|
|
withPluginApi("1.6.0", initializeAIBotReplies);
|
|
withPluginApi("1.6.0", initializePersonaDecorator);
|
|
withPluginApi("1.22.0", (api) => initializeShareButton(api, container));
|
|
withPluginApi("1.22.0", (api) =>
|
|
initializeShareTopicButton(api, container)
|
|
);
|
|
}
|
|
},
|
|
};
|