const { parseDate } = require('@/utils/parse-date'); const got = require('@/utils/got'); const cheerio = require('cheerio'); const config = require('@/config').value; const videoAPI = 'https://api.ruguoapp.com/1.0/mediaMeta/play?type=ORIGINAL_POST'; module.exports = { topicDataHanding: (data, ctx) => data.posts.map((item) => { let audioName, videoName, linkName; // 获取纯文字内容 和 即刻原文链接 let content, link; switch (item.type) { case 'ORIGINAL_POST': content = item.content; link = `https://m.okjike.com/originalPosts/${item.id}`; break; /* case 'QUESTION': content = item.title; link = `https://m.okjike.com/questions/${item.id}`; break; case 'OFFICIAL_MESSAGE': content = item.content; link = `https://web.okjike.com/message-detail/${item.id}/officialMessage`; break;*/ default: content = '未知类型,请前往GitHub提交issue'; link = 'https://github.com/DIYgod/RSSHub/issues'; } // rss内容 let description = ''; // 作者昵称 let author = ''; // 添加内容作者信息 if (item.user) { author = item.user.screenName; if (ctx.params.showUid) { description += `用户昵称:${author}
Username:${item.user.username}

`; } } if (item.linkInfo) { const linkUrl = item.linkInfo.originalLinkUrl || item.linkInfo.linkUrl; // 对于即刻抓取的微信公众号文章 特殊处理 // 此时 Rss原文链接 变为 微信公众号链接 if (new URL(linkUrl).host === 'mp.weixin.qq.com') { link = linkUrl; } // 1. 音频 const audioObject = item.linkInfo.audio || item.audio; if (audioObject) { const audioImage = audioObject.image.picUrl || audioObject.image.thumbnailUrl; const audioLink = linkUrl; const audioTitle = audioObject.title; const audioAuthor = audioObject.author; audioName = `${audioTitle} - ${audioAuthor}`; description += ` ${audioName} `; } // 2. 视频 const videoObject = item.linkInfo.video || item.video; if (videoObject) { const videoImage = videoObject.image.picUrl || videoObject.image.thumbnailUrl; const videoLink = linkUrl; const videoDuration = Math.floor(videoObject.duration / 60000); videoName = item.linkInfo.title; description += ` ${videoName || '观看视频'} - 约${videoDuration}分钟 `; } // 3. 链接 if (!audioObject && !videoObject && linkUrl) { // 部分链接有标题 linkName = item.linkInfo.title; const linkTitle = linkName || '访问原文'; // 部分链接有缩略图 const linkImage = item.linkInfo.pictureUrl; const imageTag = ``; description += ` ${linkImage ? imageTag : ''} ${linkTitle} `; } } // 4. 文字内容 description += description ? `
${content}` : content; // 5. 图片 if (item.pictures) { item.pictures.forEach((pic) => { if (pic.format === 'gif') { description += ``; } else { // jpeg, bmp, png, gif, webp // https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types // let type = 'jpeg'; // switch (pic.format) { // case 'bmp': // type = 'bmp'; // break; // case 'png': // type = 'png'; // break; // default: // break; // } const imgUrl = pic.picUrl.match(/\.[a-z0-9]+?\?imageMogr2/) ? pic.picUrl.split('?imageMogr2/')[0] : pic.picUrl.replace(/thumbnail\/.+/, ''); description += `
`; // description += `
`; } }); } // 6. 视频 if (item.video) { description += `
`; } // rss标题 // 优先将音频和视频名作为标题 // 其次将正文内容作为标题 // 若都没有 则是推送型消息,将连接标题作为主题 // “无题” fallback const title = audioName || videoName || content || linkName || '无题'; return { title, description: description.trim().replace(/\n/g, '
'), pubDate: parseDate(item.createdAt), author, link, }; }), constructTopicEntry: async (ctx, url) => { const data = await ctx.cache.tryGet( url, async () => { const resp = await got(url); const html = resp.data; const $ = cheerio.load(html); const raw = $('[type = "application/json"]').html(); const data = JSON.parse(raw).props.pageProps; data.posts = await Promise.all( data.posts.map(async (item) => { if (!item.video) { return item; } const videoUrl = `${videoAPI}&id=${item.id}`; const videoRes = await got(videoUrl); item.video = videoRes.data; return item; }) ); return data; }, false, config.cache.routeExpire ); if (data.length === 0) { ctx.state.data = { title: '主题 ID 不存在,或该主题暂无内容', }; return null; } const topic = data.topic; ctx.state.data = { title: `${topic.content} - 即刻圈子`, link: url, description: topic.briefIntro, image: topic.squarePicture.picUrl || topic.squarePicture.middlePicUrl || topic.squarePicture.thumbnailUrl, // item: topicDataHanding(data), }; return data; }, };