| @@ -907,6 +907,10 @@ export class NoteCreateService implements OnApplicationShutdown { | ||||
| 			// 自分自身以外への返信 | ||||
| 			if (note.replyId && note.replyUserId !== note.userId) { | ||||
| 				this.redisTimelineService.push(`userTimelineWithReplies:${user.id}`, note.id, note.userHost == null ? meta.perLocalUserUserTimelineCacheMax : meta.perRemoteUserUserTimelineCacheMax, r); | ||||
|  | ||||
| 				if (note.visibility === 'public' && note.userHost == null) { | ||||
| 					this.redisTimelineService.push('localTimelineWithReplies', note.id, 300, r); | ||||
| 				} | ||||
| 			} else { | ||||
| 				this.redisTimelineService.push(`userTimeline:${user.id}`, note.id, note.userHost == null ? meta.perLocalUserUserTimelineCacheMax : meta.perRemoteUserUserTimelineCacheMax, r); | ||||
| 				if (note.fileIds.length > 0) { | ||||
|   | ||||
| @@ -55,6 +55,7 @@ export const paramDef = { | ||||
| 		includeLocalRenotes: { type: 'boolean', default: true }, | ||||
| 		withFiles: { type: 'boolean', default: false }, | ||||
| 		withRenotes: { type: 'boolean', default: true }, | ||||
| 		withReplies: { type: 'boolean', default: false }, | ||||
| 	}, | ||||
| 	required: [], | ||||
| } as const; | ||||
| @@ -94,12 +95,29 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				this.cacheService.userBlockedCache.fetch(me.id), | ||||
| 			]); | ||||
|  | ||||
| 			const [htlNoteIds, ltlNoteIds] = await this.redisTimelineService.getMulti([ | ||||
| 				ps.withFiles ? `homeTimelineWithFiles:${me.id}` : `homeTimeline:${me.id}`, | ||||
| 				ps.withFiles ? 'localTimelineWithFiles' : 'localTimeline', | ||||
| 			], untilId, sinceId); | ||||
| 			let noteIds: string[]; | ||||
|  | ||||
| 			if (ps.withFiles) { | ||||
| 				const [htlNoteIds, ltlNoteIds] = await this.redisTimelineService.getMulti([ | ||||
| 					`homeTimelineWithFiles:${me.id}`, | ||||
| 					'localTimelineWithFiles', | ||||
| 				], untilId, sinceId); | ||||
| 				noteIds = Array.from(new Set([...htlNoteIds, ...ltlNoteIds])); | ||||
| 			} else if (ps.withReplies) { | ||||
| 				const [htlNoteIds, ltlNoteIds, ltlReplyNoteIds] = await this.redisTimelineService.getMulti([ | ||||
| 					`homeTimeline:${me.id}`, | ||||
| 					'localTimeline', | ||||
| 					'localTimelineWithReplies', | ||||
| 				], untilId, sinceId); | ||||
| 				noteIds = Array.from(new Set([...htlNoteIds, ...ltlNoteIds, ...ltlReplyNoteIds])); | ||||
| 			} else { | ||||
| 				const [htlNoteIds, ltlNoteIds] = await this.redisTimelineService.getMulti([ | ||||
| 					`homeTimeline:${me.id}`, | ||||
| 					'localTimeline', | ||||
| 				], untilId, sinceId); | ||||
| 				noteIds = Array.from(new Set([...htlNoteIds, ...ltlNoteIds])); | ||||
| 			} | ||||
|  | ||||
| 			let noteIds = Array.from(new Set([...htlNoteIds, ...ltlNoteIds])); | ||||
| 			noteIds.sort((a, b) => a > b ? -1 : 1); | ||||
| 			noteIds = noteIds.slice(0, ps.limit); | ||||
|  | ||||
|   | ||||
| @@ -45,6 +45,7 @@ export const paramDef = { | ||||
| 	properties: { | ||||
| 		withFiles: { type: 'boolean', default: false }, | ||||
| 		withRenotes: { type: 'boolean', default: true }, | ||||
| 		withReplies: { type: 'boolean', default: false }, | ||||
| 		excludeNsfw: { type: 'boolean', default: false }, | ||||
| 		limit: { type: 'integer', minimum: 1, maximum: 100, default: 10 }, | ||||
| 		sinceId: { type: 'string', format: 'misskey:id' }, | ||||
| @@ -90,7 +91,21 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				this.cacheService.userBlockedCache.fetch(me.id), | ||||
| 			]) : [new Set<string>(), new Set<string>(), new Set<string>()]; | ||||
|  | ||||
| 			let noteIds = await this.redisTimelineService.get(ps.withFiles ? 'localTimelineWithFiles' : 'localTimeline', untilId, sinceId); | ||||
| 			let noteIds: string[]; | ||||
|  | ||||
| 			if (ps.withFiles) { | ||||
| 				noteIds = await this.redisTimelineService.get('localTimelineWithFiles', untilId, sinceId); | ||||
| 			} else if (ps.withReplies) { | ||||
| 				const [nonReplyNoteIds, replyNoteIds] = await this.redisTimelineService.getMulti([ | ||||
| 					'localTimeline', | ||||
| 					'localTimelineWithReplies', | ||||
| 				], untilId, sinceId); | ||||
| 				noteIds = Array.from(new Set([...nonReplyNoteIds, ...replyNoteIds])); | ||||
| 				noteIds.sort((a, b) => a > b ? -1 : 1); | ||||
| 			} else { | ||||
| 				noteIds = await this.redisTimelineService.get('localTimeline', untilId, sinceId); | ||||
| 			} | ||||
|  | ||||
| 			noteIds = noteIds.slice(0, ps.limit); | ||||
|  | ||||
| 			if (noteIds.length === 0) { | ||||
|   | ||||
| @@ -19,6 +19,7 @@ class HybridTimelineChannel extends Channel { | ||||
| 	public static shouldShare = false; | ||||
| 	public static requireCredential = true; | ||||
| 	private withRenotes: boolean; | ||||
| 	private withReplies: boolean; | ||||
| 	private withFiles: boolean; | ||||
|  | ||||
| 	constructor( | ||||
| @@ -39,6 +40,7 @@ class HybridTimelineChannel extends Channel { | ||||
| 		if (!policies.ltlAvailable) return; | ||||
|  | ||||
| 		this.withRenotes = params.withRenotes ?? true; | ||||
| 		this.withReplies = params.withReplies ?? false; | ||||
| 		this.withFiles = params.withFiles ?? false; | ||||
|  | ||||
| 		// Subscribe events | ||||
| @@ -87,7 +89,7 @@ class HybridTimelineChannel extends Channel { | ||||
| 		if (isInstanceMuted(note, new Set<string>(this.userProfile!.mutedInstances ?? []))) return; | ||||
|  | ||||
| 		// 関係ない返信は除外 | ||||
| 		if (note.reply && !this.following[note.userId]?.withReplies) { | ||||
| 		if (note.reply && !this.following[note.userId]?.withReplies && !this.withReplies) { | ||||
| 			const reply = note.reply; | ||||
| 			// 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合 | ||||
| 			if (reply.userId !== this.user!.id && note.userId !== this.user!.id && reply.userId !== note.userId) return; | ||||
|   | ||||
| @@ -18,6 +18,7 @@ class LocalTimelineChannel extends Channel { | ||||
| 	public static shouldShare = false; | ||||
| 	public static requireCredential = false; | ||||
| 	private withRenotes: boolean; | ||||
| 	private withReplies: boolean; | ||||
| 	private withFiles: boolean; | ||||
|  | ||||
| 	constructor( | ||||
| @@ -38,6 +39,7 @@ class LocalTimelineChannel extends Channel { | ||||
| 		if (!policies.ltlAvailable) return; | ||||
|  | ||||
| 		this.withRenotes = params.withRenotes ?? true; | ||||
| 		this.withReplies = params.withReplies ?? false; | ||||
| 		this.withFiles = params.withFiles ?? false; | ||||
|  | ||||
| 		// Subscribe events | ||||
| @@ -66,7 +68,7 @@ class LocalTimelineChannel extends Channel { | ||||
| 		} | ||||
|  | ||||
| 		// 関係ない返信は除外 | ||||
| 		if (note.reply && this.user && !this.following[note.userId]?.withReplies) { | ||||
| 		if (note.reply && this.user && !this.following[note.userId]?.withReplies && !this.withReplies) { | ||||
| 			const reply = note.reply; | ||||
| 			// 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合 | ||||
| 			if (reply.userId !== this.user.id && note.userId !== this.user.id && reply.userId !== note.userId) return; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 syuilo
					syuilo