From b92e91183043014b640c6156d344e2fba46ed570 Mon Sep 17 00:00:00 2001 From: Richard Knoll Date: Wed, 10 Apr 2024 10:46:50 -0700 Subject: [PATCH] Upgrade to Crowdin V2 APIs (#9954) * use crowdin v2 apis * rename crowdin API * add projectid setting * fix test mode * copy pxt-translations crowdin config --- cli/cli.ts | 8 +- cli/crowdin.ts | 410 ++++++++++++++++++------------------- cli/crowdinApi.ts | 386 ++++++++++++++++++++++++++++++++++ localtypings/pxtarget.d.ts | 1 + package.json | 4 + pxtlib/crowdin.ts | 284 ------------------------- 6 files changed, 589 insertions(+), 504 deletions(-) create mode 100644 cli/crowdinApi.ts diff --git a/cli/cli.ts b/cli/cli.ts index 6418a675742..f2b5022bda2 100644 --- a/cli/cli.ts +++ b/cli/cli.ts @@ -449,14 +449,14 @@ async function ciAsync() { if (branch === "master" && isTaggedCommit) { if (uploadDocs) { await buildWebStringsAsync(); - await crowdin.execCrowdinAsync("upload", "built/webstrings.json"); + await crowdin.uploadBuiltStringsAsync("built/webstrings.json"); for (const subapp of SUB_WEBAPPS) { - await crowdin.execCrowdinAsync("upload", `built/${subapp.name}-strings.json`); + await crowdin.uploadBuiltStringsAsync(`built/${subapp.name}-strings.json`); } } if (uploadApiStrings) { - await crowdin.execCrowdinAsync("upload", "built/strings.json"); + await crowdin.uploadBuiltStringsAsync("built/strings.json"); } if (uploadDocs || uploadApiStrings) { await crowdin.internalUploadTargetTranslationsAsync(uploadApiStrings, uploadDocs); @@ -7020,7 +7020,7 @@ ${pxt.crowdin.KEY_VARIABLE} - crowdin key advanced: true, }, pc => uploadTargetRefsAsync(pc.args[0])); advancedCommand("uploadtt", "upload tagged release", uploadTaggedTargetAsync, ""); - advancedCommand("downloadtrgtranslations", "download translations from bundled projects", crowdin.downloadTargetTranslationsAsync, ""); + advancedCommand("downloadtrgtranslations", "download translations from bundled projects", crowdin.downloadTargetTranslationsAsync, "[package]"); p.defineCommand({ name: "checkdocs", diff --git a/cli/crowdin.ts b/cli/crowdin.ts index 39ccd7713cb..d09f5af43ac 100644 --- a/cli/crowdin.ts +++ b/cli/crowdin.ts @@ -5,109 +5,87 @@ import * as path from 'path'; import Map = pxt.Map; import * as commandParser from './commandparser'; - -interface CrowdinCredentials { prj: string; key: string; branch: string; } - -function crowdinCredentialsAsync(): Promise { - const prj = pxt.appTarget.appTheme.crowdinProject; - const branch = pxt.appTarget.appTheme.crowdinBranch; - if (!prj) { - pxt.log(`crowdin upload skipped, Crowdin project missing in target theme`); - return Promise.resolve(undefined); - } - - let key: string; - if (pxt.crowdin.testMode) - key = pxt.crowdin.TEST_KEY; - else - key = process.env[pxt.crowdin.KEY_VARIABLE]; - if (!key) { - pxt.log(`Crowdin operation skipped: '${pxt.crowdin.KEY_VARIABLE}' variable is missing`); - return Promise.resolve(undefined); - } - return Promise.resolve({ prj, key, branch }); -} +import { downloadFileTranslationsAsync, listFilesAsync, uploadFileAsync } from './crowdinApi'; export function uploadTargetTranslationsAsync(parsed?: commandParser.ParsedCommand) { const uploadDocs = parsed && !!parsed.flags["docs"]; const uploadApiStrings = parsed && !!parsed.flags["apis"] - if (parsed && !!parsed.flags["test"]) + if (parsed && !!parsed.flags["test"]) { pxt.crowdin.setTestMode(); + } return internalUploadTargetTranslationsAsync(uploadApiStrings, uploadDocs); } -export function internalUploadTargetTranslationsAsync(uploadApiStrings: boolean, uploadDocs: boolean) { +export async function internalUploadTargetTranslationsAsync(uploadApiStrings: boolean, uploadDocs: boolean) { pxt.log(`uploading translations (apis ${uploadApiStrings ? "yes" : "no"}, docs ${uploadDocs ? "yes" : "no"})...`); - return crowdinCredentialsAsync() - .then(cred => { - if (!cred) return Promise.resolve(); - pxt.log("got Crowdin credentials"); - const crowdinDir = pxt.appTarget.id; - if (crowdinDir == "core") { - if (!uploadDocs) { - pxt.log('missing --docs flag, skipping') - return Promise.resolve(); - } - return uploadDocsTranslationsAsync("docs", crowdinDir, cred.branch, cred.prj, cred.key) - .then(() => uploadDocsTranslationsAsync("common-docs", crowdinDir, cred.branch, cred.prj, cred.key)) - } else { - let p = Promise.resolve(); - if (uploadApiStrings) - p = p.then(() => execCrowdinAsync("upload", "built/target-strings.json", crowdinDir)) - .then(() => fs.existsSync("built/sim-strings.json") ? execCrowdinAsync("upload", "built/sim-strings.json", crowdinDir) : Promise.resolve()) - .then(() => uploadBundledTranslationsAsync(crowdinDir, cred.branch, cred.prj, cred.key)); - else - p = p.then(() => pxt.log(`translations: skipping api strings upload`)); - if (uploadDocs) - p = p.then(() => uploadDocsTranslationsAsync("docs", crowdinDir, cred.branch, cred.prj, cred.key)) - // scan for docs in bundled packages - .then(() => Promise.all(pxt.appTarget.bundleddirs - // there must be a folder under .../docs - .filter(pkgDir => nodeutil.existsDirSync(path.join(pkgDir, "docs"))) - // upload to crowdin - .map(pkgDir => uploadDocsTranslationsAsync(path.join(pkgDir, "docs"), crowdinDir, cred.branch, cred.prj, cred.key) - )).then(() => { - pxt.log("docs uploaded"); - })); - else - p = p.then(() => pxt.log(`translations: skipping docs upload`)); - return p; + + const crowdinDir = pxt.appTarget.id; + if (crowdinDir == "core") { + if (!uploadDocs) { + pxt.log('missing --docs flag, skipping') + return; + } + + await uploadDocsTranslationsAsync("docs", crowdinDir); + await uploadDocsTranslationsAsync("common-docs", crowdinDir); + } else { + if (uploadApiStrings) { + await uploadBuiltStringsAsync("built/target-strings.json", crowdinDir); + + if (fs.existsSync("built/sim-strings.json")) { + await uploadBuiltStringsAsync("built/sim-strings.json", crowdinDir); } - }); + await uploadBundledTranslationsAsync(crowdinDir); + } + else { + pxt.log(`translations: skipping api strings upload`); + } + + if (uploadDocs) { + await uploadDocsTranslationsAsync("docs", crowdinDir); + await Promise.all( + pxt.appTarget.bundleddirs + .filter(pkgDir => nodeutil.existsDirSync(path.join(pkgDir, "docs"))) + .map(pkgDir => uploadDocsTranslationsAsync(path.join(pkgDir, "docs"), crowdinDir)) + ); + pxt.log("docs uploaded"); + } + else { + pxt.log(`translations: skipping docs upload`); + } + } } -function uploadDocsTranslationsAsync(srcDir: string, crowdinDir: string, branch: string, prj: string, key: string): Promise { - pxt.log(`uploading from ${srcDir} to ${crowdinDir} under project ${prj}/${branch || ""}`) + +export async function uploadBuiltStringsAsync(filename: string, crowdinDir?: string) { + const baseName = path.basename(filename); + const crowdinFile = crowdinDir ? path.join(crowdinDir, baseName) : baseName; + const contents = fs.readFileSync(filename, "utf8"); + + pxt.log(`Uploading ${filename} to ${crowdinFile}`); + await uploadFileAsync(crowdinFile, contents); +} + +async function uploadDocsTranslationsAsync(srcDir: string, crowdinDir: string): Promise { + pxt.log(`Uploading from ${srcDir} to ${crowdinDir}`) const ignoredDirectoriesList = getIgnoredDirectories(srcDir); const todo = nodeutil.allFiles(srcDir).filter(f => /\.md$/.test(f) && !/_locales/.test(f)).reverse(); - const knownFolders: Map = {}; - const ensureFolderAsync = (crowdd: string) => { - if (!knownFolders[crowdd]) { - knownFolders[crowdd] = true; - pxt.log(`creating folder ${crowdd}`); - return pxt.crowdin.createDirectoryAsync(branch, prj, key, crowdd); - } - return Promise.resolve(); - } - const nextFileAsync = (f: string): Promise => { - if (!f) return Promise.resolve(); - const crowdf = path.join(crowdinDir, f); - const crowdd = path.dirname(crowdf); + + for (const file of todo) { + if (!file) continue; + + const crowdinFile = path.join(crowdinDir, file); + // check if file should be ignored - if (ignoredDirectoriesList.filter(d => path.dirname(f).indexOf(d) == 0).length > 0) { - pxt.log(`skipping ${f} because of .crowdinignore file`) - return nextFileAsync(todo.pop()); + if (ignoredDirectoriesList.filter(d => path.dirname(file).indexOf(d) == 0).length > 0) { + pxt.log(`skipping ${file} because of .crowdinignore file`) + continue; } - const data = fs.readFileSync(f, 'utf8'); - pxt.log(`uploading ${f} to ${crowdf}`); - return ensureFolderAsync(crowdd) - .then(() => pxt.crowdin.uploadTranslationAsync(branch, prj, key, crowdf, data)) - .then(() => nextFileAsync(todo.pop())); + pxt.log(`Uploading ${file} to ${crowdinFile}`); + await uploadFileAsync(crowdinFile, fs.readFileSync(file, "utf8")); } - return ensureFolderAsync(path.join(crowdinDir, srcDir)) - .then(() => nextFileAsync(todo.pop())); } function getIgnoredDirectories(srcDir: string) { @@ -125,43 +103,46 @@ function getIgnoredDirectories(srcDir: string) { return Object.keys(ignoredDirectories).filter(d => ignoredDirectories[d]); } -function uploadBundledTranslationsAsync(crowdinDir: string, branch: string, prj: string, key: string): Promise { +async function uploadBundledTranslationsAsync(crowdinDir: string) { const todo: string[] = []; - pxt.appTarget.bundleddirs.forEach(dir => { + for (const dir of pxt.appTarget.bundleddirs) { const locdir = path.join(dir, "_locales"); - if (fs.existsSync(locdir)) - fs.readdirSync(locdir) - .filter(f => /strings\.json$/i.test(f)) - .forEach(f => todo.push(path.join(locdir, f))); - }); + if (fs.existsSync(locdir)) { + const stringsFiles = fs.readdirSync(locdir).filter(f => /strings\.json$/i.test(f)); + + for (const file of stringsFiles) { + todo.unshift(path.join(locdir, file)); + } + } + } pxt.log(`uploading bundled translations to Crowdin (${todo.length} files)`); - const nextFileAsync = (): Promise => { - const f = todo.pop(); - if (!f) return Promise.resolve(); - const data = JSON.parse(fs.readFileSync(f, 'utf8')) as Map; - const crowdf = path.join(crowdinDir, path.basename(f)); - pxt.log(`uploading ${f} to ${crowdf}`); - return pxt.crowdin.uploadTranslationAsync(branch, prj, key, crowdf, JSON.stringify(data)) - .then(nextFileAsync); + for (const file of todo) { + const data = JSON.parse(fs.readFileSync(file, 'utf8')) as Map; + const crowdinFile = path.join(crowdinDir, path.basename(file)); + pxt.log(`Uploading ${file} to ${crowdinFile}`); + await uploadFileAsync(crowdinFile, JSON.stringify(data)); } - return nextFileAsync(); } export async function downloadTargetTranslationsAsync(parsed?: commandParser.ParsedCommand) { const name = parsed?.args[0]; - const cred = await crowdinCredentialsAsync(); - - if (!cred) - return; await buildAllTranslationsAsync(async (fileName: string) => { pxt.log(`downloading ${fileName}`); - return pxt.crowdin.downloadTranslationsAsync(cred.branch, cred.prj, cred.key, fileName, { translatedOnly: true, validatedOnly: true }); + + const translations = await downloadFileTranslationsAsync(fileName); + const parsed: pxt.Map> = {}; + + for (const file of Object.keys(translations)) { + parsed[file] = JSON.parse(translations[file]); + } + + return parsed; }, name); } -export async function buildAllTranslationsAsync(langToStringsHandlerAsync: (fileName: string) => Promise>>, singleDir?: string) { +export async function buildAllTranslationsAsync(fetchFileTranslationAsync: (fileName: string) => Promise>>, singleDir?: string) { await buildTranslationFilesAsync(["sim-strings.json"], "sim-strings.json"); await buildTranslationFilesAsync(["target-strings.json"], "target-strings.json"); await buildTranslationFilesAsync(["strings.json"], "strings.json", true); @@ -190,7 +171,7 @@ export async function buildAllTranslationsAsync(langToStringsHandlerAsync: (file const locdir = path.dirname(filePath); const projectdir = path.dirname(locdir); pxt.debug(`projectdir: ${projectdir}`); - const data = await langToStringsHandlerAsync(crowdf); + const data = await fetchFileTranslationAsync(crowdf); for (const lang of Object.keys(data)) { const dataLang = data[lang]; @@ -226,121 +207,118 @@ function stringifyTranslations(strings: pxt.Map): string { else return JSON.stringify(trg, null, 2); } -export function execCrowdinAsync(cmd: string, ...args: string[]): Promise { +export async function execCrowdinAsync(cmd: string, ...args: string[]): Promise { pxt.log(`executing Crowdin command ${cmd}...`); - const prj = pxt.appTarget.appTheme.crowdinProject; - if (!prj) { - console.log(`crowdin operation skipped, crowdin project not specified in pxtarget.json`); - return Promise.resolve(); - } - const branch = pxt.appTarget.appTheme.crowdinBranch; - return crowdinCredentialsAsync() - .then(crowdinCredentials => { - if (!crowdinCredentials) return Promise.resolve(); - const key = crowdinCredentials.key; - cmd = cmd.toLowerCase(); - if (!args[0] && (cmd != "clean" && cmd != "stats")) throw new Error(cmd == "status" ? "language missing" : "filename missing"); - switch (cmd) { - case "stats": return statsCrowdinAsync(prj, key, args[0]); - case "clean": return cleanCrowdinAsync(prj, key, args[0] || "docs"); - case "upload": return uploadCrowdinAsync(branch, prj, key, args[0], args[1]); - case "download": { - if (!args[1]) throw new Error("output path missing"); - const fn = path.basename(args[0]); - return pxt.crowdin.downloadTranslationsAsync(branch, prj, key, args[0], { translatedOnly: true, validatedOnly: true }) - .then(r => { - Object.keys(r).forEach(k => { - const rtranslations = stringifyTranslations(r[k]); - if (!rtranslations) return; - - nodeutil.mkdirP(path.join(args[1], k)); - const outf = path.join(args[1], k, fn); - console.log(`writing ${outf}`) - nodeutil.writeFileSync( - outf, - rtranslations, - { encoding: "utf8" }); - }) - }) - } - default: throw new Error("unknown command"); + + switch (cmd.toLowerCase()) { + case "stats": + // return statsCrowdinAsync(prj, key, args[0]); + throw new Error("stats command is not supported"); + case "clean": + await execCleanAsync(args[0] || "docs"); + break; + case "upload": + if (!args[0]) { + throw new Error("filename missing"); + } + await uploadBuiltStringsAsync(args[0], args[1]); + break; + case "download": + if (!args[1]) { + throw new Error("output path missing"); } - }) + await execDownloadAsync(args[0], args[1]); + break; + default: + throw new Error("unknown command"); + } } -function cleanCrowdinAsync(prj: string, key: string, dir: string): Promise { - const p = pxt.appTarget.id + "/" + dir; - return pxt.crowdin.listFilesAsync(prj, key, p) - .then(files => { - files.filter(f => !nodeutil.fileExistsSync(f.fullName.substring(pxt.appTarget.id.length + 1))) - .forEach(f => pxt.log(`crowdin: dead file: ${f.branch ? f.branch + "/" : ""}${f.fullName}`)); - }) -} +async function execDownloadAsync(filename: string, outputDir: string): Promise { + const basename = path.basename(filename); + pxt.log("Downloading translations") + const translations = await downloadFileTranslationsAsync(filename); -function statsCrowdinAsync(prj: string, key: string, preferredLang?: string): Promise { - pxt.log(`collecting crowdin stats for ${prj} ${preferredLang ? `for language ${preferredLang}` : `all languages`}`); - console.log(`context\t language\t translated%\t approved%\t phrases\t translated\t approved`) - - const fn = `crowdinstats.csv`; - let headers = 'sep=\t\r\n'; - headers += `id\t file\t language\t phrases\t translated\t approved\r\n`; - nodeutil.writeFileSync(fn, headers, { encoding: "utf8" }); - return pxt.crowdin.projectInfoAsync(prj, key) - .then(info => { - if (!info) throw new Error("info failed") - let languages = info.languages; - // remove in-context language - languages = languages.filter(l => l.code != ts.pxtc.Util.TRANSLATION_LOCALE); - if (preferredLang) - languages = languages.filter(lang => lang.code.toLowerCase() == preferredLang.toLowerCase()); - return Promise.all(languages.map(lang => langStatsCrowdinAsync(prj, key, lang.code))) - }).then(() => { - console.log(`stats written to ${fn}`) - }) - - function langStatsCrowdinAsync(prj: string, key: string, lang: string): Promise { - return pxt.crowdin.languageStatsAsync(prj, key, lang) - .then(stats => { - let uiphrases = 0; - let uitranslated = 0; - let uiapproved = 0; - let corephrases = 0; - let coretranslated = 0; - let coreapproved = 0; - let phrases = 0; - let translated = 0; - let approved = 0; - let r = ''; - stats.forEach(stat => { - const cfn = `${stat.branch ? stat.branch + "/" : ""}${stat.fullName}`; - r += `${stat.id}\t ${cfn}\t ${lang}\t ${stat.phrases}\t ${stat.translated}\t ${stat.approved}\r\n`; - if (stat.fullName == "strings.json") { - uiapproved += Number(stat.approved); - uitranslated += Number(stat.translated); - uiphrases += Number(stat.phrases); - } else if (/core-strings\.json$/.test(stat.fullName)) { - coreapproved += Number(stat.approved); - coretranslated += Number(stat.translated); - corephrases += Number(stat.phrases); - } else if (/-strings\.json$/.test(stat.fullName)) { - approved += Number(stat.approved); - translated += Number(stat.translated); - phrases += Number(stat.phrases); - } - }) - fs.appendFileSync(fn, r, { encoding: "utf8" }); - console.log(`ui\t ${lang}\t ${(uitranslated / uiphrases * 100) >> 0}%\t ${(uiapproved / uiphrases * 100) >> 0}%\t ${uiphrases}\t ${uitranslated}\t ${uiapproved}`) - console.log(`core\t ${lang}\t ${(coretranslated / corephrases * 100) >> 0}%\t ${(coreapproved / corephrases * 100) >> 0}%\t ${corephrases}\t ${coretranslated}\t ${coreapproved}`) - console.log(`blocks\t ${lang}\t ${(translated / phrases * 100) >> 0}%\t ${(approved / phrases * 100) >> 0}%\t ${phrases}\t ${translated}\t ${approved}`) - }) - } + for (const language of Object.keys(translations)) { + const langTranslations = stringifyTranslations(JSON.parse(translations[language])); + if (!langTranslations) continue; + nodeutil.mkdirP(path.join(outputDir, language)); + const outFilename = path.join(outputDir, language, basename); + console.log(`Writing ${outFilename}`); + nodeutil.writeFileSync(outFilename, langTranslations, { encoding: "utf8" }); + } } -function uploadCrowdinAsync(branch: string, prj: string, key: string, p: string, dir?: string): Promise { - let fn = path.basename(p); - if (dir) fn = dir.replace(/[\\/]*$/g, '') + '/' + fn; - const data = JSON.parse(fs.readFileSync(p, "utf8")) as Map; - pxt.log(`upload ${fn} (${Object.keys(data).length} strings) to https://crowdin.com/project/${prj}${branch ? `?branch=${branch}` : ''}`); - return pxt.crowdin.uploadTranslationAsync(branch, prj, key, fn, JSON.stringify(data)); +async function execCleanAsync(dir: string): Promise { + const directoryPath = pxt.appTarget.id + "/" + dir; + + const files = await listFilesAsync(directoryPath); + + for (const file of files) { + if (!nodeutil.fileExistsSync(file.substring(pxt.appTarget.id.length + 1))) { + pxt.log(`crowdin: dead file: ${file}`) + } + } } + +// TODO (riknoll): Either remove this or update it to work with the new crowdin API +// function statsCrowdinAsync(prj: string, key: string, preferredLang?: string): Promise { +// pxt.log(`collecting crowdin stats for ${prj} ${preferredLang ? `for language ${preferredLang}` : `all languages`}`); +// console.log(`context\t language\t translated%\t approved%\t phrases\t translated\t approved`) + +// const fn = `crowdinstats.csv`; +// let headers = 'sep=\t\r\n'; +// headers += `id\t file\t language\t phrases\t translated\t approved\r\n`; +// nodeutil.writeFileSync(fn, headers, { encoding: "utf8" }); +// return pxt.crowdin.projectInfoAsync(prj, key) +// .then(info => { +// if (!info) throw new Error("info failed") +// let languages = info.languages; +// // remove in-context language +// languages = languages.filter(l => l.code != ts.pxtc.Util.TRANSLATION_LOCALE); +// if (preferredLang) +// languages = languages.filter(lang => lang.code.toLowerCase() == preferredLang.toLowerCase()); +// return Promise.all(languages.map(lang => langStatsCrowdinAsync(prj, key, lang.code))) +// }).then(() => { +// console.log(`stats written to ${fn}`) +// }) + +// function langStatsCrowdinAsync(prj: string, key: string, lang: string): Promise { +// return pxt.crowdin.languageStatsAsync(prj, key, lang) +// .then(stats => { +// let uiphrases = 0; +// let uitranslated = 0; +// let uiapproved = 0; +// let corephrases = 0; +// let coretranslated = 0; +// let coreapproved = 0; +// let phrases = 0; +// let translated = 0; +// let approved = 0; +// let r = ''; +// stats.forEach(stat => { +// const cfn = `${stat.branch ? stat.branch + "/" : ""}${stat.fullName}`; +// r += `${stat.id}\t ${cfn}\t ${lang}\t ${stat.phrases}\t ${stat.translated}\t ${stat.approved}\r\n`; +// if (stat.fullName == "strings.json") { +// uiapproved += Number(stat.approved); +// uitranslated += Number(stat.translated); +// uiphrases += Number(stat.phrases); +// } else if (/core-strings\.json$/.test(stat.fullName)) { +// coreapproved += Number(stat.approved); +// coretranslated += Number(stat.translated); +// corephrases += Number(stat.phrases); +// } else if (/-strings\.json$/.test(stat.fullName)) { +// approved += Number(stat.approved); +// translated += Number(stat.translated); +// phrases += Number(stat.phrases); +// } +// }) +// fs.appendFileSync(fn, r, { encoding: "utf8" }); +// console.log(`ui\t ${lang}\t ${(uitranslated / uiphrases * 100) >> 0}%\t ${(uiapproved / uiphrases * 100) >> 0}%\t ${uiphrases}\t ${uitranslated}\t ${uiapproved}`) +// console.log(`core\t ${lang}\t ${(coretranslated / corephrases * 100) >> 0}%\t ${(coreapproved / corephrases * 100) >> 0}%\t ${corephrases}\t ${coretranslated}\t ${coreapproved}`) +// console.log(`blocks\t ${lang}\t ${(translated / phrases * 100) >> 0}%\t ${(approved / phrases * 100) >> 0}%\t ${phrases}\t ${translated}\t ${approved}`) +// }) +// } + +// } diff --git a/cli/crowdinApi.ts b/cli/crowdinApi.ts new file mode 100644 index 00000000000..b19226124ff --- /dev/null +++ b/cli/crowdinApi.ts @@ -0,0 +1,386 @@ +import crowdin, { Credentials, SourceFilesModel, ClientConfig } from '@crowdin/crowdin-api-client'; +import * as path from 'path'; +import axios from 'axios'; + +import * as AdmZip from "adm-zip"; + +let client: crowdin; +const KINDSCRIPT_PROJECT_ID = 157956; + +let projectId = KINDSCRIPT_PROJECT_ID; + +let fetchedFiles: SourceFilesModel.File[]; +let fetchedDirectories: SourceFilesModel.Directory[]; + +export function setProjectId(id: number) { + projectId = id; + fetchedFiles = undefined; + fetchedDirectories = undefined; +} + +export async function uploadFileAsync(fileName: string, fileContent: string): Promise { + if (pxt.crowdin.testMode) return; + + const files = await getAllFiles(); + + // If file already exists, update it + for (const file of files) { + if (normalizePath(file.path) === normalizePath(fileName)) { + await updateFile(file.id, path.basename(fileName), fileContent); + return; + } + } + + // Ensure directory exists + const parentDir = path.dirname(fileName); + let parentDirId: number; + if (parentDir && parentDir !== ".") { + parentDirId = (await mkdirAsync(parentDir)).id; + } + + // Create new file + await createFile(path.basename(fileName), fileContent, parentDirId); +} + +export async function getProjectInfoAsync() { + const { projectsGroupsApi } = getClient(); + + const project = await projectsGroupsApi.getProject(projectId); + return project.data; +} + +export async function getProjectProgressAsync(languages?: string[]) { + const { translationStatusApi } = getClient(); + + const stats = await translationStatusApi + .withFetchAll() + .getProjectProgress(projectId); + + let results = stats.data.map(stat => stat.data); + if (languages) { + results = results.filter(stat => languages.indexOf(stat.language.locale) !== -1 || languages.indexOf(stat.language.twoLettersCode) !== -1); + } + + return results; +} + +export async function getDirectoryProgressAsync(directory: string, languages?: string[]) { + const { translationStatusApi } = getClient(); + + const directoryId = await getDirectoryIdAsync(directory); + + const stats = await translationStatusApi + .withFetchAll() + .getDirectoryProgress(projectId, directoryId); + + let results = stats.data.map(stat => stat.data); + if (languages) { + results = results.filter(stat => languages.indexOf(stat.language.locale) !== -1 || languages.indexOf(stat.language.twoLettersCode) !== -1); + } + + return results; +} + +export async function getFileProgressAsync(file: string, languages?: string[]) { + const { translationStatusApi } = getClient(); + + const fileId = await getFileIdAsync(file); + + const stats = await translationStatusApi + .withFetchAll() + .getFileProgress(projectId, fileId); + + let results = stats.data.map(stat => stat.data); + if (languages) { + results = results.filter(stat => languages.indexOf(stat.language.locale) !== -1 || languages.indexOf(stat.language.twoLettersCode) !== -1); + } + + return results; +} + +export async function listFilesAsync(directory: string): Promise { + directory = normalizePath(directory); + const files = (await getAllFiles()).map(file => normalizePath(file.path)); + + return files.filter(file => file.startsWith(directory)); +} + +export async function downloadTranslationsAsync(directory?: string) { + const { translationsApi } = getClient(); + + let buildId: number; + let status: string; + + const options = { + skipUntranslatedFiles: true, + exportApprovedOnly: true + }; + + if (directory) { + pxt.log(`Building translations for directory ${directory}`); + const directoryId = await getDirectoryIdAsync(directory); + const buildResp = await translationsApi.buildProjectDirectoryTranslation(projectId, directoryId, options); + buildId = buildResp.data.id; + status = buildResp.data.status; + } + else { + pxt.log(`Building all translations`) + const buildResp = await translationsApi.buildProject(projectId, options); + buildId = buildResp.data.id; + status = buildResp.data.status; + } + + // Translation builds take a long time, so poll for progress + while (status !== "finished") { + const progress = await translationsApi.checkBuildStatus(projectId, buildId); + status = progress.data.status; + + pxt.log(`Translation build progress: ${progress.data.progress}%`) + if (status !== "finished") { + await pxt.Util.delay(5000); + } + } + + pxt.log("Fetching translation build"); + const downloadReq = await translationsApi.downloadTranslations(projectId, buildId); + + // The downloaded file is a zip of all files broken out in a directory for each language + // e.g. /en/docs/tutorial.md, /fr/docs/tutorial.md, etc. + pxt.log("Downloading translation zip"); + const zipFile = await axios.get(downloadReq.data.url, { responseType: 'arraybuffer' }); + + const zip = new AdmZip(Buffer.from(zipFile.data)); + + const entries = zip.getEntries(); + const filesystem: pxt.Map = {}; + + for (const entry of entries) { + if (entry.isDirectory) continue; + + filesystem[entry.entryName] = zip.readAsText(entry); + } + + pxt.log("Translation download complete"); + + return filesystem; +} + +export async function downloadFileTranslationsAsync(fileName: string): Promise> { + const { translationsApi } = getClient(); + const fileId = await getFileIdAsync(fileName); + const projectInfo = await getProjectInfoAsync(); + + let todo = projectInfo.targetLanguageIds.filter(id => id !== "en"); + + if (pxt.appTarget && pxt.appTarget.appTheme && pxt.appTarget.appTheme.availableLocales) { + todo = todo.filter(l => pxt.appTarget.appTheme.availableLocales.indexOf(l) > -1); + } + + const options = { + skipUntranslatedFiles: true, + exportApprovedOnly: true + }; + + const results: pxt.Map = {}; + + // There's no API to get all translations for a file, so we have to build each one individually + for (const language of todo) { + pxt.debug(`Building ${language} translation for '${fileName}'`); + + try { + const buildResp = await translationsApi.buildProjectFileTranslation(projectId, fileId, { + targetLanguageId: language, + ...options + }); + + if (!buildResp.data) { + pxt.debug(`No translation available for ${language}`); + continue; + } + + const textResp = await axios.get(buildResp.data.url, { responseType: "text" }); + results[language] = textResp.data; + } + catch (e) { + console.log(`Error building ${language} translation for '${fileName}'`, e); + continue; + } + } + + return results; +} + +async function getFileIdAsync(fileName: string): Promise { + for (const file of await getAllFiles()) { + if (normalizePath(file.path) === normalizePath(fileName)) { + return file.id; + } + } + + throw new Error(`File '${fileName}' not found in crowdin project`); +} + +async function getDirectoryIdAsync(dirName: string): Promise { + for (const dir of await getAllDirectories()) { + if (normalizePath(dir.path) === normalizePath(dirName)) { + return dir.id; + } + } + + throw new Error(`Directory '${dirName}' not found in crowdin project`); +} + +async function mkdirAsync(dirName: string): Promise { + const dirs = await getAllDirectories(); + + for (const dir of dirs) { + if (normalizePath(dir.path) === normalizePath(dirName)) { + return dir; + } + } + + let parentDirId: number; + const parentDir = path.dirname(dirName); + if (parentDir && parentDir !== ".") { + parentDirId = (await mkdirAsync(parentDir)).id; + } + + return await createDirectory(path.basename(dirName), parentDirId); +} + +async function getAllDirectories() { + // This request takes a decent amount of time, so cache the results + if (!fetchedDirectories) { + const { sourceFilesApi } = getClient(); + + pxt.debug(`Fetching directories`) + const dirsResponse = await sourceFilesApi + .withFetchAll() + .listProjectDirectories(projectId, {}); + + let dirs = dirsResponse.data.map(fileResponse => fileResponse.data); + + if (!dirs.length) { + throw new Error("No directories found!"); + } + + pxt.debug(`Directory count: ${dirs.length}`); + + fetchedDirectories = dirs; + } + + return fetchedDirectories; +} + +async function getAllFiles() { + // This request takes a decent amount of time, so cache the results + if (!fetchedFiles) { + const { sourceFilesApi } = getClient(); + + pxt.debug(`Fetching files`) + const filesResponse = await sourceFilesApi + .withFetchAll() + .listProjectFiles(projectId, {}); + + let files = filesResponse.data.map(fileResponse => fileResponse.data); + + if (!files.length) { + throw new Error("No files found!"); + } + + pxt.debug(`File count: ${files.length}`); + fetchedFiles = files; + } + + return fetchedFiles; +} + +async function createFile(fileName: string, fileContent: any, directoryId?: number): Promise { + const { uploadStorageApi, sourceFilesApi } = getClient(); + + // This request happens in two parts: first we upload the file to the storage API, + // then we actually create the file + const storageResponse = await uploadStorageApi.addStorage(fileName, fileContent); + + const file = await sourceFilesApi.createFile(projectId, { + storageId: storageResponse.data.id, + name: fileName, + directoryId + }); + + // Make sure to add the file to the cache if it exists + if (fetchedFiles) { + fetchedFiles.push(file.data); + } +} + +async function createDirectory(dirName: string, directoryId?: number): Promise { + const { sourceFilesApi } = getClient(); + + const dir = await sourceFilesApi.createDirectory(projectId, { + name: dirName, + directoryId + }); + + // Make sure to add the directory to the cache if it exists + if (fetchedDirectories) { + fetchedDirectories.push(dir.data); + } + return dir.data; +} + + +async function updateFile(fileId: number, fileName: string, fileContent: any): Promise { + const { uploadStorageApi, sourceFilesApi } = getClient(); + + const storageResponse = await uploadStorageApi.addStorage(fileName, fileContent); + + await sourceFilesApi.updateOrRestoreFile(projectId, fileId, { + storageId: storageResponse.data.id, + }); +} + +function getClient() { + if (!client) { + const crowdinConfig: ClientConfig = { + retryConfig: { + retries: 5, + waitInterval: 5000, + conditions: [ + { + test: (error: any) => { + // do not retry when result has not changed + return error?.code == 304; + } + } + ] + } + }; + + client = new crowdin(crowdinCredentials(), crowdinConfig); + } + + return client; +} + +function crowdinCredentials(): Credentials { + const token = process.env[pxt.crowdin.KEY_VARIABLE]; + + if (!token) { + throw new Error(`Crowdin token not found in environment variable ${pxt.crowdin.KEY_VARIABLE}`); + } + + if (pxt.appTarget?.appTheme?.crowdinProjectId !== undefined) { + setProjectId(pxt.appTarget.appTheme.crowdinProjectId); + } + + return { token }; +} + +// calls path.normalize and removes leading slash +function normalizePath(p: string) { + p = path.normalize(p); + if (/^[\/\\]/.test(p)) p = p.slice(1) + + return p; +} \ No newline at end of file diff --git a/localtypings/pxtarget.d.ts b/localtypings/pxtarget.d.ts index db2c71941f7..e1c77ff8e93 100644 --- a/localtypings/pxtarget.d.ts +++ b/localtypings/pxtarget.d.ts @@ -373,6 +373,7 @@ declare namespace pxt { simAnimationExit?: string; // Simulator exit animation hasAudio?: boolean; // target uses the Audio manager. if true: a mute button is added to the simulator toolbar. crowdinProject?: string; + crowdinProjectId?: number; // Crowdin project id. Can be found by going to the project page in Crowdin and selecting Tools > API crowdinBranch?: string; // optional branch specification for localization files monacoToolbox?: boolean; // if true: show the monaco toolbox when in the monaco editor blockHats?: boolean; // if true, event blocks have hats diff --git a/package.json b/package.json index af3e1248c68..bb19b6275a1 100644 --- a/package.json +++ b/package.json @@ -63,11 +63,14 @@ "dependencies": { "@blockly/keyboard-navigation": "0.5.4", "@blockly/plugin-workspace-search": "8.0.9", + "@crowdin/crowdin-api-client": "^1.33.0", "@fortawesome/fontawesome-free": "^5.15.4", "@microsoft/applicationinsights-web": "^2.8.11", "@microsoft/immersive-reader-sdk": "1.1.0", "@types/diff-match-patch": "^1.0.32", "@zip.js/zip.js": "2.4.20", + "adm-zip": "^0.5.12", + "axios": "^1.6.8", "blockly": "10.4.1", "browserify": "16.2.3", "chai": "^3.5.0", @@ -101,6 +104,7 @@ }, "devDependencies": { "@microsoft/eslint-plugin-sdl": "^0.1.5", + "@types/adm-zip": "^0.5.5", "@types/chai": "4.0.6", "@types/fuse": "2.5.2", "@types/highlight.js": "9.12.2", diff --git a/pxtlib/crowdin.ts b/pxtlib/crowdin.ts index c9ece960914..427a20b1117 100644 --- a/pxtlib/crowdin.ts +++ b/pxtlib/crowdin.ts @@ -1,296 +1,12 @@ namespace pxt.crowdin { export const KEY_VARIABLE = "CROWDIN_KEY"; export let testMode = false; - export const TEST_KEY = "!!!testmode!!!"; export function setTestMode() { pxt.crowdin.testMode = true; pxt.log(`CROWDIN TEST MODE - files will NOT be uploaded`); } - function multipartPostAsync(key: string, uri: string, data: any = {}, filename: string = null, filecontents: string = null): Promise { - if (testMode || key == TEST_KEY) { - const resp = { - success: true - } - return Promise.resolve({ statusCode: 200, headers: {}, text: JSON.stringify(resp), json: resp }) - } - return Util.multipartPostAsync(uri, data, filename, filecontents); - } - - function apiUri(branch: string, prj: string, key: string, cmd: string, args?: Map) { - Util.assert(!!prj && !!key && !!cmd); - const apiRoot = "https://api.crowdin.com/api/project/" + prj + "/"; - args = args || {}; - if (testMode) - delete args["key"]; // ensure no key is passed in test mode - else - args["key"] = key; - if (branch) - args["branch"] = branch; - return apiRoot + cmd + "?" + Object.keys(args).map(k => `${k}=${encodeURIComponent(args[k])}`).join("&"); - } - - export interface CrowdinFileInfo { - name: string; - fullName?: string; - id: number; - node_type: "file" | "directory" | "branch"; - phrases?: number; - translated?: number; - approved?: number; - branch?: string; - files?: CrowdinFileInfo[]; - } - - export interface CrowdinProjectInfo { - languages: { name: string; code: string; }[]; - files: CrowdinFileInfo[]; - } - - export interface DownloadOptions { - translatedOnly?: boolean; - validatedOnly?: boolean; - } - - export function downloadTranslationsAsync(branch: string, prj: string, key: string, filename: string, options: DownloadOptions = {}): Promise>> { - const q: Map = { json: "true" } - const infoUri = apiUri(branch, prj, key, "info", q); - - const r: Map> = {}; - filename = normalizeFileName(filename); - return Util.httpGetTextAsync(infoUri).then(respText => { - const info = JSON.parse(respText) as CrowdinProjectInfo; - if (!info) throw new Error("info failed") - - let todo = info.languages.filter(l => l.code != "en"); - if (pxt.appTarget && pxt.appTarget.appTheme && pxt.appTarget.appTheme.availableLocales) - todo = todo.filter(l => pxt.appTarget.appTheme.availableLocales.indexOf(l.code) > -1); - pxt.log('languages: ' + todo.map(l => l.code).join(', ')); - const nextFile = (): Promise => { - const item = todo.pop(); - if (!item) return Promise.resolve(); - const exportFileUri = apiUri(branch, prj, key, "export-file", { - file: filename, - language: item.code, - export_translated_only: options.translatedOnly ? "1" : "0", - export_approved_only: options.validatedOnly ? "1" : "0" - }); - pxt.log(`downloading ${item.name} - ${item.code} (${todo.length} more)`) - return Util.httpGetTextAsync(exportFileUri).then((transationsText) => { - try { - const translations = JSON.parse(transationsText) as Map; - if (translations) - r[item.code] = translations; - } catch (e) { - pxt.log(exportFileUri + ' ' + e) - } - return nextFile(); - }).then(() => Util.delay(1000)); // throttling otherwise crowdin fails - }; - - return nextFile(); - }).then(() => r); - } - - function mkIncr(filename: string): () => void { - let cnt = 0 - return function incr() { - if (cnt++ > 10) { - throw new Error("Too many API calls for " + filename); - } - } - } - - export function createDirectoryAsync(branch: string, prj: string, key: string, name: string, incr?: () => void): Promise { - name = normalizeFileName(name); - pxt.debug(`create directory ${branch || ""}/${name}`) - if (!incr) incr = mkIncr(name); - return multipartPostAsync(key, apiUri(branch, prj, key, "add-directory"), { json: "true", name: name }) - .then(resp => { - pxt.debug(`crowdin resp: ${resp.statusCode}`) - // 400 returned by folder already exists - if (resp.statusCode == 200 || resp.statusCode == 400) - return Promise.resolve(); - - if (resp.statusCode == 500 && resp.text) { - const json = JSON.parse(resp.text); - if (json.error.code === 50) { - pxt.log('directory already exists') - return Promise.resolve(); - } - } - - const data: any = resp.json || JSON.parse(resp.text) || { error: {} } - if (resp.statusCode == 404 && data.error.code == 17) { - pxt.log(`parent directory missing for ${name}`) - const par = name.replace(/\/[^\/]+$/, "") - if (par != name) { - return createDirectoryAsync(branch, prj, key, par, incr) - .then(() => createDirectoryAsync(branch, prj, key, name, incr)); // retry - } - } - - throw new Error(`cannot create directory ${branch || ""}/${name}: ${resp.statusCode} ${JSON.stringify(data)}`) - }) - } - - function normalizeFileName(filename: string): string { - return filename.replace(/\\/g, '/'); - } - - export function uploadTranslationAsync(branch: string, prj: string, key: string, filename: string, data: string) { - Util.assert(!!prj); - Util.assert(!!key); - - filename = normalizeFileName(filename); - const incr = mkIncr(filename); - - function startAsync(): Promise { - return uploadAsync("update-file", { update_option: "update_as_unapproved" }) - } - - function uploadAsync(op: string, opts: any): Promise { - opts["type"] = "auto"; - opts["json"] = ""; - opts["escape_quotes"] = "0"; - incr(); - return multipartPostAsync(key, apiUri(branch, prj, key, op), opts, filename, data) - .then(resp => handleResponseAsync(resp)) - } - - function handleResponseAsync(resp: Util.HttpResponse) { - const code = resp.statusCode; - const errorData: { - success?: boolean; - error?: { - code: number; - } - } = Util.jsonTryParse(resp.text) || {}; - - pxt.debug(`upload result: ${code}`); - if (code == 404 && errorData.error && errorData.error.code == 8) { - pxt.log(`create new translation file: ${filename}`) - return uploadAsync("add-file", {}) - } - else if (code == 404 && errorData.error?.code == 17) { - return createDirectoryAsync(branch, prj, key, filename.replace(/\/[^\/]+$/, ""), incr) - .then(() => startAsync()) - } else if (!errorData.success && errorData.error?.code == 53) { - // file is being updated - pxt.log(`${filename} being updated, waiting 5s and retry...`) - return U.delay(5000) // wait 5s and try again - .then(() => uploadTranslationAsync(branch, prj, key, filename, data)); - } else if (code == 429 && errorData.error?.code == 55) { - // Too many concurrent requests - pxt.log(`Maximum concurrent requests reached, waiting 10s and retry...`) - return U.delay(10 * 1000) // wait 10s and try again - .then(() => uploadTranslationAsync(branch, prj, key, filename, data)); - } else if (code == 200 || errorData.success) { - // something crowdin reports 500 with success=true - return Promise.resolve() - } else { - throw new Error(`Error, upload translation: ${filename}, ${code}, ${resp.text}`) - } - } - - return startAsync(); - } - - function flatten(allFiles: CrowdinFileInfo[], node: CrowdinFileInfo, parentDir: string, branch?: string) { - const n = node.name; - const d = parentDir ? parentDir + "/" + n : n; - node.fullName = d; - node.branch = branch || ""; - switch (node.node_type) { - case "file": - allFiles.push(node); - break; - case "directory": - (node.files || []).forEach(f => flatten(allFiles, f, d, branch)); - break; - case "branch": - (node.files || []).forEach(f => flatten(allFiles, f, parentDir, node.name)); - break; - } - } - - function filterAndFlattenFiles(files: CrowdinFileInfo[], crowdinPath?: string): CrowdinFileInfo[] { - const pxtCrowdinBranch = pxt.appTarget.versions.pxtCrowdinBranch || ""; - const targetCrowdinBranch = pxt.appTarget.versions.targetCrowdinBranch || ""; - - let allFiles: CrowdinFileInfo[] = []; - - // flatten the files - files.forEach(f => flatten(allFiles, f, "")); - - // top level files are for PXT, subolder are targets - allFiles = allFiles.filter(f => { - if (f.fullName.indexOf('/') < 0) return f.branch == pxtCrowdinBranch; // pxt file - else return f.branch == targetCrowdinBranch; - }) - - // folder filter - if (crowdinPath) { - // filter out crowdin folder - allFiles = allFiles.filter(f => f.fullName.indexOf(crowdinPath) == 0); - } - - // filter out non-target files - if (pxt.appTarget.id != "core") { - const id = pxt.appTarget.id + '/' - allFiles = allFiles.filter(f => { - return f.fullName.indexOf('/') < 0 // top level file - || f.fullName.substr(0, id.length) == id // from the target folder - || f.fullName.indexOf('common-docs') >= 0 // common docs - }) - } - - return allFiles; - } - - export function projectInfoAsync(prj: string, key: string): Promise { - const q: Map = { json: "true" } - const infoUri = apiUri("", prj, key, "info", q); - return Util.httpGetTextAsync(infoUri).then(respText => { - const info = JSON.parse(respText) as CrowdinProjectInfo; - return info; - }); - } - - /** - * Scans files in crowdin and report files that are not on disk anymore - */ - export function listFilesAsync(prj: string, key: string, crowdinPath: string): Promise<{ fullName: string; branch: string; }[]> { - - pxt.log(`crowdin: listing files under ${crowdinPath}`); - - return projectInfoAsync(prj, key) - .then(info => { - if (!info) throw new Error("info failed") - - let allFiles = filterAndFlattenFiles(info.files, crowdinPath); - pxt.debug(`crowdin: found ${allFiles.length} under ${crowdinPath}`) - - return allFiles.map(f => { - return { - fullName: f.fullName, - branch: f.branch || "" - }; - }) - }); - } - - export function languageStatsAsync(prj: string, key: string, lang: string): Promise { - const uri = apiUri("", prj, key, "language-status", { language: lang, json: "true" }); - - return Util.httpGetJsonAsync(uri) - .then(info => { - const allFiles = filterAndFlattenFiles(info.files); - return allFiles; - }); - } - export function inContextLoadAsync(text: string): Promise { const node = document.createElement("input") as HTMLInputElement; node.type = "text";