forked from coaidev/coai
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathworker.go
94 lines (75 loc) · 2.51 KB
/
worker.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package channel
import (
"chat/adapter"
"chat/globals"
"chat/utils"
"fmt"
"github.com/go-redis/redis/v8"
"time"
)
func NewChatRequest(group string, props *adapter.ChatProps, hook globals.Hook) error {
ticker := ConduitInstance.GetTicker(props.Model, group)
if ticker == nil || ticker.IsEmpty() {
return fmt.Errorf("cannot find channel for model %s", props.Model)
}
var err error
for !ticker.IsDone() {
if channel := ticker.Next(); channel != nil {
props.MaxRetries = utils.ToPtr(channel.GetRetry())
if err = adapter.NewChatRequest(channel, props, hook); err == nil || err.Error() == "signal" {
return nil
}
globals.Warn(fmt.Sprintf("[channel] caught error %s for model %s at channel %s", err.Error(), props.Model, channel.GetName()))
}
}
globals.Info(fmt.Sprintf("[channel] channels are exhausted for model %s", props.Model))
if err == nil {
err = fmt.Errorf("channels are exhausted for model %s", props.Model)
}
return err
}
func PreflightCache(cache *redis.Client, model string, hash string, buffer *utils.Buffer, hook globals.Hook) (int64, bool, error) {
if !utils.Contains(model, globals.CacheAcceptedModels) {
return 0, false, nil
}
idx := utils.Intn64(globals.CacheAcceptedSize)
key := fmt.Sprintf("chat-cache:%d:%s", idx, hash)
raw, err := cache.Get(cache.Context(), key).Result()
if err != nil {
return idx, false, nil
}
buf, err := utils.UnmarshalString[utils.Buffer](raw)
if err != nil {
return idx, false, nil
}
data := buf.Read()
if data == "" {
return idx, false, nil
}
buffer.SetInputTokens(buf.CountInputToken())
buffer.SetToolCalls(buf.GetToolCalls())
buffer.SetFunctionCall(buf.GetFunctionCall())
return idx, true, hook(&globals.Chunk{
Content: data,
FunctionCall: buf.GetFunctionCall(),
ToolCall: buf.GetToolCalls(),
})
}
func StoreCache(cache *redis.Client, hash string, index int64, buffer *utils.Buffer) {
key := fmt.Sprintf("chat-cache:%d:%s", index, hash)
raw := utils.Marshal(buffer)
expire := time.Duration(globals.CacheAcceptedExpire) * time.Second
cache.Set(cache.Context(), key, raw, expire)
}
func NewChatRequestWithCache(cache *redis.Client, buffer *utils.Buffer, group string, props *adapter.ChatProps, hook globals.Hook) (bool, error) {
hash := utils.Md5Encrypt(utils.Marshal(props))
idx, hit, err := PreflightCache(cache, props.Model, hash, buffer, hook)
if hit {
return true, err
}
if err = NewChatRequest(group, props, hook); err != nil {
return false, err
}
StoreCache(cache, hash, idx, buffer)
return false, nil
}