Merge pull request #1374 from akihikodaki/log
Introduce followed log and following log
This commit is contained in:
		
							
								
								
									
										11
									
								
								src/models/followed-log.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								src/models/followed-log.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| import { ObjectID } from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
|  | ||||
| const FollowedLog = db.get<IFollowedLog>('followedLogs'); | ||||
| export default FollowedLog; | ||||
|  | ||||
| export type IFollowedLog = { | ||||
| 	_id: ObjectID; | ||||
| 	userId: ObjectID; | ||||
| 	count: number; | ||||
| }; | ||||
							
								
								
									
										11
									
								
								src/models/following-log.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								src/models/following-log.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| import { ObjectID } from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
|  | ||||
| const FollowingLog = db.get<IFollowingLog>('followingLogs'); | ||||
| export default FollowingLog; | ||||
|  | ||||
| export type IFollowingLog = { | ||||
| 	_id: ObjectID; | ||||
| 	userId: ObjectID; | ||||
| 	count: number; | ||||
| }; | ||||
| @@ -8,7 +8,6 @@ export default Following; | ||||
| export type IFollowing = { | ||||
| 	_id: mongo.ObjectID; | ||||
| 	createdAt: Date; | ||||
| 	deletedAt: Date; | ||||
| 	followeeId: mongo.ObjectID; | ||||
| 	followerId: mongo.ObjectID; | ||||
| }; | ||||
|   | ||||
| @@ -234,8 +234,7 @@ export const pack = ( | ||||
| 		_user.isFollowing = (async () => { | ||||
| 			const follow = await Following.findOne({ | ||||
| 				followerId: meId, | ||||
| 				followeeId: _user.id, | ||||
| 				deletedAt: { $exists: false } | ||||
| 				followeeId: _user.id | ||||
| 			}); | ||||
| 			return follow !== null; | ||||
| 		})(); | ||||
| @@ -244,8 +243,7 @@ export const pack = ( | ||||
| 		_user.isFollowed = (async () => { | ||||
| 			const follow2 = await Following.findOne({ | ||||
| 				followerId: _user.id, | ||||
| 				followeeId: meId, | ||||
| 				deletedAt: { $exists: false } | ||||
| 				followeeId: meId | ||||
| 			}); | ||||
| 			return follow2 !== null; | ||||
| 		})(); | ||||
| @@ -275,15 +273,13 @@ export const pack = ( | ||||
| 			// Get following you know count | ||||
| 			_user.followingYouKnowCount = Following.count({ | ||||
| 				followeeId: { $in: myFollowingIds }, | ||||
| 				followerId: _user.id, | ||||
| 				deletedAt: { $exists: false } | ||||
| 				followerId: _user.id | ||||
| 			}); | ||||
|  | ||||
| 			// Get followers you know count | ||||
| 			_user.followersYouKnowCount = Following.count({ | ||||
| 				followeeId: _user.id, | ||||
| 				followerId: { $in: myFollowingIds }, | ||||
| 				deletedAt: { $exists: false } | ||||
| 				followerId: { $in: myFollowingIds } | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
| @@ -48,9 +48,7 @@ export default async (user, mentions, post) => { | ||||
| 		// Fetch all followers | ||||
| 		const followers = await Following | ||||
| 			.find({ | ||||
| 				followeeId: user._id, | ||||
| 				// 削除されたドキュメントは除く | ||||
| 				deletedAt: { $exists: false } | ||||
| 				followeeId: user._id | ||||
| 			}, { | ||||
| 				followerId: true, | ||||
| 				_id: false | ||||
|   | ||||
| @@ -3,6 +3,8 @@ import { sign } from 'http-signature'; | ||||
| import { URL } from 'url'; | ||||
| import User, { isLocalUser, pack as packUser } from '../../models/user'; | ||||
| import Following from '../../models/following'; | ||||
| import FollowingLog from '../../models/following-log'; | ||||
| import FollowedLog from '../../models/followed-log'; | ||||
| import event from '../../publishers/stream'; | ||||
| import notify from '../../publishers/notify'; | ||||
| import context from '../../remote/activitypub/renderer/context'; | ||||
| @@ -21,6 +23,11 @@ export default ({ data }, done) => Following.findOne({ _id: data.following }).th | ||||
| 			} | ||||
| 		}), | ||||
|  | ||||
| 		promisedFollower.then(({ followingCount }) => FollowingLog.insert({ | ||||
| 			userId: followerId, | ||||
| 			count: followingCount + 1 | ||||
| 		})), | ||||
|  | ||||
| 		// Increment followers count | ||||
| 		User.update({ _id: followeeId }, { | ||||
| 			$inc: { | ||||
| @@ -28,6 +35,11 @@ export default ({ data }, done) => Following.findOne({ _id: data.following }).th | ||||
| 			} | ||||
| 		}), | ||||
|  | ||||
| 		promisedFollowee.then(({ followersCount }) => FollowedLog.insert({ | ||||
| 			userId: followerId, | ||||
| 			count: followersCount + 1 | ||||
| 		})), | ||||
|  | ||||
| 		// Notify | ||||
| 		promisedFollowee.then(followee => followee.host === null ? | ||||
| 			notify(followeeId, followerId, 'follow') : null), | ||||
|   | ||||
| @@ -6,9 +6,7 @@ export default async (me: mongodb.ObjectID, includeMe: boolean = true) => { | ||||
| 	// SELECT followee | ||||
| 	const myfollowing = await Following | ||||
| 		.find({ | ||||
| 			followerId: me, | ||||
| 			// 削除されたドキュメントは除く | ||||
| 			deletedAt: { $exists: false } | ||||
| 			followerId: me | ||||
| 		}, { | ||||
| 			fields: { | ||||
| 				followeeId: true | ||||
|   | ||||
| @@ -2,8 +2,9 @@ | ||||
|  * Module dependencies | ||||
|  */ | ||||
| import $ from 'cafy'; | ||||
| import { ObjectID } from 'mongodb'; | ||||
| import User from '../../../../../models/user'; | ||||
| import Following from '../../../../../models/following'; | ||||
| import FollowedLog from '../../../../../models/followed-log'; | ||||
|  | ||||
| /** | ||||
|  * Aggregate followers of a user | ||||
| @@ -29,47 +30,36 @@ module.exports = (params) => new Promise(async (res, rej) => { | ||||
| 		return rej('user not found'); | ||||
| 	} | ||||
|  | ||||
| 	const startTime = new Date(new Date().setMonth(new Date().getMonth() - 1)); | ||||
|  | ||||
| 	const following = await Following | ||||
| 		.find({ | ||||
| 			followeeId: user._id, | ||||
| 			$or: [ | ||||
| 				{ deletedAt: { $exists: false } }, | ||||
| 				{ deletedAt: { $gt: startTime } } | ||||
| 			] | ||||
| 		}, { | ||||
| 			sort: { createdAt: -1 }, | ||||
| 			fields: { | ||||
| 				_id: false, | ||||
| 				followerId: false, | ||||
| 				followeeId: false | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 	const today = new Date(); | ||||
| 	const graph = []; | ||||
|  | ||||
| 	today.setMinutes(0); | ||||
| 	today.setSeconds(0); | ||||
| 	today.setMilliseconds(0); | ||||
|  | ||||
| 	let cursorDate = new Date(today.getTime()); | ||||
| 	let cursorTime = cursorDate.setDate(new Date(today.getTime()).getDate() + 1); | ||||
|  | ||||
| 	for (let i = 0; i < 30; i++) { | ||||
| 		let day = new Date(new Date().setDate(new Date().getDate() - i)); | ||||
| 		day = new Date(day.setMilliseconds(999)); | ||||
| 		day = new Date(day.setSeconds(59)); | ||||
| 		day = new Date(day.setMinutes(59)); | ||||
| 		day = new Date(day.setHours(23)); | ||||
| 		// day = day.getTime(); | ||||
| 		graph.push(FollowedLog.findOne({ | ||||
| 			_id: { $lt: ObjectID.createFromTime(cursorTime / 1000) }, | ||||
| 			userId: user._id | ||||
| 		}, { | ||||
| 			sort: { _id: -1 }, | ||||
| 		}).then(log => { | ||||
| 			cursorDate = new Date(today.getTime()); | ||||
| 			cursorTime = cursorDate.setDate(today.getDate() - i); | ||||
|  | ||||
| 		const count = following.filter(f => | ||||
| 			f.createdAt < day && (f.deletedAt == null || f.deletedAt > day) | ||||
| 		).length; | ||||
|  | ||||
| 		graph.push({ | ||||
| 			date: { | ||||
| 				year: day.getFullYear(), | ||||
| 				month: day.getMonth() + 1, // In JavaScript, month is zero-based. | ||||
| 				day: day.getDate() | ||||
| 			}, | ||||
| 			count: count | ||||
| 		}); | ||||
| 			return { | ||||
| 				date: { | ||||
| 					year: cursorDate.getFullYear(), | ||||
| 					month: cursorDate.getMonth() + 1, // In JavaScript, month is zero-based. | ||||
| 					day: cursorDate.getDate() | ||||
| 				}, | ||||
| 				count: log ? log.count : 0 | ||||
| 			}; | ||||
| 		})); | ||||
| 	} | ||||
|  | ||||
| 	res(graph); | ||||
| 	res(await Promise.all(graph)); | ||||
| }); | ||||
|   | ||||
| @@ -2,8 +2,9 @@ | ||||
|  * Module dependencies | ||||
|  */ | ||||
| import $ from 'cafy'; | ||||
| import { ObjectID } from 'mongodb'; | ||||
| import User from '../../../../../models/user'; | ||||
| import Following from '../../../../../models/following'; | ||||
| import FollowingLog from '../../../../../models/following-log'; | ||||
|  | ||||
| /** | ||||
|  * Aggregate following of a user | ||||
| @@ -29,46 +30,36 @@ module.exports = (params) => new Promise(async (res, rej) => { | ||||
| 		return rej('user not found'); | ||||
| 	} | ||||
|  | ||||
| 	const startTime = new Date(new Date().setMonth(new Date().getMonth() - 1)); | ||||
|  | ||||
| 	const following = await Following | ||||
| 		.find({ | ||||
| 			followerId: user._id, | ||||
| 			$or: [ | ||||
| 				{ deletedAt: { $exists: false } }, | ||||
| 				{ deletedAt: { $gt: startTime } } | ||||
| 			] | ||||
| 		}, { | ||||
| 			sort: { createdAt: -1 }, | ||||
| 			fields: { | ||||
| 				_id: false, | ||||
| 				followerId: false, | ||||
| 				followeeId: false | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 	const today = new Date(); | ||||
| 	const graph = []; | ||||
|  | ||||
| 	today.setMinutes(0); | ||||
| 	today.setSeconds(0); | ||||
| 	today.setMilliseconds(0); | ||||
|  | ||||
| 	let cursorDate = new Date(today.getTime()); | ||||
| 	let cursorTime = cursorDate.setDate(new Date(today.getTime()).getDate() + 1); | ||||
|  | ||||
| 	for (let i = 0; i < 30; i++) { | ||||
| 		let day = new Date(new Date().setDate(new Date().getDate() - i)); | ||||
| 		day = new Date(day.setMilliseconds(999)); | ||||
| 		day = new Date(day.setSeconds(59)); | ||||
| 		day = new Date(day.setMinutes(59)); | ||||
| 		day = new Date(day.setHours(23)); | ||||
| 		graph.push(FollowingLog.findOne({ | ||||
| 			_id: { $lt: ObjectID.createFromTime(cursorTime / 1000) }, | ||||
| 			userId: user._id | ||||
| 		}, { | ||||
| 			sort: { _id: -1 }, | ||||
| 		}).then(log => { | ||||
| 			cursorDate = new Date(today.getTime()); | ||||
| 			cursorTime = cursorDate.setDate(today.getDate() - i); | ||||
|  | ||||
| 		const count = following.filter(f => | ||||
| 			f.createdAt < day && (f.deletedAt == null || f.deletedAt > day) | ||||
| 		).length; | ||||
|  | ||||
| 		graph.push({ | ||||
| 			date: { | ||||
| 				year: day.getFullYear(), | ||||
| 				month: day.getMonth() + 1, // In JavaScript, month is zero-based. | ||||
| 				day: day.getDate() | ||||
| 			}, | ||||
| 			count: count | ||||
| 		}); | ||||
| 			return { | ||||
| 				date: { | ||||
| 					year: cursorDate.getFullYear(), | ||||
| 					month: cursorDate.getMonth() + 1, // In JavaScript, month is zero-based. | ||||
| 					day: cursorDate.getDate() | ||||
| 				}, | ||||
| 				count: log ? log.count : 0 | ||||
| 			}; | ||||
| 		})); | ||||
| 	} | ||||
|  | ||||
| 	res(graph); | ||||
| 	res(await Promise.all(graph)); | ||||
| }); | ||||
|   | ||||
| @@ -42,8 +42,7 @@ module.exports = (params, user) => new Promise(async (res, rej) => { | ||||
| 	// Check if already following | ||||
| 	const exist = await Following.findOne({ | ||||
| 		followerId: follower._id, | ||||
| 		followeeId: followee._id, | ||||
| 		deletedAt: { $exists: false } | ||||
| 		followeeId: followee._id | ||||
| 	}); | ||||
|  | ||||
| 	if (exist !== null) { | ||||
|   | ||||
| @@ -42,8 +42,7 @@ module.exports = (params, user) => new Promise(async (res, rej) => { | ||||
| 	// Check not following | ||||
| 	const exist = await Following.findOne({ | ||||
| 		followerId: follower._id, | ||||
| 		followeeId: followee._id, | ||||
| 		deletedAt: { $exists: false } | ||||
| 		followeeId: followee._id | ||||
| 	}); | ||||
|  | ||||
| 	if (exist === null) { | ||||
| @@ -51,12 +50,8 @@ module.exports = (params, user) => new Promise(async (res, rej) => { | ||||
| 	} | ||||
|  | ||||
| 	// Delete following | ||||
| 	await Following.update({ | ||||
| 	await Following.findOneAndDelete({ | ||||
| 		_id: exist._id | ||||
| 	}, { | ||||
| 		$set: { | ||||
| 			deletedAt: new Date() | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	// Send response | ||||
|   | ||||
| @@ -46,8 +46,7 @@ module.exports = (params, me) => new Promise(async (res, rej) => { | ||||
|  | ||||
| 	// Construct query | ||||
| 	const query = { | ||||
| 		followeeId: user._id, | ||||
| 		deletedAt: { $exists: false } | ||||
| 		followeeId: user._id | ||||
| 	} as any; | ||||
|  | ||||
| 	// ログインしていてかつ iknow フラグがあるとき | ||||
|   | ||||
| @@ -46,8 +46,7 @@ module.exports = (params, me) => new Promise(async (res, rej) => { | ||||
|  | ||||
| 	// Construct query | ||||
| 	const query = { | ||||
| 		followerId: user._id, | ||||
| 		deletedAt: { $exists: false } | ||||
| 		followerId: user._id | ||||
| 	} as any; | ||||
|  | ||||
| 	// ログインしていてかつ iknow フラグがあるとき | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 syuilo
					syuilo