Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add function to register custom keysets [WIP] #1844

Open
wants to merge 12 commits into
base: main
Choose a base branch
from
5 changes: 3 additions & 2 deletions src/components/theme/types.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Use (string & {}) for better autocomplete https://stackoverflow.com/a/61048124
export type RealTheme = 'light' | 'light-hc' | 'dark' | 'dark-hc' | (string & {});
import type {AutocompleteSafeString} from '../../utils/autocomplete-safe-string';

export type RealTheme = 'light' | 'light-hc' | 'dark' | 'dark-hc' | AutocompleteSafeString;
export type ThemeType = 'light' | 'dark';
export type Theme = 'system' | RealTheme;
export type Direction = 'ltr' | 'rtl';
Expand Down
4 changes: 3 additions & 1 deletion src/components/utils/configure.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
import type {AutocompleteSafeString} from '../../utils/autocomplete-safe-string';

export enum Lang {
Ru = 'ru',
En = 'en',
}

interface Config {
lang: `${Lang}`;
lang: `${Lang}` | AutocompleteSafeString;
Copy link
Author

@NikitaShkaruba NikitaShkaruba Sep 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change is concerning, but is needed for selecting custom keysets via configure. I've read all the codes, and I think it's alright. But what do you think, guys?

}

type Subscriber = (config: Config) => void;
Expand Down
226 changes: 226 additions & 0 deletions src/components/utils/registerCustomKeysets.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
import React from 'react';

import type {KeysetData} from '@gravity-ui/i18n';

import {render, screen} from '../../../test-utils/utils';
import {Dialog} from '../Dialog';
import {Pagination} from '../Pagination';

import {Lang, configure} from './configure';
import {registerCustomKeysets} from './registerCustomKeysets';

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it ok, that I don't have describe, or should I encapsulate everything in it? I haven't done it, because jest splits tests by files anyway

test('should render components with custom keysets', () => {
registerCustomKeysets(
'rs',
createTestCustomKeyset({
Dialog: {
close: 'Затвори дијалог',
},
Pagination: {
button_previous: 'Претходно',
button_next: 'Следеће',
button_first: 'Прво',
label_select_size: 'Изаберите величину странице',
'label_input-placeholder': 'Стр.',
'label_page-of': 'из',
},
}),
);
registerCustomKeysets(
'it',
createTestCustomKeyset({
Dialog: {
close: 'Chiudere il dialogo',
},
Pagination: {
button_previous: 'Precedente',
button_next: 'Avanti',
button_first: 'Primo',
label_select_size: 'Seleziona la dimensione della pagina',
'label_input-placeholder': 'Pagina n.',
'label_page-of': 'di',
},
}),
);

configure({lang: 'rs'});
render(<TestComponents />);
expect(screen.getByRole('button', {name: 'Затвори дијалог'})).toBeInTheDocument();
expect(screen.getByRole('button', {name: 'Претходно'})).toBeInTheDocument();

configure({lang: 'it'});
render(<TestComponents />);
expect(screen.getByRole('button', {name: 'Chiudere il dialogo'})).toBeInTheDocument();
expect(screen.getByRole('button', {name: 'Precedente'})).toBeInTheDocument();
});

test('should render components with bundled keysets after a custom keysets registration', () => {
registerCustomKeysets(
'rs',
createTestCustomKeyset({
Dialog: {
close: 'Затвори дијалог',
},
Pagination: {
button_previous: 'Претходно',
button_next: 'Следеће',
button_first: 'Прво',
label_select_size: 'Изаберите величину странице',
'label_input-placeholder': 'Стр.',
'label_page-of': 'из',
},
}),
);

configure({lang: Lang.En});
render(<TestComponents />);
expect(screen.getByRole('button', {name: 'Close dialog'})).toBeInTheDocument();
expect(screen.getByRole('button', {name: 'Previous'})).toBeInTheDocument();
});

test('should override bundled keysets', () => {
registerCustomKeysets(
Lang.En,
createTestCustomKeyset({
Dialog: {
close: '[Overriden] Close dialog',
},
Pagination: {
button_previous: '[Overriden] Previous',
button_next: '[Overriden] Next',
button_first: '[Overriden] First',
label_select_size: '[Overriden] Select page size',
'label_input-placeholder': '[Overriden] Page #',
'label_page-of': '[Overriden] of',
},
}),
);

configure({lang: Lang.En});
render(<TestComponents />);
expect(screen.getByRole('button', {name: '[Overriden] Close dialog'})).toBeInTheDocument();
expect(screen.getByRole('button', {name: '[Overriden] Previous'})).toBeInTheDocument();
});

test('should throw an error if a component keyset is not provided', () => {
const keysetData = createTestCustomKeyset({});
delete keysetData.Table;

expect(() => registerCustomKeysets('rs', keysetData)).toThrow();
});

test('should throw an error if a component key is not provided', () => {
const keysetData = createTestCustomKeyset({});
Object.assign(keysetData, {
Table: {
label_empty: 'empty',
// The values are omitted on purpose
// 'label-actions': pluggedValue,
// 'label-row-select': pluggedValue,
},
});

expect(() => registerCustomKeysets('rs', keysetData)).toThrow();
});

test('should throw an error if excess components are provided', () => {
const keysetData = createTestCustomKeyset({
NonexistentComponent1: {
label_cancel: 'cancel',
label_submit: 'submit',
},
NonexistentComponent2: {
label_load: 'load',
label_preload: 'preload',
},
});
expect(() => registerCustomKeysets('it', keysetData)).toThrow();
});

test('should throw an error if excess component keys are provided', () => {
const keysetData = createTestCustomKeyset({
Alert: {
label_close: 'cancel',
nonexistent_key: 'nonexistent',
nonexistent_key2: 'nonexistent2',
},
});
expect(() => registerCustomKeysets('it', keysetData)).toThrow();
});

function TestComponents(): React.ReactElement {
return (
<React.Fragment>
<Dialog onClose={() => {}} open={true}>
<Dialog.Header />
</Dialog>
<Pagination page={1} pageSize={1} onUpdate={() => {}} />
</React.Fragment>
);
}

// Custom keyset registration needs keysets for every component, or validation will fail.
// We don't want to provide every keyset in every test, otherwise tests will be too verbose, that's where this function comes in.
function createTestCustomKeyset(dataToOverride: KeysetData): KeysetData {
const pluggedValue = '[PLUG]';
const fullTestKeyset = {
Alert: {
label_close: pluggedValue,
},
AvatarStack: {
more: [pluggedValue, pluggedValue, pluggedValue],
},
Breadcrumbs: {
label_more: pluggedValue,
},
ClipboardButton: {
startCopy: pluggedValue,
endCopy: pluggedValue,
},
Dialog: {
close: pluggedValue,
},
Pagination: {
button_previous: pluggedValue,
button_next: pluggedValue,
button_first: pluggedValue,
label_select_size: pluggedValue,
'label_input-placeholder': pluggedValue,
'label_page-of': pluggedValue,
},
PinInput: {
'label_one-of': pluggedValue,
},
Select: {
label_clear: pluggedValue,
'label_show-error-info': pluggedValue,
},
Table: {
label_empty: pluggedValue,
'label-actions': pluggedValue,
'label-row-select': pluggedValue,
},
TableColumnSetup: {
button_switcher: pluggedValue,
},
TableColumnSetupInner: {
button_apply: pluggedValue,
button_reset: pluggedValue,
button_switcher: pluggedValue,
},
Toaster: {
'label_close-button': pluggedValue,
},
withTableSettings: {
label_settings: pluggedValue,
},
'g-clear-button': {
'label_clear-button': pluggedValue,
},
'g-user-label': {
'label_remove-button': pluggedValue,
},
};

return Object.assign(fullTestKeyset, dataToOverride);
}
65 changes: 65 additions & 0 deletions src/components/utils/registerCustomKeysets.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
import type {KeysetData} from '@gravity-ui/i18n';

import {i18n} from '../../i18n';

export const registerCustomKeysets = (language: string, data: KeysetData) => {
validateCustomKeysets(language, data);
Copy link
Author

@NikitaShkaruba NikitaShkaruba Sep 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've decided to throw errors on startup, because this way developers will not have weird translation errors in their production, when we'll add new key to our component in a new major release for example. I think that their CI or dev environment manual tests will easily catch it.

What do you guys, think?

i18n.registerKeysets(language, data);
};

function validateCustomKeysets(language: string, givenData: KeysetData): void {
const trustedData = i18n.data.en;
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We provide complete keysets in English language. So we can compare custom keysets with it.

The code below looks kinda verbose, but IDK how to improve it, and it's very reliable and transparent about what needs to be fixed.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also I think creating your first custom keyset might be daunting, because you'll only get a error one at a time. So maybe in our errors we should also show the templated English keyset. In the end our errors will look like this:

Custom keysets 'it' validation error: excess component 'Alert' keys for found ["nonexistent_key","nonexistent_key2"]

English keysets example:
{"Alert":{"label_close":"Close"},"Breadcrumbs":{"label_more":"Show more"},"ClipboardButton":{"startCopy":"Copy","endCopy":"Copied!"},"Dialog":{"close":"Close dialog"},"AvatarStack":{"more":["and {{count}} more","and {{count}} more","and {{count}} more"]},"g-clear-button":{"label_clear-button":"Clear"},"Pagination":{"button_previous":"Previous","button_next":"Next","button_first":"First","label_input-placeholder":"Page #","label_page-of":"of","label_select_size":"Select page size"},"Select":{"label_clear":"Clear","label_show-error-info":"Show popup with error info"},"g-user-label":{"label_remove-button":"Remove"},"PinInput":{"label_one-of":"{{number}} of {{count}}, "},"Table":{"label_empty":"No data","label-actions":"Actions","label-row-select":"Select"},"TableColumnSetupInner":{"button_apply":"Apply","button_reset":"Reset","button_switcher":"Columns"},"withTableSettings":{"label_settings":"Table settings"},"TableColumnSetup":{"button_switcher":"Columns"},"Toaster":{"label_close-button":"Close"}}

What do you think?


const trustedComponents = Object.keys(trustedData);
const givenComponents = Object.keys(givenData);

trustedComponents.forEach((componentName) => {
const trustedKeys = Object.keys(trustedData[componentName]);
const givenKeys = Object.keys(givenData[componentName]);

// Check that all component keys in trusted data exist in given data
trustedKeys.forEach((keyName) => {
if (givenData[componentName] === undefined) {
throw createValidationError(
language,
`keyset for component '${componentName}' is required`,
);
}
if (givenData[componentName][keyName] === undefined) {
throw createValidationError(
language,
`key '${keyName}' for component '${componentName}' is required`,
);
}
});

// Check for extra component keys
if (trustedKeys.length !== givenKeys.length) {
const keyDifference = getArrayDifference(trustedKeys, givenKeys);
throw createValidationError(
language,
`excess component '${componentName}' keys for found ${JSON.stringify(keyDifference)}`,
);
}
});

// Check for extra components
if (trustedComponents.length !== givenComponents.length) {
const componentDifference = getArrayDifference(trustedComponents, givenComponents);
throw createValidationError(
language,
`excess components found ${JSON.stringify(componentDifference)}`,
);
}
}

function createValidationError(language: string, text: string): Error {
return new Error(`Custom keysets '${language}' validation error: ${text}`);
}

function getArrayDifference(arr1: string[], arr2: string[]): string[] {
const arr1Extra = arr1.filter((x) => !arr2.includes(x));
const arr2Extra = arr2.filter((x) => !arr1.includes(x));

return arr1Extra.concat(arr2Extra);
}
2 changes: 2 additions & 0 deletions src/utils/autocomplete-safe-string.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
// Use it for better autocomplete https://stackoverflow.com/a/61048124
export type AutocompleteSafeString = string & {};
Loading