192 lines
5.2 KiB
Vue
192 lines
5.2 KiB
Vue
<script setup lang="ts">
|
||
import { deepResearch, type PartialSearchResult, type ResearchResult, type ResearchStep } from '~/lib/deep-research'
|
||
import type { TreeNode } from './Tree.vue'
|
||
|
||
const emit = defineEmits<{
|
||
(e: 'complete', results: ResearchResult): void
|
||
}>()
|
||
|
||
const tree = ref<TreeNode>({
|
||
id: '0',
|
||
label: 'Start',
|
||
children: [],
|
||
})
|
||
const selectedNode = ref<TreeNode>()
|
||
const searchResults = ref<Record<string, PartialSearchResult>>({})
|
||
|
||
function handleResearchProgress(step: ResearchStep) {
|
||
let node: TreeNode | null = null
|
||
let nodeId = ''
|
||
|
||
if (step.type !== 'complete') {
|
||
nodeId = step.nodeId
|
||
node = findNode(tree.value, step.nodeId)
|
||
if (node) {
|
||
node.status = step.type
|
||
}
|
||
}
|
||
|
||
switch (step.type) {
|
||
case 'generating_query': {
|
||
if (!node) {
|
||
// 创建新节点
|
||
node = {
|
||
id: nodeId,
|
||
label: 'Generating...',
|
||
researchGoal: 'Generating research goal...',
|
||
learnings: [],
|
||
children: [],
|
||
}
|
||
const parentNodeId = getParentNodeId(nodeId)
|
||
// 如果是根节点的直接子节点
|
||
if (parentNodeId === '0') {
|
||
tree.value.children.push(node)
|
||
} else {
|
||
// 找到父节点并添加
|
||
const parentNode = findNode(tree.value, getParentNodeId(step.nodeId))
|
||
if (parentNode) {
|
||
parentNode.children.push(node)
|
||
}
|
||
}
|
||
}
|
||
// 更新节点的查询内容
|
||
if (step.result) {
|
||
node.label = step.result.query ?? 'Generating...'
|
||
node.researchGoal = step.result.researchGoal
|
||
}
|
||
break
|
||
}
|
||
|
||
case 'generated_query': {
|
||
break
|
||
}
|
||
|
||
case 'searching': {
|
||
break
|
||
}
|
||
|
||
case 'search_complete': {
|
||
if (node) {
|
||
node.visitedUrls = step.urls
|
||
}
|
||
break
|
||
}
|
||
|
||
case 'processing_serach_result': {
|
||
if (node) {
|
||
node.learnings = step.result.learnings || []
|
||
node.followUpQuestions = step.result.followUpQuestions || []
|
||
}
|
||
break
|
||
}
|
||
|
||
case 'processed_search_result': {
|
||
if (node) {
|
||
node.learnings = step.result.learnings
|
||
searchResults.value[nodeId] = step.result
|
||
}
|
||
break
|
||
}
|
||
|
||
case 'error':
|
||
console.error(`Research error on node ${nodeId}:`, step.message)
|
||
break
|
||
|
||
case 'complete':
|
||
emit('complete', step)
|
||
break
|
||
}
|
||
}
|
||
|
||
// 辅助函数:根据节点ID查找节点
|
||
function findNode(root: TreeNode, targetId: string): TreeNode | null {
|
||
if (!targetId) return null
|
||
if (root.id === targetId) {
|
||
return root
|
||
}
|
||
for (const child of root.children) {
|
||
const found = findNode(child, targetId)
|
||
if (found) {
|
||
return found
|
||
}
|
||
}
|
||
return null
|
||
}
|
||
|
||
function selectNode(node: TreeNode) {
|
||
if (selectedNode.value?.id === node.id) {
|
||
selectedNode.value = undefined
|
||
} else {
|
||
selectedNode.value = node
|
||
}
|
||
}
|
||
|
||
// 辅助函数:获取节点的父节点 ID
|
||
function getParentNodeId(nodeId: string): string {
|
||
const parts = nodeId.split('-')
|
||
parts.pop()
|
||
return parts.join('-')
|
||
}
|
||
|
||
async function startResearch(query: string, depth: number, breadth: number) {
|
||
tree.value.children = []
|
||
selectedNode.value = undefined
|
||
searchResults.value = {}
|
||
try {
|
||
await deepResearch({
|
||
query,
|
||
maxDepth: depth,
|
||
breadth,
|
||
onProgress: handleResearchProgress,
|
||
})
|
||
} catch (error) {
|
||
console.error('Research failed:', error)
|
||
}
|
||
}
|
||
|
||
defineExpose({
|
||
startResearch,
|
||
})
|
||
</script>
|
||
|
||
<template>
|
||
<UCard>
|
||
<template #header>
|
||
<h2 class="font-bold">3. Web Browsing</h2>
|
||
<p class="text-sm text-gray-500">
|
||
The AI will then search the web based on our research goal, and iterate until the depth is reached.
|
||
<br />
|
||
Click a child node to view details.
|
||
</p>
|
||
</template>
|
||
<div class="flex flex-col">
|
||
<div class="overflow-y-auto">
|
||
<Tree :node="tree" :selected-node="selectedNode" @select="selectNode" />
|
||
</div>
|
||
<div v-if="selectedNode" class="p-4">
|
||
<USeparator label="Node Details" />
|
||
<h2 class="text-xl font-bold mt-2">{{ selectedNode.label }}</h2>
|
||
|
||
<!-- Root node has no additional information -->
|
||
<p v-if="selectedNode.id === '0'"> This is the beginning of your deep research journey! </p>
|
||
<template v-else>
|
||
<h3 class="text-lg font-semibold mt-2">Research Goal:</h3>
|
||
<p>{{ selectedNode.researchGoal }}</p>
|
||
|
||
<h3 class="text-lg font-semibold mt-2">Visited URLs:</h3>
|
||
<ul class="list-disc list-inside">
|
||
<li v-for="(url, index) in selectedNode.visitedUrls" :key="index">
|
||
<ULink :href="url" target="_blank">{{ url }}</ULink>
|
||
</li>
|
||
</ul>
|
||
|
||
<h3 class="text-lg font-semibold mt-2">Learnings:</h3>
|
||
<ul class="list-disc list-inside">
|
||
<li v-for="(learning, index) in selectedNode.learnings" :key="index">{{ learning }}</li>
|
||
</ul>
|
||
</template>
|
||
</div>
|
||
</div>
|
||
</UCard>
|
||
</template>
|