add a probably-not-terrible string-equality function #407

This commit is contained in:
dakkar 2024-02-09 18:22:32 +00:00
parent 9d6c24fb6d
commit b6c59fa98e
4 changed files with 64 additions and 5 deletions

View file

@ -11,7 +11,7 @@ import type { MiUserProfile, UserProfilesRepository, UsersRepository } from '@/m
import { bindThis } from '@/decorators.js';
import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js';
import type { MiLocalUser } from '@/models/User.js';
import * as crypto from 'node:crypto';
import { secureishCompare } from '@/misc/secure-ish-compare.js';
@Injectable()
export class UserAuthService {
@ -29,7 +29,7 @@ export class UserAuthService {
if (profile.twoFactorBackupSecret?.includes(token)) {
await this.userProfilesRepository.update({ userId: profile.userId }, {
twoFactorBackupSecret: profile.twoFactorBackupSecret.filter(
(secret) => !crypto.timingSafeEqual(secret, token)
(secret) => !secureishCompare(secret, token)
),
});
} else {

View file

@ -0,0 +1,33 @@
/*
* SPDX-FileCopyrightText: dakkar and other sharkey contributors
* SPDX-License-Identifier: AGPL-3.0-only
*/
import * as crypto from 'node:crypto';
export function secureishCompare(expected: string, given:string) {
const expectedLen = expected.length;
const givenLen = given.length;
/*
this ensures that the two strings are the same length, so
`timingSafeEqual` will not throw an error
notice that we perform the same operations regardless of the
length of the strings, we're trying not to leak timing
information!
*/
const paddedExpected = Buffer.from(expected + given);
const paddedGiven = Buffer.from(given + expected);
/*
if the two strings were equal, `sortOfEqual` will be true
if will be true in other cases, too! like `abc` and `abcabc`; but
then, `expectedLen` and `givenLen` would be different, and we'd
return false
*/
const sortOfEqual = crypto.timingSafeEqual(paddedExpected, paddedGiven);
return sortOfEqual && (givenLen === expectedLen);
}

View file

@ -38,6 +38,7 @@ import type { FastifyInstance, FastifyRequest, FastifyReply, FastifyPluginOption
import type { FindOptionsWhere } from 'typeorm';
import type Logger from '@/logger.js';
import { LoggerService } from '@/core/LoggerService.js';
import { secureishCompare } from '@/misc/secure-ish-compare.js';
const ACTIVITY_JSON = 'application/activity+json; charset=utf-8';
const LD_JSON = 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"; charset=utf-8';
@ -287,7 +288,7 @@ export class ActivityPubServerService {
const hash = crypto.createHash('sha256').update(request.rawBody).digest('base64');
if (! crypto.timingSafeEqual(hash, digestValue)) {
if (!secureishCompare(hash, digestValue)) {
// Invalid digest
reply.code(401);
return;
@ -795,7 +796,7 @@ export class ActivityPubServerService {
fastify.get<{ Params: { user: string; } }>('/users/:user', { constraints: { apOrHtml: 'ap' } }, async (request, reply) => {
if (await this.shouldRefuseGetRequest(request, reply, request.params.user)) return;
vary(reply.raw, 'Accept');
const userId = request.params.user;
@ -811,7 +812,7 @@ export class ActivityPubServerService {
fastify.get<{ Params: { user: string; } }>('/@:user', { constraints: { apOrHtml: 'ap' } }, async (request, reply) => {
if (await this.shouldRefuseGetRequest(request, reply, request.params.user)) return;
vary(reply.raw, 'Accept');
const user = await this.usersRepository.findOneBy({

View file

@ -0,0 +1,25 @@
/*
* SPDX-FileCopyrightText: dakkar and other sharkey contributors
* SPDX-License-Identifier: AGPL-3.0-only
*/
import { secureishCompare } from '@/misc/secure-ish-compare.js';
describe(secureishCompare, () => {
it('should return true if strings are equal', () => {
expect(secureishCompare('abc','abc')).toBe(true);
expect(secureishCompare('aaa','aaa')).toBe(true);
});
it('should return false if strings are different', () => {
expect(secureishCompare('abc','def')).toBe(false);
});
it('should return false if one is prefix of the other', () => {
expect(secureishCompare('abc','abcabc')).toBe(false);
expect(secureishCompare('abcabc','abc')).toBe(false);
expect(secureishCompare('aaa','aa')).toBe(false);
});
it('should return false if strings are very different', () => {
expect(secureishCompare('abc','defghi')).toBe(false);
expect(secureishCompare('defghi','abc')).toBe(false);
});
});