Files
Memoh/internal/channel/adapters/wecom/adapter_integration_test.go
T
BBQ d3bf6bc90a fix(channel,attachment): channel quality refactor & attachment pipeline fixes (#349)
* feat(channel): add DingTalk channel adapter

- Add DingTalk channel adapter (`internal/channel/adapters/dingtalk/`) using dingtalk-stream-sdk-go, supporting inbound message receiving and outbound text/markdown reply
- Register DingTalk adapter in cmd/agent and cmd/memoh
- Add go.mod dependency: github.com/memohai/dingtalk-stream-sdk-go
- Add Dingtalk and Wecom SVG icons and Vue components to @memohai/icon
- Refactor existing icon components to remove redundant inline wrappers
- Add `channelTypeDisplayName` util for consistent channel label resolution
- Add DingTalk/WeCom i18n entries (en/zh) for types and typesShort
- Extend channel-icon, bot-channels, channel-settings-panel to support dingtalk/wecom
- Use channelTypeDisplayName in profile page to replace ad-hoc i18n lookup

* fix(channel,attachment): channel quality refactor & attachment pipeline fixes

Channel module:
- Fix RemoveAdapter not cleaning connectionMeta (stale status leak)
- Fix preparedAttachmentTypeFromMime misclassifying image/gif
- Fix sleepWithContext time.After goroutine/timer leak
- Export IsDataURL/IsHTTPURL/IsDataPath, dedup across packages
- Cache OutboundPolicy in managerOutboundStream to avoid repeated lookups
- Split OutboundAttachmentStore: extract ContainerAttachmentIngester interface
- Add ManagerOption funcs (WithInboundQueueSize, WithInboundWorkers, WithRefreshInterval)
- Add thread-safety docs on OutboundStream / managerOutboundStream
- Add debug logs on successful send/edit paths
- Expand outbound_prepare_test.go with 21 new cases
- Convert no-receiver adapter helpers to package-level funcs; drop unused params

DingTalk adapter:
- Implement AttachmentResolver: download inbound media via /v1.0/robot/messageFiles/download
- Fix pure-image inbound messages failing due to missing resolver

Attachment pipeline:
- Fix images invisible to LLM in pipeline (DCP) path: inject InlineImages into
  last user message when cfg.Query is empty
- Fix public_url fallback: skip direct URL-to-LLM when ContentHash is set,
  always prefer inlined persisted asset
- Inject path: carry ImageParts through agent.InjectMessage; inline persisted
  attachments in resolver inject goroutine so mid-stream images reach the model
- Fix ResolveMime for images: prefer content-sniffed MIME over platform-declared
  MIME (fixes Feishu sending image/png header for actual JPEG content → API 400)
2026-04-09 14:36:11 +08:00

160 lines
3.5 KiB
Go

package wecom
import (
"context"
"encoding/json"
"net/http"
"net/http/httptest"
"strings"
"testing"
"time"
"github.com/gorilla/websocket"
"github.com/memohai/memoh/internal/channel"
)
func TestWeComAdapter_ReplyUsesRespondCmd(t *testing.T) {
t.Parallel()
upgrader := websocket.Upgrader{}
receivedRespond := make(chan WSFrame, 1)
serverErr := make(chan error, 1)
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
select {
case serverErr <- err:
default:
}
return
}
defer func() { _ = conn.Close() }()
var subscribeFrame WSFrame
if err := conn.ReadJSON(&subscribeFrame); err != nil {
select {
case serverErr <- err:
default:
}
return
}
if err := conn.WriteJSON(WSFrame{
Headers: WSHeaders{ReqID: subscribeFrame.Headers.ReqID},
ErrCode: 0,
}); err != nil {
select {
case serverErr <- err:
default:
}
return
}
body, _ := json.Marshal(MessageCallbackBody{
MsgID: "msg_1",
ChatID: "chat_1",
ChatType: "group",
CreateTime: time.Now().UnixMilli(),
From: CallbackFrom{UserID: "u1"},
MsgType: "text",
ResponseURL: "https://example.com/resp",
Text: &MessageText{Content: "hello"},
})
if err := conn.WriteJSON(WSFrame{
Cmd: WSCmdMsgCallback,
Headers: WSHeaders{ReqID: "callback_req_id"},
Body: body,
}); err != nil {
select {
case serverErr <- err:
default:
}
return
}
var respondFrame WSFrame
if err := conn.ReadJSON(&respondFrame); err != nil {
select {
case serverErr <- err:
default:
}
return
}
select {
case receivedRespond <- respondFrame:
default:
}
_ = conn.WriteJSON(WSFrame{
Headers: WSHeaders{ReqID: respondFrame.Headers.ReqID},
ErrCode: 0,
})
}))
defer server.Close()
adapter := NewWeComAdapter(nil)
cfg := channel.ChannelConfig{
Credentials: map[string]any{
"botId": "bot",
"secret": "sec",
"wsUrl": "ws" + strings.TrimPrefix(server.URL, "http"),
},
}
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
inboundCh := make(chan channel.InboundMessage, 1)
conn, err := adapter.Connect(ctx, cfg, func(_ context.Context, _ channel.ChannelConfig, msg channel.InboundMessage) error {
select {
case inboundCh <- msg:
default:
}
return nil
})
if err != nil {
t.Fatalf("connect error: %v", err)
}
defer func() { _ = conn.Stop(context.Background()) }()
select {
case inbound := <-inboundCh:
if inbound.Message.ID != "msg_1" {
t.Fatalf("unexpected inbound message id: %s", inbound.Message.ID)
}
case err := <-serverErr:
t.Fatalf("server error: %v", err)
case <-time.After(2 * time.Second):
t.Fatal("timeout waiting inbound callback")
}
err = adapter.Send(context.Background(), cfg, channel.PreparedOutboundMessage{
Target: "chat_id:chat_1",
Message: channel.PreparedMessage{
Message: channel.Message{
Format: channel.MessageFormatPlain,
Text: "reply content",
Reply: &channel.ReplyRef{
MessageID: "msg_1",
},
},
},
})
if err != nil {
t.Fatalf("send error: %v", err)
}
select {
case frame := <-receivedRespond:
if frame.Cmd != WSCmdRespond {
t.Fatalf("expected cmd=%s got=%s", WSCmdRespond, frame.Cmd)
}
if frame.Headers.ReqID != "callback_req_id" {
t.Fatalf("expected req_id callback_req_id got=%s", frame.Headers.ReqID)
}
case err := <-serverErr:
t.Fatalf("server error: %v", err)
case <-time.After(2 * time.Second):
t.Fatal("timeout waiting respond frame")
}
}