mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
d3bf6bc90a
* 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)
51 lines
1.1 KiB
Go
51 lines
1.1 KiB
Go
package local
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/memohai/memoh/internal/channel"
|
|
)
|
|
|
|
func TestRouteHubPublishEvent(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
hub := NewRouteHub()
|
|
_, stream, cancel := hub.Subscribe("bot-1")
|
|
defer cancel()
|
|
|
|
hub.PublishEvent("bot-1", channel.StreamEvent{
|
|
Type: channel.StreamEventDelta,
|
|
Delta: "hello",
|
|
})
|
|
|
|
select {
|
|
case item := <-stream:
|
|
if item.Target != "bot-1" {
|
|
t.Fatalf("unexpected target: %s", item.Target)
|
|
}
|
|
if item.Event.Type != channel.StreamEventDelta {
|
|
t.Fatalf("unexpected event type: %s", item.Event.Type)
|
|
}
|
|
case <-time.After(time.Second):
|
|
t.Fatal("expected event but timed out")
|
|
}
|
|
}
|
|
|
|
func TestLocalOutboundStreamClose(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
hub := NewRouteHub()
|
|
stream := newLocalOutboundStream(hub, "bot-2")
|
|
if err := stream.Close(context.Background()); err != nil {
|
|
t.Fatalf("unexpected close error: %v", err)
|
|
}
|
|
if err := stream.Push(context.Background(), channel.PreparedStreamEvent{
|
|
Type: channel.StreamEventDelta,
|
|
Delta: "should fail",
|
|
}); err == nil {
|
|
t.Fatal("expected push on closed stream to fail")
|
|
}
|
|
}
|