feat: add per user pv chart
This commit is contained in:
		
							parent
							
								
									4c4af2ae84
								
							
						
					
					
						commit
						969e9df889
					
				
					 13 changed files with 153 additions and 4 deletions
				
			
		|  | @ -34,6 +34,7 @@ You should also include the user name that made the change. | ||||||
| - Add Cloudflare Turnstile CAPTCHA support @CyberRex0 | - Add Cloudflare Turnstile CAPTCHA support @CyberRex0 | ||||||
| - Introduce retention-rate aggregation @syuilo | - Introduce retention-rate aggregation @syuilo | ||||||
| - Make possible to export favorited notes @syuilo | - Make possible to export favorited notes @syuilo | ||||||
|  | - Add per user pv chart @syuilo | ||||||
| - Server: signToActivityPubGet is set to true by default @syuilo | - Server: signToActivityPubGet is set to true by default @syuilo | ||||||
| - Server: improve syslog performance @syuilo | - Server: improve syslog performance @syuilo | ||||||
| - Server: improve note scoring for featured notes @CyberRex0 | - Server: improve note scoring for featured notes @CyberRex0 | ||||||
|  |  | ||||||
							
								
								
									
										17
									
								
								packages/backend/migration/1672562400597-PerUserPvChart.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								packages/backend/migration/1672562400597-PerUserPvChart.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | ||||||
|  | export class PerUserPvChart1672562400597 { | ||||||
|  |     name = 'PerUserPvChart1672562400597' | ||||||
|  | 
 | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`CREATE TABLE "__chart__per_user_pv" ("id" SERIAL NOT NULL, "date" integer NOT NULL, "group" character varying(128) NOT NULL, "unique_temp___upv_user" character varying array NOT NULL DEFAULT '{}', "___upv_user" smallint NOT NULL DEFAULT '0', "___pv_user" smallint NOT NULL DEFAULT '0', "unique_temp___upv_visitor" character varying array NOT NULL DEFAULT '{}', "___upv_visitor" smallint NOT NULL DEFAULT '0', "___pv_visitor" smallint NOT NULL DEFAULT '0', CONSTRAINT "UQ_f2a56da57921ca8439f45c1d95f" UNIQUE ("date", "group"), CONSTRAINT "PK_3c938a24f0203b5bd13fab51059" PRIMARY KEY ("id"))`); | ||||||
|  |         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_f2a56da57921ca8439f45c1d95" ON "__chart__per_user_pv" ("date", "group") `); | ||||||
|  |         await queryRunner.query(`CREATE TABLE "__chart_day__per_user_pv" ("id" SERIAL NOT NULL, "date" integer NOT NULL, "group" character varying(128) NOT NULL, "unique_temp___upv_user" character varying array NOT NULL DEFAULT '{}', "___upv_user" smallint NOT NULL DEFAULT '0', "___pv_user" smallint NOT NULL DEFAULT '0', "unique_temp___upv_visitor" character varying array NOT NULL DEFAULT '{}', "___upv_visitor" smallint NOT NULL DEFAULT '0', "___pv_visitor" smallint NOT NULL DEFAULT '0', CONSTRAINT "UQ_f221e45cfac5bea0ce0f3149fbb" UNIQUE ("date", "group"), CONSTRAINT "PK_0085d7542f6772e99b9dcfb0a9c" PRIMARY KEY ("id"))`); | ||||||
|  |         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_f221e45cfac5bea0ce0f3149fb" ON "__chart_day__per_user_pv" ("date", "group") `); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`DROP INDEX "public"."IDX_f221e45cfac5bea0ce0f3149fb"`); | ||||||
|  |         await queryRunner.query(`DROP TABLE "__chart_day__per_user_pv"`); | ||||||
|  |         await queryRunner.query(`DROP INDEX "public"."IDX_f2a56da57921ca8439f45c1d95"`); | ||||||
|  |         await queryRunner.query(`DROP TABLE "__chart__per_user_pv"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -57,6 +57,7 @@ import UsersChart from './chart/charts/users.js'; | ||||||
| import ActiveUsersChart from './chart/charts/active-users.js'; | import ActiveUsersChart from './chart/charts/active-users.js'; | ||||||
| import InstanceChart from './chart/charts/instance.js'; | import InstanceChart from './chart/charts/instance.js'; | ||||||
| import PerUserNotesChart from './chart/charts/per-user-notes.js'; | import PerUserNotesChart from './chart/charts/per-user-notes.js'; | ||||||
|  | import PerUserPvChart from './chart/charts/per-user-pv.js'; | ||||||
| import DriveChart from './chart/charts/drive.js'; | import DriveChart from './chart/charts/drive.js'; | ||||||
| import PerUserReactionsChart from './chart/charts/per-user-reactions.js'; | import PerUserReactionsChart from './chart/charts/per-user-reactions.js'; | ||||||
| import HashtagChart from './chart/charts/hashtag.js'; | import HashtagChart from './chart/charts/hashtag.js'; | ||||||
|  | @ -176,6 +177,7 @@ const $UsersChart: Provider = { provide: 'UsersChart', useExisting: UsersChart } | ||||||
| const $ActiveUsersChart: Provider = { provide: 'ActiveUsersChart', useExisting: ActiveUsersChart }; | const $ActiveUsersChart: Provider = { provide: 'ActiveUsersChart', useExisting: ActiveUsersChart }; | ||||||
| const $InstanceChart: Provider = { provide: 'InstanceChart', useExisting: InstanceChart }; | const $InstanceChart: Provider = { provide: 'InstanceChart', useExisting: InstanceChart }; | ||||||
| const $PerUserNotesChart: Provider = { provide: 'PerUserNotesChart', useExisting: PerUserNotesChart }; | const $PerUserNotesChart: Provider = { provide: 'PerUserNotesChart', useExisting: PerUserNotesChart }; | ||||||
|  | const $PerUserPvChart: Provider = { provide: 'PerUserPvChart', useExisting: PerUserPvChart }; | ||||||
| const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart }; | const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart }; | ||||||
| const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart }; | const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart }; | ||||||
| const $HashtagChart: Provider = { provide: 'HashtagChart', useExisting: HashtagChart }; | const $HashtagChart: Provider = { provide: 'HashtagChart', useExisting: HashtagChart }; | ||||||
|  | @ -298,6 +300,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||||
| 		ActiveUsersChart, | 		ActiveUsersChart, | ||||||
| 		InstanceChart, | 		InstanceChart, | ||||||
| 		PerUserNotesChart, | 		PerUserNotesChart, | ||||||
|  | 		PerUserPvChart, | ||||||
| 		DriveChart, | 		DriveChart, | ||||||
| 		PerUserReactionsChart, | 		PerUserReactionsChart, | ||||||
| 		HashtagChart, | 		HashtagChart, | ||||||
|  | @ -414,6 +417,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||||
| 		$ActiveUsersChart, | 		$ActiveUsersChart, | ||||||
| 		$InstanceChart, | 		$InstanceChart, | ||||||
| 		$PerUserNotesChart, | 		$PerUserNotesChart, | ||||||
|  | 		$PerUserPvChart, | ||||||
| 		$DriveChart, | 		$DriveChart, | ||||||
| 		$PerUserReactionsChart, | 		$PerUserReactionsChart, | ||||||
| 		$HashtagChart, | 		$HashtagChart, | ||||||
|  | @ -530,6 +534,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||||
| 		ActiveUsersChart, | 		ActiveUsersChart, | ||||||
| 		InstanceChart, | 		InstanceChart, | ||||||
| 		PerUserNotesChart, | 		PerUserNotesChart, | ||||||
|  | 		PerUserPvChart, | ||||||
| 		DriveChart, | 		DriveChart, | ||||||
| 		PerUserReactionsChart, | 		PerUserReactionsChart, | ||||||
| 		HashtagChart, | 		HashtagChart, | ||||||
|  | @ -645,6 +650,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||||
| 		$ActiveUsersChart, | 		$ActiveUsersChart, | ||||||
| 		$InstanceChart, | 		$InstanceChart, | ||||||
| 		$PerUserNotesChart, | 		$PerUserNotesChart, | ||||||
|  | 		$PerUserPvChart, | ||||||
| 		$DriveChart, | 		$DriveChart, | ||||||
| 		$PerUserReactionsChart, | 		$PerUserReactionsChart, | ||||||
| 		$HashtagChart, | 		$HashtagChart, | ||||||
|  |  | ||||||
|  | @ -1,11 +1,13 @@ | ||||||
| import { Injectable, Inject } from '@nestjs/common'; | import { Injectable, Inject } from '@nestjs/common'; | ||||||
| 
 | 
 | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import FederationChart from './charts/federation.js'; | import FederationChart from './charts/federation.js'; | ||||||
| import NotesChart from './charts/notes.js'; | import NotesChart from './charts/notes.js'; | ||||||
| import UsersChart from './charts/users.js'; | import UsersChart from './charts/users.js'; | ||||||
| import ActiveUsersChart from './charts/active-users.js'; | import ActiveUsersChart from './charts/active-users.js'; | ||||||
| import InstanceChart from './charts/instance.js'; | import InstanceChart from './charts/instance.js'; | ||||||
| import PerUserNotesChart from './charts/per-user-notes.js'; | import PerUserNotesChart from './charts/per-user-notes.js'; | ||||||
|  | import PerUserPvChart from './charts/per-user-pv.js'; | ||||||
| import DriveChart from './charts/drive.js'; | import DriveChart from './charts/drive.js'; | ||||||
| import PerUserReactionsChart from './charts/per-user-reactions.js'; | import PerUserReactionsChart from './charts/per-user-reactions.js'; | ||||||
| import HashtagChart from './charts/hashtag.js'; | import HashtagChart from './charts/hashtag.js'; | ||||||
|  | @ -13,7 +15,6 @@ import PerUserFollowingChart from './charts/per-user-following.js'; | ||||||
| import PerUserDriveChart from './charts/per-user-drive.js'; | import PerUserDriveChart from './charts/per-user-drive.js'; | ||||||
| import ApRequestChart from './charts/ap-request.js'; | import ApRequestChart from './charts/ap-request.js'; | ||||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | import type { OnApplicationShutdown } from '@nestjs/common'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
| 
 | 
 | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ChartManagementService implements OnApplicationShutdown { | export class ChartManagementService implements OnApplicationShutdown { | ||||||
|  | @ -27,6 +28,7 @@ export class ChartManagementService implements OnApplicationShutdown { | ||||||
| 		private activeUsersChart: ActiveUsersChart, | 		private activeUsersChart: ActiveUsersChart, | ||||||
| 		private instanceChart: InstanceChart, | 		private instanceChart: InstanceChart, | ||||||
| 		private perUserNotesChart: PerUserNotesChart, | 		private perUserNotesChart: PerUserNotesChart, | ||||||
|  | 		private perUserPvChart: PerUserPvChart, | ||||||
| 		private driveChart: DriveChart, | 		private driveChart: DriveChart, | ||||||
| 		private perUserReactionsChart: PerUserReactionsChart, | 		private perUserReactionsChart: PerUserReactionsChart, | ||||||
| 		private hashtagChart: HashtagChart, | 		private hashtagChart: HashtagChart, | ||||||
|  | @ -41,6 +43,7 @@ export class ChartManagementService implements OnApplicationShutdown { | ||||||
| 			this.activeUsersChart, | 			this.activeUsersChart, | ||||||
| 			this.instanceChart, | 			this.instanceChart, | ||||||
| 			this.perUserNotesChart, | 			this.perUserNotesChart, | ||||||
|  | 			this.perUserPvChart, | ||||||
| 			this.driveChart, | 			this.driveChart, | ||||||
| 			this.perUserReactionsChart, | 			this.perUserReactionsChart, | ||||||
| 			this.hashtagChart, | 			this.hashtagChart, | ||||||
|  |  | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | import Chart from '../../core.js'; | ||||||
|  | 
 | ||||||
|  | export const name = 'perUserPv'; | ||||||
|  | 
 | ||||||
|  | export const schema = { | ||||||
|  | 	'upv.user': { uniqueIncrement: true, range: 'small' }, | ||||||
|  | 	'pv.user': { range: 'small' }, | ||||||
|  | 	'upv.visitor': { uniqueIncrement: true, range: 'small' }, | ||||||
|  | 	'pv.visitor': { range: 'small' }, | ||||||
|  | } as const; | ||||||
|  | 
 | ||||||
|  | export const entity = Chart.schemaToEntity(name, schema, true); | ||||||
							
								
								
									
										51
									
								
								packages/backend/src/core/chart/charts/per-user-pv.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								packages/backend/src/core/chart/charts/per-user-pv.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,51 @@ | ||||||
|  | import { Injectable, Inject } from '@nestjs/common'; | ||||||
|  | import { DataSource } from 'typeorm'; | ||||||
|  | import type { User } from '@/models/entities/User.js'; | ||||||
|  | import { AppLockService } from '@/core/AppLockService.js'; | ||||||
|  | import { DI } from '@/di-symbols.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
|  | import Chart from '../core.js'; | ||||||
|  | import { ChartLoggerService } from '../ChartLoggerService.js'; | ||||||
|  | import { name, schema } from './entities/per-user-pv.js'; | ||||||
|  | import type { KVs } from '../core.js'; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * ユーザーごとのプロフィール被閲覧数に関するチャート | ||||||
|  |  */ | ||||||
|  | // eslint-disable-next-line import/no-default-export
 | ||||||
|  | @Injectable() | ||||||
|  | export default class PerUserPvChart extends Chart<typeof schema> { | ||||||
|  | 	constructor( | ||||||
|  | 		@Inject(DI.db) | ||||||
|  | 		private db: DataSource, | ||||||
|  | 
 | ||||||
|  | 		private appLockService: AppLockService, | ||||||
|  | 		private chartLoggerService: ChartLoggerService, | ||||||
|  | 	) { | ||||||
|  | 		super(db, (k) => appLockService.getChartInsertLock(k), chartLoggerService.logger, name, schema, true); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	protected async tickMajor(): Promise<Partial<KVs<typeof schema>>> { | ||||||
|  | 		return {}; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	protected async tickMinor(): Promise<Partial<KVs<typeof schema>>> { | ||||||
|  | 		return {}; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	@bindThis | ||||||
|  | 	public async commitByUser(user: { id: User['id'] }, key: string): Promise<void> { | ||||||
|  | 		await this.commit({ | ||||||
|  | 			'upv.user': [key], | ||||||
|  | 			'pv.user': 1, | ||||||
|  | 		}, user.id); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	@bindThis | ||||||
|  | 	public async commitByVisitor(user: { id: User['id'] }, key: string): Promise<void> { | ||||||
|  | 		await this.commit({ | ||||||
|  | 			'upv.visitor': [key], | ||||||
|  | 			'pv.visitor': 1, | ||||||
|  | 		}, user.id); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -4,6 +4,7 @@ import { entity as UsersChart } from './charts/entities/users.js'; | ||||||
| import { entity as ActiveUsersChart } from './charts/entities/active-users.js'; | import { entity as ActiveUsersChart } from './charts/entities/active-users.js'; | ||||||
| import { entity as InstanceChart } from './charts/entities/instance.js'; | import { entity as InstanceChart } from './charts/entities/instance.js'; | ||||||
| import { entity as PerUserNotesChart } from './charts/entities/per-user-notes.js'; | import { entity as PerUserNotesChart } from './charts/entities/per-user-notes.js'; | ||||||
|  | import { entity as PerUserPvChart } from './charts/entities/per-user-pv.js'; | ||||||
| import { entity as DriveChart } from './charts/entities/drive.js'; | import { entity as DriveChart } from './charts/entities/drive.js'; | ||||||
| import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js'; | import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js'; | ||||||
| import { entity as HashtagChart } from './charts/entities/hashtag.js'; | import { entity as HashtagChart } from './charts/entities/hashtag.js'; | ||||||
|  | @ -23,6 +24,7 @@ export const entities = [ | ||||||
| 	ActiveUsersChart.hour, ActiveUsersChart.day, | 	ActiveUsersChart.hour, ActiveUsersChart.day, | ||||||
| 	InstanceChart.hour, InstanceChart.day, | 	InstanceChart.hour, InstanceChart.day, | ||||||
| 	PerUserNotesChart.hour, PerUserNotesChart.day, | 	PerUserNotesChart.hour, PerUserNotesChart.day, | ||||||
|  | 	PerUserPvChart.hour, PerUserPvChart.day, | ||||||
| 	DriveChart.hour, DriveChart.day, | 	DriveChart.hour, DriveChart.day, | ||||||
| 	PerUserReactionsChart.hour, PerUserReactionsChart.day, | 	PerUserReactionsChart.hour, PerUserReactionsChart.day, | ||||||
| 	HashtagChart.hour, HashtagChart.day, | 	HashtagChart.hour, HashtagChart.day, | ||||||
|  |  | ||||||
|  | @ -9,15 +9,16 @@ import UsersChart from '@/core/chart/charts/users.js'; | ||||||
| import ActiveUsersChart from '@/core/chart/charts/active-users.js'; | import ActiveUsersChart from '@/core/chart/charts/active-users.js'; | ||||||
| import InstanceChart from '@/core/chart/charts/instance.js'; | import InstanceChart from '@/core/chart/charts/instance.js'; | ||||||
| import PerUserNotesChart from '@/core/chart/charts/per-user-notes.js'; | import PerUserNotesChart from '@/core/chart/charts/per-user-notes.js'; | ||||||
|  | import PerUserPvChart from '@/core/chart/charts/per-user-pv.js'; | ||||||
| import DriveChart from '@/core/chart/charts/drive.js'; | import DriveChart from '@/core/chart/charts/drive.js'; | ||||||
| import PerUserReactionsChart from '@/core/chart/charts/per-user-reactions.js'; | import PerUserReactionsChart from '@/core/chart/charts/per-user-reactions.js'; | ||||||
| import HashtagChart from '@/core/chart/charts/hashtag.js'; | import HashtagChart from '@/core/chart/charts/hashtag.js'; | ||||||
| import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | ||||||
| import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | ||||||
| import ApRequestChart from '@/core/chart/charts/ap-request.js'; | import ApRequestChart from '@/core/chart/charts/ap-request.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { QueueLoggerService } from '../QueueLoggerService.js'; | import { QueueLoggerService } from '../QueueLoggerService.js'; | ||||||
| import type Bull from 'bull'; | import type Bull from 'bull'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
| 
 | 
 | ||||||
| @Injectable() | @Injectable() | ||||||
| export class CleanChartsProcessorService { | export class CleanChartsProcessorService { | ||||||
|  | @ -33,6 +34,7 @@ export class CleanChartsProcessorService { | ||||||
| 		private activeUsersChart: ActiveUsersChart, | 		private activeUsersChart: ActiveUsersChart, | ||||||
| 		private instanceChart: InstanceChart, | 		private instanceChart: InstanceChart, | ||||||
| 		private perUserNotesChart: PerUserNotesChart, | 		private perUserNotesChart: PerUserNotesChart, | ||||||
|  | 		private perUserPvChart: PerUserPvChart, | ||||||
| 		private driveChart: DriveChart, | 		private driveChart: DriveChart, | ||||||
| 		private perUserReactionsChart: PerUserReactionsChart, | 		private perUserReactionsChart: PerUserReactionsChart, | ||||||
| 		private hashtagChart: HashtagChart, | 		private hashtagChart: HashtagChart, | ||||||
|  | @ -56,6 +58,7 @@ export class CleanChartsProcessorService { | ||||||
| 			this.activeUsersChart.clean(), | 			this.activeUsersChart.clean(), | ||||||
| 			this.instanceChart.clean(), | 			this.instanceChart.clean(), | ||||||
| 			this.perUserNotesChart.clean(), | 			this.perUserNotesChart.clean(), | ||||||
|  | 			this.perUserPvChart.clean(), | ||||||
| 			this.driveChart.clean(), | 			this.driveChart.clean(), | ||||||
| 			this.perUserReactionsChart.clean(), | 			this.perUserReactionsChart.clean(), | ||||||
| 			this.hashtagChart.clean(), | 			this.hashtagChart.clean(), | ||||||
|  |  | ||||||
|  | @ -9,15 +9,16 @@ import UsersChart from '@/core/chart/charts/users.js'; | ||||||
| import ActiveUsersChart from '@/core/chart/charts/active-users.js'; | import ActiveUsersChart from '@/core/chart/charts/active-users.js'; | ||||||
| import InstanceChart from '@/core/chart/charts/instance.js'; | import InstanceChart from '@/core/chart/charts/instance.js'; | ||||||
| import PerUserNotesChart from '@/core/chart/charts/per-user-notes.js'; | import PerUserNotesChart from '@/core/chart/charts/per-user-notes.js'; | ||||||
|  | import PerUserPvChart from '@/core/chart/charts/per-user-pv.js'; | ||||||
| import DriveChart from '@/core/chart/charts/drive.js'; | import DriveChart from '@/core/chart/charts/drive.js'; | ||||||
| import PerUserReactionsChart from '@/core/chart/charts/per-user-reactions.js'; | import PerUserReactionsChart from '@/core/chart/charts/per-user-reactions.js'; | ||||||
| import HashtagChart from '@/core/chart/charts/hashtag.js'; | import HashtagChart from '@/core/chart/charts/hashtag.js'; | ||||||
| import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | ||||||
| import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | ||||||
| import ApRequestChart from '@/core/chart/charts/ap-request.js'; | import ApRequestChart from '@/core/chart/charts/ap-request.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { QueueLoggerService } from '../QueueLoggerService.js'; | import { QueueLoggerService } from '../QueueLoggerService.js'; | ||||||
| import type Bull from 'bull'; | import type Bull from 'bull'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
| 
 | 
 | ||||||
| @Injectable() | @Injectable() | ||||||
| export class TickChartsProcessorService { | export class TickChartsProcessorService { | ||||||
|  | @ -33,6 +34,7 @@ export class TickChartsProcessorService { | ||||||
| 		private activeUsersChart: ActiveUsersChart, | 		private activeUsersChart: ActiveUsersChart, | ||||||
| 		private instanceChart: InstanceChart, | 		private instanceChart: InstanceChart, | ||||||
| 		private perUserNotesChart: PerUserNotesChart, | 		private perUserNotesChart: PerUserNotesChart, | ||||||
|  | 		private perUserPvChart: PerUserPvChart, | ||||||
| 		private driveChart: DriveChart, | 		private driveChart: DriveChart, | ||||||
| 		private perUserReactionsChart: PerUserReactionsChart, | 		private perUserReactionsChart: PerUserReactionsChart, | ||||||
| 		private hashtagChart: HashtagChart, | 		private hashtagChart: HashtagChart, | ||||||
|  | @ -56,6 +58,7 @@ export class TickChartsProcessorService { | ||||||
| 			this.activeUsersChart.tick(false), | 			this.activeUsersChart.tick(false), | ||||||
| 			this.instanceChart.tick(false), | 			this.instanceChart.tick(false), | ||||||
| 			this.perUserNotesChart.tick(false), | 			this.perUserNotesChart.tick(false), | ||||||
|  | 			this.perUserPvChart.tick(false), | ||||||
| 			this.driveChart.tick(false), | 			this.driveChart.tick(false), | ||||||
| 			this.perUserReactionsChart.tick(false), | 			this.perUserReactionsChart.tick(false), | ||||||
| 			this.hashtagChart.tick(false), | 			this.hashtagChart.tick(false), | ||||||
|  |  | ||||||
|  | @ -99,6 +99,7 @@ import * as ep___charts_notes from './endpoints/charts/notes.js'; | ||||||
| import * as ep___charts_user_drive from './endpoints/charts/user/drive.js'; | import * as ep___charts_user_drive from './endpoints/charts/user/drive.js'; | ||||||
| import * as ep___charts_user_following from './endpoints/charts/user/following.js'; | import * as ep___charts_user_following from './endpoints/charts/user/following.js'; | ||||||
| import * as ep___charts_user_notes from './endpoints/charts/user/notes.js'; | import * as ep___charts_user_notes from './endpoints/charts/user/notes.js'; | ||||||
|  | import * as ep___charts_user_pv from './endpoints/charts/user/pv.js'; | ||||||
| import * as ep___charts_user_reactions from './endpoints/charts/user/reactions.js'; | import * as ep___charts_user_reactions from './endpoints/charts/user/reactions.js'; | ||||||
| import * as ep___charts_users from './endpoints/charts/users.js'; | import * as ep___charts_users from './endpoints/charts/users.js'; | ||||||
| import * as ep___clips_addNote from './endpoints/clips/add-note.js'; | import * as ep___clips_addNote from './endpoints/clips/add-note.js'; | ||||||
|  | @ -419,6 +420,7 @@ const $charts_notes: Provider = { provide: 'ep:charts/notes', useClass: ep___cha | ||||||
| const $charts_user_drive: Provider = { provide: 'ep:charts/user/drive', useClass: ep___charts_user_drive.default }; | const $charts_user_drive: Provider = { provide: 'ep:charts/user/drive', useClass: ep___charts_user_drive.default }; | ||||||
| const $charts_user_following: Provider = { provide: 'ep:charts/user/following', useClass: ep___charts_user_following.default }; | const $charts_user_following: Provider = { provide: 'ep:charts/user/following', useClass: ep___charts_user_following.default }; | ||||||
| const $charts_user_notes: Provider = { provide: 'ep:charts/user/notes', useClass: ep___charts_user_notes.default }; | const $charts_user_notes: Provider = { provide: 'ep:charts/user/notes', useClass: ep___charts_user_notes.default }; | ||||||
|  | const $charts_user_pv: Provider = { provide: 'ep:charts/user/pv', useClass: ep___charts_user_pv.default }; | ||||||
| const $charts_user_reactions: Provider = { provide: 'ep:charts/user/reactions', useClass: ep___charts_user_reactions.default }; | const $charts_user_reactions: Provider = { provide: 'ep:charts/user/reactions', useClass: ep___charts_user_reactions.default }; | ||||||
| const $charts_users: Provider = { provide: 'ep:charts/users', useClass: ep___charts_users.default }; | const $charts_users: Provider = { provide: 'ep:charts/users', useClass: ep___charts_users.default }; | ||||||
| const $clips_addNote: Provider = { provide: 'ep:clips/add-note', useClass: ep___clips_addNote.default }; | const $clips_addNote: Provider = { provide: 'ep:clips/add-note', useClass: ep___clips_addNote.default }; | ||||||
|  | @ -743,6 +745,7 @@ const $retention: Provider = { provide: 'ep:retention', useClass: ep___retention | ||||||
| 		$charts_user_drive, | 		$charts_user_drive, | ||||||
| 		$charts_user_following, | 		$charts_user_following, | ||||||
| 		$charts_user_notes, | 		$charts_user_notes, | ||||||
|  | 		$charts_user_pv, | ||||||
| 		$charts_user_reactions, | 		$charts_user_reactions, | ||||||
| 		$charts_users, | 		$charts_users, | ||||||
| 		$clips_addNote, | 		$clips_addNote, | ||||||
|  | @ -1061,6 +1064,7 @@ const $retention: Provider = { provide: 'ep:retention', useClass: ep___retention | ||||||
| 		$charts_user_drive, | 		$charts_user_drive, | ||||||
| 		$charts_user_following, | 		$charts_user_following, | ||||||
| 		$charts_user_notes, | 		$charts_user_notes, | ||||||
|  | 		$charts_user_pv, | ||||||
| 		$charts_user_reactions, | 		$charts_user_reactions, | ||||||
| 		$charts_users, | 		$charts_users, | ||||||
| 		$clips_addNote, | 		$clips_addNote, | ||||||
|  |  | ||||||
|  | @ -98,6 +98,7 @@ import * as ep___charts_notes from './endpoints/charts/notes.js'; | ||||||
| import * as ep___charts_user_drive from './endpoints/charts/user/drive.js'; | import * as ep___charts_user_drive from './endpoints/charts/user/drive.js'; | ||||||
| import * as ep___charts_user_following from './endpoints/charts/user/following.js'; | import * as ep___charts_user_following from './endpoints/charts/user/following.js'; | ||||||
| import * as ep___charts_user_notes from './endpoints/charts/user/notes.js'; | import * as ep___charts_user_notes from './endpoints/charts/user/notes.js'; | ||||||
|  | import * as ep___charts_user_pv from './endpoints/charts/user/pv.js'; | ||||||
| import * as ep___charts_user_reactions from './endpoints/charts/user/reactions.js'; | import * as ep___charts_user_reactions from './endpoints/charts/user/reactions.js'; | ||||||
| import * as ep___charts_users from './endpoints/charts/users.js'; | import * as ep___charts_users from './endpoints/charts/users.js'; | ||||||
| import * as ep___clips_addNote from './endpoints/clips/add-note.js'; | import * as ep___clips_addNote from './endpoints/clips/add-note.js'; | ||||||
|  | @ -416,6 +417,7 @@ const eps = [ | ||||||
| 	['charts/user/drive', ep___charts_user_drive], | 	['charts/user/drive', ep___charts_user_drive], | ||||||
| 	['charts/user/following', ep___charts_user_following], | 	['charts/user/following', ep___charts_user_following], | ||||||
| 	['charts/user/notes', ep___charts_user_notes], | 	['charts/user/notes', ep___charts_user_notes], | ||||||
|  | 	['charts/user/pv', ep___charts_user_pv], | ||||||
| 	['charts/user/reactions', ep___charts_user_reactions], | 	['charts/user/reactions', ep___charts_user_reactions], | ||||||
| 	['charts/users', ep___charts_users], | 	['charts/users', ep___charts_users], | ||||||
| 	['clips/add-note', ep___clips_addNote], | 	['clips/add-note', ep___clips_addNote], | ||||||
|  |  | ||||||
							
								
								
									
										37
									
								
								packages/backend/src/server/api/endpoints/charts/user/pv.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								packages/backend/src/server/api/endpoints/charts/user/pv.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,37 @@ | ||||||
|  | import { Inject, Injectable } from '@nestjs/common'; | ||||||
|  | import { getJsonSchema } from '@/core/chart/core.js'; | ||||||
|  | import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||||
|  | import PerUserPvChart from '@/core/chart/charts/per-user-pv.js'; | ||||||
|  | import { schema } from '@/core/chart/charts/entities/per-user-notes.js'; | ||||||
|  | 
 | ||||||
|  | export const meta = { | ||||||
|  | 	tags: ['charts', 'users'], | ||||||
|  | 
 | ||||||
|  | 	res: getJsonSchema(schema), | ||||||
|  | 
 | ||||||
|  | 	allowGet: true, | ||||||
|  | 	cacheSec: 60 * 60, | ||||||
|  | } as const; | ||||||
|  | 
 | ||||||
|  | export const paramDef = { | ||||||
|  | 	type: 'object', | ||||||
|  | 	properties: { | ||||||
|  | 		span: { type: 'string', enum: ['day', 'hour'] }, | ||||||
|  | 		limit: { type: 'integer', minimum: 1, maximum: 500, default: 30 }, | ||||||
|  | 		offset: { type: 'integer', nullable: true, default: null }, | ||||||
|  | 		userId: { type: 'string', format: 'misskey:id' }, | ||||||
|  | 	}, | ||||||
|  | 	required: ['span', 'userId'], | ||||||
|  | } as const; | ||||||
|  | 
 | ||||||
|  | // eslint-disable-next-line import/no-default-export
 | ||||||
|  | @Injectable() | ||||||
|  | export default class extends Endpoint<typeof meta, typeof paramDef> { | ||||||
|  | 	constructor( | ||||||
|  | 		private perUserPvChart: PerUserPvChart, | ||||||
|  | 	) { | ||||||
|  | 		super(meta, paramDef, async (ps, me) => { | ||||||
|  | 			return await this.perUserPvChart.getChart(ps.span, ps.limit, ps.offset ? new Date(ps.offset) : null, ps.userId); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -6,6 +6,7 @@ import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { RemoteUserResolveService } from '@/core/RemoteUserResolveService.js'; | import { RemoteUserResolveService } from '@/core/RemoteUserResolveService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
|  | import PerUserPvChart from '@/core/chart/charts/per-user-pv.js'; | ||||||
| import { ApiError } from '../../error.js'; | import { ApiError } from '../../error.js'; | ||||||
| import { ApiLoggerService } from '../../ApiLoggerService.js'; | import { ApiLoggerService } from '../../ApiLoggerService.js'; | ||||||
| import type { FindOptionsWhere } from 'typeorm'; | import type { FindOptionsWhere } from 'typeorm'; | ||||||
|  | @ -90,9 +91,10 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { | ||||||
| 
 | 
 | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private remoteUserResolveService: RemoteUserResolveService, | 		private remoteUserResolveService: RemoteUserResolveService, | ||||||
|  | 		private perUserPvChart: PerUserPvChart, | ||||||
| 		private apiLoggerService: ApiLoggerService, | 		private apiLoggerService: ApiLoggerService, | ||||||
| 	) { | 	) { | ||||||
| 		super(meta, paramDef, async (ps, me) => { | 		super(meta, paramDef, async (ps, me, _1, _2, _3, ip) => { | ||||||
| 			let user; | 			let user; | ||||||
| 
 | 
 | ||||||
| 			const isAdminOrModerator = me && (me.isAdmin || me.isModerator); | 			const isAdminOrModerator = me && (me.isAdmin || me.isModerator); | ||||||
|  | @ -137,6 +139,12 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { | ||||||
| 					throw new ApiError(meta.errors.noSuchUser); | 					throw new ApiError(meta.errors.noSuchUser); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
|  | 				if (me == null && ip != null) { | ||||||
|  | 					this.perUserPvChart.commitByVisitor(user, ip); | ||||||
|  | 				} else if (me && me.id !== user.id) { | ||||||
|  | 					this.perUserPvChart.commitByUser(user, me.id); | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
| 				return await this.userEntityService.pack(user, me, { | 				return await this.userEntityService.pack(user, me, { | ||||||
| 					detail: true, | 					detail: true, | ||||||
| 				}); | 				}); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue