FEATURE: Add streaming to post AI helper's explain option (#344)
Co-authored-by: Rafael dos Santos Silva <xfalcox@gmail.com> Co-authored-by: Roman Rizzi <roman@discourse.org>
This commit is contained in:
parent
d4357c29c7
commit
6aaf1f002e
|
@ -101,12 +101,14 @@ module DiscourseAi
|
||||||
|
|
||||||
raise Discourse::InvalidParameters.new(:post_id) unless post
|
raise Discourse::InvalidParameters.new(:post_id) unless post
|
||||||
|
|
||||||
render json:
|
Jobs.enqueue(
|
||||||
DiscourseAi::AiHelper::TopicHelper.new(current_user).explain(
|
:stream_post_helper,
|
||||||
term_to_explain,
|
post_id: post.id,
|
||||||
post,
|
user_id: current_user.id,
|
||||||
),
|
term_to_explain: term_to_explain,
|
||||||
status: 200
|
)
|
||||||
|
|
||||||
|
render json: { success: true }, status: 200
|
||||||
rescue DiscourseAi::Completions::Endpoints::Base::CompletionFailed => e
|
rescue DiscourseAi::Completions::Endpoints::Base::CompletionFailed => e
|
||||||
render_json_error I18n.t("discourse_ai.ai_helper.errors.completion_request_failed"),
|
render_json_error I18n.t("discourse_ai.ai_helper.errors.completion_request_failed"),
|
||||||
status: 502
|
status: 502
|
||||||
|
|
|
@ -0,0 +1,35 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Jobs
|
||||||
|
class StreamPostHelper < ::Jobs::Base
|
||||||
|
sidekiq_options retry: false
|
||||||
|
|
||||||
|
def execute(args)
|
||||||
|
return unless post = Post.includes(:topic).find_by(id: args[:post_id])
|
||||||
|
return unless user = User.find_by(id: args[:user_id])
|
||||||
|
return unless args[:term_to_explain]
|
||||||
|
|
||||||
|
topic = post.topic
|
||||||
|
reply_to = post.reply_to_post
|
||||||
|
|
||||||
|
guardian = Guardian.new(user)
|
||||||
|
return unless guardian.can_see?(post)
|
||||||
|
|
||||||
|
prompt = CompletionPrompt.enabled_by_name("explain")
|
||||||
|
|
||||||
|
input = <<~TEXT
|
||||||
|
<term>#{args[:term_to_explain]}</term>
|
||||||
|
<context>#{post.raw}</context>
|
||||||
|
<topic>#{topic.title}</topic>
|
||||||
|
#{reply_to ? "<replyTo>#{reply_to.raw}</replyTo>" : nil}
|
||||||
|
TEXT
|
||||||
|
|
||||||
|
DiscourseAi::AiHelper::Assistant.new.stream_prompt(
|
||||||
|
prompt,
|
||||||
|
input,
|
||||||
|
user,
|
||||||
|
"/discourse-ai/ai-helper/explain/#{post.id}",
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,19 @@
|
||||||
|
import DButton from "discourse/components/d-button";
|
||||||
|
import i18n from "discourse-common/helpers/i18n";
|
||||||
|
|
||||||
|
const AiHelperLoading = <template>
|
||||||
|
<div class="ai-helper-context-menu__loading">
|
||||||
|
<div class="dot-falling"></div>
|
||||||
|
<span>
|
||||||
|
{{i18n "discourse_ai.ai_helper.context_menu.loading"}}
|
||||||
|
</span>
|
||||||
|
<DButton
|
||||||
|
@icon="times"
|
||||||
|
@title="discourse_ai.ai_helper.context_menu.cancel"
|
||||||
|
@action={{@cancel}}
|
||||||
|
class="btn-flat cancel-request"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>;
|
||||||
|
|
||||||
|
export default AiHelperLoading;
|
|
@ -53,18 +53,7 @@
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
{{else if (eq this.menuState this.CONTEXT_MENU_STATES.loading)}}
|
{{else if (eq this.menuState this.CONTEXT_MENU_STATES.loading)}}
|
||||||
<div class="ai-helper-context-menu__loading">
|
<AiHelperLoading @cancel={{this.cancelAIAction}} />
|
||||||
<div class="dot-falling"></div>
|
|
||||||
<span>
|
|
||||||
{{i18n "discourse_ai.ai_helper.context_menu.loading"}}
|
|
||||||
</span>
|
|
||||||
<DButton
|
|
||||||
@icon="times"
|
|
||||||
@title="discourse_ai.ai_helper.context_menu.cancel"
|
|
||||||
@action={{this.cancelAIAction}}
|
|
||||||
class="btn-flat cancel-request"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{{else if (eq this.menuState this.CONTEXT_MENU_STATES.review)}}
|
{{else if (eq this.menuState this.CONTEXT_MENU_STATES.review)}}
|
||||||
<ul class="ai-helper-context-menu__review">
|
<ul class="ai-helper-context-menu__review">
|
||||||
|
|
|
@ -1,20 +1,23 @@
|
||||||
import Component from "@glimmer/component";
|
import Component from "@glimmer/component";
|
||||||
import { tracked } from "@glimmer/tracking";
|
import { tracked } from "@glimmer/tracking";
|
||||||
import { action } from "@ember/object";
|
import { action } from "@ember/object";
|
||||||
|
import didInsert from "@ember/render-modifiers/modifiers/did-insert";
|
||||||
|
import willDestroy from "@ember/render-modifiers/modifiers/will-destroy";
|
||||||
|
import { inject as service } from "@ember/service";
|
||||||
import DButton from "discourse/components/d-button";
|
import DButton from "discourse/components/d-button";
|
||||||
import { ajax } from "discourse/lib/ajax";
|
import { ajax } from "discourse/lib/ajax";
|
||||||
import { popupAjaxError } from "discourse/lib/ajax-error";
|
import { popupAjaxError } from "discourse/lib/ajax-error";
|
||||||
import I18n from "I18n";
|
import { bind } from "discourse-common/utils/decorators";
|
||||||
import eq from "truth-helpers/helpers/eq";
|
import eq from "truth-helpers/helpers/eq";
|
||||||
import not from "truth-helpers/helpers/not";
|
import not from "truth-helpers/helpers/not";
|
||||||
|
import AiHelperLoading from "../../components/ai-helper-loading";
|
||||||
import { showPostAIHelper } from "../../lib/show-ai-helper";
|
import { showPostAIHelper } from "../../lib/show-ai-helper";
|
||||||
|
|
||||||
const i18n = I18n.t.bind(I18n);
|
|
||||||
|
|
||||||
export default class AIHelperOptionsMenu extends Component {
|
export default class AIHelperOptionsMenu extends Component {
|
||||||
static shouldRender(outletArgs, helper) {
|
static shouldRender(outletArgs, helper) {
|
||||||
return showPostAIHelper(outletArgs, helper);
|
return showPostAIHelper(outletArgs, helper);
|
||||||
}
|
}
|
||||||
|
@service messageBus;
|
||||||
@tracked helperOptions = [];
|
@tracked helperOptions = [];
|
||||||
@tracked menuState = this.MENU_STATES.triggers;
|
@tracked menuState = this.MENU_STATES.triggers;
|
||||||
@tracked loading = false;
|
@tracked loading = false;
|
||||||
|
@ -47,12 +50,38 @@ export default class AIHelperOptionsMenu extends Component {
|
||||||
this.menuState = this.MENU_STATES.options;
|
this.menuState = this.MENU_STATES.options;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@bind
|
||||||
|
subscribe() {
|
||||||
|
const channel = `/discourse-ai/ai-helper/explain/${this.args.outletArgs.data.quoteState.postId}`;
|
||||||
|
this.messageBus.subscribe(channel, this._updateResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
@bind
|
||||||
|
unsubscribe() {
|
||||||
|
this.messageBus.unsubscribe(
|
||||||
|
"/discourse-ai/ai-helper/explain/*",
|
||||||
|
this._updateResult
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@bind
|
||||||
|
_updateResult(result) {
|
||||||
|
const suggestion = result.result;
|
||||||
|
|
||||||
|
if (suggestion.length > 0) {
|
||||||
|
this.suggestion = suggestion;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@action
|
@action
|
||||||
async performAISuggestion(option) {
|
async performAISuggestion(option) {
|
||||||
this.menuState = this.MENU_STATES.loading;
|
this.menuState = this.MENU_STATES.loading;
|
||||||
|
|
||||||
if (option.name === "Explain") {
|
if (option.name === "Explain") {
|
||||||
this._activeAIRequest = ajax("/discourse-ai/ai-helper/explain", {
|
this.menuState = this.MENU_STATES.result;
|
||||||
|
|
||||||
|
const fetchUrl = `/discourse-ai/ai-helper/explain`;
|
||||||
|
this._activeAIRequest = ajax(fetchUrl, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
data: {
|
data: {
|
||||||
mode: option.value,
|
mode: option.value,
|
||||||
|
@ -71,15 +100,17 @@ export default class AIHelperOptionsMenu extends Component {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
this._activeAIRequest
|
if (option.name !== "Explain") {
|
||||||
.then(({ suggestions }) => {
|
this._activeAIRequest
|
||||||
this.suggestion = suggestions[0];
|
.then(({ suggestions }) => {
|
||||||
})
|
this.suggestion = suggestions[0];
|
||||||
.catch(popupAjaxError)
|
})
|
||||||
.finally(() => {
|
.catch(popupAjaxError)
|
||||||
this.loading = false;
|
.finally(() => {
|
||||||
this.menuState = this.MENU_STATES.result;
|
this.loading = false;
|
||||||
});
|
this.menuState = this.MENU_STATES.result;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return this._activeAIRequest;
|
return this._activeAIRequest;
|
||||||
}
|
}
|
||||||
|
@ -154,30 +185,35 @@ export default class AIHelperOptionsMenu extends Component {
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{{else if (eq this.menuState this.MENU_STATES.loading)}}
|
{{else if (eq this.menuState this.MENU_STATES.loading)}}
|
||||||
<div class="ai-helper-context-menu__loading">
|
<AiHelperLoading @cancel={{this.cancelAIAction}} />
|
||||||
<div class="dot-falling"></div>
|
|
||||||
<span>
|
|
||||||
{{i18n "discourse_ai.ai_helper.context_menu.loading"}}
|
|
||||||
</span>
|
|
||||||
<DButton
|
|
||||||
@icon="times"
|
|
||||||
@title="discourse_ai.ai_helper.context_menu.cancel"
|
|
||||||
@action={{this.cancelAIAction}}
|
|
||||||
class="btn-flat cancel-request"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{{else if (eq this.menuState this.MENU_STATES.result)}}
|
{{else if (eq this.menuState this.MENU_STATES.result)}}
|
||||||
<div class="ai-post-helper__suggestion">
|
<div
|
||||||
<div class="ai-post-helper__suggestion__text">
|
class="ai-post-helper__suggestion"
|
||||||
{{this.suggestion}}
|
{{didInsert this.subscribe}}
|
||||||
</div>
|
{{willDestroy this.unsubscribe}}
|
||||||
<DButton
|
>
|
||||||
@class="btn-flat ai-post-helper__suggestion__copy"
|
{{#if this.suggestion}}
|
||||||
@icon={{this.copyButtonIcon}}
|
<div class="ai-post-helper__suggestion__text">
|
||||||
@label={{this.copyButtonLabel}}
|
{{this.suggestion}}
|
||||||
@action={{this.copySuggestion}}
|
</div>
|
||||||
@disabled={{not this.suggestion}}
|
<di class="ai-post-helper__suggestion__buttons">
|
||||||
/>
|
<DButton
|
||||||
|
@class="btn-flat ai-post-helper__suggestion__cancel"
|
||||||
|
@icon="times"
|
||||||
|
@label="discourse_ai.ai_helper.post_options_menu.cancel"
|
||||||
|
@action={{this.cancelAIAction}}
|
||||||
|
/>
|
||||||
|
<DButton
|
||||||
|
@class="btn-flat ai-post-helper__suggestion__copy"
|
||||||
|
@icon={{this.copyButtonIcon}}
|
||||||
|
@label={{this.copyButtonLabel}}
|
||||||
|
@action={{this.copySuggestion}}
|
||||||
|
@disabled={{not this.suggestion}}
|
||||||
|
/>
|
||||||
|
</di>
|
||||||
|
{{else}}
|
||||||
|
<AiHelperLoading @cancel={{this.cancelAIAction}} />
|
||||||
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -334,8 +334,6 @@
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
||||||
&__copy {
|
&__copy {
|
||||||
margin-top: 0.5rem;
|
|
||||||
|
|
||||||
.d-icon-check {
|
.d-icon-check {
|
||||||
color: var(--success);
|
color: var(--success);
|
||||||
}
|
}
|
||||||
|
@ -344,5 +342,16 @@
|
||||||
&__text {
|
&__text {
|
||||||
padding: 0.5rem;
|
padding: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__buttons {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: stretch;
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
gap: 0.5rem;
|
||||||
|
.btn {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -103,6 +103,7 @@ en:
|
||||||
close: "Close"
|
close: "Close"
|
||||||
copy: "Copy"
|
copy: "Copy"
|
||||||
copied: "Copied!"
|
copied: "Copied!"
|
||||||
|
cancel: "Cancel"
|
||||||
reviewables:
|
reviewables:
|
||||||
model_used: "Model used:"
|
model_used: "Model used:"
|
||||||
accuracy: "Accuracy:"
|
accuracy: "Accuracy:"
|
||||||
|
|
|
@ -24,29 +24,76 @@ module DiscourseAi
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def generate_and_send_prompt(completion_prompt, input, user)
|
def generate_prompt(completion_prompt, input, user, &block)
|
||||||
llm = DiscourseAi::Completions::Llm.proxy(SiteSetting.ai_helper_model)
|
llm = DiscourseAi::Completions::Llm.proxy(SiteSetting.ai_helper_model)
|
||||||
|
|
||||||
generic_prompt = completion_prompt.messages_with_input(input)
|
generic_prompt = completion_prompt.messages_with_input(input)
|
||||||
|
|
||||||
completion_result = llm.completion!(generic_prompt, user)
|
llm.completion!(generic_prompt, user, &block)
|
||||||
|
end
|
||||||
|
|
||||||
|
def generate_and_send_prompt(completion_prompt, input, user)
|
||||||
|
completion_result = generate_prompt(completion_prompt, input, user)
|
||||||
result = { type: completion_prompt.prompt_type }
|
result = { type: completion_prompt.prompt_type }
|
||||||
|
|
||||||
result[:diff] = parse_diff(input, completion_result) if completion_prompt.diff?
|
result[:diff] = parse_diff(input, completion_result) if completion_prompt.diff?
|
||||||
|
|
||||||
result[:suggestions] = (
|
result[:suggestions] = (
|
||||||
if completion_prompt.list?
|
if completion_prompt.list?
|
||||||
parse_list(completion_result)
|
parse_list(completion_result).map { |suggestion| sanitize_result(suggestion) }
|
||||||
else
|
else
|
||||||
[completion_result]
|
[sanitize_result(completion_result)]
|
||||||
end
|
end
|
||||||
)
|
)
|
||||||
|
|
||||||
result
|
result
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def stream_prompt(completion_prompt, input, user, channel)
|
||||||
|
streamed_result = +""
|
||||||
|
start = Time.now
|
||||||
|
|
||||||
|
generate_prompt(completion_prompt, input, user) do |partial_response, cancel_function|
|
||||||
|
streamed_result << partial_response
|
||||||
|
|
||||||
|
# Throttle the updates
|
||||||
|
if (Time.now - start > 0.5) || Rails.env.test?
|
||||||
|
payload = { result: sanitize_result(streamed_result), done: false }
|
||||||
|
publish_update(channel, payload, user)
|
||||||
|
start = Time.now
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
sanitized_result = sanitize_result(streamed_result)
|
||||||
|
if sanitized_result.present?
|
||||||
|
publish_update(channel, { result: sanitized_result, done: true }, user)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def sanitize_result(result)
|
||||||
|
tags_to_remove = %w[
|
||||||
|
<term>
|
||||||
|
</term>
|
||||||
|
<context>
|
||||||
|
</context>
|
||||||
|
<topic>
|
||||||
|
</topic>
|
||||||
|
<replyTo>
|
||||||
|
</replyTo>
|
||||||
|
<input>
|
||||||
|
</input>
|
||||||
|
<output>
|
||||||
|
</output>
|
||||||
|
]
|
||||||
|
|
||||||
|
result.dup.tap { |dup_result| tags_to_remove.each { |tag| dup_result.gsub!(tag, "") } }
|
||||||
|
end
|
||||||
|
|
||||||
|
def publish_update(channel, payload, user)
|
||||||
|
MessageBus.publish(channel, payload, user_ids: [user.id])
|
||||||
|
end
|
||||||
|
|
||||||
def icon_map(name)
|
def icon_map(name)
|
||||||
case name
|
case name
|
||||||
when "translate"
|
when "translate"
|
||||||
|
|
|
@ -1,35 +0,0 @@
|
||||||
# frozen_string_literal: true
|
|
||||||
|
|
||||||
module DiscourseAi
|
|
||||||
module AiHelper
|
|
||||||
class TopicHelper
|
|
||||||
def initialize(user)
|
|
||||||
@user = user
|
|
||||||
end
|
|
||||||
|
|
||||||
def explain(term_to_explain, post)
|
|
||||||
return nil unless term_to_explain
|
|
||||||
return nil unless post
|
|
||||||
|
|
||||||
reply_to = post.reply_to_post
|
|
||||||
topic = post.topic
|
|
||||||
|
|
||||||
prompt = CompletionPrompt.enabled_by_name("explain")
|
|
||||||
raise Discourse::InvalidParameters.new(:mode) if !prompt
|
|
||||||
|
|
||||||
input = <<~TEXT
|
|
||||||
<term>#{term_to_explain}</term>
|
|
||||||
<context>#{post.raw}</context>
|
|
||||||
<topic>#{topic.title}</topic>
|
|
||||||
#{reply_to ? "<replyTo>#{reply_to.raw}</replyTo>" : nil}
|
|
||||||
TEXT
|
|
||||||
|
|
||||||
DiscourseAi::AiHelper::Assistant.new.generate_and_send_prompt(prompt, input, user)
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
attr_reader :user
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -0,0 +1,83 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
RSpec.describe Jobs::StreamPostHelper do
|
||||||
|
subject(:job) { described_class.new }
|
||||||
|
|
||||||
|
describe "#execute" do
|
||||||
|
fab!(:topic) { Fabricate(:topic) }
|
||||||
|
fab!(:post) do
|
||||||
|
Fabricate(
|
||||||
|
:post,
|
||||||
|
topic: topic,
|
||||||
|
raw:
|
||||||
|
"I like to eat pie. It is a very good dessert. Some people are wasteful by throwing pie at others but I do not do that. I always eat the pie.",
|
||||||
|
)
|
||||||
|
end
|
||||||
|
fab!(:user) { Fabricate(:leader) }
|
||||||
|
|
||||||
|
before do
|
||||||
|
Group.find(Group::AUTO_GROUPS[:trust_level_3]).add(user)
|
||||||
|
SiteSetting.composer_ai_helper_enabled = true
|
||||||
|
end
|
||||||
|
|
||||||
|
describe "validates params" do
|
||||||
|
it "does nothing if there is no post" do
|
||||||
|
messages =
|
||||||
|
MessageBus.track_publish("/discourse-ai/ai-helper/explain/#{post.id}") do
|
||||||
|
job.execute(post_id: nil, user_id: user.id, term_to_explain: "pie")
|
||||||
|
end
|
||||||
|
|
||||||
|
expect(messages).to be_empty
|
||||||
|
end
|
||||||
|
|
||||||
|
it "does nothing if there is no user" do
|
||||||
|
messages =
|
||||||
|
MessageBus.track_publish("/discourse-ai/ai-helper/explain/#{post.id}") do
|
||||||
|
job.execute(post_id: post.id, user_id: nil, term_to_explain: "pie")
|
||||||
|
end
|
||||||
|
|
||||||
|
expect(messages).to be_empty
|
||||||
|
end
|
||||||
|
|
||||||
|
it "does nothing if there is no term to explain" do
|
||||||
|
messages =
|
||||||
|
MessageBus.track_publish("/discourse-ai/ai-helper/explain/#{post.id}") do
|
||||||
|
job.execute(post_id: post.id, user_id: user.id, term_to_explain: nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
expect(messages).to be_empty
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "publishes updates with a partial result" do
|
||||||
|
explanation =
|
||||||
|
"In this context, \"pie\" refers to a baked dessert typically consisting of a pastry crust and filling."
|
||||||
|
|
||||||
|
DiscourseAi::Completions::Llm.with_prepared_responses([explanation]) do
|
||||||
|
messages =
|
||||||
|
MessageBus.track_publish("/discourse-ai/ai-helper/explain/#{post.id}") do
|
||||||
|
job.execute(post_id: post.id, user_id: user.id, term_to_explain: "pie")
|
||||||
|
end
|
||||||
|
|
||||||
|
partial_result_update = messages.first.data
|
||||||
|
expect(partial_result_update[:done]).to eq(false)
|
||||||
|
expect(partial_result_update[:result]).to eq(explanation)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "publishes a final update to signal we're donea" do
|
||||||
|
explanation =
|
||||||
|
"In this context, \"pie\" refers to a baked dessert typically consisting of a pastry crust and filling."
|
||||||
|
|
||||||
|
DiscourseAi::Completions::Llm.with_prepared_responses([explanation]) do
|
||||||
|
messages =
|
||||||
|
MessageBus.track_publish("/discourse-ai/ai-helper/explain/#{post.id}") do
|
||||||
|
job.execute(post_id: post.id, user_id: user.id, term_to_explain: "pie")
|
||||||
|
end
|
||||||
|
|
||||||
|
final_update = messages.last.data
|
||||||
|
expect(final_update[:done]).to eq(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,6 +1,6 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe "AI Composer helper", type: :system, js: true do
|
RSpec.describe "AI Post helper", type: :system, js: true do
|
||||||
fab!(:user) { Fabricate(:admin) }
|
fab!(:user) { Fabricate(:admin) }
|
||||||
fab!(:non_member_group) { Fabricate(:group) }
|
fab!(:non_member_group) { Fabricate(:group) }
|
||||||
fab!(:topic) { Fabricate(:topic) }
|
fab!(:topic) { Fabricate(:topic) }
|
||||||
|
@ -18,13 +18,6 @@ RSpec.describe "AI Composer helper", type: :system, js: true do
|
||||||
let(:topic_page) { PageObjects::Pages::Topic.new }
|
let(:topic_page) { PageObjects::Pages::Topic.new }
|
||||||
let(:post_ai_helper) { PageObjects::Components::AIHelperPostOptions.new }
|
let(:post_ai_helper) { PageObjects::Components::AIHelperPostOptions.new }
|
||||||
|
|
||||||
let(:explain_response) { <<~STRING }
|
|
||||||
In this context, \"pie\" refers to a baked dessert typically consisting of a pastry crust and filling.
|
|
||||||
The person states they enjoy eating pie, considering it a good dessert. They note that some people wastefully
|
|
||||||
throw pie at others, but the person themselves chooses to eat the pie rather than throwing it. Overall, \"pie\"
|
|
||||||
is being used to refer the the baked dessert food item.
|
|
||||||
STRING
|
|
||||||
|
|
||||||
before do
|
before do
|
||||||
Group.find_by(id: Group::AUTO_GROUPS[:admins]).add(user)
|
Group.find_by(id: Group::AUTO_GROUPS[:admins]).add(user)
|
||||||
SiteSetting.composer_ai_helper_enabled = true
|
SiteSetting.composer_ai_helper_enabled = true
|
||||||
|
@ -56,18 +49,23 @@ RSpec.describe "AI Composer helper", type: :system, js: true do
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when using explain mode" do
|
context "when using explain mode" do
|
||||||
skip "TODO: Fix explain mode option not appearing in spec" do
|
let(:mode) { CompletionPrompt::EXPLAIN }
|
||||||
let(:mode) { CompletionPrompt::EXPLAIN }
|
|
||||||
|
|
||||||
|
let(:explain_response) { <<~STRING }
|
||||||
|
In this context, \"pie\" refers to a baked dessert typically consisting of a pastry crust and filling.
|
||||||
|
The person states they enjoy eating pie, considering it a good dessert. They note that some people wastefully
|
||||||
|
throw pie at others, but the person themselves chooses to eat the pie rather than throwing it. Overall, \"pie\"
|
||||||
|
is being used to refer the the baked dessert food item.
|
||||||
|
STRING
|
||||||
|
|
||||||
|
skip "TODO: Fix explain option stuck in loading in test" do
|
||||||
it "shows an explanation of the selected text" do
|
it "shows an explanation of the selected text" do
|
||||||
select_post_text(post)
|
select_post_text(post)
|
||||||
post_ai_helper.click_ai_button
|
post_ai_helper.click_ai_button
|
||||||
|
|
||||||
DiscourseAi::Completions::Llm.with_prepared_responses([explain_response]) do
|
DiscourseAi::Completions::Llm.with_prepared_responses([explain_response]) do
|
||||||
post_ai_helper.select_helper_model(mode)
|
post_ai_helper.select_helper_model(mode)
|
||||||
|
|
||||||
wait_for { post_ai_helper.suggestion_value == explain_response }
|
wait_for { post_ai_helper.suggestion_value == explain_response }
|
||||||
|
|
||||||
expect(post_ai_helper.suggestion_value).to eq(explain_response)
|
expect(post_ai_helper.suggestion_value).to eq(explain_response)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -75,22 +73,20 @@ RSpec.describe "AI Composer helper", type: :system, js: true do
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when using translate mode" do
|
context "when using translate mode" do
|
||||||
skip "TODO: Fix WebMock request for translate mode not working" do
|
let(:mode) { CompletionPrompt::TRANSLATE }
|
||||||
let(:mode) { CompletionPrompt::TRANSLATE }
|
|
||||||
|
|
||||||
let(:translated_input) { "The rain in Spain, stays mainly in the Plane." }
|
let(:translated_input) { "The rain in Spain, stays mainly in the Plane." }
|
||||||
|
|
||||||
it "shows a translation of the selected text" do
|
it "shows a translation of the selected text" do
|
||||||
select_post_text(post_2)
|
select_post_text(post_2)
|
||||||
post_ai_helper.click_ai_button
|
post_ai_helper.click_ai_button
|
||||||
|
|
||||||
DiscourseAi::Completions::Llm.with_prepared_responses([translated_input]) do
|
DiscourseAi::Completions::Llm.with_prepared_responses([translated_input]) do
|
||||||
post_ai_helper.select_helper_model(mode)
|
post_ai_helper.select_helper_model(mode)
|
||||||
|
|
||||||
wait_for { post_ai_helper.suggestion_value == translated_input }
|
wait_for { post_ai_helper.suggestion_value == translated_input }
|
||||||
|
|
||||||
expect(post_ai_helper.suggestion_value).to eq(translated_input)
|
expect(post_ai_helper.suggestion_value).to eq(translated_input)
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -23,7 +23,7 @@ module PageObjects
|
||||||
end
|
end
|
||||||
|
|
||||||
def suggestion_value
|
def suggestion_value
|
||||||
find(SUGGESTION_SELECTOR).text
|
find("#{SUGGESTION_SELECTOR}__text").text
|
||||||
end
|
end
|
||||||
|
|
||||||
def has_post_ai_helper?
|
def has_post_ai_helper?
|
||||||
|
|
Loading…
Reference in New Issue