/* * SPDX-FileCopyrightText: syuilo and misskey-project * SPDX-License-Identifier: AGPL-3.0-only */ import { Injectable, Inject } from '@nestjs/common'; import { DataSource } from 'typeorm'; import * as Redis from 'ioredis'; import type { DriveFilesRepository } from '@/models/_.js'; import type { MiDriveFile } from '@/models/DriveFile.js'; import { DI } from '@/di-symbols.js'; import { DriveFileEntityService } from '@/core/entities/DriveFileEntityService.js'; import { bindThis } from '@/decorators.js'; import { acquireChartInsertLock } from '@/misc/distributed-lock.js'; import Chart from '../core.js'; import { ChartLoggerService } from '../ChartLoggerService.js'; import { name, schema } from './entities/per-user-drive.js'; import type { KVs } from '../core.js'; /** * ユーザーごとのドライブに関するチャート */ @Injectable() export default class PerUserDriveChart extends Chart { // eslint-disable-line import/no-default-export constructor( @Inject(DI.db) private db: DataSource, @Inject(DI.redis) private redisClient: Redis.Redis, @Inject(DI.driveFilesRepository) private driveFilesRepository: DriveFilesRepository, private driveFileEntityService: DriveFileEntityService, private chartLoggerService: ChartLoggerService, ) { super(db, (k) => acquireChartInsertLock(redisClient, k), chartLoggerService.logger, name, schema, true); } protected async tickMajor(group: string): Promise>> { const [count, size] = await Promise.all([ this.driveFilesRepository.countBy({ userId: group }), this.driveFileEntityService.calcDriveUsageOf(group), ]); return { 'totalCount': count, 'totalSize': size, }; } protected async tickMinor(): Promise>> { return {}; } @bindThis public async update(file: MiDriveFile, isAdditional: boolean): Promise { const fileSizeKb = file.size / 1000; await this.commit({ 'totalCount': isAdditional ? 1 : -1, 'totalSize': isAdditional ? fileSizeKb : -fileSizeKb, 'incCount': isAdditional ? 1 : 0, 'incSize': isAdditional ? fileSizeKb : 0, 'decCount': isAdditional ? 0 : 1, 'decSize': isAdditional ? 0 : fileSizeKb, }, file.userId); } }