mirror of
https://github.com/LifeArchiveProject/WeChatDataAnalysis.git
synced 2026-02-19 22:30:49 +08:00
- 新增后台 autosync:监听 db_storage 变化后触发 realtime -> decrypted 增量同步(去抖/限频)\n- 优化 WCDB realtime 关闭:支持锁超时,避免 busy 时强行 shutdown\n- 新增消息搜索索引相关接口(status/build/senders)\n- 前端关闭 realtime 前改为 sync_all,减少切回解密库后的列表/消息落后\n- 增加解密库消息表/索引创建相关单测
381 lines
16 KiB
JavaScript
381 lines
16 KiB
JavaScript
// API请求组合式函数
|
||
export const useApi = () => {
|
||
const config = useRuntimeConfig()
|
||
|
||
// 基础请求函数
|
||
const request = async (url, options = {}) => {
|
||
try {
|
||
// 在客户端使用完整的API路径
|
||
const baseURL = process.client ? 'http://localhost:8000/api' : '/api'
|
||
|
||
const response = await $fetch(url, {
|
||
baseURL,
|
||
...options,
|
||
onResponseError({ response }) {
|
||
if (response.status === 400) {
|
||
throw new Error(response._data?.detail || '请求参数错误')
|
||
} else if (response.status === 500) {
|
||
throw new Error('服务器错误,请稍后重试')
|
||
}
|
||
}
|
||
})
|
||
return response
|
||
} catch (error) {
|
||
console.error('API请求错误:', error)
|
||
throw error
|
||
}
|
||
}
|
||
|
||
// 微信检测API
|
||
const detectWechat = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.data_root_path) {
|
||
query.set('data_root_path', params.data_root_path)
|
||
}
|
||
const url = '/wechat-detection' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 检测当前登录账号API
|
||
const detectCurrentAccount = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.data_root_path) {
|
||
query.set('data_root_path', params.data_root_path)
|
||
}
|
||
const url = '/current-account' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 数据库解密API
|
||
const decryptDatabase = async (data) => {
|
||
return await request('/decrypt', {
|
||
method: 'POST',
|
||
body: data
|
||
})
|
||
}
|
||
|
||
// 健康检查API
|
||
const healthCheck = async () => {
|
||
return await request('/health')
|
||
}
|
||
|
||
const listChatAccounts = async () => {
|
||
return await request('/chat/accounts')
|
||
}
|
||
|
||
const listChatSessions = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.include_hidden != null) query.set('include_hidden', String(!!params.include_hidden))
|
||
if (params && params.include_official != null) query.set('include_official', String(!!params.include_official))
|
||
if (params && params.source) query.set('source', params.source)
|
||
const url = '/chat/sessions' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const listChatMessages = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.offset != null) query.set('offset', String(params.offset))
|
||
if (params && params.order) query.set('order', params.order)
|
||
if (params && params.render_types) query.set('render_types', params.render_types)
|
||
if (params && params.source) query.set('source', params.source)
|
||
const url = '/chat/messages' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const getChatRealtimeStatus = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
const url = '/chat/realtime/status' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const syncChatRealtimeMessages = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.max_scan != null) query.set('max_scan', String(params.max_scan))
|
||
if (params && params.backfill_limit != null) query.set('backfill_limit', String(params.backfill_limit))
|
||
const url = '/chat/realtime/sync' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url, { method: 'POST' })
|
||
}
|
||
|
||
const syncChatRealtimeAll = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.max_scan != null) query.set('max_scan', String(params.max_scan))
|
||
if (params && params.priority_username) query.set('priority_username', params.priority_username)
|
||
if (params && params.priority_max_scan != null) query.set('priority_max_scan', String(params.priority_max_scan))
|
||
if (params && params.include_hidden != null) query.set('include_hidden', String(!!params.include_hidden))
|
||
if (params && params.include_official != null) query.set('include_official', String(!!params.include_official))
|
||
const url = '/chat/realtime/sync_all' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url, { method: 'POST' })
|
||
}
|
||
|
||
const searchChatMessages = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.q) query.set('q', params.q)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.sender) query.set('sender', params.sender)
|
||
if (params && params.session_type) query.set('session_type', params.session_type)
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.offset != null) query.set('offset', String(params.offset))
|
||
if (params && params.start_time != null) query.set('start_time', String(params.start_time))
|
||
if (params && params.end_time != null) query.set('end_time', String(params.end_time))
|
||
if (params && params.render_types) query.set('render_types', params.render_types)
|
||
if (params && params.include_hidden != null) query.set('include_hidden', String(!!params.include_hidden))
|
||
if (params && params.include_official != null) query.set('include_official', String(!!params.include_official))
|
||
if (params && params.session_limit != null) query.set('session_limit', String(params.session_limit))
|
||
if (params && params.per_chat_scan != null) query.set('per_chat_scan', String(params.per_chat_scan))
|
||
if (params && params.scan_limit != null) query.set('scan_limit', String(params.scan_limit))
|
||
const url = '/chat/search' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const listChatSearchSenders = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.session_type) query.set('session_type', params.session_type)
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.q) query.set('q', params.q)
|
||
if (params && params.message_q) query.set('message_q', params.message_q)
|
||
if (params && params.start_time != null) query.set('start_time', String(params.start_time))
|
||
if (params && params.end_time != null) query.set('end_time', String(params.end_time))
|
||
if (params && params.render_types) query.set('render_types', params.render_types)
|
||
if (params && params.include_hidden != null) query.set('include_hidden', String(!!params.include_hidden))
|
||
if (params && params.include_official != null) query.set('include_official', String(!!params.include_official))
|
||
const url = '/chat/search-index/senders' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const getChatSearchIndexStatus = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
const url = '/chat/search-index/status' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
const buildChatSearchIndex = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.rebuild != null) query.set('rebuild', String(!!params.rebuild))
|
||
const url = '/chat/search-index/build' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url, { method: 'POST' })
|
||
}
|
||
|
||
const getChatMessagesAround = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.anchor_id) query.set('anchor_id', params.anchor_id)
|
||
if (params && params.before != null) query.set('before', String(params.before))
|
||
if (params && params.after != null) query.set('after', String(params.after))
|
||
const url = '/chat/messages/around' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 朋友圈时间线
|
||
const listSnsTimeline = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.offset != null) query.set('offset', String(params.offset))
|
||
if (params && params.usernames && Array.isArray(params.usernames) && params.usernames.length > 0) {
|
||
query.set('usernames', params.usernames.join(','))
|
||
} else if (params && params.usernames && typeof params.usernames === 'string') {
|
||
query.set('usernames', params.usernames)
|
||
}
|
||
if (params && params.keyword) query.set('keyword', params.keyword)
|
||
const url = '/sns/timeline' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 朋友圈图片本地缓存候选(用于错图时手动选择)
|
||
const listSnsMediaCandidates = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.create_time != null) query.set('create_time', String(params.create_time))
|
||
if (params && params.width != null) query.set('width', String(params.width))
|
||
if (params && params.height != null) query.set('height', String(params.height))
|
||
if (params && params.limit != null) query.set('limit', String(params.limit))
|
||
if (params && params.offset != null) query.set('offset', String(params.offset))
|
||
const url = '/sns/media_candidates' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 保存朋友圈图片手动匹配结果(本机)
|
||
const saveSnsMediaPicks = async (data = {}) => {
|
||
return await request('/sns/media_picks', {
|
||
method: 'POST',
|
||
body: {
|
||
account: data.account || null,
|
||
picks: (data && data.picks && typeof data.picks === 'object' && !Array.isArray(data.picks)) ? data.picks : {}
|
||
}
|
||
})
|
||
}
|
||
|
||
const openChatMediaFolder = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
if (params && params.username) query.set('username', params.username)
|
||
if (params && params.kind) query.set('kind', params.kind)
|
||
if (params && params.md5) query.set('md5', params.md5)
|
||
if (params && params.file_id) query.set('file_id', params.file_id)
|
||
if (params && params.server_id != null) query.set('server_id', String(params.server_id))
|
||
const url = '/chat/media/open_folder' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url, { method: 'POST' })
|
||
}
|
||
|
||
const downloadChatEmoji = async (data = {}) => {
|
||
return await request('/chat/media/emoji/download', {
|
||
method: 'POST',
|
||
body: {
|
||
account: data.account || null,
|
||
md5: data.md5 || '',
|
||
emoji_url: data.emoji_url || '',
|
||
force: !!data.force
|
||
}
|
||
})
|
||
}
|
||
|
||
// 保存图片解密密钥
|
||
const saveMediaKeys = async (params = {}) => {
|
||
return await request('/media/keys', {
|
||
method: 'POST',
|
||
body: {
|
||
account: params.account || null,
|
||
xor_key: params.xor_key || '',
|
||
aes_key: params.aes_key || null
|
||
}
|
||
})
|
||
}
|
||
|
||
// 获取已保存的密钥(数据库密钥 + 图片密钥)
|
||
const getSavedKeys = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.account) query.set('account', params.account)
|
||
const url = '/keys' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// 批量解密所有图片
|
||
const decryptAllMedia = async (params = {}) => {
|
||
return await request('/media/decrypt_all', {
|
||
method: 'POST',
|
||
body: {
|
||
account: params.account || null,
|
||
xor_key: params.xor_key || null,
|
||
aes_key: params.aes_key || null
|
||
}
|
||
})
|
||
}
|
||
|
||
// 聊天记录导出(离线zip)
|
||
const createChatExport = async (data = {}) => {
|
||
return await request('/chat/exports', {
|
||
method: 'POST',
|
||
body: {
|
||
account: data.account || null,
|
||
scope: data.scope || 'selected',
|
||
usernames: Array.isArray(data.usernames) ? data.usernames : [],
|
||
format: data.format || 'json',
|
||
start_time: data.start_time != null ? Number(data.start_time) : null,
|
||
end_time: data.end_time != null ? Number(data.end_time) : null,
|
||
include_hidden: !!data.include_hidden,
|
||
include_official: !!data.include_official,
|
||
message_types: Array.isArray(data.message_types) ? data.message_types : [],
|
||
include_media: data.include_media == null ? true : !!data.include_media,
|
||
media_kinds: Array.isArray(data.media_kinds) ? data.media_kinds : ['image', 'emoji', 'video', 'video_thumb', 'voice', 'file'],
|
||
allow_process_key_extract: !!data.allow_process_key_extract,
|
||
privacy_mode: !!data.privacy_mode,
|
||
file_name: data.file_name || null
|
||
}
|
||
})
|
||
}
|
||
|
||
const getChatExport = async (exportId) => {
|
||
if (!exportId) throw new Error('Missing exportId')
|
||
return await request(`/chat/exports/${encodeURIComponent(String(exportId))}`)
|
||
}
|
||
|
||
const listChatExports = async () => {
|
||
return await request('/chat/exports')
|
||
}
|
||
|
||
const cancelChatExport = async (exportId) => {
|
||
if (!exportId) throw new Error('Missing exportId')
|
||
return await request(`/chat/exports/${encodeURIComponent(String(exportId))}`, { method: 'DELETE' })
|
||
}
|
||
|
||
// WeChat Wrapped(年度总结)
|
||
const getWrappedAnnual = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.year != null) query.set('year', String(params.year))
|
||
if (params && params.account) query.set('account', String(params.account))
|
||
if (params && params.refresh != null) query.set('refresh', String(!!params.refresh))
|
||
const url = '/wrapped/annual' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// WeChat Wrapped(年度总结)- 目录/元信息(轻量,用于按页懒加载)
|
||
const getWrappedAnnualMeta = async (params = {}) => {
|
||
const query = new URLSearchParams()
|
||
if (params && params.year != null) query.set('year', String(params.year))
|
||
if (params && params.account) query.set('account', String(params.account))
|
||
if (params && params.refresh != null) query.set('refresh', String(!!params.refresh))
|
||
const url = '/wrapped/annual/meta' + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
// WeChat Wrapped(年度总结)- 单张卡片(按页加载)
|
||
const getWrappedAnnualCard = async (cardId, params = {}) => {
|
||
if (cardId == null) throw new Error('Missing cardId')
|
||
const query = new URLSearchParams()
|
||
if (params && params.year != null) query.set('year', String(params.year))
|
||
if (params && params.account) query.set('account', String(params.account))
|
||
if (params && params.refresh != null) query.set('refresh', String(!!params.refresh))
|
||
const safeId = encodeURIComponent(String(cardId))
|
||
const url = `/wrapped/annual/cards/${safeId}` + (query.toString() ? `?${query.toString()}` : '')
|
||
return await request(url)
|
||
}
|
||
|
||
return {
|
||
detectWechat,
|
||
detectCurrentAccount,
|
||
decryptDatabase,
|
||
healthCheck,
|
||
listChatAccounts,
|
||
listChatSessions,
|
||
listChatMessages,
|
||
getChatRealtimeStatus,
|
||
syncChatRealtimeMessages,
|
||
syncChatRealtimeAll,
|
||
searchChatMessages,
|
||
getChatSearchIndexStatus,
|
||
buildChatSearchIndex,
|
||
listChatSearchSenders,
|
||
getChatMessagesAround,
|
||
listSnsTimeline,
|
||
listSnsMediaCandidates,
|
||
saveSnsMediaPicks,
|
||
openChatMediaFolder,
|
||
downloadChatEmoji,
|
||
saveMediaKeys,
|
||
getSavedKeys,
|
||
decryptAllMedia,
|
||
createChatExport,
|
||
getChatExport,
|
||
listChatExports,
|
||
cancelChatExport,
|
||
getWrappedAnnual,
|
||
getWrappedAnnualMeta,
|
||
getWrappedAnnualCard
|
||
}
|
||
}
|