gd-utils/src/tg.js

260 lines
11 KiB
JavaScript
Raw Normal View History

2020-06-27 14:19:43 +08:00
const Table = require('cli-table3')
const dayjs = require('dayjs')
const axios = require('@viegg/axios')
const HttpsProxyAgent = require('https-proxy-agent')
const { db } = require('../db')
2020-07-04 01:12:20 +08:00
const { gen_count_body, validate_fid, real_copy, get_name_by_id } = require('./gd')
2020-06-27 14:19:43 +08:00
const { AUTH, DEFAULT_TARGET } = require('../config')
const { tg_token } = AUTH
2020-07-04 01:12:20 +08:00
const gen_link = (fid, text) => `<a href="https://drive.google.com/drive/folders/${fid}">${text || fid}</a>`
2020-06-27 14:19:43 +08:00
if (!tg_token) throw new Error('请先在auth.js里设置tg_token')
const { https_proxy } = process.env
const axins = axios.create(https_proxy ? { httpsAgent: new HttpsProxyAgent(https_proxy) } : {})
2020-07-04 01:12:20 +08:00
const FID_TO_NAME = {}
async function get_folder_name (fid) {
let name = FID_TO_NAME[fid]
if (name) return name
name = await get_name_by_id(fid)
return FID_TO_NAME[fid] = name
}
module.exports = { send_count, send_help, sm, extract_fid, reply_cb_query, send_choice, send_task_info, send_all_tasks, tg_copy, extract_from_text }
2020-06-27 14:19:43 +08:00
function send_help (chat_id) {
const text = `<pre>[使用帮助]
命令 说明
/help | 使
2020-07-04 01:12:20 +08:00
/count shareID [-u] | sourceID, sourceIDgoogle driveID -u线
2020-06-27 14:19:43 +08:00
2020-07-04 01:12:20 +08:00
/copy sourceID targetID [-u] | sourceIDtargetIDtargetIDconfig.js -u线taskID
2020-06-27 14:19:43 +08:00
/task taskID | all
</pre>`
return sm({ chat_id, text, parse_mode: 'HTML' })
}
function send_choice ({ fid, chat_id }) {
return sm({
chat_id,
text: `识别出分享ID ${fid},请选择动作`,
reply_markup: {
inline_keyboard: [
[
{ text: '文件统计', callback_data: `count ${fid}` },
{ text: '开始复制', callback_data: `copy ${fid}` }
]
]
}
})
}
async function send_all_tasks (chat_id) {
let records = db.prepare('select id, status, ctime from task').all()
if (!records.length) return sm({ chat_id, text: '数据库中没有任务记录' })
const tb = new Table({ style: { head: [], border: [] } })
const headers = ['ID', 'status', 'ctime']
records = records.map(v => {
const { id, status, ctime } = v
2020-06-27 14:25:50 +08:00
return [id, status, dayjs(ctime).format('YYYY-MM-DD HH:mm:ss')]
2020-06-27 14:19:43 +08:00
})
tb.push(headers, ...records)
const text = tb.toString().replace(/─/g, '—')
const url = `https://api.telegram.org/bot${tg_token}/sendMessage`
return axins.post(url, {
chat_id,
parse_mode: 'HTML',
text: `所有拷贝任务:\n<pre>${text}</pre>`
2020-07-04 01:12:20 +08:00
}).catch(err => {
// const description = err.response && err.response.data && err.response.data.description
// if (description && description.includes('message is too long')) {
if (true) {
2020-06-27 14:19:43 +08:00
const text = [headers].concat(records).map(v => v.join('\t')).join('\n')
return sm({ chat_id, parse_mode: 'HTML', text: `所有拷贝任务:\n<pre>${text}</pre>` })
}
console.error(err)
})
}
2020-07-04 01:12:20 +08:00
async function get_task_info (task_id) {
2020-06-27 14:19:43 +08:00
const record = db.prepare('select * from task where id=?').get(task_id)
2020-07-04 01:12:20 +08:00
if (!record) return {}
2020-06-27 14:19:43 +08:00
const { source, target, status, copied, mapping, ctime, ftime } = record
2020-07-04 01:12:20 +08:00
const folder_mapping = mapping && mapping.trim().split('\n')
const new_folder = folder_mapping && folder_mapping[0].split(' ')[1]
2020-06-27 14:19:43 +08:00
const { summary } = db.prepare('select summary from gd where fid=?').get(source) || {}
const { file_count, folder_count, total_size } = summary ? JSON.parse(summary) : {}
const copied_files = copied ? copied.trim().split('\n').length : 0
2020-07-04 01:12:20 +08:00
const copied_folders = folder_mapping ? (folder_mapping.length - 1) : 0
2020-06-27 14:19:43 +08:00
let text = '任务编号:' + task_id + '\n'
2020-07-04 01:12:20 +08:00
const folder_name = await get_folder_name(source)
text += '源文件夹:' + gen_link(source, folder_name) + '\n'
text += '目的位置:' + gen_link(target) + '\n'
text += '新文件夹:' + (new_folder ? gen_link(new_folder) : '暂未创建') + '\n'
2020-06-27 14:19:43 +08:00
text += '任务状态:' + status + '\n'
2020-06-27 14:25:50 +08:00
text += '创建时间:' + dayjs(ctime).format('YYYY-MM-DD HH:mm:ss') + '\n'
text += '完成时间:' + (ftime ? dayjs(ftime).format('YYYY-MM-DD HH:mm:ss') : '未完成') + '\n'
2020-06-27 14:19:43 +08:00
text += '目录进度:' + copied_folders + '/' + (folder_count === undefined ? '未知数量' : folder_count) + '\n'
text += '文件进度:' + copied_files + '/' + (file_count === undefined ? '未知数量' : file_count) + '\n'
2020-07-04 01:12:20 +08:00
text += '合计大小:' + (total_size || '未知大小')
const total_count = (folder_count || 0) + (file_count || 0)
return { text, status, total_count }
2020-06-27 14:19:43 +08:00
}
2020-07-04 01:12:20 +08:00
async function send_task_info ({ task_id, chat_id }) {
const { text, status, total_count } = await get_task_info(task_id)
if (!text) return sm({ chat_id, text: '数据库不存在此任务ID' + task_id })
const url = `https://api.telegram.org/bot${tg_token}/sendMessage`
let message_id
try {
const { data } = await axins.post(url, { chat_id, text, parse_mode: 'HTML' })
message_id = data && data.result && data.result.message_id
} catch (e) {
console.log('fail to send message to tg', e.message)
}
// get_task_info 在task文件数超大时比较吃cpu如果超5万就不每10秒更新了
if (!message_id || status !== 'copying' || total_count > 50000) return
const loop = setInterval(async () => {
const url = `https://api.telegram.org/bot${tg_token}/editMessageText`
const { text, status } = await get_task_info(task_id)
if (status !== 'copying') clearInterval(loop)
axins.post(url, { chat_id, message_id, text, parse_mode: 'HTML' }).catch(e => console.error(e.message))
}, 10 * 1000)
}
async function tg_copy ({ fid, target, chat_id, update }) { // return task_id
2020-06-27 14:19:43 +08:00
target = target || DEFAULT_TARGET
if (!target) {
sm({ chat_id, text: '请输入目的地ID或先在config.js里设置默认复制目的地ID(DEFAULT_TARGET)' })
return
}
let record = db.prepare('select id, status from task where source=? and target=?').get(fid, target)
if (record) {
if (record.status === 'copying') {
sm({ chat_id, text: '已有相同源ID和目的ID的任务正在进行查询进度可输入 /task ' + record.id })
return
} else if (record.status === 'finished') {
2020-07-04 01:12:20 +08:00
sm({ chat_id, text: `检测到已存在的任务 ${record.id},开始继续拷贝` })
2020-06-27 14:19:43 +08:00
}
}
2020-07-04 01:12:20 +08:00
real_copy({ source: fid, update, target, not_teamdrive: true, service_account: true, is_server: true })
.then(async info => {
2020-06-27 14:19:43 +08:00
if (!record) record = {} // 防止无限循环
2020-07-04 01:12:20 +08:00
if (!info) return
const { task_id } = info
const row = db.prepare('select * from task where id=?').get(task_id)
const { source, target, status, copied, mapping, ctime, ftime } = row
const { summary } = db.prepare('select summary from gd where fid=?').get(source) || {}
const { file_count, folder_count, total_size } = summary ? JSON.parse(summary) : {}
const copied_files = copied ? copied.trim().split('\n').length : 0
const copied_folders = mapping ? (mapping.trim().split('\n').length - 1) : 0
let text = `任务 ${task_id} 复制完成\n`
const name = await get_folder_name(source)
text += '源文件夹:' + gen_link(source, name) + '\n'
text += '目录完成数:' + copied_folders + '/' + folder_count + '\n'
text += '文件完成数:' + copied_files + '/' + file_count + '\n'
sm({ chat_id, text, parse_mode: 'HTML' })
2020-06-27 14:19:43 +08:00
})
.catch(err => {
if (!record) record = {}
console.error('复制失败', fid, '-->', target)
console.error(err)
sm({ chat_id, text: '复制失败,失败消息:' + err.message })
})
while (!record) {
record = db.prepare('select id from task where source=? and target=?').get(fid, target)
await sleep(1000)
}
return record.id
}
function sleep (ms) {
return new Promise((resolve, reject) => {
setTimeout(resolve, ms)
})
}
function reply_cb_query ({ id, data }) {
const url = `https://api.telegram.org/bot${tg_token}/answerCallbackQuery`
return axins.post(url, {
callback_query_id: id,
text: '开始执行 ' + data
})
}
2020-07-04 01:12:20 +08:00
async function send_count ({ fid, chat_id, update }) {
const table = await gen_count_body({ fid, update, type: 'tg', service_account: true })
if (!table) return sm({ chat_id, parse_mode: 'HTML', text: gen_link(fid) + ' 信息获取失败' })
2020-06-27 14:19:43 +08:00
const url = `https://api.telegram.org/bot${tg_token}/sendMessage`
const gd_link = `https://drive.google.com/drive/folders/${fid}`
2020-07-04 01:12:20 +08:00
const name = await get_folder_name(fid)
2020-06-27 14:19:43 +08:00
return axins.post(url, {
chat_id,
parse_mode: 'HTML',
2020-07-04 01:12:20 +08:00
text: `<pre>源文件夹名称:${name}
源链接${gd_link}
2020-06-27 14:19:43 +08:00
${table}</pre>`
}).catch(async err => {
2020-07-04 01:12:20 +08:00
// const description = err.response && err.response.data && err.response.data.description
// const too_long_msgs = ['request entity too large', 'message is too long']
// if (description && too_long_msgs.some(v => description.toLowerCase().includes(v))) {
if (true) {
2020-06-27 14:19:43 +08:00
const smy = await gen_count_body({ fid, type: 'json', service_account: true })
const { file_count, folder_count, total_size } = JSON.parse(smy)
return sm({
chat_id,
parse_mode: 'HTML',
2020-07-04 01:12:20 +08:00
text: `链接:<a href="https://drive.google.com/drive/folders/${fid}">${fid}</a>\n<pre>
2020-06-27 14:19:43 +08:00
表格太长超出telegram消息限制只显示概要
2020-07-04 01:12:20 +08:00
目录名称${name}
2020-06-27 14:19:43 +08:00
文件总数${file_count}
目录总数${folder_count}
合计大小${total_size}
</pre>`
})
}
throw err
})
}
function sm (data) {
const url = `https://api.telegram.org/bot${tg_token}/sendMessage`
return axins.post(url, data).catch(err => {
2020-07-04 01:12:20 +08:00
// console.error('fail to post', url, data)
console.error('fail to send message to tg:', err.message)
2020-06-27 14:19:43 +08:00
})
}
function extract_fid (text) {
2020-07-04 01:12:20 +08:00
text = text.replace(/^\/count/, '').replace(/^\/copy/, '').replace(/\\/g, '').trim()
2020-06-27 14:19:43 +08:00
const [source, target] = text.split(' ').map(v => v.trim())
if (validate_fid(source)) return source
try {
if (!text.startsWith('http')) text = 'https://' + text
const u = new URL(text)
if (u.pathname.includes('/folders/')) {
2020-07-04 01:12:20 +08:00
const reg = /[^\/?]+$/
2020-06-27 14:19:43 +08:00
const match = u.pathname.match(reg)
2020-07-04 01:12:20 +08:00
return match && match[0]
2020-06-27 14:19:43 +08:00
}
return u.searchParams.get('id')
} catch (e) {
return ''
}
}
2020-07-04 01:12:20 +08:00
function extract_from_text (text) {
const reg = /https?:\/\/drive.google.com\/[^\s]+/g
const m = text.match(reg)
return m && extract_fid(m[0])
}