172 lines
4.8 KiB
Go
172 lines
4.8 KiB
Go
package main
|
||
|
||
import (
|
||
"bytes"
|
||
"encoding/json"
|
||
"io"
|
||
"net/http"
|
||
"strings"
|
||
|
||
"github.com/gofiber/fiber/v2"
|
||
)
|
||
|
||
var DeepseekErrorCodes map[string]string
|
||
|
||
func init() {
|
||
DeepseekErrorCodes = make(map[string]string)
|
||
DeepseekErrorCodes["400"] = "Invalid Request - Please contact the support."
|
||
DeepseekErrorCodes["401"] = "Invalid Authentication - Ensure that the API key is still valid."
|
||
DeepseekErrorCodes["402"] = "You have run out of balance. Please check your account's balance."
|
||
DeepseekErrorCodes["422"] = "Request exceeds the maximum allowed number of bytes."
|
||
DeepseekErrorCodes["429"] = "Your account has hit a rate limit. You are sending requests too quickly."
|
||
DeepseekErrorCodes["500"] = "An unexpected error has occurred internal to Deepseek’s systems."
|
||
DeepseekErrorCodes["503"] = "Deepseek’s server is temporarily overloaded."
|
||
}
|
||
|
||
func TestDeepseekKey(apiKey string) bool {
|
||
url := "https://api.deepseek.com/chat/completions"
|
||
|
||
// Convert messages to OpenAI format
|
||
deepseekMessages := []RequestMessage{
|
||
{
|
||
Role: "user",
|
||
Content: "Hello",
|
||
},
|
||
}
|
||
|
||
requestBody := OpenaiChatCompletionRequest{
|
||
Model: "deepseek-chat",
|
||
Messages: deepseekMessages,
|
||
Temperature: 0,
|
||
MaxTokens: 10,
|
||
}
|
||
|
||
jsonBody, err := json.Marshal(requestBody)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
|
||
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonBody))
|
||
if err != nil {
|
||
return false
|
||
}
|
||
|
||
req.Header.Set("Content-Type", "application/json")
|
||
req.Header.Set("Authorization", "Bearer "+apiKey)
|
||
|
||
client := &http.Client{}
|
||
resp, err := client.Do(req)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
defer resp.Body.Close()
|
||
|
||
body, err := io.ReadAll(resp.Body)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
|
||
var chatCompletionResponse OpenaiChatCompletionResponse
|
||
err = json.Unmarshal(body, &chatCompletionResponse)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
|
||
if chatCompletionResponse.Usage.CompletionTokens == 0 {
|
||
return false
|
||
}
|
||
|
||
return true
|
||
}
|
||
|
||
func RequestDeepseek(c *fiber.Ctx, llm LLM, messages []Message) string {
|
||
model := llm.Model.ModelID
|
||
temperature := float64(llm.Temperature)
|
||
context := llm.Context
|
||
maxTokens := int(llm.MaxToken)
|
||
|
||
url := "https://api.deepseek.com/chat/completions"
|
||
|
||
var apiKey string
|
||
err := edgeGlobalClient.WithGlobals(map[string]interface{}{"ext::auth::client_token": c.Cookies("jade-edgedb-auth-token")}).QuerySingle(edgeCtx, `
|
||
with
|
||
filtered_keys := (
|
||
select Key {
|
||
key
|
||
} filter .company.name = <str>$0 AND .<keys[is Setting].<setting[is User] = global currentUser
|
||
)
|
||
select filtered_keys.key limit 1
|
||
`, &apiKey, "deepseek")
|
||
if err != nil {
|
||
return "JADE internal error: 08-00-0000. Please contact the support."
|
||
}
|
||
|
||
requestBody := OpenaiChatCompletionRequest{
|
||
Model: model,
|
||
Messages: Message2RequestMessage(messages, context),
|
||
MaxTokens: maxTokens,
|
||
Temperature: temperature,
|
||
}
|
||
|
||
jsonBody, err := json.Marshal(requestBody)
|
||
if err != nil {
|
||
return "JADE internal error: 08-01-0001. Please contact the support."
|
||
}
|
||
|
||
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonBody))
|
||
if err != nil {
|
||
return "JADE internal error: 08-02-0002. Please contact the support."
|
||
}
|
||
|
||
req.Header.Set("Content-Type", "application/json")
|
||
req.Header.Set("Authorization", "Bearer "+apiKey)
|
||
|
||
client := &http.Client{}
|
||
resp, err := client.Do(req)
|
||
if err != nil {
|
||
return "JADE internal error: 08-02-0003. Please contact the support."
|
||
}
|
||
defer resp.Body.Close()
|
||
|
||
body, err := io.ReadAll(resp.Body)
|
||
if err != nil {
|
||
return "JADE internal error: 08-01-0004. Please contact the support."
|
||
}
|
||
|
||
for key, value := range DeepseekErrorCodes {
|
||
if strings.Contains(resp.Status, key) {
|
||
return value
|
||
}
|
||
}
|
||
|
||
var chatCompletionResponse OpenaiChatCompletionResponse
|
||
err = json.Unmarshal(body, &chatCompletionResponse)
|
||
if err != nil {
|
||
return "JADE internal error: 08-01-0005. Please contact the support."
|
||
}
|
||
|
||
var usedModelInfo ModelInfo
|
||
err = edgeGlobalClient.WithGlobals(map[string]interface{}{"ext::auth::client_token": c.Cookies("jade-edgedb-auth-token")}).QuerySingle(edgeCtx, `
|
||
SELECT ModelInfo {
|
||
inputPrice,
|
||
outputPrice
|
||
}
|
||
FILTER .modelID = <str>$0
|
||
LIMIT 1
|
||
`, &usedModelInfo, model)
|
||
if err != nil {
|
||
return "JADE internal error: 08-00-0006. Please contact the support."
|
||
}
|
||
|
||
var inputCost float32 = float32(chatCompletionResponse.Usage.PromptTokens) * usedModelInfo.InputPrice
|
||
var outputCost float32 = float32(chatCompletionResponse.Usage.CompletionTokens) * usedModelInfo.OutputPrice
|
||
addUsage(c, inputCost, outputCost, chatCompletionResponse.Usage.PromptTokens, chatCompletionResponse.Usage.CompletionTokens, model)
|
||
|
||
if len(chatCompletionResponse.Choices) == 0 {
|
||
logMissingErrorCode.Println("Deepseek -", resp.Status, "-", string(body))
|
||
return "JADE internal error: 08-03-0007. Please contact the support."
|
||
}
|
||
|
||
return chatCompletionResponse.Choices[0].Message.Content
|
||
}
|