REFACTOR: update embeddings to formkit (#1188)
This commit is contained in:
parent
ec8018333e
commit
51ca942d7d
|
@ -1,23 +1,19 @@
|
|||
import Component from "@glimmer/component";
|
||||
import { tracked } from "@glimmer/tracking";
|
||||
import { Input, Textarea } from "@ember/component";
|
||||
import { cached, tracked } from "@glimmer/tracking";
|
||||
import { concat, fn, get } from "@ember/helper";
|
||||
import { on } from "@ember/modifier";
|
||||
import { action, computed } from "@ember/object";
|
||||
import didInsert from "@ember/render-modifiers/modifiers/did-insert";
|
||||
import didUpdate from "@ember/render-modifiers/modifiers/did-update";
|
||||
import { action } from "@ember/object";
|
||||
import { later } from "@ember/runloop";
|
||||
import { service } from "@ember/service";
|
||||
import { eq, not } from "truth-helpers";
|
||||
import BackButton from "discourse/components/back-button";
|
||||
import ConditionalLoadingSpinner from "discourse/components/conditional-loading-spinner";
|
||||
import DButton from "discourse/components/d-button";
|
||||
import Form from "discourse/components/form";
|
||||
import icon from "discourse/helpers/d-icon";
|
||||
import { popupAjaxError } from "discourse/lib/ajax-error";
|
||||
import { i18n } from "discourse-i18n";
|
||||
import AdminSectionLandingItem from "admin/components/admin-section-landing-item";
|
||||
import AdminSectionLandingWrapper from "admin/components/admin-section-landing-wrapper";
|
||||
import ComboBox from "select-kit/components/combo-box";
|
||||
import DTooltip from "float-kit/components/d-tooltip";
|
||||
import not from "truth-helpers/helpers/not";
|
||||
|
||||
export default class AiEmbeddingEditor extends Component {
|
||||
@service toasts;
|
||||
|
@ -27,12 +23,35 @@ export default class AiEmbeddingEditor extends Component {
|
|||
|
||||
@tracked isSaving = false;
|
||||
@tracked selectedPreset = null;
|
||||
|
||||
@tracked testRunning = false;
|
||||
@tracked testResult = null;
|
||||
@tracked testError = null;
|
||||
@tracked apiKeySecret = true;
|
||||
@tracked editingModel = null;
|
||||
@tracked currentProvider = null;
|
||||
|
||||
constructor() {
|
||||
super(...arguments);
|
||||
if (this.args.model) {
|
||||
this.currentProvider = this.args.model.provider;
|
||||
}
|
||||
}
|
||||
|
||||
@cached
|
||||
get formData() {
|
||||
let data;
|
||||
|
||||
if (this.selectedPreset) {
|
||||
data = this.store
|
||||
.createRecord("ai-embedding", this.selectedPreset)
|
||||
.workingCopy();
|
||||
} else {
|
||||
data = this.args.model.workingCopy();
|
||||
}
|
||||
|
||||
const originalData = JSON.parse(JSON.stringify(data));
|
||||
this._originalFormData = originalData;
|
||||
|
||||
return originalData;
|
||||
}
|
||||
|
||||
get selectedProviders() {
|
||||
const t = (provName) => {
|
||||
|
@ -50,18 +69,9 @@ export default class AiEmbeddingEditor extends Component {
|
|||
};
|
||||
|
||||
return this.args.embeddings.resultSetMeta.distance_functions.map((df) => {
|
||||
let iconName;
|
||||
|
||||
if (df === "<=>") {
|
||||
iconName = "discourse-spaceship-operator";
|
||||
} else if (df === "<#>") {
|
||||
iconName = "discourse-negative-inner-product";
|
||||
}
|
||||
|
||||
return {
|
||||
id: df,
|
||||
name: t(df),
|
||||
icon: iconName,
|
||||
};
|
||||
});
|
||||
}
|
||||
|
@ -88,13 +98,17 @@ export default class AiEmbeddingEditor extends Component {
|
|||
return !this.selectedPreset && this.args.model.isNew;
|
||||
}
|
||||
|
||||
@computed("editingModel.provider")
|
||||
get metaProviderParams() {
|
||||
return (
|
||||
this.args.embeddings.resultSetMeta.provider_params[
|
||||
this.editingModel?.provider
|
||||
] || {}
|
||||
);
|
||||
const provider = this.currentProvider;
|
||||
if (!provider) {
|
||||
return {};
|
||||
}
|
||||
|
||||
const embeddings = this.args.embeddings || {};
|
||||
const meta = embeddings.resultSetMeta || {};
|
||||
const providerParams = meta.provider_params || {};
|
||||
|
||||
return providerParams[provider] || {};
|
||||
}
|
||||
|
||||
get testErrorMessage() {
|
||||
|
@ -105,6 +119,10 @@ export default class AiEmbeddingEditor extends Component {
|
|||
return this.testRunning || this.testResult !== null;
|
||||
}
|
||||
|
||||
get seeded() {
|
||||
return this.args.model.id < 0;
|
||||
}
|
||||
|
||||
@action
|
||||
configurePreset(preset) {
|
||||
this.selectedPreset =
|
||||
|
@ -113,40 +131,96 @@ export default class AiEmbeddingEditor extends Component {
|
|||
preset.id
|
||||
) || {};
|
||||
|
||||
this.editingModel = this.store
|
||||
.createRecord("ai-embedding", this.selectedPreset)
|
||||
.workingCopy();
|
||||
if (this.selectedPreset.provider) {
|
||||
this.currentProvider = this.selectedPreset.provider;
|
||||
}
|
||||
}
|
||||
|
||||
@action
|
||||
updateModel() {
|
||||
this.editingModel = this.args.model.workingCopy();
|
||||
setProvider(provider, { set }) {
|
||||
set("provider", provider);
|
||||
|
||||
this.currentProvider = provider;
|
||||
|
||||
const providerParams =
|
||||
this.args.embeddings?.resultSetMeta?.provider_params || {};
|
||||
const params = providerParams[provider] || {};
|
||||
|
||||
const initialParams = {};
|
||||
|
||||
if (params) {
|
||||
const keys = Object.keys(params);
|
||||
keys.forEach((key) => {
|
||||
initialParams[key] = null;
|
||||
});
|
||||
}
|
||||
|
||||
set("provider_params", initialParams);
|
||||
}
|
||||
|
||||
get providerParams() {
|
||||
const normalizeParam = (value) => {
|
||||
if (!value) {
|
||||
return { type: "text" };
|
||||
}
|
||||
|
||||
if (typeof value === "string") {
|
||||
return { type: value };
|
||||
}
|
||||
|
||||
return {
|
||||
type: value.type || "text",
|
||||
values: (value.values || []).map((v) => ({ id: v, name: v })),
|
||||
default: value.default,
|
||||
};
|
||||
};
|
||||
|
||||
return Object.entries(this.metaProviderParams).reduce(
|
||||
(acc, [field, value]) => {
|
||||
acc[field] = normalizeParam(value);
|
||||
return acc;
|
||||
},
|
||||
{}
|
||||
);
|
||||
}
|
||||
|
||||
@action
|
||||
makeApiKeySecret() {
|
||||
this.apiKeySecret = true;
|
||||
resetForm() {
|
||||
this.selectedPreset = null;
|
||||
this.currentProvider = null;
|
||||
}
|
||||
|
||||
@action
|
||||
toggleApiKeySecret() {
|
||||
this.apiKeySecret = !this.apiKeySecret;
|
||||
}
|
||||
|
||||
@action
|
||||
async save() {
|
||||
async save(formData) {
|
||||
this.isSaving = true;
|
||||
const isNew = this.args.model.isNew;
|
||||
|
||||
try {
|
||||
await this.editingModel.save();
|
||||
const dataToSave = { ...formData };
|
||||
|
||||
if (this.selectedPreset) {
|
||||
// new embeddings
|
||||
const newModel = this.store.createRecord("ai-embedding", {
|
||||
...this.selectedPreset,
|
||||
...dataToSave,
|
||||
});
|
||||
await newModel.save();
|
||||
this.args.embeddings.addObject(newModel);
|
||||
} else {
|
||||
// existing embeddings
|
||||
await this.args.model.save(dataToSave);
|
||||
}
|
||||
|
||||
if (isNew) {
|
||||
this.args.embeddings.addObject(this.editingModel);
|
||||
this.router.transitionTo(
|
||||
"adminPlugins.show.discourse-ai-embeddings.index"
|
||||
);
|
||||
} else {
|
||||
const savedProvider = this.currentProvider;
|
||||
|
||||
this._originalFormData = JSON.parse(JSON.stringify(dataToSave));
|
||||
this.currentProvider = savedProvider;
|
||||
|
||||
this.toasts.success({
|
||||
data: { message: i18n("discourse_ai.embeddings.saved") },
|
||||
duration: 2000,
|
||||
|
@ -162,11 +236,11 @@ export default class AiEmbeddingEditor extends Component {
|
|||
}
|
||||
|
||||
@action
|
||||
async test() {
|
||||
async test(data) {
|
||||
this.testRunning = true;
|
||||
|
||||
try {
|
||||
const configTestResult = await this.editingModel.testConfig();
|
||||
const configTestResult = await this.args.model.testConfig(data);
|
||||
this.testResult = configTestResult.success;
|
||||
|
||||
if (this.testResult) {
|
||||
|
@ -191,7 +265,7 @@ export default class AiEmbeddingEditor extends Component {
|
|||
return this.args.model
|
||||
.destroyRecord()
|
||||
.then(() => {
|
||||
this.args.llms.removeObject(this.args.model);
|
||||
this.args.embeddings.removeObject(this.args.model);
|
||||
this.router.transitionTo(
|
||||
"adminPlugins.show.discourse-ai-embeddings.index"
|
||||
);
|
||||
|
@ -201,52 +275,44 @@ export default class AiEmbeddingEditor extends Component {
|
|||
});
|
||||
}
|
||||
|
||||
@action
|
||||
resetForm() {
|
||||
this.selectedPreset = null;
|
||||
this.editingModel = null;
|
||||
}
|
||||
|
||||
<template>
|
||||
<form
|
||||
{{didInsert this.updateModel @model.id}}
|
||||
{{didUpdate this.updateModel @model.id}}
|
||||
class="form-horizontal ai-embedding-editor"
|
||||
>
|
||||
{{#if this.showPresets}}
|
||||
<BackButton
|
||||
@route="adminPlugins.show.discourse-ai-embeddings"
|
||||
@label="discourse_ai.embeddings.back"
|
||||
/>
|
||||
<div class="control-group">
|
||||
<h2>{{i18n "discourse_ai.embeddings.presets"}}</h2>
|
||||
<AdminSectionLandingWrapper>
|
||||
{{#each this.presets as |preset|}}
|
||||
<AdminSectionLandingItem
|
||||
@titleLabelTranslated={{preset.name}}
|
||||
@taglineLabel={{concat
|
||||
"discourse_ai.embeddings.providers."
|
||||
preset.provider
|
||||
}}
|
||||
data-preset-id={{preset.id}}
|
||||
class="ai-llms-list-editor__templates-list-item"
|
||||
>
|
||||
<:buttons as |buttons|>
|
||||
<buttons.Default
|
||||
@action={{fn this.configurePreset preset}}
|
||||
@icon="gear"
|
||||
@label="discourse_ai.llms.preconfigured.button"
|
||||
/>
|
||||
</:buttons>
|
||||
</AdminSectionLandingItem>
|
||||
|
||||
{{/each}}
|
||||
</AdminSectionLandingWrapper>
|
||||
|
||||
</div>
|
||||
|
||||
{{else}}
|
||||
{{#if this.editingModel.isNew}}
|
||||
{{#if this.showPresets}}
|
||||
<BackButton
|
||||
@route="adminPlugins.show.discourse-ai-embeddings"
|
||||
@label="discourse_ai.embeddings.back"
|
||||
/>
|
||||
<div class="control-group">
|
||||
<h2>{{i18n "discourse_ai.embeddings.presets"}}</h2>
|
||||
<AdminSectionLandingWrapper>
|
||||
{{#each this.presets as |preset|}}
|
||||
<AdminSectionLandingItem
|
||||
@titleLabelTranslated={{preset.name}}
|
||||
@taglineLabel={{concat
|
||||
"discourse_ai.embeddings.providers."
|
||||
preset.provider
|
||||
}}
|
||||
data-preset-id={{preset.id}}
|
||||
class="ai-llms-list-editor__templates-list-item"
|
||||
>
|
||||
<:buttons as |buttons|>
|
||||
<buttons.Default
|
||||
@action={{fn this.configurePreset preset}}
|
||||
@icon="gear"
|
||||
@label="discourse_ai.llms.preconfigured.button"
|
||||
/>
|
||||
</:buttons>
|
||||
</AdminSectionLandingItem>
|
||||
{{/each}}
|
||||
</AdminSectionLandingWrapper>
|
||||
</div>
|
||||
{{else}}
|
||||
<Form
|
||||
@onSubmit={{this.save}}
|
||||
@data={{this.formData}}
|
||||
class="form-horizontal ai-embedding-editor {{if this.seeded 'seeded'}}"
|
||||
as |form data|
|
||||
>
|
||||
{{#if @model.isNew}}
|
||||
<DButton
|
||||
@action={{this.resetForm}}
|
||||
@label="back_button"
|
||||
|
@ -259,194 +325,231 @@ export default class AiEmbeddingEditor extends Component {
|
|||
@label="discourse_ai.embeddings.back"
|
||||
/>
|
||||
{{/if}}
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.display_name"}}</label>
|
||||
<Input
|
||||
class="ai-embedding-editor-input ai-embedding-editor__display-name"
|
||||
@type="text"
|
||||
@value={{this.editingModel.display_name}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.provider"}}</label>
|
||||
<ComboBox
|
||||
@value={{this.editingModel.provider}}
|
||||
@content={{this.selectedProviders}}
|
||||
@class="ai-embedding-editor__provider"
|
||||
/>
|
||||
</div>
|
||||
<form.Field
|
||||
@name="display_name"
|
||||
@title={{i18n "discourse_ai.embeddings.display_name"}}
|
||||
@validation="required|length:1,100"
|
||||
@format="large"
|
||||
class="ai-embedding-editor__display-name"
|
||||
as |field|
|
||||
>
|
||||
<field.Input />
|
||||
</form.Field>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.url"}}</label>
|
||||
<Input
|
||||
class="ai-embedding-editor-input ai-embedding-editor__url"
|
||||
@type="text"
|
||||
@value={{this.editingModel.url}}
|
||||
required="true"
|
||||
/>
|
||||
</div>
|
||||
<form.Field
|
||||
@name="provider"
|
||||
@title={{i18n "discourse_ai.embeddings.provider"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
@onSet={{this.setProvider}}
|
||||
class="ai-embedding-editor__provider"
|
||||
as |field|
|
||||
>
|
||||
<field.Select as |select|>
|
||||
{{#each this.selectedProviders as |provider|}}
|
||||
<select.Option
|
||||
@value={{provider.id}}
|
||||
>{{provider.name}}</select.Option>
|
||||
{{/each}}
|
||||
</field.Select>
|
||||
</form.Field>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.api_key"}}</label>
|
||||
<div class="ai-embedding-editor__secret-api-key-group">
|
||||
<Input
|
||||
@value={{this.editingModel.api_key}}
|
||||
class="ai-embedding-editor-input ai-embedding-editor__api-key"
|
||||
@type={{if this.apiKeySecret "password" "text"}}
|
||||
required="true"
|
||||
{{on "focusout" this.makeApiKeySecret}}
|
||||
/>
|
||||
<DButton
|
||||
@action={{this.toggleApiKeySecret}}
|
||||
@icon="far-eye-slash"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<form.Field
|
||||
@name="url"
|
||||
@title={{i18n "discourse_ai.embeddings.url"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
class="ai-embedding-editor__url"
|
||||
as |field|
|
||||
>
|
||||
<field.Input />
|
||||
</form.Field>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.tokenizer"}}</label>
|
||||
<ComboBox
|
||||
@value={{this.editingModel.tokenizer_class}}
|
||||
@content={{@embeddings.resultSetMeta.tokenizers}}
|
||||
@class="ai-embedding-editor__tokenizer"
|
||||
/>
|
||||
</div>
|
||||
<form.Field
|
||||
@name="api_key"
|
||||
@title={{i18n "discourse_ai.embeddings.api_key"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
class="ai-embedding-editor__api-key"
|
||||
as |field|
|
||||
>
|
||||
<field.Password />
|
||||
</form.Field>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.dimensions"}}</label>
|
||||
<Input
|
||||
<form.Field
|
||||
@name="tokenizer_class"
|
||||
@title={{i18n "discourse_ai.embeddings.tokenizer"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
class="ai-embedding-editor__tokenizer"
|
||||
as |field|
|
||||
>
|
||||
<field.Select as |select|>
|
||||
{{#each @embeddings.resultSetMeta.tokenizers as |tokenizer|}}
|
||||
<select.Option
|
||||
@value={{tokenizer.id}}
|
||||
>{{tokenizer.name}}</select.Option>
|
||||
{{/each}}
|
||||
</field.Select>
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="dimensions"
|
||||
@title={{i18n "discourse_ai.embeddings.dimensions"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
@tooltip={{if
|
||||
@model.isNew
|
||||
(i18n "discourse_ai.embeddings.hints.dimensions_warning")
|
||||
}}
|
||||
class="ai-embedding-editor__dimensions"
|
||||
as |field|
|
||||
>
|
||||
<field.Input
|
||||
@type="number"
|
||||
class="ai-embedding-editor-input ai-embedding-editor__dimensions"
|
||||
step="any"
|
||||
min="0"
|
||||
lang="en"
|
||||
@value={{this.editingModel.dimensions}}
|
||||
required="true"
|
||||
disabled={{not this.editingModel.isNew}}
|
||||
disabled={{not @model.isNew}}
|
||||
/>
|
||||
{{#if this.editingModel.isNew}}
|
||||
<DTooltip
|
||||
@icon="circle-exclamation"
|
||||
@content={{i18n
|
||||
"discourse_ai.embeddings.hints.dimensions_warning"
|
||||
}}
|
||||
/>
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="matryoshka_dimensions"
|
||||
@title={{i18n "discourse_ai.embeddings.matryoshka_dimensions"}}
|
||||
@tooltip={{i18n
|
||||
"discourse_ai.embeddings.hints.matryoshka_dimensions"
|
||||
}}
|
||||
@format="large"
|
||||
class="ai-embedding-editor__matryoshka_dimensions"
|
||||
as |field|
|
||||
>
|
||||
<field.Checkbox />
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="embed_prompt"
|
||||
@title={{i18n "discourse_ai.embeddings.embed_prompt"}}
|
||||
@tooltip={{i18n "discourse_ai.embeddings.hints.embed_prompt"}}
|
||||
@format="large"
|
||||
class="ai-embedding-editor__embed_prompt"
|
||||
as |field|
|
||||
>
|
||||
<field.Textarea />
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="search_prompt"
|
||||
@title={{i18n "discourse_ai.embeddings.search_prompt"}}
|
||||
@tooltip={{i18n "discourse_ai.embeddings.hints.search_prompt"}}
|
||||
@format="large"
|
||||
class="ai-embedding-editor__search_prompt"
|
||||
as |field|
|
||||
>
|
||||
<field.Textarea />
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="max_sequence_length"
|
||||
@title={{i18n "discourse_ai.embeddings.max_sequence_length"}}
|
||||
@tooltip={{i18n "discourse_ai.embeddings.hints.sequence_length"}}
|
||||
@validation="required"
|
||||
@format="large"
|
||||
class="ai-embedding-editor__max_sequence_length"
|
||||
as |field|
|
||||
>
|
||||
<field.Input @type="number" step="any" min="0" lang="en" />
|
||||
</form.Field>
|
||||
|
||||
<form.Field
|
||||
@name="pg_function"
|
||||
@title={{i18n "discourse_ai.embeddings.distance_function"}}
|
||||
@tooltip={{i18n "discourse_ai.embeddings.hints.distance_function"}}
|
||||
@format="large"
|
||||
@validation="required"
|
||||
class="ai-embedding-editor__distance_functions"
|
||||
as |field|
|
||||
>
|
||||
<field.Select @includeNone={{false}} as |select|>
|
||||
{{#each this.distanceFunctions as |df|}}
|
||||
<select.Option @value={{df.id}}>{{df.name}}</select.Option>
|
||||
{{/each}}
|
||||
</field.Select>
|
||||
</form.Field>
|
||||
|
||||
{{! provider-specific content }}
|
||||
{{#if this.currentProvider}}
|
||||
{{#if data.provider_params}}
|
||||
<form.Object @name="provider_params" as |object name|>
|
||||
{{#let (get this.providerParams name) as |params|}}
|
||||
{{#if params}}
|
||||
<object.Field
|
||||
@name={{name}}
|
||||
@title={{i18n
|
||||
(concat "discourse_ai.embeddings.provider_fields." name)
|
||||
}}
|
||||
@format="large"
|
||||
@validation="required"
|
||||
class="ai-embedding-editor-provider-param__{{params.type}}"
|
||||
as |field|
|
||||
>
|
||||
{{#if (eq params.type "enum")}}
|
||||
<field.Select @includeNone={{false}} as |select|>
|
||||
{{#each params.values as |option|}}
|
||||
<select.Option
|
||||
@value={{option.id}}
|
||||
>{{option.name}}</select.Option>
|
||||
{{/each}}
|
||||
</field.Select>
|
||||
{{else if (eq params.type "checkbox")}}
|
||||
<field.Checkbox />
|
||||
{{else}}
|
||||
<field.Input @type={{params.type}} />
|
||||
{{/if}}
|
||||
</object.Field>
|
||||
{{/if}}
|
||||
{{/let}}
|
||||
</form.Object>
|
||||
{{/if}}
|
||||
</div>
|
||||
{{/if}}
|
||||
|
||||
<div class="control-group ai-embedding-editor__matryoshka_dimensions">
|
||||
<Input
|
||||
@type="checkbox"
|
||||
@checked={{this.editingModel.matryoshka_dimensions}}
|
||||
/>
|
||||
<label>{{i18n "discourse_ai.embeddings.matryoshka_dimensions"}}
|
||||
</label>
|
||||
<DTooltip
|
||||
@icon="circle-question"
|
||||
@content={{i18n
|
||||
"discourse_ai.embeddings.hints.matryoshka_dimensions"
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.embed_prompt"}}</label>
|
||||
<Textarea
|
||||
class="ai-embedding-editor-input ai-embedding-editor__embed_prompt"
|
||||
@value={{this.editingModel.embed_prompt}}
|
||||
/>
|
||||
<DTooltip
|
||||
@icon="circle-question"
|
||||
@content={{i18n "discourse_ai.embeddings.hints.embed_prompt"}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.search_prompt"}}</label>
|
||||
<Textarea
|
||||
class="ai-embedding-editor-input ai-embedding-editor__search_prompt"
|
||||
@value={{this.editingModel.search_prompt}}
|
||||
/>
|
||||
<DTooltip
|
||||
@icon="circle-question"
|
||||
@content={{i18n "discourse_ai.embeddings.hints.search_prompt"}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.max_sequence_length"}}</label>
|
||||
<Input
|
||||
@type="number"
|
||||
class="ai-embedding-editor-input ai-embedding-editor__max_sequence_length"
|
||||
step="any"
|
||||
min="0"
|
||||
lang="en"
|
||||
@value={{this.editingModel.max_sequence_length}}
|
||||
required="true"
|
||||
/>
|
||||
<DTooltip
|
||||
@icon="circle-question"
|
||||
@content={{i18n "discourse_ai.embeddings.hints.sequence_length"}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="control-group">
|
||||
<label>{{i18n "discourse_ai.embeddings.distance_function"}}</label>
|
||||
<ComboBox
|
||||
@value={{this.editingModel.pg_function}}
|
||||
@content={{this.distanceFunctions}}
|
||||
@class="ai-embedding-editor__distance_functions"
|
||||
/>
|
||||
<DTooltip
|
||||
@icon="circle-question"
|
||||
@content={{i18n "discourse_ai.embeddings.hints.distance_function"}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{{#each-in this.metaProviderParams as |field type|}}
|
||||
<div
|
||||
class="control-group ai-embedding-editor-provider-param__{{type}}"
|
||||
>
|
||||
<label>
|
||||
{{i18n (concat "discourse_ai.embeddings.provider_fields." field)}}
|
||||
</label>
|
||||
<Input
|
||||
@type="text"
|
||||
class="ai-embedding-editor-input ai-embedding-editor__{{field}}"
|
||||
@value={{mut (get this.editingModel.provider_params field)}}
|
||||
/>
|
||||
</div>
|
||||
{{/each-in}}
|
||||
|
||||
<div class="control-group ai-embedding-editor__action_panel">
|
||||
<DButton
|
||||
class="ai-embedding-editor__test"
|
||||
@action={{this.test}}
|
||||
<form.Actions class="ai-embedding-editor__action_panel">
|
||||
<form.Button
|
||||
@action={{fn this.test data}}
|
||||
@disabled={{this.testRunning}}
|
||||
@label="discourse_ai.embeddings.tests.title"
|
||||
class="ai-embedding-editor__test"
|
||||
/>
|
||||
|
||||
<DButton
|
||||
class="btn-primary ai-embedding-editor__save"
|
||||
@action={{this.save}}
|
||||
@disabled={{this.isSaving}}
|
||||
<form.Submit
|
||||
@label="discourse_ai.embeddings.save"
|
||||
class="btn-primary ai-embedding-editor__save"
|
||||
/>
|
||||
{{#unless this.editingModel.isNew}}
|
||||
<DButton
|
||||
|
||||
{{#unless data.isNew}}
|
||||
<form.Button
|
||||
@action={{this.delete}}
|
||||
class="btn-danger ai-embedding-editor__delete"
|
||||
@label="discourse_ai.embeddings.delete"
|
||||
class="btn-danger ai-embedding-editor__delete"
|
||||
/>
|
||||
{{/unless}}
|
||||
</form.Actions>
|
||||
|
||||
<div class="control-group ai-embedding-editor-tests">
|
||||
{{#if this.displayTestResult}}
|
||||
{{#if this.testRunning}}
|
||||
<div class="spinner small"></div>
|
||||
{{i18n "discourse_ai.embeddings.tests.running"}}
|
||||
{{else}}
|
||||
{{#if this.displayTestResult}}
|
||||
<form.Field
|
||||
@showTitle={{false}}
|
||||
@name="test_results"
|
||||
@title="test_results"
|
||||
@format="full"
|
||||
class="ai-embedding-editor-tests"
|
||||
as |field|
|
||||
>
|
||||
<field.Custom>
|
||||
<ConditionalLoadingSpinner
|
||||
@size="small"
|
||||
@condition={{this.testRunning}}
|
||||
>
|
||||
{{#if this.testResult}}
|
||||
<div class="ai-embedding-editor-tests__success">
|
||||
{{icon "check"}}
|
||||
|
@ -458,11 +561,11 @@ export default class AiEmbeddingEditor extends Component {
|
|||
{{this.testErrorMessage}}
|
||||
</div>
|
||||
{{/if}}
|
||||
{{/if}}
|
||||
{{/if}}
|
||||
</div>
|
||||
</div>
|
||||
{{/if}}
|
||||
</form>
|
||||
</ConditionalLoadingSpinner>
|
||||
</field.Custom>
|
||||
</form.Field>
|
||||
{{/if}}
|
||||
</Form>
|
||||
{{/if}}
|
||||
</template>
|
||||
}
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
RSpec.describe "Managing Embeddings configurations", type: :system, js: true do
|
||||
fab!(:admin)
|
||||
let(:page_header) { PageObjects::Components::DPageHeader.new }
|
||||
let(:form) { PageObjects::Components::FormKit.new("form") }
|
||||
|
||||
before { sign_in(admin) }
|
||||
|
||||
|
@ -16,8 +17,8 @@ RSpec.describe "Managing Embeddings configurations", type: :system, js: true do
|
|||
|
||||
find("[data-preset-id='text-embedding-3-small'] button").click()
|
||||
|
||||
find("input.ai-embedding-editor__api-key").fill_in(with: api_key)
|
||||
find(".ai-embedding-editor__save").click()
|
||||
form.field("api_key").fill_in(api_key)
|
||||
form.submit
|
||||
|
||||
expect(page).to have_current_path("/admin/plugins/discourse-ai/ai-embeddings")
|
||||
|
||||
|
@ -45,35 +46,27 @@ RSpec.describe "Managing Embeddings configurations", type: :system, js: true do
|
|||
|
||||
find("[data-preset-id='manual'] button").click()
|
||||
|
||||
find("input.ai-embedding-editor__display-name").fill_in(with: "text-embedding-3-small")
|
||||
|
||||
select_kit = PageObjects::Components::SelectKit.new(".ai-embedding-editor__provider")
|
||||
select_kit.expand
|
||||
select_kit.select_row_by_value(EmbeddingDefinition::OPEN_AI)
|
||||
|
||||
find("input.ai-embedding-editor__url").fill_in(with: "https://api.openai.com/v1/embeddings")
|
||||
find("input.ai-embedding-editor__api-key").fill_in(with: api_key)
|
||||
|
||||
select_kit = PageObjects::Components::SelectKit.new(".ai-embedding-editor__tokenizer")
|
||||
select_kit.expand
|
||||
select_kit.select_row_by_value("DiscourseAi::Tokenizer::OpenAiTokenizer")
|
||||
form.field("display_name").fill_in("text-embedding-3-small")
|
||||
form.field("provider").select(EmbeddingDefinition::OPEN_AI)
|
||||
form.field("url").fill_in("https://api.openai.com/v1/embeddings")
|
||||
form.field("api_key").fill_in(api_key)
|
||||
form.field("tokenizer_class").select("DiscourseAi::Tokenizer::OpenAiTokenizer")
|
||||
|
||||
embed_prefix = "On creation:"
|
||||
search_prefix = "On search:"
|
||||
find(".ai-embedding-editor__embed_prompt").fill_in(with: embed_prefix)
|
||||
find(".ai-embedding-editor__search_prompt").fill_in(with: search_prefix)
|
||||
form.field("embed_prompt").fill_in(embed_prefix)
|
||||
form.field("search_prompt").fill_in(search_prefix)
|
||||
form.field("dimensions").fill_in(1536)
|
||||
form.field("max_sequence_length").fill_in(8191)
|
||||
form.field("pg_function").select("<=>")
|
||||
form.field("provider_params.model_name").fill_in("text-embedding-3-small")
|
||||
|
||||
find("input.ai-embedding-editor__dimensions").fill_in(with: 1536)
|
||||
find("input.ai-embedding-editor__max_sequence_length").fill_in(with: 8191)
|
||||
|
||||
select_kit = PageObjects::Components::SelectKit.new(".ai-embedding-editor__distance_functions")
|
||||
select_kit.expand
|
||||
select_kit.select_row_by_value("<=>")
|
||||
find(".ai-embedding-editor__save").click()
|
||||
form.submit
|
||||
|
||||
expect(page).to have_current_path("/admin/plugins/discourse-ai/ai-embeddings")
|
||||
|
||||
embedding_def = EmbeddingDefinition.order(:id).last
|
||||
|
||||
expect(embedding_def.api_key).to eq(api_key)
|
||||
|
||||
preset = EmbeddingDefinition.presets.find { |p| p[:preset_id] == "text-embedding-3-small" }
|
||||
|
|
|
@ -14,16 +14,6 @@
|
|||
<path d="M448 309.64V413H64V290h190c12.23 0 18.46-14.71 9.94-23.48-7.56-7.78-14.48-16.2-20.66-25.16-5.36-7.77-14.24-12.36-23.69-12.36H64V93h130c10.65 0 20.14-6.92 23.18-17.12a193.4 193.4 0 0 1 9.74-25.76c4.56-9.86-2.77-21.12-13.63-21.12H64C28.65 29 0 57.65 0 93v320c0 35.35 28.65 64 64 64h384c35.35 0 64-28.65 64-64V263.8c0-9.16-10.7-14.14-17.7-8.25a194.056 194.056 0 0 1-25.76 18.32c-12.66 7.53-20.53 21.05-20.53 35.77Z"/><path d="M247.46 122.15c-6.45 2.44-10.75 8.6-10.75 15.49s4.3 13.05 10.75 15.49l81.01 30.4 30.4 81.01c2.44 6.45 8.6 10.75 15.49 10.75s13.05-4.3 15.49-10.75l30.4-81.01 81.01-30.4c6.45-2.44 10.75-8.6 10.75-15.49s-4.3-13.05-10.75-15.49l-81.01-30.4-30.4-81.01c-2.44-6.45-8.6-10.75-15.49-10.75s-13.05 4.3-15.49 10.75l-30.4 81.01-81.01 30.4Z"/>
|
||||
</symbol>
|
||||
|
||||
<!-- "Discourse Spaceship Operator" is a Discourse derivative of https://fontawesome.com/icons/equals?f=classic&s=solid and https://fontawesome.com/icons/less-than?f=classic&s=solid -->
|
||||
<symbol id="discourse-spaceship-operator" viewBox="0 0 448 512">
|
||||
<path d="M132.3 195.8c2.7 5.5.5 12.2-5 14.9L36 256.4l91.3 45.7c5.5 2.7 7.7 9.4 5 14.9s-9.4 7.7-14.9 5L6.2 266.4c-3.8-1.9-6.2-5.7-6.2-9.9s2.4-8.1 6.2-9.9L117.5 191c5.5-2.7 12.2-.5 14.9 5ZM315.7 317c-2.7-5.5-.5-12.2 5-14.9l91.3-45.7-91.3-45.7c-5.5-2.7-7.7-9.4-5-14.9s9.4-7.7 14.9-5l111.3 55.6c3.8 1.9 6.2 5.7 6.2 9.9s-2.4 8.1-6.2 9.9l-111.3 55.6c-5.5 2.7-12.2.5-14.9-5ZM169.2 211.5c-5.6 0-10.1 5.3-10.1 11.8s4.5 11.8 10.1 11.8h110.9c5.6 0 10.1-5.3 10.1-11.8s-4.5-11.8-10.1-11.8H169.2Zm0 65.9c-5.6 0-10.1 5.3-10.1 11.8s4.5 11.8 10.1 11.8h110.9c5.6 0 10.1-5.3 10.1-11.8s-4.5-11.8-10.1-11.8H169.2Z"/>
|
||||
</symbol>
|
||||
|
||||
<!-- "Discourse Negative Inner Product" is a Discourse derivative of https://fontawesome.com/icons/hashtag?f=classic&s=solid and https://fontawesome.com/icons/less-than?f=classic&s=solid -->
|
||||
<symbol id="discourse-negative-inner-product" viewBox="0 0 448 512">
|
||||
<path d="M132.3 195.8c2.7 5.5.5 12.2-5 14.9L36 256.4l91.3 45.7c5.5 2.7 7.7 9.4 5 14.9s-9.4 7.7-14.9 5L6.2 266.4c-3.8-1.9-6.2-5.7-6.2-9.9s2.4-8.1 6.2-9.9L117.5 191c5.5-2.7 12.2-.5 14.9 5ZM315.7 317c-2.7-5.5-.5-12.2 5-14.9l91.3-45.7-91.3-45.7c-5.5-2.7-7.7-9.4-5-14.9s9.4-7.7 14.9-5l111.3 55.6c3.8 1.9 6.2 5.7 6.2 9.9s-2.4 8.1-6.2 9.9l-111.3 55.6c-5.5 2.7-12.2.5-14.9-5ZM211.9 187.6c5.2.9 8.8 5.8 7.9 11.1l-3 17.7h28.6l3.5-20.9c.9-5.2 5.8-8.8 11.1-7.9 5.2.9 8.8 5.8 7.9 11.1l-2.9 17.7h17.5c5.3 0 9.6 4.3 9.6 9.6s-4.3 9.6-9.6 9.6h-20.7l-6.4 38.5h17.5c5.3 0 9.6 4.3 9.6 9.6s-4.3 9.6-9.6 9.6h-20.7l-3.5 20.9c-.9 5.2-5.8 8.8-11.1 7.9s-8.8-5.8-7.9-11.1l3-17.7h-28.6l-3.5 20.9c-.9 5.2-5.8 8.8-11.1 7.9-5.2-.9-8.8-5.8-7.9-11.1l2.9-17.7H167c-5.3 0-9.6-4.3-9.6-9.6s4.3-9.6 9.6-9.6h20.7l6.4-38.5h-17.5c-5.3 0-9.6-4.3-9.6-9.6s4.3-9.6 9.6-9.6h20.7l3.5-20.9c.9-5.2 5.8-8.8 11.1-7.9Zm1.7 48-6.4 38.5h28.6l6.4-38.5h-28.6Z"/>
|
||||
</symbol>
|
||||
|
||||
<symbol id="discobot" viewBox="0 0 512 512">
|
||||
<path d="M175.61 481.26H26.03c-6.75 0-13.36-2.67-18.42-7.59-5.06-5.06-7.87-11.81-7.59-18.84V290.76C.3 242.12 19.42 196 54.43 161.28c35.43-35.43 82.67-54.83 132.71-54.83h.56c49.77 0 96.3 19.54 131.17 55.11a187.257 187.257 0 0 1 48.5 88.57c5.06 21.79-6.19 44.14-26.71 53.14-14.62 6.47-69.17 27.7-170.67 37.96 4.5 11.25 16.17 23.34 46.68 29.66 29.8 6.19 64.53 4.22 87.59 2.95 9.84-.56 16.87-.98 21.79-.56 6.33.42 11.81 4.22 14.62 9.98 2.67 5.62 2.25 12.37-1.41 17.57-6.19 9-13.22 17.57-20.95 25.31-33.88 34.44-79.29 53.99-127.79 54.97h-14.76c0 .14-.14.14-.28.14h.14ZM35.17 446.11h154.65c34.87-.7 67.9-13.36 94.19-35.85-22.07.7-49.49.42-74.51-4.78-58.48-12.23-74.23-45.41-77.18-71.28-.7-6.47 1.12-12.79 5.2-17.85s10.12-8.15 16.59-8.72c82.81-7.31 143.26-23.62 172.22-36.27 5.06-2.25 7.87-7.73 6.61-13.07-6.33-27.13-19.82-51.88-39.36-71.84-28.12-28.68-65.8-44.57-106.14-44.57h-.56c-40.91 0-79.15 15.89-107.97 44.57-28.26 28.12-43.86 65.37-43.86 104.6v155.07c-.14 0 0 0 0 0h.14Zm-10.13 0ZM17.87 290.9Z"/><circle cx="171.11" cy="217.94" r="32.05"/><path d="M511.76 293.71c-.7-5.2-3.51-9.84-7.73-12.93s-9.42-4.5-14.62-3.66l-52.3 7.73c-5.2.7-9.84 3.51-12.93 7.73-3.09 4.22-4.5 9.42-3.66 14.62.98 6.61 5.2 12.23 11.39 15.04 2.53 1.12 5.34 1.69 8.01 1.69s1.97 0 2.95-.28l52.3-7.73c5.2-.7 9.84-3.51 12.93-7.73s4.5-9.42 3.66-14.62v.14ZM490.67 388.19l-71.98-36.41c-9.28-5.34-21.37-2.25-26.71 7.03-2.67 4.5-3.37 9.84-2.11 14.9 1.27 5.06 4.5 9.42 9.14 11.95l71.98 36.41c1.27.7 2.53 1.27 3.94 1.69 1.97.56 3.94.84 5.9.84 6.89 0 13.36-3.51 16.87-9.7 5.48-9.42 2.39-21.51-7.03-26.99v.28Z"/>
|
||||
</symbol>
|
||||
|
|
Before Width: | Height: | Size: 5.1 KiB After Width: | Height: | Size: 3.1 KiB |
Loading…
Reference in New Issue