FEATURE: Configure persona backed features in admin panel (#1245)

In this feature update, we add the UI for the ability to easily configure persona backed AI-features. The feature will still be hidden until structured responses are complete.
This commit is contained in:
Keegan George 2025-04-10 08:16:31 -07:00 committed by GitHub
parent 129ced9088
commit 4de39a07e5
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
24 changed files with 713 additions and 12 deletions

View File

@ -0,0 +1,27 @@
import { ajax } from "discourse/lib/ajax";
import DiscourseRoute from "discourse/routes/discourse";
import SiteSetting from "admin/models/site-setting";
export default class AdminPluginsShowDiscourseAiFeaturesEdit extends DiscourseRoute {
async model(params) {
const allFeatures = this.modelFor(
"adminPlugins.show.discourse-ai-features"
);
const id = parseInt(params.id, 10);
const currentFeature = allFeatures.find((feature) => feature.id === id);
const { site_settings } = await ajax("/admin/config/site_settings.json", {
data: {
filter_area: `ai-features/${currentFeature.ref}`,
plugin: "discourse-ai",
category: "discourse_ai",
},
});
currentFeature.feature_settings = site_settings.map((setting) =>
SiteSetting.create(setting)
);
return currentFeature;
}
}

View File

@ -0,0 +1,10 @@
import { service } from "@ember/service";
import DiscourseRoute from "discourse/routes/discourse";
export default class AdminPluginsShowDiscourseAiFeatures extends DiscourseRoute {
@service store;
async model() {
return this.store.findAll("ai-feature");
}
}

View File

@ -0,0 +1,24 @@
import RouteTemplate from "ember-route-template";
import BackButton from "discourse/components/back-button";
import SiteSettingComponent from "admin/components/site-setting";
export default RouteTemplate(
<template>
<BackButton
@route="adminPlugins.show.discourse-ai-features"
@label="discourse_ai.features.back"
/>
<section class="ai-feature-editor__header">
<h2>{{@model.name}}</h2>
<p>{{@model.description}}</p>
</section>
<section class="ai-feature-editor">
{{#each @model.feature_settings as |setting|}}
<div>
<SiteSettingComponent @setting={{setting}} />
</div>
{{/each}}
</section>
</template>
);

View File

@ -0,0 +1,156 @@
import Component from "@glimmer/component";
import { service } from "@ember/service";
import RouteTemplate from "ember-route-template";
import { gt } from "truth-helpers";
import DBreadcrumbsItem from "discourse/components/d-breadcrumbs-item";
import DButton from "discourse/components/d-button";
import DPageSubheader from "discourse/components/d-page-subheader";
import { i18n } from "discourse-i18n";
export default RouteTemplate(
class extends Component {
@service adminPluginNavManager;
get tableHeaders() {
const prefix = "discourse_ai.features.list.header";
return [
i18n(`${prefix}.name`),
i18n(`${prefix}.persona`),
i18n(`${prefix}.groups`),
"",
];
}
get configuredFeatures() {
return this.args.model.filter(
(feature) => feature.enable_setting.value === true
);
}
get unconfiguredFeatures() {
return this.args.model.filter(
(feature) => feature.enable_setting.value === false
);
}
<template>
<DBreadcrumbsItem
@path="/admin/plugins/{{this.adminPluginNavManager.currentPlugin.name}}/ai-features"
@label={{i18n "discourse_ai.features.short_title"}}
/>
<section class="ai-feature-list admin-detail">
<DPageSubheader
@titleLabel={{i18n "discourse_ai.features.short_title"}}
@descriptionLabel={{i18n "discourse_ai.features.description"}}
@learnMoreUrl="todo"
/>
{{#if (gt this.configuredFeatures.length 0)}}
<div class="ai-feature-list__configured-features">
<h3>{{i18n "discourse_ai.features.list.configured_features"}}</h3>
<table class="d-admin-table">
<thead>
<tr>
{{#each this.tableHeaders as |header|}}
<th>{{header}}</th>
{{/each}}
</tr>
</thead>
<tbody>
{{#each this.configuredFeatures as |feature|}}
<tr
class="ai-feature-list__row d-admin-row__content"
data-feature-name={{feature.name}}
>
<td class="d-admin-row__overview ai-feature-list__row-item">
<span class="ai-feature-list__row-item-name">
<strong>
{{feature.name}}
</strong>
</span>
<span class="ai-feature-list__row-item-description">
{{feature.description}}
</span>
</td>
<td
class="d-admin-row__detail ai-feature-list__row-item ai-feature-list__persona"
>
<DButton
class="btn-flat btn-small ai-feature-list__row-item-persona"
@translatedLabel={{feature.persona.name}}
@route="adminPlugins.show.discourse-ai-personas.edit"
@routeModels={{feature.persona.id}}
/>
</td>
<td
class="d-admin-row__detail ai-feature-list__row-item ai-feature-list__groups"
>
{{#if (gt feature.persona.allowed_groups.length 0)}}
<ul class="ai-feature-list__row-item-groups">
{{#each feature.persona.allowed_groups as |group|}}
<li>{{group.name}}</li>
{{/each}}
</ul>
{{/if}}
</td>
<td class="d-admin-row_controls">
<DButton
class="btn-small edit"
@label="discourse_ai.features.list.edit"
@route="adminPlugins.show.discourse-ai-features.edit"
@routeModels={{feature.id}}
/>
</td>
</tr>
{{/each}}
</tbody>
</table>
</div>
{{/if}}
{{#if (gt this.unconfiguredFeatures.length 0)}}
<div class="ai-feature-list__unconfigured-features">
<h3>{{i18n "discourse_ai.features.list.unconfigured_features"}}</h3>
<table class="d-admin-table">
<thead>
<tr>
<th>{{i18n "discourse_ai.features.list.header.name"}}</th>
<th></th>
</tr>
</thead>
<tbody>
{{#each this.unconfiguredFeatures as |feature|}}
<tr class="ai-feature-list__row d-admin-row__content">
<td class="d-admin-row__overview ai-feature-list__row-item">
<span class="ai-feature-list__row-item-name">
<strong>
{{feature.name}}
</strong>
</span>
<span class="ai-feature-list__row-item-description">
{{feature.description}}
</span>
</td>
<td class="d-admin-row_controls">
<DButton
class="btn-small"
@label="discourse_ai.features.list.set_up"
@route="adminPlugins.show.discourse-ai-features.edit"
@routeModels={{feature.id}}
/>
</td>
</tr>
{{/each}}
</tbody>
</table>
</div>
{{/if}}
</section>
</template>
}
);

View File

@ -0,0 +1,36 @@
# frozen_string_literal: true
module DiscourseAi
module Admin
class AiFeaturesController < ::Admin::AdminController
requires_plugin ::DiscourseAi::PLUGIN_NAME
def index
render json: serialize_features(DiscourseAi::Features.features)
end
def edit
raise Discourse::InvalidParameters.new(:id) if params[:id].blank?
render json: serialize_feature(DiscourseAi::Features.find_feature_by_id(params[:id].to_i))
end
private
def serialize_features(features)
features.map { |feature| feature.merge(persona: serialize_persona(feature[:persona])) }
end
def serialize_feature(feature)
return nil if feature.blank?
feature.merge(persona: serialize_persona(feature[:persona]))
end
def serialize_persona(persona)
return nil if persona.blank?
serialize_data(persona, AiFeaturesPersonaSerializer, root: false)
end
end
end
end

View File

@ -7,6 +7,8 @@ module Jobs
def execute(args)
interaction = args[:interaction]
return unless SiteSetting.ai_discord_search_enabled
if SiteSetting.ai_discord_search_mode == "persona"
DiscourseAi::Discord::Bot::PersonaReplier.new(interaction).handle_interaction!
else

View File

@ -0,0 +1,12 @@
# frozen_string_literal: true
class AiFeaturesPersonaSerializer < ApplicationSerializer
attributes :id, :name, :system_prompt, :allowed_groups, :enabled
def allowed_groups
Group
.where(id: object.allowed_group_ids)
.pluck(:id, :name)
.map { |id, name| { id: id, name: name } }
end
end

View File

@ -29,5 +29,9 @@ export default {
this.route("edit", { path: "/:id/edit" });
}
);
this.route("discourse-ai-features", { path: "ai-features" }, function () {
this.route("edit", { path: "/:id/edit" });
});
},
};

View File

@ -0,0 +1,21 @@
import RestAdapter from "discourse/adapters/rest";
export default class AiFeatureAdapter extends RestAdapter {
jsonMode = true;
basePath() {
return "/admin/plugins/discourse-ai/";
}
pathFor(store, type, findArgs) {
// removes underscores which are implemented in base
let path =
this.basePath(store, type, findArgs) +
store.pluralize(this.apiNameFor(type));
return this.appendQueryParams(path, findArgs);
}
apiNameFor() {
return "ai-feature";
}
}

View File

@ -0,0 +1,15 @@
import RestModel from "discourse/models/rest";
export default class AiFeature extends RestModel {
createProperties() {
return this.getProperties(
"id",
"name",
"ref",
"description",
"enable_setting",
"persona",
"persona_setting"
);
}
}

View File

@ -41,6 +41,12 @@ export default {
route: "adminPlugins.show.discourse-ai-spam",
description: "discourse_ai.spam.spam_description",
},
// TODO(@keegan / @roman): Uncomment this when structured output is merged
// {
// label: "discourse_ai.features.short_title",
// route: "adminPlugins.show.discourse-ai-features",
// description: "discourse_ai.features.description",
// },
]);
});
},

View File

@ -0,0 +1,65 @@
.ai-feature-list {
&__configured-features {
margin-block: 2rem;
}
&__row-item-name,
&__row-item-description {
display: block;
}
&__row-item-persona {
padding: 0;
text-align: left;
@include ellipsis;
}
&__row-item-groups {
list-style: none;
margin: 0.5em 0 0 0;
display: flex;
li {
font-size: var(--font-down-2);
border-radius: var(--d-border-radius);
background: var(--primary-very-low);
border: 1px solid var(--primary-low);
padding: 1px 3px;
margin-right: 0.5em;
}
}
}
.ai-feature-editor {
&__header {
border-bottom: 1px solid var(--primary-low);
}
.setting {
margin-block: 1.5rem;
}
.setting-label {
font-size: var(--font-down-1-rem);
color: var(--primary-high);
a[title="View change history"],
.history-icon {
display: none;
}
}
.setting-value {
.desc {
font-size: var(--font-down-1-rem);
color: var(--primary-high-or-secondary-low);
}
}
.setting-controls,
.setting-controls__undo {
font-size: var(--font-down-1-rem);
margin-top: 0.5rem;
}
}

View File

@ -165,6 +165,20 @@ en:
discourse_ai:
title: "AI"
features:
short_title: "Features"
description: "These are the AI features available to visitors on your site. These can be configured to use specific personas and LLMs, and can be access controlled by groups."
back: "Back"
list:
header:
name: "Name"
persona: "Persona"
groups: "Groups"
edit: "Edit"
set_up: "Set up"
configured_features: "Configured features"
unconfigured_features: "Unconfigured features"
modals:
select_option: "Select an option..."

View File

@ -81,11 +81,12 @@ en:
ai_embeddings_semantic_search_hyde_model: "Model used to expand keywords to get better results during a semantic search"
ai_embeddings_per_post_enabled: Generate embeddings for each post
ai_summarization_enabled: "Enable the topic summarization module."
ai_summarization_model: "Model to use for summarization."
ai_summarization_enabled: "Enable the summarize feature"
ai_summarization_model: "Model to use for summarization"
ai_summarization_persona: "Persona to use for summarize feature"
ai_custom_summarization_allowed_groups: "Groups allowed to use create new summaries."
ai_pm_summarization_allowed_groups: "Groups allowed to create and view summaries in PMs."
ai_summary_gists_enabled: "Generate brief summaries of latest replies in topics automatically."
ai_summary_gists_enabled: "Generate brief summaries of latest replies in topics automatically"
ai_summary_gists_allowed_groups: "Groups allowed to see gists in the hot topics list."
ai_summary_backfill_maximum_topics_per_hour: "Number of topic summaries to backfill per hour."
@ -104,6 +105,13 @@ en:
ai_google_custom_search_api_key: "API key for the Google Custom Search API see: https://developers.google.com/custom-search"
ai_google_custom_search_cx: "CX for Google Custom Search API"
ai_discord_search_enabled: "Enables the Discord search feature"
ai_discord_app_id: "The ID of the Discord application you would like to connect Discord search to"
ai_discord_app_public_key: "The public key of the Discord application you would like to connect Discord search to"
ai_discord_search_mode: "Select the search mode to use for Discord search"
ai_discord_search_persona: "The persona to use for Discord search."
ai_discord_allowed_guilds: "Discord guilds (servers) where the bot is allowed to search"
reviewables:
reasons:
flagged_by_toxicity: The AI plugin flagged this after classifying it as toxic.
@ -487,6 +495,20 @@ en:
missing_provider_param: "%{param} can't be blank"
bedrock_invalid_url: "Please complete all the fields to use this model."
features:
summarization:
name: "Summaries"
description: "Makes a summarization button available that allows visitors to summarize topics"
gists:
name: "Short Summaries"
description: "Adds the ability to view short summaries of topics on the topic list"
discoveries:
name: "Discobot Discoveries"
description: "Enhances search experience by providing AI-generated answers to queries"
discord_search:
name: "Discord Search"
description: "Adds the ability to search Discord channels"
errors:
quota_exceeded: "You have exceeded the quota for this model. Please try again in %{relative_time}."
quota_required: "You must specify maximum tokens or usages for this model"

View File

@ -110,6 +110,11 @@ Discourse::Application.routes.draw do
controller: "discourse_ai/admin/ai_embeddings" do
collection { get :test }
end
resources :ai_features,
only: %i[index edit],
path: "ai-features",
controller: "discourse_ai/admin/ai_features"
end
end

View File

@ -234,6 +234,7 @@ discourse_ai:
default: false
client: true
validator: "DiscourseAi::Configuration::LlmDependencyValidator"
area: "ai-features/summarization"
ai_summarization_model:
default: ""
allow_any: false
@ -245,11 +246,12 @@ discourse_ai:
default: "-11"
type: enum
enum: "DiscourseAi::Configuration::PersonaEnumerator"
area: "ai-features/summarization"
ai_pm_summarization_allowed_groups:
type: group_list
list_type: compact
default: ""
area: "ai-features/summarization"
ai_custom_summarization_allowed_groups: # Deprecated. TODO(roman): Remove 2025-09-01
type: group_list
list_type: compact
@ -257,12 +259,12 @@ discourse_ai:
hidden: true
ai_summary_gists_enabled:
default: false
hidden: true
area: "ai-features/gists"
ai_summary_gists_persona:
default: "-12"
type: enum
enum: "DiscourseAi::Configuration::PersonaEnumerator"
hidden: true
area: "ai-features/gists"
ai_summary_gists_allowed_groups: # Deprecated. TODO(roman): Remove 2025-09-01
type: group_list
list_type: compact
@ -277,17 +279,20 @@ discourse_ai:
default: 30
min: 1
max: 10000
area: "ai-features/summarization"
ai_summary_backfill_maximum_topics_per_hour:
default: 0
min: 0
max: 10000
area: "ai-features/summarization"
ai_summary_backfill_minimum_word_count:
default: 200
hidden: true
area: "ai-features/summarization"
ai_bot_enabled:
default: false
client: true
area: "ai-features/discoveries"
ai_bot_enable_chat_warning:
default: false
client: true
@ -326,9 +331,9 @@ discourse_ai:
ai_bot_discover_persona:
default: ""
type: enum
hidden: true
client: true
enum: "DiscourseAi::Configuration::PersonaEnumerator"
area: "ai-features/discoveries"
ai_automation_max_triage_per_minute:
default: 60
hidden: true
@ -341,26 +346,35 @@ discourse_ai:
type: list
list_type: compact
ai_discord_search_enabled:
default: false
client: true
area: "ai-features/discord_search"
ai_discord_app_id:
default: ""
client: false
area: "ai-features/discord_search"
ai_discord_app_public_key:
default: ""
client: false
area: "ai-features/discord_search"
ai_discord_search_mode:
default: "search"
type: enum
choices:
- search
- persona
area: "ai-features/discord_search"
ai_discord_search_persona:
default: ""
type: enum
enum: "DiscourseAi::Configuration::PersonaEnumerator"
area: "ai-features/discord_search"
ai_discord_allowed_guilds:
type: list
list_type: compact
default: ""
area: "ai-features/discord_search"
ai_spam_detection_enabled:
default: false

View File

@ -105,10 +105,7 @@ module DiscourseAi
plugin.add_to_serializer(
:current_user,
:ai_enabled_personas,
include_condition: -> do
SiteSetting.ai_bot_enabled && scope.authenticated? &&
scope.user.in_any_groups?(SiteSetting.ai_bot_allowed_groups_map)
end,
include_condition: -> { scope.authenticated? },
) do
DiscourseAi::Personas::Persona
.all(user: scope.user)

84
lib/features.rb Normal file
View File

@ -0,0 +1,84 @@
# frozen_string_literal: true
module DiscourseAi
module Features
def self.feature_config
[
{
id: 1,
name_ref: "summarization",
name_key: "discourse_ai.features.summarization.name",
description_key: "discourse_ai.features.summarization.description",
persona_setting_name: "ai_summarization_persona",
enable_setting_name: "ai_summarization_enabled",
},
{
id: 2,
name_ref: "gists",
name_key: "discourse_ai.features.gists.name",
description_key: "discourse_ai.features.gists.description",
persona_setting_name: "ai_summary_gists_persona",
enable_setting_name: "ai_summary_gists_enabled",
},
{
id: 3,
name_ref: "discoveries",
name_key: "discourse_ai.features.discoveries.name",
description_key: "discourse_ai.features.discoveries.description",
persona_setting_name: "ai_bot_discover_persona",
enable_setting_name: "ai_bot_enabled",
},
{
id: 4,
name_ref: "discord_search",
name_key: "discourse_ai.features.discord_search.name",
description_key: "discourse_ai.features.discord_search.description",
persona_setting_name: "ai_discord_search_persona",
enable_setting_name: "ai_discord_search_enabled",
},
]
end
def self.features
feature_config.map do |feature|
{
id: feature[:id],
ref: feature[:name_ref],
name: I18n.t(feature[:name_key]),
description: I18n.t(feature[:description_key]),
persona: AiPersona.find_by(id: SiteSetting.get(feature[:persona_setting_name])),
persona_setting: {
name: feature[:persona_setting_name],
value: SiteSetting.get(feature[:persona_setting_name]),
type: SiteSetting.type_supervisor.get_type(feature[:persona_setting_name]),
},
enable_setting: {
name: feature[:enable_setting_name],
value: SiteSetting.get(feature[:enable_setting_name]),
type: SiteSetting.type_supervisor.get_type(feature[:enable_setting_name]),
},
}
end
end
def self.find_feature_by_id(id)
lookup = features.index_by { |f| f[:id] }
lookup[id]
end
def self.find_feature_by_ref(name_ref)
lookup = features.index_by { |f| f[:ref] }
lookup[name_ref]
end
def self.find_feature_id_by_ref(name_ref)
find_feature_by_ref(name_ref)&.dig(:id)
end
def self.feature_area(name_ref)
name_ref = name_ref.to_s if name_ref.is_a?(Symbol)
find_feature_by_ref(name_ref) || raise(ArgumentError, "Feature not found: #{name_ref}")
"ai-features/#{name_ref}"
end
end
end

View File

@ -27,6 +27,7 @@ enabled_site_setting :discourse_ai_enabled
register_asset "stylesheets/common/streaming.scss"
register_asset "stylesheets/common/ai-blinking-animation.scss"
register_asset "stylesheets/common/ai-user-settings.scss"
register_asset "stylesheets/common/ai-features.scss"
register_asset "stylesheets/modules/ai-helper/common/ai-helper.scss"
register_asset "stylesheets/modules/ai-helper/desktop/ai-helper-fk-modals.scss", :desktop
@ -69,6 +70,11 @@ end
Rails.autoloaders.main.push_dir(File.join(__dir__, "lib"), namespace: ::DiscourseAi)
require_relative "lib/engine"
require_relative "lib/features"
DiscourseAi::Features.feature_config.each do |feature|
register_site_setting_area("ai-features/#{feature[:name_ref]}")
end
after_initialize do
if defined?(Rack::MiniProfiler)

View File

@ -17,6 +17,7 @@ RSpec.describe Jobs::StreamDiscordReply, type: :job do
fab!(:persona) { Fabricate(:ai_persona, default_llm_id: llm_model.id) }
before do
SiteSetting.ai_discord_search_enabled = true
SiteSetting.ai_discord_search_mode = "persona"
SiteSetting.ai_discord_search_persona = persona.id
end

View File

@ -0,0 +1,32 @@
# frozen_string_literal: true
RSpec.describe DiscourseAi::Admin::AiFeaturesController do
let(:controller) { described_class.new }
fab!(:admin)
fab!(:group)
fab!(:llm_model)
fab!(:summarizer_persona) { Fabricate(:ai_persona) }
fab!(:alternate_summarizer_persona) { Fabricate(:ai_persona) }
before do
sign_in(admin)
SiteSetting.ai_bot_enabled = true
SiteSetting.discourse_ai_enabled = true
end
describe "#index" do
it "lists all features backed by personas" do
get "/admin/plugins/discourse-ai/ai-features.json"
expect(response.status).to eq(200)
expect(response.parsed_body["ai_features"].count).to eq(4)
end
end
describe "#edit" do
it "returns a success response" do
get "/admin/plugins/discourse-ai/ai-features/1/edit.json"
expect(response.parsed_body["name"]).to eq(I18n.t "discourse_ai.features.summarization.name")
end
end
end

View File

@ -0,0 +1,33 @@
# frozen_string_literal: true
RSpec.describe AiFeaturesPersonaSerializer do
fab!(:admin)
fab!(:ai_persona)
fab!(:group)
fab!(:group_2) { Fabricate(:group) }
describe "serialized attributes" do
before do
ai_persona.allowed_group_ids = [group.id, group_2.id]
ai_persona.save!
end
context "when there is a persona with allowed groups" do
let(:allowed_groups) do
Group
.where(id: ai_persona.allowed_group_ids)
.pluck(:id, :name)
.map { |id, name| { id: id, name: name } }
end
it "display every participant" do
serialized = described_class.new(ai_persona, scope: Guardian.new(admin), root: nil)
expect(serialized.id).to eq(ai_persona.id)
expect(serialized.name).to eq(ai_persona.name)
expect(serialized.system_prompt).to eq(ai_persona.system_prompt)
expect(serialized.allowed_groups).to eq(allowed_groups)
expect(serialized.enabled).to eq(ai_persona.enabled)
end
end
end
end

View File

@ -0,0 +1,64 @@
# frozen_string_literal: true
RSpec.describe "Admin AI features configuration", type: :system, js: true do
fab!(:admin)
fab!(:llm_model)
fab!(:summarization_persona) { Fabricate(:ai_persona) }
fab!(:group_1) { Fabricate(:group) }
fab!(:group_2) { Fabricate(:group) }
let(:page_header) { PageObjects::Components::DPageHeader.new }
let(:form) { PageObjects::Components::FormKit.new("form") }
let(:ai_features_page) { PageObjects::Pages::AdminAiFeatures.new }
before do
summarization_persona.allowed_group_ids = [group_1.id, group_2.id]
summarization_persona.save!
assign_fake_provider_to(:ai_summarization_model)
SiteSetting.ai_summarization_enabled = true
SiteSetting.ai_summarization_persona = summarization_persona.id
sign_in(admin)
end
it "lists all persona backed AI features separated by configured/unconfigured" do
ai_features_page.visit
expect(
ai_features_page
.configured_features_table
.find(".ai-feature-list__row-item .ai-feature-list__row-item-name")
.text,
).to eq(I18n.t("discourse_ai.features.summarization.name"))
expect(ai_features_page).to have_configured_feature_items(1)
expect(ai_features_page).to have_unconfigured_feature_items(3)
end
it "lists the persona used for the corresponding AI feature" do
ai_features_page.visit
expect(ai_features_page).to have_feature_persona(summarization_persona.name)
end
it "lists the groups allowed to use the AI feature" do
ai_features_page.visit
expect(ai_features_page).to have_feature_groups([group_1.name, group_2.name])
end
it "can navigate the AI plugin with breadcrumbs" do
visit "/admin/plugins/discourse-ai/ai-features"
expect(page).to have_css(".d-breadcrumbs")
expect(page).to have_css(".d-breadcrumbs__item", count: 4)
find(".d-breadcrumbs__item", text: I18n.t("admin_js.admin.plugins.title")).click
expect(page).to have_current_path("/admin/plugins")
end
it "shows edit page with settings" do
ai_features_page.visit
ai_features_page.click_edit_feature(I18n.t("discourse_ai.features.summarization.name"))
expect(page).to have_current_path("/admin/plugins/discourse-ai/ai-features/1/edit")
expect(page).to have_css(
".ai-feature-editor__header h2",
text: I18n.t("discourse_ai.features.summarization.name"),
)
expect(page).to have_css(".setting")
end
end

View File

@ -0,0 +1,51 @@
# frozen_string_literal: true
module PageObjects
module Pages
class AdminAiFeatures < PageObjects::Pages::Base
CONFIGURED_FEATURES_TABLE = ".ai-feature-list__configured-features .d-admin-table"
UNCONFIGURED_FEATURES_TABLE = ".ai-feature-list__unconfigured-features .d-admin-table"
def visit
page.visit("/admin/plugins/discourse-ai/ai-features")
self
end
def configured_features_table
page.find(CONFIGURED_FEATURES_TABLE)
end
def unconfigured_features_table
page.find(UNCONFIGURED_FEATURES_TABLE)
end
def has_configured_feature_items?(count)
page.has_css?("#{CONFIGURED_FEATURES_TABLE} .ai-feature-list__row", count: count)
end
def has_unconfigured_feature_items?(count)
page.has_css?("#{UNCONFIGURED_FEATURES_TABLE} .ai-feature-list__row", count: count)
end
def has_feature_persona?(name)
page.has_css?(
"#{CONFIGURED_FEATURES_TABLE} .ai-feature-list__persona .d-button-label ",
text: name,
)
end
def has_feature_groups?(groups)
listed_groups = page.find("#{CONFIGURED_FEATURES_TABLE} .ai-feature-list__groups")
list_items = listed_groups.all("li", visible: true).map(&:text)
list_items.sort == groups.sort
end
def click_edit_feature(feature_name)
page.find(
"#{CONFIGURED_FEATURES_TABLE} .ai-feature-list__row[data-feature-name='#{feature_name}'] .edit",
).click
end
end
end
end