mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-22 10:09:47 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			405 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			405 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
| const del = require("del");
 | |
| const path = require("path");
 | |
| const gulp = require("gulp");
 | |
| const fs = require("fs");
 | |
| const foreach = require("gulp-foreach");
 | |
| const hash = require("gulp-hash");
 | |
| const hashFilename = require("gulp-hash-filename");
 | |
| const merge = require("gulp-merge-json");
 | |
| const minify = require("gulp-jsonminify");
 | |
| const rename = require("gulp-rename");
 | |
| const transform = require("gulp-json-transform");
 | |
| 
 | |
| const inDir = "translations";
 | |
| const workDir = "build-translations";
 | |
| const fullDir = workDir + "/full";
 | |
| const coreDir = workDir + "/core";
 | |
| const outDir = workDir + "/output";
 | |
| 
 | |
| String.prototype.rsplit = function(sep, maxsplit) {
 | |
|   var split = this.split(sep);
 | |
|   return maxsplit
 | |
|     ? [split.slice(0, -maxsplit).join(sep)].concat(split.slice(-maxsplit))
 | |
|     : split;
 | |
| };
 | |
| 
 | |
| // Panel translations which should be split from the core translations. These
 | |
| // should mirror the fragment definitions in polymer.json, so that we load
 | |
| // additional resources at equivalent points.
 | |
| const TRANSLATION_FRAGMENTS = [
 | |
|   "config",
 | |
|   "history",
 | |
|   "logbook",
 | |
|   "mailbox",
 | |
|   "profile",
 | |
|   "shopping-list",
 | |
|   "page-authorize",
 | |
|   "page-demo",
 | |
|   "page-onboarding",
 | |
|   "developer-tools",
 | |
| ];
 | |
| 
 | |
| const tasks = [];
 | |
| 
 | |
| function recursiveFlatten(prefix, data) {
 | |
|   let output = {};
 | |
|   Object.keys(data).forEach(function(key) {
 | |
|     if (typeof data[key] === "object") {
 | |
|       output = Object.assign(
 | |
|         {},
 | |
|         output,
 | |
|         recursiveFlatten(prefix + key + ".", data[key])
 | |
|       );
 | |
|     } else {
 | |
|       output[prefix + key] = data[key];
 | |
|     }
 | |
|   });
 | |
|   return output;
 | |
| }
 | |
| 
 | |
| function flatten(data) {
 | |
|   return recursiveFlatten("", data);
 | |
| }
 | |
| 
 | |
| function emptyFilter(data) {
 | |
|   const newData = {};
 | |
|   Object.keys(data).forEach((key) => {
 | |
|     if (data[key]) {
 | |
|       if (typeof data[key] === "object") {
 | |
|         newData[key] = emptyFilter(data[key]);
 | |
|       } else {
 | |
|         newData[key] = data[key];
 | |
|       }
 | |
|     }
 | |
|   });
 | |
|   return newData;
 | |
| }
 | |
| 
 | |
| function recursiveEmpty(data) {
 | |
|   const newData = {};
 | |
|   Object.keys(data).forEach((key) => {
 | |
|     if (data[key]) {
 | |
|       if (typeof data[key] === "object") {
 | |
|         newData[key] = recursiveEmpty(data[key]);
 | |
|       } else {
 | |
|         newData[key] = "TRANSLATED";
 | |
|       }
 | |
|     }
 | |
|   });
 | |
|   return newData;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Replace Lokalise key placeholders with their actual values.
 | |
|  *
 | |
|  * We duplicate the behavior of Lokalise here so that placeholders can
 | |
|  * be included in src/translations/en.json, but still be usable while
 | |
|  * developing locally.
 | |
|  *
 | |
|  * @link https://docs.lokalise.co/article/KO5SZWLLsy-key-referencing
 | |
|  */
 | |
| const re_key_reference = /\[%key:([^%]+)%\]/;
 | |
| function lokalise_transform(data, original) {
 | |
|   const output = {};
 | |
|   Object.entries(data).forEach(([key, value]) => {
 | |
|     if (value instanceof Object) {
 | |
|       output[key] = lokalise_transform(value, original);
 | |
|     } else {
 | |
|       output[key] = value.replace(re_key_reference, (match, key) => {
 | |
|         const replace = key.split("::").reduce((tr, k) => tr[k], original);
 | |
|         if (typeof replace !== "string") {
 | |
|           throw Error(
 | |
|             `Invalid key placeholder ${key} in src/translations/en.json`
 | |
|           );
 | |
|         }
 | |
|         return replace;
 | |
|       });
 | |
|     }
 | |
|   });
 | |
|   return output;
 | |
| }
 | |
| 
 | |
| let taskName = "clean-translations";
 | |
| gulp.task(taskName, function() {
 | |
|   return del([`${outDir}/**/*.json`]);
 | |
| });
 | |
| tasks.push(taskName);
 | |
| 
 | |
| gulp.task("ensure-translations-build-dir", (done) => {
 | |
|   if (!fs.existsSync(workDir)) {
 | |
|     fs.mkdirSync(workDir);
 | |
|   }
 | |
|   done();
 | |
| });
 | |
| 
 | |
| taskName = "create-test-metadata";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("ensure-translations-build-dir", function writeTestMetaData(cb) {
 | |
|     fs.writeFile(
 | |
|       workDir + "/testMetadata.json",
 | |
|       JSON.stringify({
 | |
|         test: {
 | |
|           nativeName: "Test",
 | |
|         },
 | |
|       }),
 | |
|       cb
 | |
|     );
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| taskName = "create-test-translation";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("create-test-metadata", function() {
 | |
|     return gulp
 | |
|       .src("src/translations/en.json")
 | |
|       .pipe(
 | |
|         transform(function(data, file) {
 | |
|           return recursiveEmpty(data);
 | |
|         })
 | |
|       )
 | |
|       .pipe(rename("test.json"))
 | |
|       .pipe(gulp.dest(workDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| /**
 | |
|  * This task will build a master translation file, to be used as the base for
 | |
|  * all languages. This starts with src/translations/en.json, and replaces all
 | |
|  * Lokalise key placeholders with their target values. Under normal circumstances,
 | |
|  * this will be the same as translations/en.json However, we build it here to
 | |
|  * facilitate both making changes in development mode, and to ensure that the
 | |
|  * project is buildable immediately after merging new translation keys, since
 | |
|  * the Lokalise update to translations/en.json will not happen immediately.
 | |
|  */
 | |
| taskName = "build-master-translation";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("clean-translations", function() {
 | |
|     return gulp
 | |
|       .src("src/translations/en.json")
 | |
|       .pipe(
 | |
|         transform(function(data, file) {
 | |
|           return lokalise_transform(data, data);
 | |
|         })
 | |
|       )
 | |
|       .pipe(rename("translationMaster.json"))
 | |
|       .pipe(gulp.dest(workDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| taskName = "build-merged-translations";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("build-master-translation", function() {
 | |
|     return gulp
 | |
|       .src([inDir + "/*.json", workDir + "/test.json"], { allowEmpty: true })
 | |
|       .pipe(
 | |
|         foreach(function(stream, file) {
 | |
|           // For each language generate a merged json file. It begins with the master
 | |
|           // translation as a failsafe for untranslated strings, and merges all parent
 | |
|           // tags into one file for each specific subtag
 | |
|           //
 | |
|           // TODO: This is a naive interpretation of BCP47 that should be improved.
 | |
|           //       Will be OK for now as long as we don't have anything more complicated
 | |
|           //       than a base translation + region.
 | |
|           const tr = path.basename(file.history[0], ".json");
 | |
|           const subtags = tr.split("-");
 | |
|           const src = [workDir + "/translationMaster.json"];
 | |
|           for (let i = 1; i <= subtags.length; i++) {
 | |
|             const lang = subtags.slice(0, i).join("-");
 | |
|             if (lang === "test") {
 | |
|               src.push(workDir + "/test.json");
 | |
|             } else {
 | |
|               src.push(inDir + "/" + lang + ".json");
 | |
|             }
 | |
|           }
 | |
|           return gulp
 | |
|             .src(src, { allowEmpty: true })
 | |
|             .pipe(transform((data) => emptyFilter(data)))
 | |
|             .pipe(
 | |
|               merge({
 | |
|                 fileName: tr + ".json",
 | |
|               })
 | |
|             )
 | |
|             .pipe(gulp.dest(fullDir));
 | |
|         })
 | |
|       );
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| const splitTasks = [];
 | |
| TRANSLATION_FRAGMENTS.forEach((fragment) => {
 | |
|   taskName = "build-translation-fragment-" + fragment;
 | |
|   gulp.task(
 | |
|     taskName,
 | |
|     gulp.series("build-merged-translations", function() {
 | |
|       // Return only the translations for this fragment.
 | |
|       return gulp
 | |
|         .src(fullDir + "/*.json")
 | |
|         .pipe(
 | |
|           transform((data) => ({
 | |
|             ui: {
 | |
|               panel: {
 | |
|                 [fragment]: data.ui.panel[fragment],
 | |
|               },
 | |
|             },
 | |
|           }))
 | |
|         )
 | |
|         .pipe(gulp.dest(workDir + "/" + fragment));
 | |
|     })
 | |
|   );
 | |
|   tasks.push(taskName);
 | |
|   splitTasks.push(taskName);
 | |
| });
 | |
| 
 | |
| taskName = "build-translation-core";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("build-merged-translations", function() {
 | |
|     // Remove the fragment translations from the core translation.
 | |
|     return gulp
 | |
|       .src(fullDir + "/*.json")
 | |
|       .pipe(
 | |
|         transform((data) => {
 | |
|           TRANSLATION_FRAGMENTS.forEach((fragment) => {
 | |
|             delete data.ui.panel[fragment];
 | |
|           });
 | |
|           return data;
 | |
|         })
 | |
|       )
 | |
|       .pipe(gulp.dest(coreDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| splitTasks.push(taskName);
 | |
| 
 | |
| taskName = "build-flattened-translations";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series(...splitTasks, function() {
 | |
|     // Flatten the split versions of our translations, and move them into outDir
 | |
|     return gulp
 | |
|       .src(
 | |
|         TRANSLATION_FRAGMENTS.map(
 | |
|           (fragment) => workDir + "/" + fragment + "/*.json"
 | |
|         ).concat(coreDir + "/*.json"),
 | |
|         { base: workDir }
 | |
|       )
 | |
|       .pipe(
 | |
|         transform(function(data) {
 | |
|           // Polymer.AppLocalizeBehavior requires flattened json
 | |
|           return flatten(data);
 | |
|         })
 | |
|       )
 | |
|       .pipe(minify())
 | |
|       .pipe(hashFilename())
 | |
|       .pipe(
 | |
|         rename((filePath) => {
 | |
|           if (filePath.dirname === "core") {
 | |
|             filePath.dirname = "";
 | |
|           }
 | |
|         })
 | |
|       )
 | |
|       .pipe(gulp.dest(outDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| taskName = "build-translation-fingerprints";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("build-flattened-translations", function() {
 | |
|     return gulp
 | |
|       .src(outDir + "/**/*.json")
 | |
|       .pipe(
 | |
|         rename({
 | |
|           extname: "",
 | |
|         })
 | |
|       )
 | |
|       .pipe(
 | |
|         hash({
 | |
|           algorithm: "md5",
 | |
|           hashLength: 32,
 | |
|           template: "<%= name %>.json",
 | |
|         })
 | |
|       )
 | |
|       .pipe(hash.manifest("translationFingerprints.json"))
 | |
|       .pipe(
 | |
|         transform(function(data) {
 | |
|           // After generating fingerprints of our translation files, consolidate
 | |
|           // all translation fragment fingerprints under the translation name key
 | |
|           const newData = {};
 | |
|           Object.entries(data).forEach(([key, value]) => {
 | |
|             const [path, _md5] = key.rsplit("-", 1);
 | |
|             // let translation = key;
 | |
|             let translation = path;
 | |
|             const parts = translation.split("/");
 | |
|             if (parts.length === 2) {
 | |
|               translation = parts[1];
 | |
|             }
 | |
|             if (!(translation in newData)) {
 | |
|               newData[translation] = {
 | |
|                 fingerprints: {},
 | |
|               };
 | |
|             }
 | |
|             newData[translation].fingerprints[path] = value;
 | |
|           });
 | |
|           return newData;
 | |
|         })
 | |
|       )
 | |
|       .pipe(gulp.dest(workDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| taskName = "build-translations";
 | |
| gulp.task(
 | |
|   taskName,
 | |
|   gulp.series("build-translation-fingerprints", function() {
 | |
|     return gulp
 | |
|       .src(
 | |
|         [
 | |
|           "src/translations/translationMetadata.json",
 | |
|           workDir + "/testMetadata.json",
 | |
|           workDir + "/translationFingerprints.json",
 | |
|         ],
 | |
|         { allowEmpty: true }
 | |
|       )
 | |
|       .pipe(merge({}))
 | |
|       .pipe(
 | |
|         transform(function(data) {
 | |
|           const newData = {};
 | |
|           Object.entries(data).forEach(([key, value]) => {
 | |
|             // Filter out translations without native name.
 | |
|             if (data[key].nativeName) {
 | |
|               newData[key] = data[key];
 | |
|             } else {
 | |
|               console.warn(
 | |
|                 `Skipping language ${key}. Native name was not translated.`
 | |
|               );
 | |
|             }
 | |
|             if (data[key]) newData[key] = value;
 | |
|           });
 | |
|           return newData;
 | |
|         })
 | |
|       )
 | |
|       .pipe(
 | |
|         transform((data) => ({
 | |
|           fragments: TRANSLATION_FRAGMENTS,
 | |
|           translations: data,
 | |
|         }))
 | |
|       )
 | |
|       .pipe(rename("translationMetadata.json"))
 | |
|       .pipe(gulp.dest(workDir));
 | |
|   })
 | |
| );
 | |
| tasks.push(taskName);
 | |
| 
 | |
| module.exports = tasks;
 | 
