const art = require('art-template'); const path = require('path'); const config = require('@/config'); const typeRegrx = /\.(atom|rss)$/; const unsupportedRegrx = /\.json$/; module.exports = async (ctx, next) => { if (ctx.request.path.match(unsupportedRegrx)) { throw Error('JSON output had been removed, see: https://github.com/DIYgod/RSSHub/issues/1114'); } ctx.state.type = ctx.request.path.match(typeRegrx) || ['', '']; ctx.request.path = ctx.request.path.replace(typeRegrx, ''); await next(); if (!ctx.body) { let template; switch (ctx.state.type[1]) { case 'atom': template = path.resolve(__dirname, '../views/atom.art'); break; case 'rss': template = path.resolve(__dirname, '../views/rss.art'); break; default: template = path.resolve(__dirname, '../views/rss.art'); break; } if (ctx.state.data) { ctx.state.data.item && ctx.state.data.item.forEach((item) => { if (item.title) { item.title = item.title.trim(); // trim title length for (let length = 0, i = 0; i < item.title.length; i++) { length += Buffer.from(item.title[i]).length !== 1 ? 2 : 1; if (length > config.titleLengthLimit) { item.title = `${item.title.slice(0, i)}...`; break; } } } if (item.enclosure_length) { const itunes_duration = Math.floor(item.enclosure_length / 3600) + ':' + Math.floor((item.enclosure_length % 3600) / 60) + ':' + (((item.enclosure_length % 3600) % 60) / 100).toFixed(2).slice(-2); item.itunes_duration = itunes_duration; } }); } const data = { lastBuildDate: new Date().toUTCString(), updated: new Date().toISOString(), ttl: config.cache.routeExpire, ...ctx.state.data, }; if (template) { ctx.body = art(template, data); } } };