Feed (#3698)
* wip * Implement feed * Update feed.ts * Update index.ts * Update feed.ts
This commit is contained in:
		
							parent
							
								
									2a8f984db7
								
							
						
					
					
						commit
						1395cf89ce
					
				
					 3 changed files with 102 additions and 0 deletions
				
			
		| 
						 | 
					@ -114,6 +114,7 @@
 | 
				
			||||||
		"eslint": "5.8.0",
 | 
							"eslint": "5.8.0",
 | 
				
			||||||
		"eslint-plugin-vue": "4.7.1",
 | 
							"eslint-plugin-vue": "4.7.1",
 | 
				
			||||||
		"eventemitter3": "3.1.0",
 | 
							"eventemitter3": "3.1.0",
 | 
				
			||||||
 | 
							"feed": "2.0.2",
 | 
				
			||||||
		"file-loader": "2.0.0",
 | 
							"file-loader": "2.0.0",
 | 
				
			||||||
		"file-type": "10.6.0",
 | 
							"file-type": "10.6.0",
 | 
				
			||||||
		"fuckadblock": "3.2.1",
 | 
							"fuckadblock": "3.2.1",
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										54
									
								
								src/server/web/feed.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								src/server/web/feed.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,54 @@
 | 
				
			||||||
 | 
					import { Feed } from 'feed';
 | 
				
			||||||
 | 
					import config from '../../config';
 | 
				
			||||||
 | 
					import Note from '../../models/note';
 | 
				
			||||||
 | 
					import { IUser } from '../../models/user';
 | 
				
			||||||
 | 
					import { getOriginalUrl } from '../../misc/get-drive-file-url';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default async function(user: IUser) {
 | 
				
			||||||
 | 
						const author: Author = {
 | 
				
			||||||
 | 
							link: `${config.url}/@${user.username}`,
 | 
				
			||||||
 | 
							name: user.name || user.username
 | 
				
			||||||
 | 
						};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const notes = await Note.find({
 | 
				
			||||||
 | 
							userId: user._id,
 | 
				
			||||||
 | 
							renoteId: null,
 | 
				
			||||||
 | 
							$or: [
 | 
				
			||||||
 | 
								{ visibility: 'public' },
 | 
				
			||||||
 | 
								{ visibility: 'home' }
 | 
				
			||||||
 | 
							]
 | 
				
			||||||
 | 
						}, {
 | 
				
			||||||
 | 
							sort: { createdAt: -1 },
 | 
				
			||||||
 | 
							limit: 20
 | 
				
			||||||
 | 
						});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const feed = new Feed({
 | 
				
			||||||
 | 
							id: author.link,
 | 
				
			||||||
 | 
							title: `${author.name} (@${user.username}@${config.host})`,
 | 
				
			||||||
 | 
							updated: notes[0].createdAt,
 | 
				
			||||||
 | 
							generator: 'Misskey',
 | 
				
			||||||
 | 
							description: `${user.notesCount} Notes, ${user.followingCount} Following, ${user.followersCount} Followers${user.description ? ` · ${user.description}` : ''}`,
 | 
				
			||||||
 | 
							link: author.link,
 | 
				
			||||||
 | 
							image: user.avatarUrl,
 | 
				
			||||||
 | 
							feedLinks: {
 | 
				
			||||||
 | 
								json: `${author.link}.json`,
 | 
				
			||||||
 | 
								atom: `${author.link}.atom`,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							author
 | 
				
			||||||
 | 
						} as FeedOptions);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (const note of notes) {
 | 
				
			||||||
 | 
							const file = note._files && note._files.find(file => file.contentType.startsWith('image/'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							feed.addItem({
 | 
				
			||||||
 | 
								title: `New note by ${author.name}`,
 | 
				
			||||||
 | 
								link: `${config.url}/notes/${note._id}`,
 | 
				
			||||||
 | 
								date: note.createdAt,
 | 
				
			||||||
 | 
								description: note.cw,
 | 
				
			||||||
 | 
								content: note.text,
 | 
				
			||||||
 | 
								image: file && getOriginalUrl(file)
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return feed;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -10,6 +10,7 @@ import * as favicon from 'koa-favicon';
 | 
				
			||||||
import * as views from 'koa-views';
 | 
					import * as views from 'koa-views';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import docs from './docs';
 | 
					import docs from './docs';
 | 
				
			||||||
 | 
					import packFeed from './feed';
 | 
				
			||||||
import User from '../../models/user';
 | 
					import User from '../../models/user';
 | 
				
			||||||
import parseAcct from '../../misc/acct/parse';
 | 
					import parseAcct from '../../misc/acct/parse';
 | 
				
			||||||
import config from '../../config';
 | 
					import config from '../../config';
 | 
				
			||||||
| 
						 | 
					@ -82,6 +83,52 @@ router.use('/docs', docs.routes());
 | 
				
			||||||
// URL preview endpoint
 | 
					// URL preview endpoint
 | 
				
			||||||
router.get('/url', require('./url-preview'));
 | 
					router.get('/url', require('./url-preview'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const getFeed = async (acct: string) => {
 | 
				
			||||||
 | 
						const { username, host } = parseAcct(acct);
 | 
				
			||||||
 | 
						const user = await User.findOne({
 | 
				
			||||||
 | 
							usernameLower: username.toLowerCase(),
 | 
				
			||||||
 | 
							host
 | 
				
			||||||
 | 
						});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return user && await packFeed(user);
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Atom
 | 
				
			||||||
 | 
					router.get('/@:user.atom', async ctx => {
 | 
				
			||||||
 | 
						const feed = await getFeed(ctx.params.user);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (feed) {
 | 
				
			||||||
 | 
							ctx.set('Content-Type', 'application/atom+xml; charset=utf-8');
 | 
				
			||||||
 | 
							ctx.body = feed.atom1();
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							ctx.status = 404;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RSS
 | 
				
			||||||
 | 
					router.get('/@:user.rss', async ctx => {
 | 
				
			||||||
 | 
						const feed = await getFeed(ctx.params.user);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (feed) {
 | 
				
			||||||
 | 
							ctx.set('Content-Type', 'application/rss+xml; charset=utf-8');
 | 
				
			||||||
 | 
							ctx.body = feed.rss2();
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							ctx.status = 404;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// JSON
 | 
				
			||||||
 | 
					router.get('/@:user.json', async ctx => {
 | 
				
			||||||
 | 
						const feed = await getFeed(ctx.params.user);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (feed) {
 | 
				
			||||||
 | 
							ctx.set('Content-Type', 'application/json; charset=utf-8');
 | 
				
			||||||
 | 
							ctx.body = feed.json1();
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							ctx.status = 404;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//#region for crawlers
 | 
					//#region for crawlers
 | 
				
			||||||
// User
 | 
					// User
 | 
				
			||||||
router.get('/@:user', async (ctx, next) => {
 | 
					router.get('/@:user', async (ctx, next) => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue