2023-09-22 14:12:33 +09:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
|
|
import * as OTPAuth from 'otpauth';
|
|
|
|
import { DI } from '@/di-symbols.js';
|
|
|
|
import type { MiUserProfile, UserProfilesRepository, UsersRepository } from '@/models/_.js';
|
|
|
|
import { bindThis } from '@/decorators.js';
|
2023-11-20 05:32:24 +09:00
|
|
|
import { IdentifiableError } from "@/misc/identifiable-error.js";
|
2023-09-22 14:12:33 +09:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class UserAuthService {
|
|
|
|
constructor(
|
|
|
|
@Inject(DI.usersRepository)
|
|
|
|
private usersRepository: UsersRepository,
|
|
|
|
|
|
|
|
@Inject(DI.userProfilesRepository)
|
|
|
|
private userProfilesRepository: UserProfilesRepository,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public async twoFactorAuthenticate(profile: MiUserProfile, token: string): Promise<void> {
|
|
|
|
if (profile.twoFactorBackupSecret?.includes(token)) {
|
|
|
|
await this.userProfilesRepository.update({ userId: profile.userId }, {
|
|
|
|
twoFactorBackupSecret: profile.twoFactorBackupSecret.filter((secret) => secret !== token),
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
const delta = OTPAuth.TOTP.validate({
|
|
|
|
secret: OTPAuth.Secret.fromBase32(profile.twoFactorSecret!),
|
|
|
|
digits: 6,
|
|
|
|
token,
|
|
|
|
window: 5,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (delta === null) {
|
2023-11-20 05:32:24 +09:00
|
|
|
throw new IdentifiableError('7d0a7d85-206c-4d16-8cf3-8af92249a082', 'authentication failed');
|
2023-09-22 14:12:33 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|