You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
242 lines
5.2 KiB
Go
242 lines
5.2 KiB
Go
/*
|
|
* Copyright 2025 CloudWeGo Authors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package agent
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"embed"
|
|
"errors"
|
|
"io"
|
|
"log"
|
|
"mime"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/cloudwego/hertz/pkg/app"
|
|
"github.com/cloudwego/hertz/pkg/protocol/consts"
|
|
"github.com/cloudwego/hertz/pkg/route"
|
|
"github.com/hertz-contrib/sse"
|
|
|
|
"github.com/cloudwego/eino-examples/quickstart/eino_assistant/pkg/mem"
|
|
)
|
|
|
|
//go:embed web
|
|
var webContent embed.FS
|
|
|
|
type ChatRequest struct {
|
|
ID string `json:"id"`
|
|
Message string `json:"message"`
|
|
}
|
|
|
|
func BindRoutes(r *route.RouterGroup) error {
|
|
if err := Init(); err != nil {
|
|
return err
|
|
}
|
|
|
|
// API 路由
|
|
r.GET("/api/chat", HandleChat)
|
|
r.GET("/api/log", HandleLog)
|
|
r.GET("/api/history", HandleHistory)
|
|
r.DELETE("/api/history", HandleDeleteHistory)
|
|
|
|
// 静态文件服务
|
|
r.GET("/", func(ctx context.Context, c *app.RequestContext) {
|
|
content, err := webContent.ReadFile("web/index.html")
|
|
if err != nil {
|
|
c.String(consts.StatusNotFound, "File not found")
|
|
return
|
|
}
|
|
c.Header("Content-Type", "text/html")
|
|
c.Write(content)
|
|
})
|
|
|
|
r.GET("/:file", func(ctx context.Context, c *app.RequestContext) {
|
|
file := c.Param("file")
|
|
content, err := webContent.ReadFile("web/" + file)
|
|
if err != nil {
|
|
c.String(consts.StatusNotFound, "File not found")
|
|
return
|
|
}
|
|
|
|
contentType := mime.TypeByExtension(filepath.Ext(file))
|
|
if contentType == "" {
|
|
contentType = "application/octet-stream"
|
|
}
|
|
c.Header("Content-Type", contentType)
|
|
c.Write(content)
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
func HandleChat(ctx context.Context, c *app.RequestContext) {
|
|
id := c.Query("id")
|
|
message := c.Query("message")
|
|
if id == "" || message == "" {
|
|
c.JSON(consts.StatusBadRequest, map[string]string{
|
|
"status": "error",
|
|
"error": "missing id or message parameter",
|
|
})
|
|
return
|
|
}
|
|
|
|
log.Printf("[Chat] Starting chat with ID: %s, Message: %s\n", id, message)
|
|
|
|
sr, err := RunAgent(ctx, id, message)
|
|
if err != nil {
|
|
log.Printf("[Chat] Error running agent: %v\n", err)
|
|
c.JSON(consts.StatusInternalServerError, map[string]string{
|
|
"status": "error",
|
|
"error": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
s := sse.NewStream(c)
|
|
defer func() {
|
|
sr.Close()
|
|
c.Flush()
|
|
|
|
log.Printf("[Chat] Finished chat with ID: %s\n", id)
|
|
}()
|
|
|
|
outer:
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
log.Printf("[Chat] Context done for chat ID: %s\n", id)
|
|
return
|
|
default:
|
|
msg, err := sr.Recv()
|
|
if errors.Is(err, io.EOF) {
|
|
log.Printf("[Chat] EOF received for chat ID: %s\n", id)
|
|
break outer
|
|
}
|
|
if err != nil {
|
|
log.Printf("[Chat] Error receiving message: %v\n", err)
|
|
break outer
|
|
}
|
|
|
|
err = s.Publish(&sse.Event{
|
|
Data: []byte(msg.Content),
|
|
})
|
|
if err != nil {
|
|
log.Printf("[Chat] Error publishing message: %v\n", err)
|
|
break outer
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func HandleHistory(ctx context.Context, c *app.RequestContext) {
|
|
// query: id => get history, none => list all
|
|
id := c.Query("id")
|
|
|
|
if id == "" {
|
|
ids := mem.GetDefaultMemory().ListConversations()
|
|
|
|
c.JSON(consts.StatusOK, map[string]interface{}{
|
|
"ids": ids,
|
|
})
|
|
return
|
|
}
|
|
|
|
conversation := mem.GetDefaultMemory().GetConversation(id, false)
|
|
if conversation == nil {
|
|
c.JSON(consts.StatusNotFound, map[string]string{
|
|
"error": "conversation not found",
|
|
})
|
|
return
|
|
}
|
|
|
|
c.JSON(consts.StatusOK, map[string]interface{}{
|
|
"conversation": conversation,
|
|
})
|
|
}
|
|
|
|
func HandleDeleteHistory(ctx context.Context, c *app.RequestContext) {
|
|
id := c.Query("id")
|
|
if id == "" {
|
|
c.JSON(consts.StatusBadRequest, map[string]string{
|
|
"error": "missing id parameter",
|
|
})
|
|
return
|
|
}
|
|
|
|
mem.GetDefaultMemory().DeleteConversation(id)
|
|
c.JSON(consts.StatusOK, map[string]string{
|
|
"status": "success",
|
|
})
|
|
}
|
|
|
|
func HandleLog(ctx context.Context, c *app.RequestContext) {
|
|
file, err := os.Open("log/eino.log")
|
|
if err != nil {
|
|
c.JSON(consts.StatusInternalServerError, map[string]string{
|
|
"status": "error",
|
|
"error": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
defer file.Close()
|
|
|
|
// Create a new SSE stream
|
|
s := sse.NewStream(c)
|
|
defer c.Flush()
|
|
|
|
// Seek to the end of the file
|
|
_, err = file.Seek(0, io.SeekEnd)
|
|
if err != nil {
|
|
log.Println("error seeking file:", err)
|
|
return
|
|
}
|
|
|
|
// Use a goroutine to continuously read new lines
|
|
go func() {
|
|
reader := bufio.NewReader(file)
|
|
for {
|
|
line, err := reader.ReadString('\n')
|
|
if err != nil && err != io.EOF {
|
|
log.Println("error reading log:", err)
|
|
break
|
|
}
|
|
|
|
// If we got a line, publish it
|
|
if line != "" {
|
|
err = s.Publish(&sse.Event{
|
|
Data: []byte(line),
|
|
})
|
|
if err != nil {
|
|
log.Println("error publishing log:", err)
|
|
break
|
|
}
|
|
}
|
|
|
|
// If we hit EOF, wait a bit and try again
|
|
if err == io.EOF {
|
|
time.Sleep(100 * time.Millisecond)
|
|
continue
|
|
}
|
|
}
|
|
}()
|
|
|
|
// Keep the connection open
|
|
<-ctx.Done()
|
|
}
|