Skip to content

Commit

Permalink
Add utilities and utility types to package
Browse files Browse the repository at this point in the history
  • Loading branch information
t3chguy committed Sep 1, 2023
1 parent 2eb122a commit 8f35375
Show file tree
Hide file tree
Showing 9 changed files with 172 additions and 19 deletions.
8 changes: 6 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,18 +9,22 @@
"files": [
"README.md",
"package.json",
"scripts/*.js"
"scripts/*.js",
"lib/*.js"
],
"main": "lib/index.js",
"types": "lib/index.d.ts",
"bin": {
"matrix-gen-i18n": "scripts/gen-i18n.js",
"matrix-compare-i18n-files": "scripts/compare-file.js",
"matrix-i18n-rekey": "scripts/rekey.js"
},
"scripts": {
"build:ts": "tsc",
"build:ts-scripts": "tsc -p tsconfig-scripts.json",
"build:chmod": "chmod +x scripts/*.js",
"build:shebang": "perl -i -pe 'print \"#!/usr/bin/env node\\n\\n\" if $. == 1 && !/^#/; close ARGV if eof' scripts/*.js",
"build": "yarn build:ts && yarn build:chmod && yarn build:shebang",
"build": "yarn build:ts && yarn build:ts-scripts && yarn build:chmod && yarn build:shebang",
"lint": "yarn lint:types",
"lint:types": "tsc --noEmit",
"prepare": "yarn build"
Expand Down
12 changes: 2 additions & 10 deletions scripts/common.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,22 +16,14 @@ limitations under the License.

import fs from "fs";
import { isBinaryExpression, isStringLiteral, isTemplateLiteral, Node } from "@babel/types";
import { Translation, Translations } from "../src";

export const NESTING_KEY = process.env["NESTING_KEY"] || "|";
export const INPUT_FILE = process.env["INPUT_FILE"] || 'src/i18n/strings/en_EN.json';
export const OUTPUT_FILE = process.env["OUTPUT_FILE"] || 'src/i18n/strings/en_EN.json';

export type Translation = string | {
one?: string;
other: string;
};

export interface Translations {
[key: string]: Translation | Translations;
}

export function getPath(key: string): string[] {
return key.split(NESTING_KEY);
return key.replace(/\\n/g, "\n").split(NESTING_KEY);
}

export function getKeys(translations: Translations | Translation, path = ""): string[] {
Expand Down
3 changes: 1 addition & 2 deletions scripts/gen-i18n.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,8 @@ import {
getTranslations,
OUTPUT_FILE,
putTranslations,
Translation,
Translations
} from "./common";
import { Translation, Translations } from "../src";

// Find the package.json for the project we're running gen-18n against
const projectPackageJsonPath = path.join(process.cwd(), 'package.json');
Expand Down
6 changes: 4 additions & 2 deletions scripts/rekey.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,16 +16,18 @@ limitations under the License.

import parseArgs from "minimist";
import _ from "lodash";
import { getPath, getTranslations, putTranslations, Translations } from "./common";
import fs from "fs";
import path from "path";

import { getPath, getTranslations, putTranslations } from "./common";
import { Translations } from "../src";

const I18NDIR = "src/i18n/strings";

const argv = parseArgs<{
copy: boolean;
}>(process.argv.slice(2), {
boolean: ["copy", "move", "case-insensitive", "find-and-replace"],
boolean: ["copy"],
});

const [oldPath, newPath] = argv._.map(getPath);
Expand Down
18 changes: 18 additions & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
/*
Copyright 2023 The Matrix.org Foundation C.I.C.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

export * from "./utils";
export * from "./types";
69 changes: 69 additions & 0 deletions src/types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
/*
Copyright 2023 The Matrix.org Foundation C.I.C.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

/**
* Utility type for string dot notation for accessing nested object properties.
* Based on https://stackoverflow.com/a/58436959
* @example
* {
* "a": {
* "b": {
* "c": "value"
* },
* "d": "foobar"
* }
* }
* will yield a type of `"a.b.c" | "a.d"` with Separator="."
* @typeParam Target the target type to generate leaf keys for
* @typeParam Separator the separator to use between key segments when accessing nested objects
* @typeParam LeafType the type which leaves of this object extend, used to determine when to stop recursion
* @typeParam MaxDepth the maximum depth to recurse to
* @returns a union type representing all dot (Separator) string notation keys which can access a Leaf (of LeafType)
*/
export type Leaves<Target, Separator extends string = ".", LeafType = string, MaxDepth extends number = 3> = [
MaxDepth,
] extends [never]
? never
: Target extends LeafType
? ""
: {
[K in keyof Target]-?: Join<K, Leaves<Target[K], Separator, LeafType, Prev[MaxDepth]>, Separator>;
}[keyof Target];
type Prev = [never, 0, 1, 2, 3, ...0[]];
type Join<K, P, S extends string = "."> = K extends string | number
? P extends string | number
? `${K}${"" extends P ? "" : S}${P}`
: never
: never;

/**
* Utility type for |-separated keys indexing into a translations file
* @typeParam Translations the target type to generate leaf keys for
* @typeParam Separator the separator to use between key segments when accessing nested objects
* @typeParam LeafType the type which leaves of this object extend, used to determine when to stop recursion
* @typeParam MaxDepth the maximum depth to recurse to
* @returns a union type representing all dot (Separator) string notation keys which can access a Leaf (of LeafType)
*/
export type TranslationKey<T extends Translations> = Leaves<T, "|", string | { other: string }>;

export type Translation = string | {
one?: string;
other: string;
};

export interface Translations {
[key: string]: Translation | Translations;
}
53 changes: 53 additions & 0 deletions src/utils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
/*
Copyright 2017 MTRNord and Cooperative EITA
Copyright 2017 Vector Creations Ltd.
Copyright 2019 Michael Telatynski <[email protected]>
Copyright 2019 - 2022 The Matrix.org Foundation C.I.C.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

/**
* Returns a language string with underscores replaced with
* hyphens, and lowercased.
*
* @param {string} language The language string to be normalized
* @returns {string} The normalized language string
*/
export function normalizeLanguageKey(language: string): string {
return language.toLowerCase().replace("_", "-");
}

/**
* Turns a language string, normalises it,
* (see normalizeLanguageKey) into an array of language strings
* with fallback to generic languages
* (eg. 'pt-BR' => ['pt-br', 'pt'])
*
* @param {string} language The input language string
* @return {string[]} List of normalised languages
*/
export function getNormalizedLanguageKeys(language: string): string[] {
const languageKeys: string[] = [];
const normalizedLanguage = normalizeLanguageKey(language);
const languageParts = normalizedLanguage.split("-");
if (languageParts.length === 2 && languageParts[0] === languageParts[1]) {
languageKeys.push(languageParts[0]);
} else {
languageKeys.push(normalizedLanguage);
if (languageParts.length === 2) {
languageKeys.push(languageParts[0]);
}
}
return languageKeys;
}
13 changes: 13 additions & 0 deletions tsconfig-scripts.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
{
"compilerOptions": {
"target": "es2016",
"esModuleInterop": true,
"module": "commonjs",
"moduleResolution": "node",
"strict": true,
"noUnusedLocals": true
},
"include": [
"./scripts/*.ts"
]
}
9 changes: 6 additions & 3 deletions tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,14 @@
"esModuleInterop": true,
"module": "commonjs",
"moduleResolution": "node",
"noImplicitAny": true,
"strict": true,
"noUnusedLocals": true
"noUnusedLocals": true,
"outDir": "lib",
"rootDir": "src",
"declaration": true,
"declarationMap": true
},
"include": [
"./scripts/*.ts",
"./src/*.ts"
]
}

0 comments on commit 8f35375

Please sign in to comment.