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
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)
|
|
}
|
|
}
|