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.

147 lines
4.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 main
import (
"bufio"
"context"
"fmt"
"log"
"os"
"strings"
"time"
clc "github.com/cloudwego/eino-ext/callbacks/cozeloop"
"github.com/cloudwego/eino/adk"
"github.com/cloudwego/eino/callbacks"
"github.com/coze-dev/cozeloop-go"
"github.com/cloudwego/eino-examples/adk/common/prints"
"github.com/cloudwego/eino-examples/adk/common/store"
"github.com/cloudwego/eino-examples/adk/common/tool"
)
func main() {
ctx := context.Background()
cozeloopApiToken := os.Getenv("COZELOOP_API_TOKEN")
cozeloopWorkspaceID := os.Getenv("COZELOOP_WORKSPACE_ID") // use cozeloop trace, from https://loop.coze.cn/open/docs/cozeloop/go-sdk#4a8c980e
var handlers []callbacks.Handler
if cozeloopApiToken != "" && cozeloopWorkspaceID != "" {
client, err := cozeloop.NewClient(
cozeloop.WithAPIToken(cozeloopApiToken),
cozeloop.WithWorkspaceID(cozeloopWorkspaceID),
)
if err != nil {
panic(err)
}
defer func() {
time.Sleep(5 * time.Second)
client.Close(ctx)
}()
handlers = append(handlers, clc.NewLoopHandler(client))
}
callbacks.AppendGlobalHandlers(handlers...)
a := NewTicketBookingAgent()
runner := adk.NewRunner(context.Background(), adk.RunnerConfig{
EnableStreaming: true, // you can disable streaming here
Agent: a,
// provide a CheckPointStore for eino to persist the execution state of the agent for later resumption.
// Here we use an in-memory store for simplicity.
// In the real world, you can use a distributed store like Redis to persist the checkpoints.
CheckPointStore: store.NewInMemoryStore(),
})
iter := runner.Query(context.Background(), "book a ticket for Martin, to Beijing, on 2025-12-01, the phone number is 1234567. directly call tool.", adk.WithCheckPointID("1"))
var lastEvent *adk.AgentEvent
for {
event, ok := iter.Next()
if !ok {
break
}
if event.Err != nil {
log.Fatal(event.Err)
}
prints.Event(event)
lastEvent = event
}
if lastEvent == nil {
log.Fatal("last event is nil")
}
if lastEvent.Action == nil || lastEvent.Action.Interrupted == nil {
log.Fatal("last event is not an interrupt event")
}
// this interruptID is crucial 'locator' for Eino to know where the interrupt happens,
// so when resuming later, you have to provide this same `interruptID` along with the approval result back to Eino
interruptID := lastEvent.Action.Interrupted.InterruptContexts[0].ID
var apResult *tool.ApprovalResult
for {
scanner := bufio.NewScanner(os.Stdin)
fmt.Print("your input here: ")
scanner.Scan()
fmt.Println()
nInput := scanner.Text()
if strings.ToUpper(nInput) == "Y" {
apResult = &tool.ApprovalResult{Approved: true}
break
} else if strings.ToUpper(nInput) == "N" {
// Prompt for reason when denying
fmt.Print("Please provide a reason for denial: ")
scanner.Scan()
reason := scanner.Text()
fmt.Println()
apResult = &tool.ApprovalResult{Approved: false, DisapproveReason: &reason}
break
}
fmt.Println("invalid input, please input Y or N")
}
// here we directly resumes right in the same instance where the original `Runner.Query` happened.
// In the real world, the original `Runner.Run/Query` and the subsequent `Runner.ResumeWithParams`
// can happen in different processes or machines, as long as you use the same `CheckPointID`,
// and you provided a distributed `CheckPointStore` when creating the `Runner` instance.
iter, err := runner.ResumeWithParams(context.Background(), "1", &adk.ResumeParams{
Targets: map[string]any{
interruptID: apResult,
},
})
if err != nil {
log.Fatal(err)
}
for {
event, ok := iter.Next()
if !ok {
break
}
if event.Err != nil {
log.Fatal(event.Err)
}
prints.Event(event)
}
}