mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
refactor(core): restructure conversation, channel and message domains
- Rename chat module to conversation with flow-based architecture - Move channelidentities into channel/identities subpackage - Add channel/route for routing logic - Add message service with event hub - Add MCP providers: container, directory, schedule - Refactor Feishu/Telegram adapters with directory and stream support - Add platform management page and channel badges in web UI - Update database schema for conversations, messages and channel routes - Add @memoh/shared package for cross-package type definitions
This commit is contained in:
+60
-16
@@ -24,7 +24,6 @@ export const createAgent = ({
|
||||
currentChannel = 'Unknown Channel',
|
||||
identity = {
|
||||
botId: '',
|
||||
sessionId: '',
|
||||
containerId: '',
|
||||
channelIdentityId: '',
|
||||
displayName: '',
|
||||
@@ -53,18 +52,43 @@ export const createAgent = ({
|
||||
toolsContent: '',
|
||||
}
|
||||
}
|
||||
const fetchFile = async (path: string) => {
|
||||
const response = await fetch(`/bots/${identity.botId}/container/fs/file?path=${encodeURIComponent(path)}`)
|
||||
if (!response.ok) {
|
||||
return ''
|
||||
const readViaMCP = async (path: string): Promise<string> => {
|
||||
const url = `${auth.baseUrl.replace(/\/$/, '')}/bots/${identity.botId}/tools`
|
||||
const headers: Record<string, string> = {
|
||||
'Content-Type': 'application/json',
|
||||
'Accept': 'application/json, text/event-stream',
|
||||
'Authorization': `Bearer ${auth.bearer}`,
|
||||
}
|
||||
const data = await response.json().catch(() => ({} as { content?: string }))
|
||||
return typeof data?.content === 'string' ? data.content : ''
|
||||
if (identity.channelIdentityId) {
|
||||
headers['X-Memoh-Channel-Identity-Id'] = identity.channelIdentityId
|
||||
}
|
||||
const body = JSON.stringify({
|
||||
jsonrpc: '2.0',
|
||||
id: `read-${path}`,
|
||||
method: 'tools/call',
|
||||
params: { name: 'read', arguments: { path } },
|
||||
})
|
||||
const response = await fetch(url, { method: 'POST', headers, body })
|
||||
if (!response.ok) return ''
|
||||
const data = await response.json().catch(() => ({} as any))
|
||||
const structured = data?.result?.structuredContent ?? data?.result?.content?.[0]?.text
|
||||
if (typeof structured === 'string') {
|
||||
try {
|
||||
const parsed = JSON.parse(structured)
|
||||
return typeof parsed?.content === 'string' ? parsed.content : ''
|
||||
} catch {
|
||||
return structured
|
||||
}
|
||||
}
|
||||
if (typeof structured === 'object' && structured?.content) {
|
||||
return typeof structured.content === 'string' ? structured.content : ''
|
||||
}
|
||||
return ''
|
||||
}
|
||||
const [identityContent, soulContent, toolsContent] = await Promise.all([
|
||||
fetchFile('IDENTITY.md'),
|
||||
fetchFile('SOUL.md'),
|
||||
fetchFile('TOOLS.md'),
|
||||
readViaMCP('IDENTITY.md'),
|
||||
readViaMCP('SOUL.md'),
|
||||
readViaMCP('TOOLS.md'),
|
||||
])
|
||||
return {
|
||||
identityContent,
|
||||
@@ -80,6 +104,7 @@ export const createAgent = ({
|
||||
language,
|
||||
maxContextLoadTime: activeContextTime,
|
||||
channels,
|
||||
currentChannel,
|
||||
skills,
|
||||
enabledSkills,
|
||||
identityContent,
|
||||
@@ -100,9 +125,6 @@ export const createAgent = ({
|
||||
const headers: Record<string, string> = {
|
||||
'Authorization': `Bearer ${auth.bearer}`,
|
||||
}
|
||||
if (identity.sessionId) {
|
||||
headers['X-Memoh-Chat-Id'] = identity.sessionId
|
||||
}
|
||||
if (identity.channelIdentityId) {
|
||||
headers['X-Memoh-Channel-Identity-Id'] = identity.channelIdentityId
|
||||
}
|
||||
@@ -115,9 +137,6 @@ export const createAgent = ({
|
||||
if (identity.replyTarget) {
|
||||
headers['X-Memoh-Reply-Target'] = identity.replyTarget
|
||||
}
|
||||
if (identity.displayName) {
|
||||
headers['X-Memoh-Display-Name'] = identity.displayName
|
||||
}
|
||||
const { tools: mcpTools, close: closeMCP } = await getMCPTools(`${baseUrl}/bots/${botId}/tools`, headers)
|
||||
return {
|
||||
tools: mcpTools,
|
||||
@@ -257,6 +276,28 @@ export const createAgent = ({
|
||||
}
|
||||
}
|
||||
|
||||
const resolveStreamErrorMessage = (raw: unknown): string => {
|
||||
if (raw instanceof Error && raw.message.trim()) {
|
||||
return raw.message
|
||||
}
|
||||
if (typeof raw === 'string' && raw.trim()) {
|
||||
return raw
|
||||
}
|
||||
if (raw && typeof raw === 'object') {
|
||||
const candidate = raw as { message?: unknown; error?: unknown }
|
||||
if (typeof candidate.message === 'string' && candidate.message.trim()) {
|
||||
return candidate.message
|
||||
}
|
||||
if (typeof candidate.error === 'string' && candidate.error.trim()) {
|
||||
return candidate.error
|
||||
}
|
||||
if (candidate.error instanceof Error && candidate.error.message.trim()) {
|
||||
return candidate.error.message
|
||||
}
|
||||
}
|
||||
return 'Model stream failed'
|
||||
}
|
||||
|
||||
async function* stream(input: AgentInput): AsyncGenerator<AgentAction> {
|
||||
const userPrompt = generateUserPrompt(input)
|
||||
const messages = [...input.messages, userPrompt]
|
||||
@@ -296,6 +337,9 @@ export const createAgent = ({
|
||||
input,
|
||||
}
|
||||
for await (const chunk of fullStream) {
|
||||
if (chunk.type === 'error') {
|
||||
throw new Error(resolveStreamErrorMessage((chunk as { error?: unknown }).error))
|
||||
}
|
||||
switch (chunk.type) {
|
||||
case 'reasoning-start': yield {
|
||||
type: 'reasoning_start',
|
||||
|
||||
Reference in New Issue
Block a user