mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-25 07:00:48 +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)
150 lines
4.3 KiB
Go
150 lines
4.3 KiB
Go
package feishu
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/memohai/memoh/internal/channel"
|
|
)
|
|
|
|
// TestFeishuGateway_Integration runs Feishu channel integration test.
|
|
// Required env: FEISHU_APP_ID, FEISHU_APP_SECRET.
|
|
// Optional: FEISHU_ENCRYPT_KEY, FEISHU_VERIFICATION_TOKEN.
|
|
func TestFeishuGateway_Integration(t *testing.T) {
|
|
appID := os.Getenv("FEISHU_APP_ID")
|
|
appSecret := os.Getenv("FEISHU_APP_SECRET")
|
|
|
|
if appID == "" || appSecret == "" {
|
|
t.Skip("skipping integration test: FEISHU_APP_ID or FEISHU_APP_SECRET not set")
|
|
}
|
|
|
|
encryptKey := os.Getenv("FEISHU_ENCRYPT_KEY")
|
|
verificationToken := os.Getenv("FEISHU_VERIFICATION_TOKEN")
|
|
|
|
logger := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{
|
|
Level: slog.LevelInfo,
|
|
}))
|
|
adapter := NewFeishuAdapter(logger)
|
|
|
|
cfg := channel.ChannelConfig{
|
|
ID: "integration-test-bot",
|
|
Credentials: map[string]any{
|
|
"app_id": appID,
|
|
"app_secret": appSecret,
|
|
"encrypt_key": encryptKey,
|
|
"verification_token": verificationToken,
|
|
},
|
|
}
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Minute)
|
|
defer cancel()
|
|
|
|
receivedChan := make(chan channel.InboundMessage, 1)
|
|
|
|
handler := func(ctx context.Context, c channel.ChannelConfig, msg channel.InboundMessage) error {
|
|
plainText := msg.Message.PlainText()
|
|
logger.Info("received message in test",
|
|
slog.String("text", plainText),
|
|
slog.String("user_id", msg.Sender.Attribute("user_id")),
|
|
slog.String("route_key", msg.RoutingKey()))
|
|
|
|
select {
|
|
case receivedChan <- msg:
|
|
default:
|
|
}
|
|
|
|
reply := channel.OutboundMessage{
|
|
Target: msg.ReplyTarget,
|
|
Message: channel.Message{
|
|
Text: fmt.Sprintf("【Memoh 集成测试】已收到消息: %s\n测试时间: %s", plainText, time.Now().Format("15:04:05")),
|
|
},
|
|
}
|
|
|
|
if err := adapter.Send(ctx, c, channel.PreparedOutboundMessage{
|
|
Target: reply.Target,
|
|
Message: channel.PreparedMessage{
|
|
Message: reply.Message,
|
|
},
|
|
}); err != nil {
|
|
return fmt.Errorf("failed to send reply: %w", err)
|
|
}
|
|
|
|
go func() {
|
|
time.Sleep(1 * time.Second)
|
|
pushMsg := channel.OutboundMessage{
|
|
Target: msg.ReplyTarget,
|
|
Message: channel.Message{
|
|
Text: "【Memoh 集成测试】主动推送验证成功。",
|
|
},
|
|
}
|
|
pushCtx, pushCancel := context.WithTimeout(context.WithoutCancel(ctx), 5*time.Second)
|
|
defer pushCancel()
|
|
_ = adapter.Send(pushCtx, c, channel.PreparedOutboundMessage{
|
|
Target: pushMsg.Target,
|
|
Message: channel.PreparedMessage{
|
|
Message: pushMsg.Message,
|
|
},
|
|
})
|
|
}()
|
|
|
|
return nil
|
|
}
|
|
|
|
logger.Info("starting Feishu adapter", slog.String("app_id", appID))
|
|
runner, err := adapter.Connect(ctx, cfg, handler)
|
|
if err != nil {
|
|
t.Fatalf("adapter connect failed: %v", err)
|
|
}
|
|
defer func() {
|
|
_ = runner.Stop(context.Background())
|
|
}()
|
|
|
|
fmt.Println("==================================================================")
|
|
fmt.Println("Feishu integration test ready. Send a message in Feishu client to verify.")
|
|
fmt.Println("Test ends on first message received or 10 min timeout.")
|
|
fmt.Println("==================================================================")
|
|
|
|
select {
|
|
case msg := <-receivedChan:
|
|
logger.Info("integration test passed", slog.String("received_text", msg.Message.PlainText()))
|
|
time.Sleep(2 * time.Second)
|
|
case <-ctx.Done():
|
|
if ctx.Err() == context.DeadlineExceeded {
|
|
t.Log("test timed out")
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestFeishuDiscoverSelf_Integration verifies the bot info API call.
|
|
// Required env: FEISHU_APP_ID, FEISHU_APP_SECRET.
|
|
func TestFeishuDiscoverSelf_Integration(t *testing.T) {
|
|
appID := os.Getenv("FEISHU_APP_ID")
|
|
appSecret := os.Getenv("FEISHU_APP_SECRET")
|
|
if appID == "" || appSecret == "" {
|
|
t.Skip("skipping integration test: FEISHU_APP_ID or FEISHU_APP_SECRET not set")
|
|
}
|
|
adapter := NewFeishuAdapter(nil)
|
|
credentials := map[string]any{
|
|
"app_id": appID,
|
|
"app_secret": appSecret,
|
|
}
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
identity, extID, err := adapter.DiscoverSelf(ctx, credentials)
|
|
if err != nil {
|
|
t.Fatalf("discover self failed: %v", err)
|
|
}
|
|
openID, _ := identity["open_id"].(string)
|
|
if openID == "" {
|
|
t.Fatalf("expected non-empty open_id")
|
|
}
|
|
if extID != openID {
|
|
t.Fatalf("expected external_id=%s, got %s", openID, extID)
|
|
}
|
|
t.Logf("bot identity: %+v", identity)
|
|
}
|