eslint-plugin-i18n-json
data:image/s3,"s3://crabby-images/61e73/61e73a132486819bf955240febb34d8b313ec700" alt="Build Status"
Fully extendable eslint plugin for JSON i18n translation files.
🎉 Check out the introductory blog post!
Table of Contents
Features 🚀
-
lint JSON translation files
- rule:
i18n-json/valid-json
- configure a custom linter in case the default doesn't fit your needs.
-
validate syntax per message
- rule:
i18n-json/valid-message-syntax
- default syntax check is for ICU Message Syntax
- can support any message syntax through custom validators. Example
-
ensure translation files have identical keys
- rule:
i18n-json/identical-keys
- supports different custom mappings and on the fly key structure generation
-
sort translation keys in ascending order through eslint auto-fix (case-sensitive)
- rule:
i18n-json/sorted-keys
-
ability to ignore certain keys. Example: metadata keys, in progress translations, etc.
- setting:
i18n-json/ignore-keys
Example
-
The plugin supports any level of nesting in the translation file. (escapes .
in key names)
Note: Check out the Examples folder to see different use cases and project setups.
Requires
- eslint >= 4.0.0
- node >= 6.0.0
Getting Started
Right out of the box you get the following through our recommended ruleset i18n-json/recommended
:
- i18n-json/valid-json
- linting of each JSON translation file
- default severity: error | 2
- i18n-json/valid-message-syntax
- default ICU Message syntax validation (using
intl-messageformat-parser
) - default severity: error | 2
- i18n-json/sorted-keys
- automatic case-sensitive ascending sort of all keys in the translation file.
- configurable to sort in descending order
- Does a level order traversal of keys, and supports sorting nested objects
Let's say your translations project directory looks like the following, (project name: simple)
> tree simple -I node_modules
simple
├── package.json
├── readme.md
└── translations
├── en-US
│ └── index.json
└── es-MX
└── index.json
In this project directory, do the following:
-
npm install --save-dev eslint-plugin-i18n-json
-
Create a .eslintrc.js
file in the root dir of your project. For this example: /simple/.eslintrc.js
.
-
paste in the following:
module.exports = {
extends: [
'plugin:i18n-json/recommended',
],
};
-
add this npm script to your package.json
file.
- note:
- without the
--fix
option, sorting the translation file won't work - the default eslint report formatter,
stylish
, doesn't handle lint messages of varying length well. Hence, we have also built a custom report formatter
well suited for this plugin.
{
"scripts": {
"lint": "eslint --fix --ext .json --format node_modules/eslint-plugin-i18n-json/formatter.js translations/"
}
}
- Also, the following builtin formatters provided by eslint also work well:
compact
, unix
, visualstudio
, json
. Learn more here
- Example usage:
eslint --fix --ext .json --format compact translations/
-
npm run lint
-
Profit! Relax knowing that each change to the translations project will go through strict checks by the eslint plugin.
Example where we have invalid ICU message syntax.
data:image/s3,"s3://crabby-images/0eab1/0eab158506e0a289feb5facf55171bee68510f06" alt=""
Examples
Check out the Examples folder to see different use cases.
Configuring your .eslintrc file
-
Simply update your .eslintrc.*
with overrides for the individual rules.
-
Eslint severities: 2 = error, 1 = warning, 0 = off
-
Example of the module's default rule configuration:
- see below for more information about how to further configure each rule. (some options may require switching to a
.eslintrc.js
file)
{
"rules": {
"i18n-json/valid-message-syntax": [2, {
"syntax": "icu"
}],
"i18n-json/valid-json": 2,
"i18n-json/sorted-keys": [2, {
"order": "asc",
"indentSpaces": 2,
}],
"i18n-json/identical-keys": 0
}
}
module.exports = {
rules: {
'i18n-json/valid-message-syntax': [2, {
syntax: 'icu',
}],
'i18n-json/valid-json': 2,
'i18n-json/sorted-keys': [2, {
order: 'asc',
indentSpaces: 2,
}],
'i18n-json/identical-keys': 0,
},
};
Rules
i18n-json/valid-json
-
linting of each JSON translation file
-
builtin linter uses json-lint
-
default severity: error | 2
-
options
linter
: String (Optional)
- Absolute path to a module which exports a JSON linting function.
Function(source: String)
- This function will be passed the source of the current file being processed.
- It should throw an Error, just like
JSON.parse
.
module.exports = {
rules: {
'i18n-json/valid-json': [2, {
linter: path.resolve('path/to/custom-linter.js'),
}],
},
};
module.exports = (source) => {
if (isBad(source)) {
throw new SyntaxError('invalid syntax');
}
};
Example output for Invalid JSON.
data:image/s3,"s3://crabby-images/b2380/b2380d12be7cfa07d013805bc2345a1640e25e97" alt=""
i18n-json/valid-message-syntax
-
default ICU Message syntax validation (using intl-messageformat-parser
)
-
default severity: error | 2
-
options
syntax
: String (Optional). Default value: icu
.
-
Can be a built in validator: icu
, non-empty-string
.
module.exports = {
rules: {
'i18n-json/valid-message-syntax': [2, {
syntax: 'non-empty-string',
}],
},
};
-
Can be an absolute path to a module which exports a Syntax Validator Function.
Function(message: String, key: String)
- This function will be invoked with each
message
and its corresponding key
- It should throw an Error, just like
JSON.parse
on invalid syntax.
module.exports = {
rules: {
'i18n-json/valid-message-syntax': [2, {
syntax: path.resolve('path/to/custom-syntax-validator.js'),
}],
},
};
module.exports = (message, key) => {
if (message !== message.toUpperCase()) {
throw new SyntaxError('MESSAGE MUST BE IN ALL CAPS!');
}
};
Output from the custom-message-syntax example where each message must have the word 'PIZZA' prepended to it.
data:image/s3,"s3://crabby-images/8ada1/8ada182cc2bf76a04ca6adf825d1505fd6d196cf" alt=""
i18n-json/identical-keys
-
compare each translation file's key structure with a reference translation file to ensure consistency
-
severity: 0 | off , this rule is OFF by default
-
Can turn this rule on by specifying options for it through your .eslintrc.*
file.
-
options
Output from the slightly advanced identical keys example where some keys from the reference translation file (en-US
) were not found during comparison.
data:image/s3,"s3://crabby-images/ab4c3/ab4c3b0ac61e4750e52c2a200019e67ef0b1a128" alt=""
i18n-json/sorted-keys
-
automatic case-sensitive ascending sort of all keys in the translation file
-
default severity: error | 2
-
options
order
: String (Optional). Possible values: asc|desc
. Default value: asc
. Case-sensitive sort order of translation keys. The rule does a level order traversal of object keys. Supports nested objects.indentSpaces
: Number (Optional). Default value: 2
. The number of spaces to indent the emitted sorted translations with. (Will be passed to JSON.stringify
when generating fixed output).
In the case --fix
is not supplied to eslint, and the i18n-json/sorted-keys
rule is not switched off, it will emit an
error
(or warning
) if it detects an invalid sort order for translation keys.
data:image/s3,"s3://crabby-images/21142/21142cbd57dc5833284db1aa2e21b69d3cc59afc" alt=""
Settings
i18n-json/ignore-keys
- list of key paths (case sensitive) to ignore when checking syntax and doing key structure comparisons. Example
- this setting is used by the following rules:
i18n-json/identical-keys
and i18n-json/valid-syntax
- if the key path points to an object, the nested paths are also ignored.
- example usage: metadata keys with values not corresponding to the syntax specified or work-in-progress translation keys which should not be used in comparisons.
Example setting configuration:
{
settings: {
'i18n-json/ignore-keys': [
'translationMetadata',
'login.form.inProgressTranslationKey',
'some-key'
],
},
}
Disclaimer
- None of the translations in the examples provided reflect actual GoDaddy translations. They were just created using Google Translate for example's sake 😉.
Special Thanks 👏
License 📋
MIT