The tool is used to extract default messages from all *.{js, ts, jsx, tsx, html, pug}
files in provided plugins directories to a JSON file.
It uses Babel to parse code and build an AST for each file or a single JS expression if whole file parsing is impossible. The tool is able to validate, extract and match IDs, default messages and descriptions only if they are defined statically and together, otherwise it will fail with detailed explanation. That means one can't define ID in one place and default message in another, or use function call to dynamically create default message etc.
Global restrictions:
The defaultMessage
and description
must be string literals, template literals w/o expressions or string-only concatenation expressions, anything else is not allowed. The id
can be only a string literal.
The defaultMessage
must contain ICU references to all keys in the values
and vice versa.
The description
is optional, values
is optional too unless defaultMessage
references to it.
-
Angular (.html)
-
Filter
{{ ::'pluginNamespace.messageId' | i18n: { defaultMessage: 'Default message string literal, {key}', values: { key: 'value' }, description: 'Message context or description' } }}
- Don't break
| i18n: {
with line breaks, and don't skip whitespaces aroundi18n:
. ::
operator is optional. Omit it if you need data binding for thevalues
.
- Don't break
-
Directive
<p i18n-id="pluginNamespace.messageId" i18n-default-message="Default message string literal, {key}. {emphasizedText}" i18n-values="{ key: value, html_emphasizedText: htmlString }" i18n-description="Message context or description" ></p>
html_
prefixes will be removed fromi18n-values
keys before validation.
-
-
React (.jsx, .tsx)
-
<FormattedMessage>
... return ( <p> <FormattedMessage id="pluginNamespace.messageId" defaultMessage="Default message string literal, {key}" values={{ key: 'value' }} description="Message context or description" /> ... </p> ); ...
- JSX element can be parsed only if it is located in a JSX code block (not in a string).
-
intl.formatMessage
const MyComponentContent = ({ intl }) => ( <input type="text" placeholder={intl.formatMessage( { id: 'pluginNamespace.messageId', defaultMessage: 'Default message string literal, {key}', description: 'Message context or description' }, { key: 'value', } )} /> );
- Callee of call expression should be either
intl.formatMessage
or*.intl.formatMessage
.
- Callee of call expression should be either
-
-
JavaScript or TypeScript (primarily server-side) (.js, .ts, .jsx, .tsx)
i18n('pluginNamespace.messageId', { defaultMessage: 'Default message string literal, {key}', values: { key: 'value', }, description: 'Message context or description', });
or
i18n.translate('pluginNamespace.messageId', { defaultMessage: 'Default message string literal, {key}', values: { key: 'value', }, description: 'Message context or description', });
- Expression can be parsed only if it is located in syntactically valid JS/TS code. Do not use type assertions in TypeScript for
defaultMessage
ordescription
properties, id argument or the second argument ofi18n*
call expression. It is never needed for i18n engine use cases.
- Expression can be parsed only if it is located in syntactically valid JS/TS code. Do not use type assertions in TypeScript for
-
Pug (.pug)
#{i18n('pluginNamespace.messageId', { defaultMessage: 'Default message string literal, {key}', values: { key: 'value' }, description: 'Message context or description', })}
- Expression in
#{...}
is parsed as a JS expression.
- Expression in
node scripts/i18n_extract --path path/to/plugin --path path/to/another/plugin --output-dir ./translations --output-format json5
path/to/plugin
is an example of path to a directory(-es) where messages searching should start. By default--path
is.
, it means that messages from all paths in.i18nrc.json
will be parsed. Each specified path should start with any path in.i18nrc.json
or be a part of it.--output-dir
specifies a path to a directory, whereen.json
will be created.
In case of parsing issues, exception with the necessary information will be thrown to console and extraction will be aborted.--output-format
specifies format of generateden.json
. By default it isjson
. Use it only if you need a JSON5 file.--include-config
specifies additional paths to.i18nrc.json
files (may be useful for 3rd-party plugins)
<output_path>/en.json
The generated JSON/JSON5 file contains formats
object and messages
object with id: message
or id: {text, comment}
pairs. Messages are sorted by id.
Example:
{
"formats": {},
"messages": {
"pluginNamespace.message.id-1": "Default message text 1",
"pluginNamespace.message.id-2": {
"text": "Default message text 2",
"comment": "Message context or description"
}
}
}
The tool is used for verifying locale file, finding unused / missing messages, key duplications and value references mismatches. If all these checks are passing, the tool groups messages by namespaces and creates JSON files in right folders.
The tool throws an exception if formats
object is missing in locale file.
node scripts/i18n_integrate --source path/to/locale.json --target x-pack/legacy/plugins/translations/translations/locale.json
--source
path to the JSON file with translations that should be integrated.--target
defines a single path to the JSON file where translations should be integrated to, path mappings from .i18nrc.json are ignored in this case. It's currently used for integrating of Kibana built-in translations that are located in a single JSON file withinx-pack/translations
plugin.--dry-run
tells the tool to exit after verification phase and not write translations to the disk.--ignore-incompatible
specifies whether tool should ignore incompatible translations. It may be useful when the code base you're integrating translations to has changed and some default messages switched to ICU structure that is incompatible with the one used in corresponding translation.--ignore-missing
specifies whether tool should ignore missing translations. It may be useful when the code base you're integrating translations to has moved forward since the revision translations were created for.--ignore-unused
specifies whether tool should ignore unused translations. It may be useful when the code base you're integrating translations to has changed and some translations are not needed anymore.--include-config
specifies additional paths to.i18nrc.json
files (may be useful for 3rd-party plugins)
Unless --target
is specified, the tool generates locale files in plugin folders and few other special locations based on namespaces and corresponding mappings defined in .i18nrc.json.
The tool performs a number of checks on internationalized labels and verifies whether they don't conflict with the existing translations.
We don't catch every possible misuse of i18n framework, but only the most common and critical ones.
To perform translations compatibility checks tool relies on existing translations referenced in translations
section of .i18nrc.json.
Currently auto-fixer (--fix
) can only automatically fix two types of errors, and for both of them the fix is just removing of conflicting translation entries from JSON files:
-
incompatible translation - this error means that the value references in internationalized label differ from the ones in the existing translation
-
unused translation - this error means that the translations file includes label that doesn't exist anymore.
node scripts/i18n_check --fix
--fix
tells the tool to try to fix as much violations as possible. All errors that tool won't be able to fix will be reported.--ignore-incompatible
specifies whether tool should ignore incompatible translations.--ignore-missing
specifies whether tool should ignore missing translations.--ignore-unused
specifies whether tool should ignore unused translations.--include-config
specifies additional paths to.i18nrc.json
files (may be useful for 3rd-party plugins)