mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
17cd077f34
* feat: add thinking support * feat: improve thinking block render in web and filter thinking content in channels * fix: migrate
277 lines
7.9 KiB
Go
277 lines
7.9 KiB
Go
package settings
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/jackc/pgx/v5"
|
|
"github.com/jackc/pgx/v5/pgtype"
|
|
|
|
"github.com/memohai/memoh/internal/db"
|
|
"github.com/memohai/memoh/internal/db/sqlc"
|
|
)
|
|
|
|
type Service struct {
|
|
queries *sqlc.Queries
|
|
logger *slog.Logger
|
|
}
|
|
|
|
var ErrPersonalBotGuestAccessUnsupported = errors.New("personal bots do not support guest access")
|
|
var ErrModelIDAmbiguous = errors.New("model_id is ambiguous across providers")
|
|
var ErrInvalidModelRef = errors.New("invalid model reference")
|
|
|
|
func NewService(log *slog.Logger, queries *sqlc.Queries) *Service {
|
|
return &Service{
|
|
queries: queries,
|
|
logger: log.With(slog.String("service", "settings")),
|
|
}
|
|
}
|
|
|
|
func (s *Service) GetBot(ctx context.Context, botID string) (Settings, error) {
|
|
pgID, err := db.ParseUUID(botID)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
row, err := s.queries.GetSettingsByBotID(ctx, pgID)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
return normalizeBotSettingsReadRow(row), nil
|
|
}
|
|
|
|
func (s *Service) UpsertBot(ctx context.Context, botID string, req UpsertRequest) (Settings, error) {
|
|
if s.queries == nil {
|
|
return Settings{}, fmt.Errorf("settings queries not configured")
|
|
}
|
|
pgID, err := db.ParseUUID(botID)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
botRow, err := s.queries.GetBotByID(ctx, pgID)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
isPersonalBot := strings.EqualFold(strings.TrimSpace(botRow.Type), "personal")
|
|
|
|
current := normalizeBotSetting(botRow.MaxContextLoadTime, botRow.MaxContextTokens, botRow.MaxInboxItems, botRow.Language, botRow.AllowGuest, botRow.ReasoningEnabled, botRow.ReasoningEffort)
|
|
if req.MaxContextLoadTime != nil && *req.MaxContextLoadTime > 0 {
|
|
current.MaxContextLoadTime = *req.MaxContextLoadTime
|
|
}
|
|
if req.MaxContextTokens != nil && *req.MaxContextTokens >= 0 {
|
|
current.MaxContextTokens = *req.MaxContextTokens
|
|
}
|
|
if req.MaxInboxItems != nil && *req.MaxInboxItems >= 0 {
|
|
current.MaxInboxItems = *req.MaxInboxItems
|
|
}
|
|
if strings.TrimSpace(req.Language) != "" {
|
|
current.Language = strings.TrimSpace(req.Language)
|
|
}
|
|
if isPersonalBot {
|
|
if req.AllowGuest != nil && *req.AllowGuest {
|
|
return Settings{}, ErrPersonalBotGuestAccessUnsupported
|
|
}
|
|
current.AllowGuest = false
|
|
} else if req.AllowGuest != nil {
|
|
current.AllowGuest = *req.AllowGuest
|
|
}
|
|
if req.ReasoningEnabled != nil {
|
|
current.ReasoningEnabled = *req.ReasoningEnabled
|
|
}
|
|
if req.ReasoningEffort != nil && isValidReasoningEffort(*req.ReasoningEffort) {
|
|
current.ReasoningEffort = *req.ReasoningEffort
|
|
}
|
|
|
|
chatModelUUID := pgtype.UUID{}
|
|
if value := strings.TrimSpace(req.ChatModelID); value != "" {
|
|
modelID, err := s.resolveModelUUID(ctx, value)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
chatModelUUID = modelID
|
|
}
|
|
memoryModelUUID := pgtype.UUID{}
|
|
if value := strings.TrimSpace(req.MemoryModelID); value != "" {
|
|
modelID, err := s.resolveModelUUID(ctx, value)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
memoryModelUUID = modelID
|
|
}
|
|
embeddingModelUUID := pgtype.UUID{}
|
|
if value := strings.TrimSpace(req.EmbeddingModelID); value != "" {
|
|
modelID, err := s.resolveModelUUID(ctx, value)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
embeddingModelUUID = modelID
|
|
}
|
|
searchProviderUUID := pgtype.UUID{}
|
|
if value := strings.TrimSpace(req.SearchProviderID); value != "" {
|
|
providerID, err := db.ParseUUID(value)
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
searchProviderUUID = providerID
|
|
}
|
|
|
|
updated, err := s.queries.UpsertBotSettings(ctx, sqlc.UpsertBotSettingsParams{
|
|
ID: pgID,
|
|
MaxContextLoadTime: int32(current.MaxContextLoadTime),
|
|
MaxContextTokens: int32(current.MaxContextTokens),
|
|
MaxInboxItems: int32(current.MaxInboxItems),
|
|
Language: current.Language,
|
|
AllowGuest: current.AllowGuest,
|
|
ReasoningEnabled: current.ReasoningEnabled,
|
|
ReasoningEffort: current.ReasoningEffort,
|
|
ChatModelID: chatModelUUID,
|
|
MemoryModelID: memoryModelUUID,
|
|
EmbeddingModelID: embeddingModelUUID,
|
|
SearchProviderID: searchProviderUUID,
|
|
})
|
|
if err != nil {
|
|
return Settings{}, err
|
|
}
|
|
return normalizeBotSettingsWriteRow(updated), nil
|
|
}
|
|
|
|
func (s *Service) Delete(ctx context.Context, botID string) error {
|
|
if s.queries == nil {
|
|
return fmt.Errorf("settings queries not configured")
|
|
}
|
|
pgID, err := db.ParseUUID(botID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return s.queries.DeleteSettingsByBotID(ctx, pgID)
|
|
}
|
|
|
|
func normalizeBotSetting(maxContextLoadTime int32, maxContextTokens int32, maxInboxItems int32, language string, allowGuest bool, reasoningEnabled bool, reasoningEffort string) Settings {
|
|
settings := Settings{
|
|
MaxContextLoadTime: int(maxContextLoadTime),
|
|
MaxContextTokens: int(maxContextTokens),
|
|
MaxInboxItems: int(maxInboxItems),
|
|
Language: strings.TrimSpace(language),
|
|
AllowGuest: allowGuest,
|
|
ReasoningEnabled: reasoningEnabled,
|
|
ReasoningEffort: strings.TrimSpace(reasoningEffort),
|
|
}
|
|
if settings.MaxContextLoadTime <= 0 {
|
|
settings.MaxContextLoadTime = DefaultMaxContextLoadTime
|
|
}
|
|
if settings.MaxContextTokens < 0 {
|
|
settings.MaxContextTokens = 0
|
|
}
|
|
if settings.MaxInboxItems <= 0 {
|
|
settings.MaxInboxItems = DefaultMaxInboxItems
|
|
}
|
|
if settings.Language == "" {
|
|
settings.Language = DefaultLanguage
|
|
}
|
|
if !isValidReasoningEffort(settings.ReasoningEffort) {
|
|
settings.ReasoningEffort = DefaultReasoningEffort
|
|
}
|
|
return settings
|
|
}
|
|
|
|
func isValidReasoningEffort(effort string) bool {
|
|
switch effort {
|
|
case "low", "medium", "high":
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func normalizeBotSettingsReadRow(row sqlc.GetSettingsByBotIDRow) Settings {
|
|
return normalizeBotSettingsFields(
|
|
row.MaxContextLoadTime,
|
|
row.MaxContextTokens,
|
|
row.MaxInboxItems,
|
|
row.Language,
|
|
row.AllowGuest,
|
|
row.ReasoningEnabled,
|
|
row.ReasoningEffort,
|
|
row.ChatModelID,
|
|
row.MemoryModelID,
|
|
row.EmbeddingModelID,
|
|
row.SearchProviderID,
|
|
)
|
|
}
|
|
|
|
func normalizeBotSettingsWriteRow(row sqlc.UpsertBotSettingsRow) Settings {
|
|
return normalizeBotSettingsFields(
|
|
row.MaxContextLoadTime,
|
|
row.MaxContextTokens,
|
|
row.MaxInboxItems,
|
|
row.Language,
|
|
row.AllowGuest,
|
|
row.ReasoningEnabled,
|
|
row.ReasoningEffort,
|
|
row.ChatModelID,
|
|
row.MemoryModelID,
|
|
row.EmbeddingModelID,
|
|
row.SearchProviderID,
|
|
)
|
|
}
|
|
|
|
func normalizeBotSettingsFields(
|
|
maxContextLoadTime int32,
|
|
maxContextTokens int32,
|
|
maxInboxItems int32,
|
|
language string,
|
|
allowGuest bool,
|
|
reasoningEnabled bool,
|
|
reasoningEffort string,
|
|
chatModelID pgtype.UUID,
|
|
memoryModelID pgtype.UUID,
|
|
embeddingModelID pgtype.UUID,
|
|
searchProviderID pgtype.UUID,
|
|
) Settings {
|
|
settings := normalizeBotSetting(maxContextLoadTime, maxContextTokens, maxInboxItems, language, allowGuest, reasoningEnabled, reasoningEffort)
|
|
if chatModelID.Valid {
|
|
settings.ChatModelID = uuid.UUID(chatModelID.Bytes).String()
|
|
}
|
|
if memoryModelID.Valid {
|
|
settings.MemoryModelID = uuid.UUID(memoryModelID.Bytes).String()
|
|
}
|
|
if embeddingModelID.Valid {
|
|
settings.EmbeddingModelID = uuid.UUID(embeddingModelID.Bytes).String()
|
|
}
|
|
if searchProviderID.Valid {
|
|
settings.SearchProviderID = uuid.UUID(searchProviderID.Bytes).String()
|
|
}
|
|
return settings
|
|
}
|
|
|
|
func (s *Service) resolveModelUUID(ctx context.Context, modelID string) (pgtype.UUID, error) {
|
|
modelID = strings.TrimSpace(modelID)
|
|
if modelID == "" {
|
|
return pgtype.UUID{}, fmt.Errorf("%w: model_id is required", ErrInvalidModelRef)
|
|
}
|
|
|
|
// Preferred path: when caller already passes the model UUID.
|
|
if parsed, err := db.ParseUUID(modelID); err == nil {
|
|
if _, err := s.queries.GetModelByID(ctx, parsed); err == nil {
|
|
return parsed, nil
|
|
} else if !errors.Is(err, pgx.ErrNoRows) {
|
|
return pgtype.UUID{}, err
|
|
}
|
|
}
|
|
|
|
rows, err := s.queries.ListModelsByModelID(ctx, modelID)
|
|
if err != nil {
|
|
return pgtype.UUID{}, err
|
|
}
|
|
if len(rows) == 0 {
|
|
return pgtype.UUID{}, fmt.Errorf("%w: model not found: %s", ErrInvalidModelRef, modelID)
|
|
}
|
|
if len(rows) > 1 {
|
|
return pgtype.UUID{}, fmt.Errorf("%w: %s", ErrModelIDAmbiguous, modelID)
|
|
}
|
|
return rows[0].ID, nil
|
|
}
|