465 lines
9.6 KiB
Go
465 lines
9.6 KiB
Go
// translations downloads translations, uploads translations, prints summary
|
|
// for translations, prints unused strings.
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/aghio"
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
"github.com/AdguardTeam/golibs/log"
|
|
"golang.org/x/exp/maps"
|
|
"golang.org/x/exp/slices"
|
|
)
|
|
|
|
const (
|
|
twoskyConfFile = "./.twosky.json"
|
|
localesDir = "./client/src/__locales"
|
|
defaultBaseFile = "en.json"
|
|
defaultProjectID = "home"
|
|
srcDir = "./client/src"
|
|
twoskyURI = "https://twosky.int.agrd.dev/api/v1"
|
|
|
|
readLimit = 1 * 1024 * 1024
|
|
)
|
|
|
|
// langCode is a language code.
|
|
type langCode string
|
|
|
|
// languages is a map, where key is language code and value is display name.
|
|
type languages map[langCode]string
|
|
|
|
// textlabel is a text label of localization.
|
|
type textLabel string
|
|
|
|
// locales is a map, where key is text label and value is translation.
|
|
type locales map[textLabel]string
|
|
|
|
func main() {
|
|
if len(os.Args) == 1 {
|
|
usage("need a command")
|
|
}
|
|
|
|
if os.Args[1] == "help" {
|
|
usage("")
|
|
}
|
|
|
|
uriStr := os.Getenv("TWOSKY_URI")
|
|
if uriStr == "" {
|
|
uriStr = twoskyURI
|
|
}
|
|
|
|
uri, err := url.Parse(uriStr)
|
|
check(err)
|
|
|
|
projectID := os.Getenv("TWOSKY_PROJECT_ID")
|
|
if projectID == "" {
|
|
projectID = defaultProjectID
|
|
}
|
|
|
|
conf, err := readTwoskyConf()
|
|
check(err)
|
|
|
|
switch os.Args[1] {
|
|
case "summary":
|
|
err = summary(conf.Languages)
|
|
check(err)
|
|
case "download":
|
|
err = download(uri, projectID, conf.Languages)
|
|
check(err)
|
|
case "unused":
|
|
err = unused()
|
|
check(err)
|
|
case "upload":
|
|
err = upload(uri, projectID, conf.BaseLangcode)
|
|
check(err)
|
|
default:
|
|
usage("unknown command")
|
|
}
|
|
}
|
|
|
|
// check is a simple error-checking helper for scripts.
|
|
func check(err error) {
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// usage prints usage. If addStr is not empty print addStr and exit with code
|
|
// 1, otherwise exit with code 0.
|
|
func usage(addStr string) {
|
|
const usageStr = `Usage: go run main.go <command> [<args>]
|
|
Commands:
|
|
help
|
|
Print usage.
|
|
summary
|
|
Print summary.
|
|
download [-n <count>]
|
|
Download translations. count is a number of concurrent downloads.
|
|
unused
|
|
Print unused strings.
|
|
upload
|
|
Upload translations.`
|
|
|
|
if addStr != "" {
|
|
fmt.Printf("%s\n%s\n", addStr, usageStr)
|
|
|
|
os.Exit(1)
|
|
}
|
|
|
|
fmt.Println(usageStr)
|
|
|
|
os.Exit(0)
|
|
}
|
|
|
|
// twoskyConf is the configuration structure for localization.
|
|
type twoskyConf struct {
|
|
Languages languages `json:"languages"`
|
|
ProjectID string `json:"project_id"`
|
|
BaseLangcode langCode `json:"base_locale"`
|
|
LocalizableFiles []string `json:"localizable_files"`
|
|
}
|
|
|
|
// readTwoskyConf returns configuration.
|
|
func readTwoskyConf() (t twoskyConf, err error) {
|
|
b, err := os.ReadFile(twoskyConfFile)
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return twoskyConf{}, err
|
|
}
|
|
|
|
var tsc []twoskyConf
|
|
err = json.Unmarshal(b, &tsc)
|
|
if err != nil {
|
|
err = fmt.Errorf("unmarshalling %q: %w", twoskyConfFile, err)
|
|
|
|
return twoskyConf{}, err
|
|
}
|
|
|
|
if len(tsc) == 0 {
|
|
err = fmt.Errorf("%q is empty", twoskyConfFile)
|
|
|
|
return twoskyConf{}, err
|
|
}
|
|
|
|
conf := tsc[0]
|
|
|
|
for _, lang := range conf.Languages {
|
|
if lang == "" {
|
|
return twoskyConf{}, errors.Error("language is empty")
|
|
}
|
|
}
|
|
|
|
return conf, nil
|
|
}
|
|
|
|
// readLocales reads file with name fn and returns a map, where key is text
|
|
// label and value is localization.
|
|
func readLocales(fn string) (loc locales, err error) {
|
|
b, err := os.ReadFile(fn)
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return nil, err
|
|
}
|
|
|
|
loc = make(locales)
|
|
err = json.Unmarshal(b, &loc)
|
|
if err != nil {
|
|
err = fmt.Errorf("unmarshalling %q: %w", fn, err)
|
|
|
|
return nil, err
|
|
}
|
|
|
|
return loc, nil
|
|
}
|
|
|
|
// summary prints summary for translations.
|
|
func summary(langs languages) (err error) {
|
|
basePath := filepath.Join(localesDir, defaultBaseFile)
|
|
baseLoc, err := readLocales(basePath)
|
|
if err != nil {
|
|
return fmt.Errorf("summary: %w", err)
|
|
}
|
|
|
|
size := float64(len(baseLoc))
|
|
|
|
keys := maps.Keys(langs)
|
|
slices.Sort(keys)
|
|
|
|
for _, lang := range keys {
|
|
name := filepath.Join(localesDir, string(lang)+".json")
|
|
if name == basePath {
|
|
continue
|
|
}
|
|
|
|
var loc locales
|
|
loc, err = readLocales(name)
|
|
if err != nil {
|
|
return fmt.Errorf("summary: reading locales: %w", err)
|
|
}
|
|
|
|
f := float64(len(loc)) * 100 / size
|
|
|
|
fmt.Printf("%s\t %6.2f %%\n", lang, f)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// download and save all translations. uri is the base URL. projectID is the
|
|
// name of the project.
|
|
func download(uri *url.URL, projectID string, langs languages) (err error) {
|
|
var numWorker int
|
|
|
|
flagSet := flag.NewFlagSet("download", flag.ExitOnError)
|
|
flagSet.Usage = func() {
|
|
usage("download command error")
|
|
}
|
|
flagSet.IntVar(&numWorker, "n", 1, "number of concurrent downloads")
|
|
|
|
err = flagSet.Parse(os.Args[2:])
|
|
if err != nil {
|
|
// Don't wrap the error since there is exit on error.
|
|
return err
|
|
}
|
|
|
|
if numWorker < 1 {
|
|
usage("count must be positive")
|
|
}
|
|
|
|
downloadURI := uri.JoinPath("download")
|
|
|
|
client := &http.Client{
|
|
Timeout: 10 * time.Second,
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
uriCh := make(chan *url.URL, len(langs))
|
|
|
|
for i := 0; i < numWorker; i++ {
|
|
wg.Add(1)
|
|
go downloadWorker(&wg, client, uriCh)
|
|
}
|
|
|
|
for lang := range langs {
|
|
uri = translationURL(downloadURI, defaultBaseFile, projectID, lang)
|
|
|
|
uriCh <- uri
|
|
}
|
|
|
|
close(uriCh)
|
|
wg.Wait()
|
|
|
|
return nil
|
|
}
|
|
|
|
// downloadWorker downloads translations by received urls and saves them.
|
|
func downloadWorker(wg *sync.WaitGroup, client *http.Client, uriCh <-chan *url.URL) {
|
|
defer wg.Done()
|
|
|
|
for uri := range uriCh {
|
|
data, err := getTranslation(client, uri.String())
|
|
if err != nil {
|
|
log.Error("download worker: getting translation: %s", err)
|
|
|
|
continue
|
|
}
|
|
|
|
q := uri.Query()
|
|
code := q.Get("language")
|
|
|
|
name := filepath.Join(localesDir, code+".json")
|
|
err = os.WriteFile(name, data, 0o664)
|
|
if err != nil {
|
|
log.Error("download worker: writing file: %s", err)
|
|
|
|
continue
|
|
}
|
|
|
|
fmt.Println(name)
|
|
}
|
|
}
|
|
|
|
// getTranslation returns received translation data or error.
|
|
func getTranslation(client *http.Client, url string) (data []byte, err error) {
|
|
resp, err := client.Get(url)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("requesting: %w", err)
|
|
}
|
|
|
|
defer log.OnCloserError(resp.Body, log.ERROR)
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
err = fmt.Errorf("url: %q; status code: %s", url, http.StatusText(resp.StatusCode))
|
|
|
|
return nil, err
|
|
}
|
|
|
|
limitReader, err := aghio.LimitReader(resp.Body, readLimit)
|
|
if err != nil {
|
|
err = fmt.Errorf("limit reading: %w", err)
|
|
|
|
return nil, err
|
|
}
|
|
|
|
data, err = io.ReadAll(limitReader)
|
|
if err != nil {
|
|
err = fmt.Errorf("reading all: %w", err)
|
|
|
|
return nil, err
|
|
}
|
|
|
|
return data, nil
|
|
}
|
|
|
|
// translationURL returns a new url.URL with provided query parameters.
|
|
func translationURL(oldURL *url.URL, baseFile, projectID string, lang langCode) (uri *url.URL) {
|
|
uri = &url.URL{}
|
|
*uri = *oldURL
|
|
|
|
q := uri.Query()
|
|
q.Set("format", "json")
|
|
q.Set("filename", baseFile)
|
|
q.Set("project", projectID)
|
|
q.Set("language", string(lang))
|
|
|
|
uri.RawQuery = q.Encode()
|
|
|
|
return uri
|
|
}
|
|
|
|
// unused prints unused text labels.
|
|
func unused() (err error) {
|
|
fileNames := []string{}
|
|
basePath := filepath.Join(localesDir, defaultBaseFile)
|
|
baseLoc, err := readLocales(basePath)
|
|
if err != nil {
|
|
return fmt.Errorf("unused: %w", err)
|
|
}
|
|
|
|
locDir := filepath.Clean(localesDir)
|
|
|
|
err = filepath.Walk(srcDir, func(name string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
log.Info("accessing a path %q: %s", name, err)
|
|
|
|
return nil
|
|
}
|
|
|
|
if info.IsDir() {
|
|
return nil
|
|
}
|
|
|
|
if strings.HasPrefix(name, locDir) {
|
|
return nil
|
|
}
|
|
|
|
ext := filepath.Ext(name)
|
|
if ext == ".js" || ext == ".json" {
|
|
fileNames = append(fileNames, name)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
if err != nil {
|
|
return fmt.Errorf("filepath walking %q: %w", srcDir, err)
|
|
}
|
|
|
|
err = removeUnused(fileNames, baseLoc)
|
|
|
|
return errors.Annotate(err, "removing unused: %w")
|
|
}
|
|
|
|
func removeUnused(fileNames []string, loc locales) (err error) {
|
|
knownUsed := []textLabel{
|
|
"blocking_mode_refused",
|
|
"blocking_mode_nxdomain",
|
|
"blocking_mode_custom_ip",
|
|
}
|
|
|
|
for _, v := range knownUsed {
|
|
delete(loc, v)
|
|
}
|
|
|
|
for _, fn := range fileNames {
|
|
var buf []byte
|
|
buf, err = os.ReadFile(fn)
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return err
|
|
}
|
|
|
|
for k := range loc {
|
|
if bytes.Contains(buf, []byte(k)) {
|
|
delete(loc, k)
|
|
}
|
|
}
|
|
}
|
|
|
|
printUnused(loc)
|
|
|
|
return nil
|
|
}
|
|
|
|
// printUnused text labels to stdout.
|
|
func printUnused(loc locales) {
|
|
keys := maps.Keys(loc)
|
|
slices.Sort(keys)
|
|
|
|
for _, v := range keys {
|
|
fmt.Println(v)
|
|
}
|
|
}
|
|
|
|
// upload base translation. uri is the base URL. projectID is the name of the
|
|
// project. baseLang is the base language code.
|
|
func upload(uri *url.URL, projectID string, baseLang langCode) (err error) {
|
|
uploadURI := uri.JoinPath("upload")
|
|
|
|
lang := baseLang
|
|
|
|
langStr := os.Getenv("UPLOAD_LANGUAGE")
|
|
if langStr != "" {
|
|
lang = langCode(langStr)
|
|
}
|
|
|
|
basePath := filepath.Join(localesDir, defaultBaseFile)
|
|
b, err := os.ReadFile(basePath)
|
|
if err != nil {
|
|
return fmt.Errorf("upload: %w", err)
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
buf.Write(b)
|
|
|
|
uri = translationURL(uploadURI, defaultBaseFile, projectID, lang)
|
|
|
|
var client http.Client
|
|
resp, err := client.Post(uri.String(), "application/json", &buf)
|
|
if err != nil {
|
|
return fmt.Errorf("upload: client post: %w", err)
|
|
}
|
|
|
|
defer func() {
|
|
err = errors.WithDeferred(err, resp.Body.Close())
|
|
}()
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
return fmt.Errorf("status code is not ok: %q", http.StatusText(resp.StatusCode))
|
|
}
|
|
|
|
return nil
|
|
}
|