initial commit
This commit is contained in:
28
backend/internal/api/convert.go
Normal file
28
backend/internal/api/convert.go
Normal file
@@ -0,0 +1,28 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"reichard.io/aethera/internal/store"
|
||||
"reichard.io/aethera/pkg/slices"
|
||||
)
|
||||
|
||||
func toChat(c *store.Chat) *Chat {
|
||||
chat := &Chat{
|
||||
ID: c.ID,
|
||||
CreatedAt: c.CreatedAt,
|
||||
Title: c.Title,
|
||||
MessageCount: len(c.Messages),
|
||||
Messages: c.Messages,
|
||||
}
|
||||
|
||||
if firstMessage, found := slices.First(c.Messages); found {
|
||||
chat.InitialMessage = firstMessage.Content
|
||||
}
|
||||
|
||||
return chat
|
||||
}
|
||||
|
||||
func toChatNoMessages(c *store.Chat) *Chat {
|
||||
chat := toChat(c)
|
||||
chat.Messages = []*store.Message{}
|
||||
return chat
|
||||
}
|
||||
27
backend/internal/api/flush_writer.go
Normal file
27
backend/internal/api/flush_writer.go
Normal file
@@ -0,0 +1,27 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
)
|
||||
|
||||
type flushWriter struct {
|
||||
w http.ResponseWriter
|
||||
f http.Flusher
|
||||
}
|
||||
|
||||
func (fw *flushWriter) Write(p []byte) (n int, err error) {
|
||||
// Write Data
|
||||
n, err = fw.w.Write(p)
|
||||
if err == nil && fw.f != nil {
|
||||
fw.f.Flush()
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func newFlushWriter(w http.ResponseWriter) *flushWriter {
|
||||
flusher, _ := w.(http.Flusher)
|
||||
return &flushWriter{
|
||||
w: w,
|
||||
f: flusher,
|
||||
}
|
||||
}
|
||||
549
backend/internal/api/handlers.go
Normal file
549
backend/internal/api/handlers.go
Normal file
@@ -0,0 +1,549 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/base64"
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/openai/openai-go/v3"
|
||||
"github.com/sirupsen/logrus"
|
||||
"reichard.io/aethera/internal/client"
|
||||
"reichard.io/aethera/internal/store"
|
||||
"reichard.io/aethera/pkg/slices"
|
||||
)
|
||||
|
||||
type API struct {
|
||||
logger *logrus.Entry
|
||||
store store.Store
|
||||
client *client.Client
|
||||
dataDir string
|
||||
}
|
||||
|
||||
func New(s store.Store, dataDir string, logger *logrus.Logger) *API {
|
||||
return &API{
|
||||
store: s,
|
||||
dataDir: dataDir,
|
||||
logger: logger.WithField("service", "api"),
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) GetSettings(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "GetSettingsHandler")
|
||||
|
||||
settings, err := a.store.GetSettings()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to retrieve settings")
|
||||
http.Error(w, "Failed to retrieve application settings", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
if err := json.NewEncoder(w).Encode(settings); err != nil {
|
||||
log.WithError(err).Error("failed to encode application settings response")
|
||||
http.Error(w, "Failed to encode application settings response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) PostSettings(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "PostSettingsHandler")
|
||||
|
||||
var newSettings store.Settings
|
||||
if err := json.NewDecoder(r.Body).Decode(&newSettings); err != nil {
|
||||
log.WithError(err).Error("invalid JSON in settings update request")
|
||||
http.Error(w, "Invalid request body format for settings", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
if apiEndpoint := newSettings.APIEndpoint; apiEndpoint != "" {
|
||||
baseURL, err := url.Parse(apiEndpoint)
|
||||
if err != nil {
|
||||
errMsg := fmt.Sprintf("Invalid API Endpoint URL: %q", baseURL)
|
||||
log.WithError(err).Error(errMsg)
|
||||
http.Error(w, errMsg, http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
testClient := client.NewClient(baseURL)
|
||||
if _, err := testClient.GetModels(r.Context()); err != nil {
|
||||
log.WithError(err).Error("failed to access configured API endpoint")
|
||||
http.Error(w, "API endpoint inaccessible", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
a.client = nil
|
||||
}
|
||||
|
||||
if err := a.store.SaveSettings(&newSettings); err != nil {
|
||||
log.WithError(err).Error("failed to save settings")
|
||||
http.Error(w, "Failed to save application settings", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
w.WriteHeader(http.StatusOK)
|
||||
}
|
||||
|
||||
func (a *API) GetModels(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "GetModelsHandler")
|
||||
|
||||
client, err := a.getClient()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to initialize API client")
|
||||
http.Error(w, "Failed to initialize API client", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
models, err := client.GetModels(r.Context())
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to retrieve available models")
|
||||
http.Error(w, "Failed to retrieve available models from API", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
if err := json.NewEncoder(w).Encode(models); err != nil {
|
||||
log.WithError(err).Error("failed to encode available models response")
|
||||
http.Error(w, "Failed to encode available models response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) GetImages(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "GetImagesHandler")
|
||||
|
||||
files, err := os.ReadDir(path.Join(a.dataDir, "generated/images"))
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to read images directory")
|
||||
http.Error(w, "Failed to read images directory", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
imageList := make([]ImageRecord, 0)
|
||||
for _, file := range files {
|
||||
if !file.IsDir() && strings.HasSuffix(strings.ToLower(file.Name()), ".png") {
|
||||
info, err := file.Info()
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
imageList = append(imageList, ImageRecord{
|
||||
Name: file.Name(),
|
||||
Path: "/generated/images/" + file.Name(),
|
||||
Size: info.Size(),
|
||||
Date: info.ModTime().Format(time.RFC3339),
|
||||
})
|
||||
}
|
||||
}
|
||||
sort.Slice(imageList, func(i, j int) bool {
|
||||
return imageList[i].Date > imageList[j].Date
|
||||
})
|
||||
|
||||
if err := json.NewEncoder(w).Encode(imageList); err != nil {
|
||||
log.WithError(err).Error("failed to encode image list metadata response")
|
||||
http.Error(w, "Failed to encode image list metadata response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) PostImage(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "PostImageHandler")
|
||||
|
||||
client, err := a.getClient()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to initialize API client")
|
||||
http.Error(w, "Failed to initialize API client", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
var genReq GenerateImageRequest
|
||||
if err := json.NewDecoder(r.Body).Decode(&genReq); err != nil {
|
||||
log.WithError(err).Error("invalid JSON in image generation request")
|
||||
http.Error(w, "Invalid request body format for image generation", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
if err := genReq.Validate(); err != nil {
|
||||
log.WithError(err).Error("invalid request")
|
||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
// Edit vs Generate Request
|
||||
var images []openai.Image
|
||||
var reqErr error
|
||||
if genReq.isEdit() {
|
||||
editParams, err := genReq.getEditParams()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("invalid image edit parameters")
|
||||
http.Error(w, "Invalid image edit parameters", http.StatusBadRequest)
|
||||
return
|
||||
|
||||
}
|
||||
images, reqErr = client.EditImage(r.Context(), *editParams)
|
||||
} else {
|
||||
genParams, err := genReq.getGenerateParams()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("invalid image generation parameters")
|
||||
http.Error(w, "Invalid image generation parameters", http.StatusBadRequest)
|
||||
return
|
||||
|
||||
}
|
||||
images, reqErr = client.GenerateImages(r.Context(), *genParams)
|
||||
}
|
||||
|
||||
// Check Error
|
||||
if reqErr != nil {
|
||||
log.WithError(reqErr).Error("failed to generate images")
|
||||
http.Error(w, "Failed to generate images via API", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
// Normalize Responses
|
||||
imageRecords := make([]ImageRecord, 0)
|
||||
for i, img := range images {
|
||||
if img.B64JSON == "" {
|
||||
log.Warnf("empty image data at index %d, skipping", i)
|
||||
continue
|
||||
}
|
||||
|
||||
// Decode Image
|
||||
imgBytes, err := base64.StdEncoding.DecodeString(img.B64JSON)
|
||||
if err != nil {
|
||||
log.WithError(err).WithField("index", i).Error("failed to decode image")
|
||||
continue
|
||||
}
|
||||
|
||||
// Save Image
|
||||
filename := fmt.Sprintf("image_%d_%d.png", time.Now().Unix(), i)
|
||||
filePath := path.Join(a.dataDir, "generated/images", filename)
|
||||
if err := os.WriteFile(filePath, imgBytes, 0644); err != nil {
|
||||
log.WithError(err).WithField("file", filePath).Error("failed to save generated image")
|
||||
continue
|
||||
}
|
||||
|
||||
// Record Image
|
||||
imageRecords = append(imageRecords, ImageRecord{
|
||||
Name: filename,
|
||||
Path: fmt.Sprintf("/generated/images/%s", filename),
|
||||
Date: time.Now().Format(time.RFC3339),
|
||||
Size: int64(len(imgBytes)),
|
||||
})
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
if err := json.NewEncoder(w).Encode(imageRecords); err != nil {
|
||||
log.WithError(err).Error("failed to encode generated images response")
|
||||
http.Error(w, "Failed to encode generated images response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) DeleteImage(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "DeleteImageHandler")
|
||||
|
||||
filename := r.PathValue("filename")
|
||||
if filename == "" {
|
||||
log.Error("missing filename parameter")
|
||||
http.Error(w, "Filename parameter is required", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
// Delete Image
|
||||
imgDir := path.Join(a.dataDir, "generated/images")
|
||||
safePath := path.Join(imgDir, filepath.Base(filename))
|
||||
if err := os.Remove(safePath); err != nil {
|
||||
log.WithError(err).WithField("file", safePath).Error("failed to delete image file")
|
||||
http.Error(w, "Failed to delete image file", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
func (a *API) GetChats(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "GetChatsHandler")
|
||||
|
||||
chats, err := a.store.ListChats()
|
||||
if err != nil {
|
||||
log.WithError(err).Error("failed to list chats")
|
||||
http.Error(w, "Failed to retrieve chats", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
sort.Slice(chats, func(i, j int) bool {
|
||||
iLast, iFound := slices.Last(chats[i].Messages)
|
||||
if !iFound {
|
||||
return false
|
||||
}
|
||||
|
||||
jLast, jFound := slices.Last(chats[j].Messages)
|
||||
if !jFound {
|
||||
return true
|
||||
}
|
||||
|
||||
return iLast.CreatedAt.After(jLast.CreatedAt)
|
||||
})
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
if err := json.NewEncoder(w).Encode(ChatListResponse{Chats: slices.Map(chats, toChatNoMessages)}); err != nil {
|
||||
log.WithError(err).Error("failed to encode chats list response")
|
||||
http.Error(w, "Failed to encode chats list response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) PostChat(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "PostChatHandler")
|
||||
|
||||
// Decode Request
|
||||
var genReq GenerateTextRequest
|
||||
if err := json.NewDecoder(r.Body).Decode(&genReq); err != nil {
|
||||
log.WithError(err).Error("invalid JSON in text generation request")
|
||||
http.Error(w, "Invalid request body format for new chat", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
if err := genReq.Validate(); err != nil {
|
||||
log.WithError(err).Error("invalid request")
|
||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
// Create Chat
|
||||
var chat store.Chat
|
||||
if err := a.store.SaveChat(&chat); err != nil {
|
||||
log.WithError(err).Error("failed to create new chat")
|
||||
http.Error(w, "Failed to create new chat", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
// Send Message
|
||||
if err := a.sendMessage(r.Context(), w, chat.ID, genReq.Model, genReq.Prompt); err != nil {
|
||||
log.WithError(err).WithField("chat_id", chat.ID).Error("failed to send message")
|
||||
http.Error(w, "Failed to send message", http.StatusInternalServerError)
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) DeleteChat(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "DeleteChatHandler")
|
||||
|
||||
chatIDStr := r.PathValue("chatId")
|
||||
if chatIDStr == "" {
|
||||
log.Error("missing chat ID parameter")
|
||||
http.Error(w, "Chat ID is required", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
chatID, err := uuid.Parse(chatIDStr)
|
||||
if err != nil {
|
||||
log.WithError(err).Error("invalid chat ID format")
|
||||
http.Error(w, "Invalid chat ID format", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
// Delete Chat
|
||||
if err := a.store.DeleteChat(chatID); err != nil {
|
||||
log.WithError(err).WithField("chat_id", chatID).Error("failed to delete chat")
|
||||
if errors.Is(err, store.ErrChatNotFound) {
|
||||
http.Error(w, "Chat not found", http.StatusNotFound)
|
||||
} else {
|
||||
http.Error(w, "Failed to delete chat", http.StatusInternalServerError)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
w.WriteHeader(http.StatusNoContent)
|
||||
}
|
||||
|
||||
func (a *API) GetChat(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "GetChatHandler")
|
||||
|
||||
chatID := r.PathValue("chatId")
|
||||
if chatID == "" {
|
||||
log.Error("missing chat ID parameter")
|
||||
http.Error(w, "Chat ID is required", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
parsedChatID, err := uuid.Parse(chatID)
|
||||
if err != nil {
|
||||
log.WithError(err).Error("invalid chat ID format")
|
||||
http.Error(w, "Invalid chat ID format", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
chat, err := a.store.GetChat(parsedChatID)
|
||||
if err != nil {
|
||||
log.WithError(err).WithField("chat_id", parsedChatID).Error("failed to get chat")
|
||||
http.Error(w, "Failed to get chat", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
if err := json.NewEncoder(w).Encode(toChat(chat)); err != nil {
|
||||
log.WithError(err).Error("failed to encode chat messages response")
|
||||
http.Error(w, "Failed to encode chat messages response", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) PostChatMessage(w http.ResponseWriter, r *http.Request) {
|
||||
log := a.logger.WithField("handler", "PostChatMessageHandler")
|
||||
|
||||
rawChatID := r.PathValue("chatId")
|
||||
if rawChatID == "" {
|
||||
log.Error("missing chat ID parameter")
|
||||
http.Error(w, "Chat ID is required", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
chatID, err := uuid.Parse(rawChatID)
|
||||
if err != nil {
|
||||
log.WithError(err).Error("invalid chat ID format")
|
||||
http.Error(w, "Invalid chat ID format", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
var genReq GenerateTextRequest
|
||||
if err := json.NewDecoder(r.Body).Decode(&genReq); err != nil {
|
||||
log.WithError(err).Error("invalid JSON in text generation request")
|
||||
http.Error(w, "Invalid request body format for text generation", http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
if err := genReq.Validate(); err != nil {
|
||||
log.WithError(err).Error("invalid request")
|
||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||
return
|
||||
}
|
||||
|
||||
if err := a.sendMessage(r.Context(), w, chatID, genReq.Model, genReq.Prompt); err != nil {
|
||||
log.WithError(err).WithField("chat_id", chatID).Error("failed to send message")
|
||||
http.Error(w, "Failed to send message", http.StatusInternalServerError)
|
||||
}
|
||||
}
|
||||
|
||||
func (a *API) getClient() (*client.Client, error) {
|
||||
if a.client != nil {
|
||||
return a.client, nil
|
||||
}
|
||||
|
||||
// Get Settings & Validate Endpoint
|
||||
settings, err := a.store.GetSettings()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to retrieve application settings: %w", err)
|
||||
} else if settings.APIEndpoint == "" {
|
||||
return nil, errors.New("no API endpoint configured in settings")
|
||||
}
|
||||
|
||||
baseURL, err := url.Parse(settings.APIEndpoint)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("invalid API endpoint URL: %w", err)
|
||||
}
|
||||
|
||||
a.client = client.NewClient(baseURL)
|
||||
return a.client, nil
|
||||
}
|
||||
|
||||
func (a *API) sendMessage(ctx context.Context, w http.ResponseWriter, chatID uuid.UUID, chatModel, userMessage string) error {
|
||||
apiClient, err := a.getClient()
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to get client: %w", err)
|
||||
}
|
||||
|
||||
// Detach Request Context
|
||||
ctx, cancel := context.WithTimeout(context.WithoutCancel(ctx), time.Minute*5)
|
||||
defer cancel()
|
||||
|
||||
// Create User Message
|
||||
userMsg := &store.Message{ChatID: chatID, Role: "user", Content: userMessage}
|
||||
if err := a.store.SaveChatMessage(userMsg); err != nil {
|
||||
return fmt.Errorf("failed to add user message to chat: %w", err)
|
||||
}
|
||||
|
||||
// Add Assistant Response - TODO: Ensure InProgress Flag?
|
||||
assistantMsg := &store.Message{ChatID: chatID, Role: "assistant"}
|
||||
if err := a.store.SaveChatMessage(assistantMsg); err != nil {
|
||||
return fmt.Errorf("failed to add assistant message to chat: %w", err)
|
||||
}
|
||||
|
||||
// Get Chat
|
||||
chat, err := a.store.GetChat(chatID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to get chat: %w", err)
|
||||
}
|
||||
|
||||
// Set Headers
|
||||
w.Header().Set("Content-Type", "application/x-ndjson")
|
||||
w.Header().Set("Cache-Control", "no-cache")
|
||||
w.Header().Set("Connection", "keep-alive")
|
||||
w.Header().Set("Transfer-Encoding", "chunked")
|
||||
|
||||
// Create Flush Writer
|
||||
flushWriter := newFlushWriter(w)
|
||||
|
||||
// Send Initial Chunk - User Message & Chat
|
||||
if err := json.NewEncoder(flushWriter).Encode(&MessageChunk{
|
||||
Chat: toChatNoMessages(chat),
|
||||
UserMessage: userMsg,
|
||||
}); err != nil {
|
||||
return fmt.Errorf("failed to send initial chunk: %w", err)
|
||||
}
|
||||
|
||||
// Send Message
|
||||
if _, err := apiClient.SendMessage(ctx, chat.Messages, chatModel, func(m *client.MessageChunk) error {
|
||||
var apiMsgChunk MessageChunk
|
||||
|
||||
if m.Stats != nil {
|
||||
assistantMsg.Stats = m.Stats
|
||||
}
|
||||
|
||||
if m.Message != nil {
|
||||
assistantMsg.Content += *m.Message
|
||||
apiMsgChunk.AssistantMessage = assistantMsg
|
||||
}
|
||||
|
||||
if m.Thinking != nil {
|
||||
assistantMsg.Thinking += *m.Thinking
|
||||
apiMsgChunk.AssistantMessage = assistantMsg
|
||||
}
|
||||
|
||||
// Send Progress Chunk
|
||||
if err := json.NewEncoder(flushWriter).Encode(apiMsgChunk); err != nil {
|
||||
return fmt.Errorf("failed to send progress chunk: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}); err != nil {
|
||||
return fmt.Errorf("failed to generate text stream: %w", err)
|
||||
}
|
||||
|
||||
// Summarize & Update Chat Title
|
||||
if chat.Title == "" {
|
||||
chat.Title, err = apiClient.CreateTitle(ctx, chat.Messages[0].Content, chatModel)
|
||||
if err != nil {
|
||||
a.logger.WithError(err).WithField("chat_id", chat.ID).Error("failed to create chat title")
|
||||
} else if err := a.store.SaveChat(chat); err != nil {
|
||||
a.logger.WithError(err).WithField("chat_id", chat.ID).Error("failed to update chat")
|
||||
}
|
||||
}
|
||||
|
||||
// Update Assistant Message
|
||||
if err := a.store.SaveChatMessage(assistantMsg); err != nil {
|
||||
return fmt.Errorf("failed to save assistant message to chat: %w", err)
|
||||
}
|
||||
|
||||
// Send Final Chunk
|
||||
if err := json.NewEncoder(flushWriter).Encode(&MessageChunk{
|
||||
Chat: toChatNoMessages(chat),
|
||||
AssistantMessage: assistantMsg,
|
||||
}); err != nil {
|
||||
return fmt.Errorf("failed to send final chunk: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
163
backend/internal/api/types.go
Normal file
163
backend/internal/api/types.go
Normal file
@@ -0,0 +1,163 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/base64"
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/openai/openai-go/v3"
|
||||
"github.com/openai/openai-go/v3/packages/param"
|
||||
"reichard.io/aethera/internal/store"
|
||||
)
|
||||
|
||||
type ChatListResponse struct {
|
||||
Chats []*Chat `json:"chats"`
|
||||
}
|
||||
|
||||
type Chat struct {
|
||||
ID uuid.UUID `json:"id"`
|
||||
CreatedAt time.Time `json:"created_at"`
|
||||
Title string `json:"title"`
|
||||
InitialMessage string `json:"initial_message"`
|
||||
MessageCount int `json:"message_count"`
|
||||
Messages []*store.Message `json:"messages"`
|
||||
}
|
||||
|
||||
type MessageChunk struct {
|
||||
Chat *Chat `json:"chat,omitempty"`
|
||||
UserMessage *store.Message `json:"user_message,omitempty"`
|
||||
AssistantMessage *store.Message `json:"assistant_message,omitempty"`
|
||||
}
|
||||
|
||||
type GenerateImageRequest struct {
|
||||
Model string `json:"model"`
|
||||
Prompt string `json:"prompt"`
|
||||
N int64 `json:"n"`
|
||||
Size string `json:"size"`
|
||||
|
||||
Mask *string `json:"mask"` // Data URL (e.g. "data:image/png;base64,...")
|
||||
Image *string `json:"image"` // Data URL (e.g. "data:image/png;base64,...")
|
||||
|
||||
GenerateImageRequestExtraArgs
|
||||
}
|
||||
|
||||
func (r *GenerateImageRequest) Validate() error {
|
||||
if r.Model == "" {
|
||||
return errors.New("model is required")
|
||||
}
|
||||
if r.Prompt == "" {
|
||||
return errors.New("prompt is required")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type GenerateImageRequestExtraArgs struct {
|
||||
Seed *int32 `json:"seed,omitempty"`
|
||||
}
|
||||
|
||||
type ImageRecord struct {
|
||||
Name string `json:"name"`
|
||||
Path string `json:"path"`
|
||||
Size int64 `json:"size"`
|
||||
Date string `json:"date"`
|
||||
}
|
||||
|
||||
type GenerateTextRequest struct {
|
||||
Model string `json:"model"`
|
||||
Prompt string `json:"prompt"`
|
||||
}
|
||||
|
||||
func (r *GenerateTextRequest) Validate() error {
|
||||
if r.Model == "" {
|
||||
return errors.New("model is required")
|
||||
}
|
||||
if r.Prompt == "" {
|
||||
return errors.New("prompt is required")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (r *GenerateImageRequest) getPrompt() string {
|
||||
prompt := r.Prompt
|
||||
d, _ := json.Marshal(r.GenerateImageRequestExtraArgs)
|
||||
if extraArgs := string(d); extraArgs != "" {
|
||||
prompt += fmt.Sprintf(" <sd_cpp_extra_args>%s</sd_cpp_extra_args>", extraArgs)
|
||||
}
|
||||
return strings.TrimSpace(prompt)
|
||||
}
|
||||
|
||||
func (r *GenerateImageRequest) isEdit() bool {
|
||||
return r.Image != nil
|
||||
}
|
||||
|
||||
func (r *GenerateImageRequest) getEditParams() (*openai.ImageEditParams, error) {
|
||||
if !r.isEdit() {
|
||||
return nil, errors.New("not an edit request")
|
||||
}
|
||||
strippedImage, err := stripDataURLPrefix(*r.Image)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to strip data url prefix for image: %w", err)
|
||||
}
|
||||
imageBytes, err := base64.StdEncoding.DecodeString(strippedImage)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to decode image: %w", err)
|
||||
}
|
||||
iFile := openai.File(bytes.NewReader(imageBytes), "main.png", "image/png")
|
||||
|
||||
editReq := &openai.ImageEditParams{
|
||||
Model: r.Model,
|
||||
Prompt: r.getPrompt(),
|
||||
Size: openai.ImageEditParamsSize(r.Size),
|
||||
N: param.NewOpt(r.N),
|
||||
OutputFormat: openai.ImageEditParamsOutputFormatPNG,
|
||||
Image: openai.ImageEditParamsImageUnion{OfFileArray: []io.Reader{iFile}},
|
||||
}
|
||||
if r.Mask == nil {
|
||||
return editReq, nil
|
||||
}
|
||||
|
||||
strippedMask, err := stripDataURLPrefix(*r.Mask)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to strip data url prefix for mask: %w", err)
|
||||
}
|
||||
maskBytes, err := base64.StdEncoding.DecodeString(strippedMask)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to decode image: %w", err)
|
||||
}
|
||||
|
||||
editReq.Mask = openai.File(bytes.NewReader(maskBytes), "mask.png", "image/png")
|
||||
return editReq, nil
|
||||
}
|
||||
|
||||
func (r *GenerateImageRequest) getGenerateParams() (*openai.ImageGenerateParams, error) {
|
||||
if r.isEdit() {
|
||||
return nil, errors.New("not a generate request")
|
||||
}
|
||||
return &openai.ImageGenerateParams{
|
||||
Model: r.Model,
|
||||
Prompt: r.getPrompt(),
|
||||
Size: openai.ImageGenerateParamsSize(r.Size),
|
||||
N: param.NewOpt(r.N),
|
||||
OutputFormat: openai.ImageGenerateParamsOutputFormatPNG,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func stripDataURLPrefix(dataURL string) (string, error) {
|
||||
if !strings.Contains(dataURL, ",") {
|
||||
return dataURL, nil
|
||||
}
|
||||
parts := strings.SplitN(dataURL, ",", 2)
|
||||
prefix := parts[0]
|
||||
switch prefix {
|
||||
case "data:image/png;base64", "data:image/jpeg;base64":
|
||||
return parts[1], nil
|
||||
default:
|
||||
return "", fmt.Errorf("unsupported image type: %s", prefix)
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user