Join Us
We welcome all pull requests. Suggestions and feedback are also welcomed here (opens new window).
# Join the discussion
# Submit new RSS rule
Before you start writing an RSS rule, please make sure that the source site does not provide RSS. Some web pages will include a link element with type application/atom+xml
or application/rss+xml
in the HTML header to indicate the RSS link.
# Debug
First, yarn
or npm install
to install dependencies, then execute yarn dev
or npm run dev
, open http://localhost:1200
to see the effect, and the page will refresh automatically if files are modified.
# Add route
First, create the corresponding route path in /lib/v2/ (opens new window) and add the route in /lib/v2/:path/router.js
# Code the script
Create a new js script in the corresponding /lib/v2/ (opens new window) path
# Acquiring Data
Typically, the data are acquired via HTTP requests (via API or webpage) sent by got (opens new window)
Occasionally puppeteer (opens new window) is required for browser stimulation and page rendering in order to acquire the data
The acquired data are most likely in JSON or HTML format
For HTML format, cheerio (opens new window) is used for further processing
Below is a list of data acquisition methods, ordered by the 「level of recommendation」
- Acquire data via API using got
Example: /lib/routes/bilibili/coin.js (opens new window)。
Acquiring data via the official API provided by the data source using got:
// Initiate a HTTP GET request const response = await got({ method: 'get', url: `https://api.bilibili.com/x/space/coin/video?vmid=${uid}&jsonp=jsonp`, headers: { Referer: `https://space.bilibili.com/${uid}/`, }, }); const data = response.data.data; // response.data is the data object returned from the previous request // The object contains a nested object called data, thus response.data.data is the actual data needed here
One of the leaf objects (response.data.data[0]):
{ "aid": 33614333, "videos": 2, "tid": 20, "tname": "宅舞", "copyright": 1, "pic": "http://i0.hdslb.com/bfs/archive/5649d7fe6ff7f7b431300fc1a0db80d3f174cacd.jpg", "title": "【赤九玖】响喜乱舞【和我一起狂舞吧,团长大人(✧◡✧)】", "pubdate": 1539259203, "ctime": 1539249536, "desc": "编舞出处:av31984673\n真心好喜欢这个舞和这首歌,居然恰巧被邀请跳了,感谢《苍之纪元》官方的邀请。这次cos的是游戏的新角色缪斯。然而时间有限很多地方还有很多不足。也没跳够,以后私下还会继续练习,希望能学到更多动作,也能为了有机会把它跳的更好。 \n摄影:绯山圣瞳九命猫 \n后期:炉火" // some more data.... }
Processing the data further to generate objects in accordance with RSS specification, mainly title, link, description, publish time, then assign them to ctx.state.data, produce RSS feed:
ctx.state.data = { // the source title title: `${name} 的 bilibili 投币视频`, // the source link link: `https://space.bilibili.com/${uid}`, // the source description description: `${name} 的 bilibili 投币视频`, // iterate through all leaf objects item: data.map((item) => ({ // the article title title: item.title, // the article content description: `${item.desc}<br><img src="${item.pic}">`, // the article publish time pubDate: new Date(item.time * 1000).toUTCString(), // the article link link: `https://www.bilibili.com/video/av${item.aid}`, })), }; // the route is now done
- Acquire data via HTML webpage using got
Data have to be acquired via HTML webpage if no API was provided, for example: /lib/routes/douban/explore.js (opens new window)
Acquiring data by scrapping the HTML using got:
// Initiate a HTTP GET request const response = await got({ method: 'get', url: 'https://www.douban.com/explore', }); const data = response.data; // response.data is the entire HTML source of the target page, returned from the previous request
Parsing the HTML using cheerio:
const $ = cheerio.load(data); // Load the HTML returned into cheerio const list = $('div[data-item_id]'); // use cheerio selector, select all 'div' elements with 'data-item_id' attribute, the result is an array of cheerio node objects // use cheerio get() method to transform a cheerio node object array into a node array // PS: every cheerio node is a HTML DOM // PPS: cheerio selector is almost identical to jquery selector // Refer to cheerio docs: https://cheerio.js.org/
Use map to traverse the array and parse out the result of each item
ctx.state.data = { title: '豆瓣-浏览发现', link: 'https://www.douban.com/explore', item: list && list .map((index, item) => { item = $(item); itemPicUrl = item.find('a.cover').attr('style').replace('background-image:url(', '').replace(')', ''); return { title: item.find('.title a').first().text(), description: `作者:${item.find('.usr-pic a').last().text()}<br>描述:${item.find('.content p').text()}<br><img src="${itemPicUrl}">`, link: item.find('.title a').attr('href'), }; }) .get(), }; // the route is now done
- Acquire data via page rendering using puppeteer
Tips
This method consumes more resources and is less performant, use only when the above methods are failed to acquire data. Otherwise, your pull requests will be rejected!
Seldomly, data source provides no API and the page requires rendering to acquire data, for example: /lib/routes/sspai/series.js (opens new window)
// use puppeteer util class, initialise a browser instance const browser = await require('@/utils/puppeteer')(); // open a new page const page = await browser.newPage(); // access the target link const link = 'https://sspai.com/series'; await page.goto(link); // render the page const html = await page.evaluate( () => // process on the rendered page document.querySelector('div.new-series-wrapper').innerHTML ); // shutdown the browser browser.close();
Parsing the HTML using cheerio:
const $ = cheerio.load(html); // Load the HTML returned into cheerio const list = $('div.item'); // // use cheerio selector, select all 'div class="item"' elements, the result is an array of cheerio node objects
Assign the value to
ctx.state.data
ctx.state.data = { title: '少数派 -- 最新上架付费专栏', link, description: '少数派 -- 最新上架付费专栏', item: list .map((i, item) => ({ // the article title title: $(item).find('.item-title a').text().trim(), // the article link link: url.resolve(link, $(item).find('.item-title a').attr('href')), // the article author author: $(item).find('.item-author').text().trim(), })) .get(), // use cheerio get() method to transform a cheerio node object array into a node array }; // the route is now done // PS: the route acts as a notifier of new articles. It does not provide access to the content behind the paywall. Thus no content was fetched
- Use general configuration routing
A large number of websites can generate RSS through a configuration paradigm.
The general configuration is to generate RSS with ease by reading json data through cheerio (CSS selector, jQuery function)
First, we need a few data:
- RSS source link
- Data source link
- RSS title (not item title)
const buildData = require('@/utils/common-config'); module.exports = async (ctx) => { ctx.state.data = await buildData({ link: '', // RSS source link url: '', // Data source link title: '%title%', // Variables are used here, such as **% xxx%** will be parsed into variables with values of the same name under **params** params: { title: '', // RSS title }, }); };
Our RSS does not have any content for now. The content needs to be completed by
item
. Here is an exampleconst buildData = require('@/utils/common-config'); module.exports = async (ctx) => { const link = `https://www.uraaka-joshi.com/`; ctx.state.data = await buildData({ link, url: link, title: `%title%`, params: { title: '裏垢女子まとめ', }, item: { item: '.content-main .stream .stream-item', title: `$('.post-account-group').text() + ' - %title%'`, // Only supports js statements like $().xxx() link: `$('.post-account-group').attr('href')`, // .text() means get the text of the element, .attr() means get the specified attribute description: `$('.post .context').html()`, // .text() means get the text of the the html code pubDate: `new Date($('.post-time').attr('datetime')).toUTCString()`, guid: `new Date($('.post-time').attr('datetime')).getTime()`, }, }); };
So far we have completed a simplest route
# Use Cache
All routes have a cache, the global cache time is set in lib/config.js
, but the content returned by some interfaces is updated less frequently. For that, you should set a longer cache time for these data like the full text of an article.
For example, the bilibili column needs to get the full text of the article: /lib/routes/bilibili/followings_article.js (opens new window)
Since the full text of all articles cannot be got from one API, each article needs to be requested once, and these data are generally unchanged, so these data should be stored in the cache to avoid requesting repeatedly
const description = await ctx.cache.tryGet(link, async () => {
const result = await got.get(link);
const $ = cheerio.load(result.data);
$('img').each(function (i, e) {
$(e).attr('src', $(e).attr('data-src'));
});
return $('.article-holder').html();
});
The implementation of tryGet can be seen here (opens new window). The 1st parameter is the cache key; the 2nd parameter is the cache data acquisition method (executed when cache miss); the 3rd parameter is the cache time, it should not be passed in normally and defaults to CACHE_CONTENT_EXPIRE; the 4th parameter determines whether to recalculate the expiration time ("renew" the cache) when the current attempt cache hits, true
is on, false
is off, default is on
# Produce RSS Feed
Assign the acquired data to ctx.state.data, the middleware template.js (opens new window) will then process the data and render the RSS output /views/rss.art (opens new window), the list of parameters:
ctx.state.data = {
title: '', // The feed title
link: '', // The feed link
description: '', // The feed description
language: '', // The language of the channel
allowEmpty: false, // default to false, set to true to allow empty item
item: [
// An article of the feed
{
title: '', // The article title
author: '', // Author of the article
category: '', // Article category
// category: [''], // Multiple category
description: '', // The article summary or content
pubDate: '', // The article publishing datetime
guid: '', // The article unique identifier, optional, default to the article link below
link: '', // The article link
},
],
};
Warning
title
, subtitle
(only for atom), author
(only for atom), item.title
, and item.author
should not contain linebreaks, consecutive white spaces, or start/end with white space(s).
Most RSS readers will automatically trim them, so they make no sense. However, some readers may not process them properly, so we will trim them before outputting to ensure these fields contain no linebreaks, consecutive white spaces, or start/end with white space(s).
If the route you are writing can not tolerate these trimmings, you should consider change the format of these fields.
In addition, although other fields will not be forced trimmed, you should also try to avoid violations of the above rules. Especially when using Cheerio to extract web pages, you need to keep in mind that Cheerio will retain wraps and indentation. In particular, for item.description
, any intended linebreaks should be converted to <br>
, otherwise the RSS reader is likely to trim them; especially if you extract the RSS feed from JSON, the JSON returned by the source website is very likely to contain linebreaks that need to be displayed, so it must be converted in this case.
# Podcast feed
Used for audio feed, these additional data are in accordance with many podcast players' subscription format:
ctx.state.data = {
itunes_author: '', // The channel's author, you must fill this data.
itunes_category: '', // Channel category
image: '', // Channel's image
item: [
{
itunes_item_image: '', // The item image
itunes_duration: '', // The audio length in seconds (or H:mm:ss), optional
enclosure_url: '', // The item's audio link
enclosure_length: '', // The file size in Bytes, optional
enclosure_type: '', // Common types are: 'audio/mpeg' for .mp3, 'audio/x-m4a' for .m4a 'video/mp4' for .mp4
},
],
};
# BT/Magnet feed
Used for downloader feed, these additional data are in accordance with many downloaders' subscription format to trigger automated download:
ctx.state.data = {
item: [
{
enclosure_url: '', // Magnet URI
enclosure_length: '', // The file size in Bytes, optional
enclosure_type: 'application/x-bittorrent', // Fixed to 'application/x-bittorrent'
},
],
};
# Media RSS
These additional data are in accordance with many Media RSS (opens new window) softwares' subscription format:
For example:
ctx.state.data = {
item: [
{
media: {
content: {
url: post.file_url,
type: `image/${mime[post.file_ext]}`,
},
thumbnail: {
url: post.preview_url,
},
},
},
],
};
# Interactions
These additional data are in accordance with some softwares' subscription format:
ctx.state.data = {
item: [
{
upvotes: 0, // default to undefined, how many upvotes for this article,
downvotes: 0, // default to undefined, how many downvotes for this article,
comments: 0, // default to undefined, how many comments for this article
},
],
};
# Add the documentation
Update the corresponding file in documentation (/docs/en/) (opens new window) directory, preview the docs via
npm run docs:dev
Documentation uses vue component:
author
: route authors, separated by a single spaceexample
: route examplepath
: route path:paramsDesc
: route parameters description, in array, supports markdown- parameter description must be in the order of its appearance in route path
- missing description will cause errors in
npm run docs:dev
'
"
must be escaped as\'
\"
- route parameters ending with
?
,*
,+
, and a word representoptional
,zero or more
,one or more
, andmandatory
respectively. They are automatically determined by Vue component and do not need to be explicitly mentioned in the description
Documentation examples:
- No parameter:
<RouteEn author="HenryQW" example="/sspai/series" path="/sspai/series" />
Preview:
- Multiple parameters:
<RouteEn author="HenryQW" example="/github/issue/DIYgod/RSSHub" path="/github/issue/:user/:repo" :paramsDesc="['GitHub username', 'GitHub repo name']" />
Preview:
Example: https://rsshub.app/github/issue/DIYgod/RSSHub
Route:
/github/issue/:user/:repo
Parameters:
user
, required - GitHub usernamerepo
, required - GitHub repo name
- Use component slot for complicated description:
<RouteEn author="DIYgod" example="/juejin/category/frontend" path="/juejin/category/:category" :paramsDesc="['分类名']"> | 前端 | Android | iOS | 后端 | 设计 | 产品 | 工具资源 | 阅读 | 人工智能 | | -------- | ------- | --- | ------- | ------ | ------- | -------- | ------- | -------- | | frontend | android | ios | backend | design | product | freebie | article | ai | </RouteEn>
Preview:
Example: https://rsshub.app/juejin/category/frontend
Route:
/juejin/category/:category
Parameters:
category
, required - 分类名
前端 Android iOS 后端 设计 产品 工具资源 阅读 人工智能 frontend android ios backend design product freebie article ai
Please be sure to close the tag of
<Route>
!Execute
npm run format
to lint the code before you commit and open a pull request
# Submit new RSSHub Radar rule
# Debug
Open browser's RSSHub Radar extension settings, switch to rules list, scroll down and you will see a text box, copy the new rules into the text box and start debugging
# Code the rule
Create radar.js
under the corresponding /lib/v2/ (opens new window) route and the rules
Simplified rules will be used for the following illustration:
{
'bilibili.com': {
_name: 'bilibili',
www: [{
title: '分区视频',
docs: 'https://docs.rsshub.app/social-media.html#bilibili',
source: '/v/*tpath',
target: (params) => {
let tid;
switch (params.tpath) {
case 'douga/mad':
tid = '24';
break;
default:
return false;
}
return `/bilibili/partion/${tid}`;
},
}],
},
'twitter.com': {
_name: 'Twitter',
'.': [{ // for twitter.com
title: 'User timeline',
docs: 'https://docs.rsshub.app/en/social-media.html#twitter',
source: '/:id',
target: (params) => {
if (params.id !== 'home') {
return '/twitter/user/:id';
}
},
}],
},
'pixiv.net': {
_name: 'Pixiv',
'www': [{
title: 'User Bookmark',
docs: 'https://docs.rsshub.app/en/social-media.html#pixiv',
source: '/bookmark.php',
target: (params, url) => `/pixiv/user/bookmarks/${new URL(url).searchParams.get('id')}`,
}],
},
'weibo.com': {
_name: '微博',
'.': [{
title: '博主',
docs: 'https://docs.rsshub.app/social-media.html#%E5%BE%AE%E5%8D%9A',
source: ['/u/:id', '/:id'],
target: (params, url, document) => {
const uid = document && document.documentElement.innerHTML.match(/\$CONFIG\['oid']='(\d+)'/)[1];
return uid ? `/weibo/user/${uid}` : '';
},
}],
},
}
The definition and usage of these fields are explained below:
# title
Required, route name
The corresponding name in RSSHub docs, e.g. the title
for Twitter User timeline
is User timeline
# docs
Required, docs link
e.g. the docs
for Twitter User timeline
is https://docs.rsshub.app/en/social-media.html#twitter
Note that it is not https://docs.rsshub.app/en/social-media.html#twitter-user-timeline
, hash should be positioned to the H1 heading
# source
Optional, source path, leave it blank and it will never match, only appears in RSSHub for current website
e.g. the source
for Twitter User timeline
is /:id
Let's say we are in https://twitter.com/DIYgod
, which matchestwitter.com/:id
, the resulting params is {id: 'DIYgod'}
, the extension will then generate an RSSHub subscription address based on the params target
Please note that source
can only match URL Path, if the parameters are in URL Param and URL Hash please use target
# target
Optional, RSSHub path, leave it blank to not generate an RSSHub path
The corresponding path in RSSHub docs, e.g. the target
for Twitter User timeline
is /twitter/user/:id
The source path in the previous step matches id
with DIYgod
, the :id
in RSSHub path will be replaced with DIYgod
, resulting in /twitter/user/DIYgod
, which is the result we want
Furthermore, if the source path does not match the desired parameters, we can use target
as a function with params
, url
and document
parameters
params
are the parameters matched by source
in the previous step, url
is the page url, document
is the page document
Note that the target
method runs in a sandbox, any changes to document
will not be reflected in the web page
# RSSBud
RSSBud (opens new window) supports RSSHub Radar rules and will also be updated automatically, but please note that:
Using
'.'
subdomain allows RSSBud to support common mobile domains such asm
/mobile
Using
document
intarget
does not apply to RSSBud: RSSBud is not a browser extension, it only fetches and analyzes the URL of a website
# Additional Information
Adding radar="1"
in the RSSHub docs will show a support browser extension
badge
If RSSBud is also supported, adding rssbud="1"
will show a support rssbud
badge