Working swaping llm icons

This commit is contained in:
Adrien Bouvais 2024-05-24 17:13:57 +02:00
parent 435149ad5e
commit 09193a227e
17 changed files with 138 additions and 67 deletions

1
.dockerignore Normal file
View File

@ -0,0 +1 @@
data/

View File

@ -3,7 +3,6 @@ package main
import ( import (
"context" "context"
"encoding/json" "encoding/json"
"fmt"
"net/url" "net/url"
"sort" "sort"
"strings" "strings"
@ -560,6 +559,7 @@ func GenerateModelPopoverHTML(refresh bool) string {
} }
} }
FILTER .user = global currentUser AND .name != 'none' AND .to_delete = false FILTER .user = global currentUser AND .name != 'none' AND .to_delete = false
ORDER BY .position
`, &llms) `, &llms)
if err != nil { if err != nil {
panic(err) panic(err)
@ -614,7 +614,6 @@ func LoadSettingsHandler(c *fiber.Ctx) error {
user.Email = url.QueryEscape(user.Email) user.Email = url.QueryEscape(user.Email)
stripeSubLink := "https://billing.stripe.com/p/login/test_eVa5kC1q7dogaaIcMM?prefilled_email=" + user.Email stripeSubLink := "https://billing.stripe.com/p/login/test_eVa5kC1q7dogaaIcMM?prefilled_email=" + user.Email
fmt.Println(stripeSubLink)
openaiExists, anthropicExists, mistralExists, groqExists, gooseaiExists, googleExists := getExistingKeys() openaiExists, anthropicExists, mistralExists, groqExists, gooseaiExists, googleExists := getExistingKeys()

38
LLM.go
View File

@ -62,10 +62,13 @@ func createLLM(c *fiber.Ctx) error {
if modelID == "custom" { if modelID == "custom" {
err := edgeClient.Execute(edgeCtx, ` err := edgeClient.Execute(edgeCtx, `
WITH
countLLM := count((SELECT LLM FILTER .user = global currentUser))
INSERT LLM { INSERT LLM {
name := <str>$0, name := <str>$0,
context := <str>$1, context := <str>$1,
temperature := <float32>$2, temperature := <float32>$2,
position := countLLM + 1,
modelInfo := (INSERT ModelInfo { modelInfo := (INSERT ModelInfo {
name := <str>$0, name := <str>$0,
modelID := <str>$5, modelID := <str>$5,
@ -86,10 +89,13 @@ func createLLM(c *fiber.Ctx) error {
} }
} else { } else {
err := edgeClient.Execute(edgeCtx, ` err := edgeClient.Execute(edgeCtx, `
WITH
countLLM := count((SELECT LLM FILTER .user = global currentUser))
INSERT LLM { INSERT LLM {
name := <str>$0, name := <str>$0,
context := <str>$1, context := <str>$1,
temperature := <float32>$2, temperature := <float32>$2,
position := countLLM + 1,
modelInfo := (SELECT ModelInfo FILTER .modelID = <str>$3 LIMIT 1), modelInfo := (SELECT ModelInfo FILTER .modelID = <str>$3 LIMIT 1),
user := global currentUser user := global currentUser
} }
@ -101,3 +107,35 @@ func createLLM(c *fiber.Ctx) error {
return c.SendString(GenerateModelPopoverHTML(true)) return c.SendString(GenerateModelPopoverHTML(true))
} }
type PositionUpdate struct {
Position int `json:"position"`
ID string `json:"id"`
}
func updateLLMPositionBatch(c *fiber.Ctx) error {
var positionUpdates []PositionUpdate
if err := c.BodyParser(&positionUpdates); err != nil {
return err
}
for _, update := range positionUpdates {
idUUID, err := edgedb.ParseUUID(update.ID)
if err != nil {
panic(err)
}
err = edgeClient.Execute(edgeCtx, `
UPDATE LLM
FILTER .id = <uuid>$0 AND .user = global currentUser
SET {
position := <int32>$1
};
`, idUUID, int32(update.Position))
if err != nil {
panic(err)
}
}
return nil
}

View File

@ -86,6 +86,7 @@ module default {
required name: str; required name: str;
required context: str; required context: str;
required temperature: float32; required temperature: float32;
required position: int32;
required to_delete: bool { required to_delete: bool {
default := false; default := false;
}; };

View File

@ -0,0 +1,9 @@
CREATE MIGRATION m1cmvjy3ikuh5ii6b7l7gckttjmqk554llocwqx7n4aibtzngybvoq
ONTO m1wlzna3nrrigx2urda3turnj6dcvcjgexeuxxqundupqet5jrkh7q
{
ALTER TYPE default::LLM {
CREATE REQUIRED PROPERTY position: std::int32 {
SET REQUIRED USING (<std::int32>{0});
};
};
};

View File

@ -89,6 +89,7 @@ func main() {
app.Get("/loadUsageKPI", LoadUsageKPIHandler) app.Get("/loadUsageKPI", LoadUsageKPIHandler)
app.Get("/loadKeys", LoadKeysHandler) app.Get("/loadKeys", LoadKeysHandler)
app.Get("/loadSettings", LoadSettingsHandler) app.Get("/loadSettings", LoadSettingsHandler)
app.Post("/updateLLMPositionBatch", updateLLMPositionBatch)
// Authentication // Authentication
app.Get("/signin", handleUiSignIn) app.Get("/signin", handleUiSignIn)

Binary file not shown.

Before

Width:  |  Height:  |  Size: 168 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 119 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 169 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 114 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 84 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

View File

@ -0,0 +1,54 @@
let lastSelectedIndex = null;
function toggleSelection(element) {
const elements = Array.from(document.getElementsByClassName('icon-text'));
const index = elements.indexOf(element);
if (document.body.classList.contains('shift-pressed') && lastSelectedIndex !== null) {
const [start, end] = [lastSelectedIndex, index].sort((a, b) => a - b);
let allSelected = true;
for (let i = start; i <= end; i++) {
if (!elements[i].classList.contains('selected')) {
allSelected = false;
break;
}
}
for (let i = start; i <= end; i++) {
if (allSelected) {
elements[i].classList.remove('selected');
elements[i].classList.add('unselected');
} else {
elements[i].classList.add('selected');
elements[i].classList.remove('unselected');
}
}
lastSelectedIndex = null;
const elements2 = Array.from(document.getElementsByClassName('icon-text'));
for (let i = 0; i < elements2.length; i++) {
elements2[i].classList.remove('shiftselected');
}
} else if (document.body.classList.contains('shift-pressed') && lastSelectedIndex === null) {
lastSelectedIndex = index;
element.classList.toggle('shiftselected');
} else {
element.classList.toggle('selected');
element.classList.toggle('unselected');
}
toggleSendButton();
}
function getSelectedModelsIDs() {
var selectedModelsIDs = [];
var selectedModels = document.getElementsByClassName('selected');
for (var i = 0; i < selectedModels.length; i++) {
selectedModelsIDs.push(selectedModels[i].getAttribute('data-id'));
}
return selectedModelsIDs.length > 0 ? JSON.stringify(selectedModelsIDs) : '[]';
}
function toggleSendButton() {
var selectedModels = document.getElementsByClassName('selected');
var sendButton = document.querySelector('button[disabled]');
sendButton.disabled = selectedModels.length === 0;
}

View File

@ -14,6 +14,8 @@
<script src="https://unpkg.com/htmx.org@2.0.0-beta4/dist/htmx.js"></script> <script src="https://unpkg.com/htmx.org@2.0.0-beta4/dist/htmx.js"></script>
<script src="https://unpkg.com/htmx-ext-sse@2.0.0/sse.js"></script> <script src="https://unpkg.com/htmx-ext-sse@2.0.0/sse.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/Sortable/1.14.0/Sortable.min.js"></script>
<script src="/model-selection-menu.js"></script>
<script async src="https://js.stripe.com/v3/pricing-table.js"></script> <script async src="https://js.stripe.com/v3/pricing-table.js"></script>
</head> </head>

View File

@ -8,15 +8,17 @@
<div class="dropdown-menu" id="dropdown-menu3" role="menu"> <div class="dropdown-menu" id="dropdown-menu3" role="menu">
<div class="dropdown-content"> <div class="dropdown-content">
<div class="dropdown-item" id="models-list"> <div class="dropdown-item" id="models-list">
{% for LLM in LLMs %} <div id="llm-list">
<div class="icon-text has-text unselected" onclick="toggleSelection(this)" style="cursor: pointer;" {% for LLM in LLMs %}
value="{{ LLM.ID.String() }}"> <div class="icon-text has-text unselected" data-id="{{ LLM.ID.String() }}" style="cursor: pointer;"
<span class="icon"> onclick="toggleSelection(this)">
<img src="{{ LLM.Model.Company.Icon }}" /> <span class="icon">
</span> <img src="{{ LLM.Model.Company.Icon }}" />
<span>{{ LLM.Name }}</span> </span>
<span>{{ LLM.Name }}</span>
</div>
{% endfor %}
</div> </div>
{% endfor %}
<div class="is-flex is-justify-content-space-between mt-4"> <div class="is-flex is-justify-content-space-between mt-4">
<button class="button is-small is-danger is-outlined" hx-get="/deleteLLM" hx-swap="outerHTML" <button class="button is-small is-danger is-outlined" hx-get="/deleteLLM" hx-swap="outerHTML"
hx-target="#models-dropdown" hx-confirm="Are you sure?" hx-trigger="click" hx-target="#models-dropdown" hx-confirm="Are you sure?" hx-trigger="click"
@ -85,6 +87,26 @@
</div> </div>
<script> <script>
var sortable = new Sortable(document.getElementById('llm-list'), {
animation: 150,
onEnd: function (evt) {
var items = evt.to.children;
var updates = [];
for (var i = 0; i < items.length; i++) {
var id = items[i].getAttribute('data-id');
var position = i + 1;
updates.push({ id: id, position: position });
}
var xhr = new XMLHttpRequest();
xhr.open('POST', '/updateLLMPositionBatch', true);
xhr.setRequestHeader('Content-Type', 'application/json;charset=UTF-8');
xhr.send(JSON.stringify(updates));
}
});
document.getElementById('model-name-input').addEventListener('input', function () { document.getElementById('model-name-input').addEventListener('input', function () {
document.getElementById('confirm-create-model-button').disabled = this.value === ''; document.getElementById('confirm-create-model-button').disabled = this.value === '';
}) })
@ -102,7 +124,6 @@
document.getElementById('model-cid-input').classList.toggle('is-hidden', !customEndpoint); document.getElementById('model-cid-input').classList.toggle('is-hidden', !customEndpoint);
}); });
document.getElementById('temperature-slider').addEventListener('input', function () { document.getElementById('temperature-slider').addEventListener('input', function () {
document.getElementById('temperature-slider-output').innerHTML = this.value; document.getElementById('temperature-slider-output').innerHTML = this.value;
}) })
@ -124,12 +145,10 @@
}); });
document.addEventListener('keyup', function (event) { document.addEventListener('keyup', function (event) {
// If Shift is press and id="chat-input-textarea" not focused
if (event.key === 'Shift' && document.activeElement.id !== 'chat-input-textarea' && document.getElementById('models-creation').classList.contains('is-hidden') && document.getElementById('models-dropdown').classList.contains('is-active')) { if (event.key === 'Shift' && document.activeElement.id !== 'chat-input-textarea' && document.getElementById('models-creation').classList.contains('is-hidden') && document.getElementById('models-dropdown').classList.contains('is-active')) {
document.body.classList.remove('shift-pressed'); document.body.classList.remove('shift-pressed');
lastSelectedIndex = null; lastSelectedIndex = null;
// Remove all "shiftselected" classes
const elements = Array.from(document.getElementsByClassName('icon-text')); const elements = Array.from(document.getElementsByClassName('icon-text'));
for (let i = 0; i < elements.length; i++) { for (let i = 0; i < elements.length; i++) {
elements[i].classList.remove('shiftselected'); elements[i].classList.remove('shiftselected');
@ -146,58 +165,6 @@
}); });
</script> </script>
{% if not DeleteUpdate %}
<script>
let lastSelectedIndex = null;
function toggleSelection(element) {
const elements = Array.from(document.getElementsByClassName('icon-text'));
const index = elements.indexOf(element);
if (document.body.classList.contains('shift-pressed') && lastSelectedIndex !== null) {
const [start, end] = [lastSelectedIndex, index].sort((a, b) => a - b);
let allSelected = true;
for (let i = start; i <= end; i++) {
if (!elements[i].classList.contains('selected')) {
allSelected = false;
break;
}
}
for (let i = start; i <= end; i++) {
if (allSelected) {
elements[i].classList.remove('selected');
elements[i].classList.add('unselected');
} else {
elements[i].classList.add('selected');
elements[i].classList.remove('unselected');
}
}
lastSelectedIndex = null;
const elements2 = Array.from(document.getElementsByClassName('icon-text'));
for (let i = 0; i < elements2.length; i++) {
elements2[i].classList.remove('shiftselected');
}
} else if (document.body.classList.contains('shift-pressed') && lastSelectedIndex === null) {
lastSelectedIndex = index;
element.classList.toggle('shiftselected');
} else {
element.classList.toggle('selected');
element.classList.toggle('unselected');
}
toggleSendButton();
}
function getSelectedModelsIDs() {
var selectedModelsIDs = [];
var selectedModels = document.getElementsByClassName('selected');
for (var i = 0; i < selectedModels.length; i++) {
selectedModelsIDs.push(selectedModels[i].getAttribute('value'));
}
return selectedModelsIDs.length > 0 ? JSON.stringify(selectedModelsIDs) : '[]';
}
</script>
<style> <style>
.selected { .selected {
border: 2px solid #126d0f; border: 2px solid #126d0f;
@ -248,5 +215,4 @@
background: #ffffff; background: #ffffff;
cursor: pointer; cursor: pointer;
} }
</style> </style>
{% endif %}