2023-07-27 05:31:52 +00:00
|
|
|
/*
|
2024-02-12 02:37:45 +00:00
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 05:31:52 +00:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-06-10 04:45:11 +00:00
|
|
|
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
|
2022-09-17 18:27:08 +00:00
|
|
|
import { DI } from '@/di-symbols.js';
|
2023-09-15 05:28:29 +00:00
|
|
|
import type { AccessTokensRepository, AppsRepository, UsersRepository } from '@/models/_.js';
|
2023-09-20 02:33:36 +00:00
|
|
|
import type { MiLocalUser } from '@/models/User.js';
|
|
|
|
import type { MiAccessToken } from '@/models/AccessToken.js';
|
2023-04-04 06:56:47 +00:00
|
|
|
import { MemoryKVCache } from '@/misc/cache.js';
|
2023-09-20 02:33:36 +00:00
|
|
|
import type { MiApp } from '@/models/App.js';
|
2023-04-04 08:32:09 +00:00
|
|
|
import { CacheService } from '@/core/CacheService.js';
|
2022-09-17 18:27:08 +00:00
|
|
|
import isNativeToken from '@/misc/is-native-token.js';
|
2022-12-04 06:03:09 +00:00
|
|
|
import { bindThis } from '@/decorators.js';
|
2022-09-17 18:27:08 +00:00
|
|
|
|
|
|
|
export class AuthenticationError extends Error {
|
|
|
|
constructor(message: string) {
|
|
|
|
super(message);
|
|
|
|
this.name = 'AuthenticationError';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Injectable()
|
2023-06-10 04:45:11 +00:00
|
|
|
export class AuthenticateService implements OnApplicationShutdown {
|
2023-08-16 08:51:28 +00:00
|
|
|
private appCache: MemoryKVCache<MiApp>;
|
2022-09-17 18:27:08 +00:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@Inject(DI.usersRepository)
|
|
|
|
private usersRepository: UsersRepository,
|
|
|
|
|
|
|
|
@Inject(DI.accessTokensRepository)
|
|
|
|
private accessTokensRepository: AccessTokensRepository,
|
|
|
|
|
|
|
|
@Inject(DI.appsRepository)
|
|
|
|
private appsRepository: AppsRepository,
|
|
|
|
|
2023-04-04 08:32:09 +00:00
|
|
|
private cacheService: CacheService,
|
2022-09-17 18:27:08 +00:00
|
|
|
) {
|
2023-08-16 08:51:28 +00:00
|
|
|
this.appCache = new MemoryKVCache<MiApp>(Infinity);
|
2022-09-17 18:27:08 +00:00
|
|
|
}
|
|
|
|
|
2022-12-04 06:03:09 +00:00
|
|
|
@bindThis
|
2023-08-16 08:51:28 +00:00
|
|
|
public async authenticate(token: string | null | undefined): Promise<[MiLocalUser | null, MiAccessToken | null]> {
|
2022-09-17 18:27:08 +00:00
|
|
|
if (token == null) {
|
|
|
|
return [null, null];
|
|
|
|
}
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (isNativeToken(token)) {
|
2023-04-04 08:32:09 +00:00
|
|
|
const user = await this.cacheService.localUserByNativeTokenCache.fetch(token,
|
2023-08-16 08:51:28 +00:00
|
|
|
() => this.usersRepository.findOneBy({ token }) as Promise<MiLocalUser | null>);
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (user == null) {
|
|
|
|
throw new AuthenticationError('user not found');
|
|
|
|
}
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
return [user, null];
|
|
|
|
} else {
|
|
|
|
const accessToken = await this.accessTokensRepository.findOne({
|
|
|
|
where: [{
|
|
|
|
hash: token.toLowerCase(), // app
|
|
|
|
}, {
|
|
|
|
token: token, // miauth
|
|
|
|
}],
|
|
|
|
});
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (accessToken == null) {
|
|
|
|
throw new AuthenticationError('invalid signature');
|
|
|
|
}
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
this.accessTokensRepository.update(accessToken.id, {
|
|
|
|
lastUsedAt: new Date(),
|
|
|
|
});
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2023-04-04 08:32:09 +00:00
|
|
|
const user = await this.cacheService.localUserByIdCache.fetch(accessToken.userId,
|
2022-09-17 18:27:08 +00:00
|
|
|
() => this.usersRepository.findOneBy({
|
|
|
|
id: accessToken.userId,
|
2023-08-16 08:51:28 +00:00
|
|
|
}) as Promise<MiLocalUser>);
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
if (accessToken.appId) {
|
2022-09-18 18:11:50 +00:00
|
|
|
const app = await this.appCache.fetch(accessToken.appId,
|
2022-09-17 18:27:08 +00:00
|
|
|
() => this.appsRepository.findOneByOrFail({ id: accessToken.appId! }));
|
2023-07-07 22:08:16 +00:00
|
|
|
|
2022-09-17 18:27:08 +00:00
|
|
|
return [user, {
|
|
|
|
id: accessToken.id,
|
|
|
|
permission: app.permission,
|
2023-08-16 08:51:28 +00:00
|
|
|
} as MiAccessToken];
|
2022-09-17 18:27:08 +00:00
|
|
|
} else {
|
|
|
|
return [user, accessToken];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-10 04:45:11 +00:00
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public dispose(): void {
|
|
|
|
this.appCache.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
|
|
this.dispose();
|
|
|
|
}
|
2022-09-17 18:27:08 +00:00
|
|
|
}
|