2021-05-03 23:24:40 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2024-08-14 23:58:01 +00:00
|
|
|
import { readFileSync } from 'node:fs';
|
|
|
|
import { join } from 'node:path';
|
2021-05-03 23:24:40 +00:00
|
|
|
import { assert } from 'chai';
|
|
|
|
import * as moment from 'moment';
|
2021-09-18 00:30:08 +00:00
|
|
|
import { setupI18n } from '../../util/setupI18n';
|
2022-11-16 20:18:02 +00:00
|
|
|
import { DurationInSeconds } from '../../util/durations';
|
2024-08-14 23:58:01 +00:00
|
|
|
import type { LocaleMessagesType } from '../../types/I18N';
|
2021-05-03 23:24:40 +00:00
|
|
|
|
|
|
|
import * as expirationTimer from '../../util/expirationTimer';
|
|
|
|
|
2024-08-14 23:58:01 +00:00
|
|
|
function loadMessages(locale: string): LocaleMessagesType {
|
|
|
|
const localePath = join(
|
|
|
|
__dirname,
|
|
|
|
'..',
|
|
|
|
'..',
|
|
|
|
'..',
|
|
|
|
'_locales',
|
|
|
|
locale,
|
|
|
|
'messages.json'
|
|
|
|
);
|
|
|
|
const json = readFileSync(localePath, 'utf8');
|
|
|
|
return JSON.parse(json) as LocaleMessagesType;
|
|
|
|
}
|
|
|
|
|
|
|
|
describe('expiration timer utilities', async () => {
|
|
|
|
const enMessages = loadMessages('en');
|
2021-05-03 23:24:40 +00:00
|
|
|
const i18n = setupI18n('en', enMessages);
|
|
|
|
|
|
|
|
describe('DEFAULT_DURATIONS_IN_SECONDS', () => {
|
|
|
|
const { DEFAULT_DURATIONS_IN_SECONDS } = expirationTimer;
|
|
|
|
|
|
|
|
it('includes at least 3 durations', () => {
|
|
|
|
assert.isAtLeast(DEFAULT_DURATIONS_IN_SECONDS.length, 3);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('includes 1 hour as seconds', () => {
|
2022-11-16 20:18:02 +00:00
|
|
|
const oneHour = DurationInSeconds.fromHours(1);
|
2021-05-03 23:24:40 +00:00
|
|
|
assert.include(DEFAULT_DURATIONS_IN_SECONDS, oneHour);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('format', () => {
|
|
|
|
const { format } = expirationTimer;
|
|
|
|
|
|
|
|
it('handles an undefined duration', () => {
|
|
|
|
assert.strictEqual(format(i18n, undefined), 'off');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('handles no duration', () => {
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(format(i18n, DurationInSeconds.ZERO), 'off');
|
2021-05-03 23:24:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('formats durations', () => {
|
|
|
|
new Map<number, string>([
|
|
|
|
[1, '1 second'],
|
|
|
|
[2, '2 seconds'],
|
|
|
|
[30, '30 seconds'],
|
|
|
|
[59, '59 seconds'],
|
|
|
|
[moment.duration(1, 'm').asSeconds(), '1 minute'],
|
|
|
|
[moment.duration(5, 'm').asSeconds(), '5 minutes'],
|
|
|
|
[moment.duration(1, 'h').asSeconds(), '1 hour'],
|
|
|
|
[moment.duration(8, 'h').asSeconds(), '8 hours'],
|
|
|
|
[moment.duration(1, 'd').asSeconds(), '1 day'],
|
|
|
|
[moment.duration(6, 'd').asSeconds(), '6 days'],
|
|
|
|
[moment.duration(8, 'd').asSeconds(), '8 days'],
|
|
|
|
[moment.duration(30, 'd').asSeconds(), '30 days'],
|
|
|
|
[moment.duration(365, 'd').asSeconds(), '365 days'],
|
|
|
|
[moment.duration(1, 'w').asSeconds(), '1 week'],
|
|
|
|
[moment.duration(3, 'w').asSeconds(), '3 weeks'],
|
|
|
|
[moment.duration(52, 'w').asSeconds(), '52 weeks'],
|
|
|
|
]).forEach((expected, input) => {
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(input)),
|
|
|
|
expected
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('formats other languages successfully', () => {
|
2024-08-14 23:58:01 +00:00
|
|
|
const esMessages = loadMessages('es');
|
2021-05-03 23:24:40 +00:00
|
|
|
const esI18n = setupI18n('es', esMessages);
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(esI18n, DurationInSeconds.fromSeconds(120)),
|
|
|
|
'2 minutos'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
|
2024-08-14 23:58:01 +00:00
|
|
|
const zhCnMessages = loadMessages('zh-CN');
|
2022-09-27 21:01:06 +00:00
|
|
|
const zhCnI18n = setupI18n('zh-CN', zhCnMessages);
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(zhCnI18n, DurationInSeconds.fromSeconds(60)),
|
|
|
|
'1 分钟'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
|
|
|
|
// The underlying library supports the "pt" locale, not the "pt_BR" locale. That's
|
|
|
|
// what we're testing here.
|
2024-08-14 23:58:01 +00:00
|
|
|
const ptBrMessages = loadMessages('pt-BR');
|
2021-05-03 23:24:40 +00:00
|
|
|
const ptBrI18n = setupI18n('pt_BR', ptBrMessages);
|
|
|
|
assert.strictEqual(
|
2022-11-16 20:18:02 +00:00
|
|
|
format(ptBrI18n, DurationInSeconds.fromDays(5)),
|
2021-05-03 23:24:40 +00:00
|
|
|
'5 dias'
|
|
|
|
);
|
|
|
|
|
|
|
|
// The underlying library supports the Norwegian language, which is a macrolanguage
|
|
|
|
// for Bokmål and Nynorsk.
|
2024-08-14 23:58:01 +00:00
|
|
|
const nbMessages = loadMessages('nb');
|
|
|
|
const nlMessages = loadMessages('nl');
|
2022-09-27 21:01:06 +00:00
|
|
|
[setupI18n('nb', nbMessages), setupI18n('nn', nlMessages)].forEach(
|
2021-05-03 23:24:40 +00:00
|
|
|
norwegianI18n => {
|
|
|
|
assert.strictEqual(
|
2022-11-16 20:18:02 +00:00
|
|
|
format(norwegianI18n, DurationInSeconds.fromHours(6)),
|
2021-05-03 23:24:40 +00:00
|
|
|
'6 timer'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('falls back to English if the locale is not supported', () => {
|
|
|
|
const badI18n = setupI18n('bogus', {});
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(badI18n, DurationInSeconds.fromSeconds(120)),
|
|
|
|
'2 minutes'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('handles a "mix" of units gracefully', () => {
|
|
|
|
// We don't expect there to be a "mix" of units, but we shouldn't choke if a bad
|
|
|
|
// client gives us an unexpected timestamp.
|
2022-11-16 20:18:02 +00:00
|
|
|
const mix = DurationInSeconds.fromSeconds(
|
|
|
|
moment.duration(6, 'days').add(moment.duration(2, 'hours')).asSeconds()
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
assert.strictEqual(format(i18n, mix), '6 days, 2 hours');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('handles negative numbers gracefully', () => {
|
|
|
|
// The proto helps enforce non-negative numbers by specifying a u32, but because
|
|
|
|
// JavaScript lacks such a type, we test it here.
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-1)),
|
|
|
|
'1 second'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-120)),
|
|
|
|
'2 minutes'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-0)),
|
|
|
|
'off'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('handles fractional seconds gracefully', () => {
|
|
|
|
// The proto helps enforce integer numbers by specifying a u32, but this function
|
|
|
|
// shouldn't choke if bad data is passed somehow.
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(4.2)),
|
|
|
|
'4 seconds'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(4.8)),
|
|
|
|
'4 seconds'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(0.2)),
|
|
|
|
'1 second'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(0.8)),
|
|
|
|
'1 second'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
|
|
|
|
// If multiple things go wrong and we pass a fractional negative number, we still
|
|
|
|
// shouldn't explode.
|
2022-11-16 20:18:02 +00:00
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-4.2)),
|
|
|
|
'4 seconds'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-4.8)),
|
|
|
|
'4 seconds'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-0.2)),
|
|
|
|
'1 second'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
format(i18n, DurationInSeconds.fromSeconds(-0.8)),
|
|
|
|
'1 second'
|
|
|
|
);
|
2021-05-03 23:24:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|