import base from './base.js' import fetch from 'node-fetch' import lodash from 'lodash' import puppeteer from '../../../lib/puppeteer/puppeteer.js' import common from '../../../lib/common/common.js' import { segment } from 'icqq' import gsCfg from '../model/gsCfg.js' const _path = process.cwd() let emoticon export default class MysNews extends base { constructor (e) { super(e) this.model = 'mysNews' } async getNews () { let type = 1 let typeName = '公告' if (this.e.msg.includes('资讯')) { type = '3' typeName = '资讯' } if (this.e.msg.includes('活动')) { type = '2' typeName = '活动' } const res = await this.postData('getNewsList', { gids: 2, page_size: 20, type }) if (!res) return const data = res.data.list if (data.length == 0) { return true } const page = this.e.msg.replace(/#|#|官方|原神|公告|资讯|活动/g, '').trim() || 1 if (page > data.length) { await this.e.reply('目前只查前20条最新的公告,请输入1-20之间的整数。') return true } const postId = data[page - 1].post.post_id const param = await this.newsDetail(postId) const img = await this.rander(param) return await this.replyMsg(img, `原神${typeName}:${param.data.post.subject}`) } async rander (param) { const pageHeight = 7000 await puppeteer.browserInit() if (!puppeteer.browser) return false const savePath = puppeteer.dealTpl('mysNews', param) if (!savePath) return false const page = await puppeteer.browser.newPage() try { await page.goto(`file://${_path}${lodash.trim(savePath, '.')}`, { timeout: 120000 }) const body = await page.$('#container') || await page.$('body') const boundingBox = await body.boundingBox() const num = Math.round(boundingBox.height / pageHeight) || 1 if (num > 1) { await page.setViewport({ width: boundingBox.width, height: pageHeight + 100 }) } const img = [] for (let i = 1; i <= num; i++) { const randData = { type: 'jpeg', quality: 90 } if (i != 1 && i == num) { await page.setViewport({ width: boundingBox.width, height: parseInt(boundingBox.height) - pageHeight * (num - 1) }) } if (i != 1 && i <= num) { await page.evaluate(() => window.scrollBy(0, 7000)) } let buff if (num == 1) { buff = await body.screenshot(randData) } else { buff = await page.screenshot(randData) } if (num > 2) await common.sleep(200) puppeteer.renderNum++ /** 计算图片大小 */ const kb = (buff.length / 1024).toFixed(2) + 'kb' logger.mark(`[图片生成][${this.model}][${puppeteer.renderNum}次] ${kb}`) img.push(segment.image(buff)) } await page.close().catch((err) => logger.error(err)) if (num > 1) { logger.mark(`[图片生成][${this.model}] 处理完成`) } return img } catch (error) { logger.error(`图片生成失败:${this.model}:${error}`) /** 关闭浏览器 */ if (puppeteer.browser) { await puppeteer.browser.close().catch((err) => logger.error(err)) } puppeteer.browser = false } } async newsDetail (postId) { const res = await this.postData('getPostFull', { gids: 2, read: 1, post_id: postId }) if (!res) return const data = await this.detalData(res.data.post) return { ...this.screenData, saveId: postId, dataConent: data.post.content, data } } postApi (type, data) { let host = 'https://bbs-api-static.mihoyo.com/' let param = [] lodash.forEach(data, (v, i) => param.push(`${i}=${v}`)) param = param.join('&') switch (type) { // 搜索 case 'searchPosts': host = 'https://bbs-api.mihoyo.com/post/wapi/searchPosts?' break // 帖子详情 case 'getPostFull': host += 'post/wapi/getPostFull?' break // 公告列表 case 'getNewsList': host += 'post/wapi/getNewsList?' break case 'emoticon': host += 'misc/api/emoticon_set?' break } return host + param } async postData (type, data) { const url = this.postApi(type, data) const headers = { Referer: 'https://bbs.mihoyo.com/', 'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36' } let response try { response = await fetch(url, { method: 'get', headers }) } catch (error) { logger.error(error.toString()) return false } if (!response.ok) { logger.error(`[米游社接口错误][${type}] ${response.status} ${response.statusText}`) return false } const res = await response.json() return res } async detalData (data) { let json try { json = JSON.parse(data.post.content) } catch (error) { } if (typeof json == 'object') { if (json.imgs && json.imgs.length > 0) { for (const val of json.imgs) { data.post.content = `
` } } } else { for (const img of data.post.images) { data.post.content = data.post.content.replace(img, img + '?x-oss-process=image//resize,s_600/quality,q_80/auto-orient,0/interlace,1/format,jpg') } if (!emoticon) { emoticon = await this.mysEmoticon() } data.post.content = data.post.content.replace(/_\([^)]*\)/g, function (t, e) { t = t.replace(/_\(|\)/g, '') if (emoticon.has(t)) { return `` } else { return '' } }) const arrEntities = { lt: '<', gt: '>', nbsp: ' ', amp: '&', quot: '"' } data.post.content = data.post.content.replace(/&(lt|gt|nbsp|amp|quot);/ig, function (all, t) { return arrEntities[t] }) } data.post.created_time = new Date(data.post.created_at * 1000).toLocaleString() for (const i in data.stat) { data.stat[i] = data.stat[i] > 10000 ? (data.stat[i] / 10000).toFixed(2) + '万' : data.stat[i] } return data } async mysEmoticon () { const emp = new Map() const res = await this.postData('emoticon', { gids: 2 }) if (res.retcode != 0) { return emp } for (const val of res.data.list) { if (!val.icon) continue for (const list of val.list) { if (!list.icon) continue emp.set(list.name, list.icon) } } return emp } async mysSearch () { let msg = this.e.msg msg = msg.replace(/#|米游社|mys/g, '') if (!msg) { await this.e.reply('请输入关键字,如#米游社七七') return false } let page = msg.match(/.*(\d){1}$/) || 0 if (page && page[1]) { page = page[1] } msg = lodash.trim(msg, page) let res = await this.postData('searchPosts', { gids: 2, size: 20, keyword: msg }) if (!res) return if (res?.data?.posts.length <= 0) { await this.e.reply('搜索不到您要的结果,换个关键词试试呗~') return false } let postId = res.data.posts[page].post.post_id const param = await this.newsDetail(postId) const img = await this.rander(param) return await this.replyMsg(img, `${param.data.post.subject}`) } async mysUrl () { let msg = this.e.msg let postId = /[0-9]+/g.exec(msg)[0] if (!postId) return false const param = await this.newsDetail(postId) const img = await this.rander(param) return await this.replyMsg(img, `${param.data.post.subject}`) } async ysEstimate () { let msg = '版本原石盘点' let res = await this.postData('searchPosts', { gids: 2, size: 20, keyword: msg }) if (res?.data?.posts.length <= 0) { await this.e.reply('暂无数据') return false } let postId = '' for (let post of res.data.posts) { if (post.user.uid == '218945821') { postId = post.post.post_id break } } if (!postId) { await this.e.reply('暂无数据') return false } const param = await this.newsDetail(postId) const img = await this.rander(param) if (img.length > 1) { img.push(segment.image(param.data.post.images[0] + '?x-oss-process=image//resize,s_600/quality,q_80/auto-orient,0/interlace,1/format,jpg')) } return await this.replyMsg(img, `${param.data.post.subject}`) } async replyMsg (img, titile) { if (!img || img.length <= 0) return false if (img.length == 1) { return img[0] } else { let msg = [titile, ...img] return await common.makeForwardMsg(this.e, msg, titile).catch((err) => { logger.error(err) }) } } async mysNewsTask (type = 1) { let cfg = gsCfg.getConfig('mys', 'pushNews') // 推送2小时内的公告资讯 let interval = 7200 // 最多同时推送两条 this.maxNum = cfg.maxNum // 包含关键字不推送 let banWord = /冒险助力礼包|纪行|预下载|脚本外挂|集中反馈|已开奖|云·原神|魔神任务|传说任务说明/g let anno = await this.postData('getNewsList', { gids: 2, page_size: 10, type: 1 }) let info = await this.postData('getNewsList', { gids: 2, page_size: 10, type: 3 }) let news = [] if (anno) anno.data.list.forEach(v => { news.push({ ...v, typeName: '公告', post_id: v.post.post_id }) }) if (info) info.data.list.forEach(v => { news.push({ ...v, typeName: '资讯', post_id: v.post.post_id }) }) if (news.length <= 0) return news = lodash.orderBy(news, ['post_id'], ['asc']) let now = Date.now() / 1000 this.key = 'Yz:genshin:mys:newPush:' this.e.isGroup = true this.pushGroup = [] for (let val of news) { if (Number(now - val.post.created_at) > interval) { continue } if (new RegExp(banWord).test(val.post.subject)) { continue } if (val.typeName == '公告') { for (let groupId of cfg.announceGroup) { await this.sendNews(groupId, val.typeName, val.post.post_id) } } if (val.typeName == '资讯') { for (let groupId of cfg.infoGroup) { await this.sendNews(groupId, val.typeName, val.post.post_id) } } } } async sendNews (groupId, typeName, postId) { if (!this.pushGroup[groupId]) this.pushGroup[groupId] = 0 if (this.pushGroup[groupId] >= this.maxNum) return let sended = await redis.get(`${this.key}${groupId}:${postId}`) if (sended) return if (!this[postId]) { const param = await this.newsDetail(postId) logger.mark(`[米游社${typeName}推送] ${param.data.post.subject}`) this[postId] = { img: await this.rander(param), title: param.data.post.subject } } this.pushGroup[groupId]++ this.e.group = Bot.pickGroup(Number(groupId)) this.e.group_id = Number(groupId) let tmp = await this.replyMsg(this[postId].img, `原神${typeName}推送:${this[postId].title}`) await common.sleep(1000) if (!tmp) return if (tmp?.type != 'xml') { tmp = [`原神${typeName}推送\n`, tmp] } redis.set(`${this.key}${groupId}:${postId}`, '1', { EX: 3600 * 10 }) await this.e.group.sendMsg(tmp) } }