You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

backport-locales.go 3.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. // Copyright 2023 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. //go:build ignore
  4. package main
  5. import (
  6. "fmt"
  7. "os"
  8. "os/exec"
  9. "path/filepath"
  10. "strings"
  11. "code.gitea.io/gitea/modules/setting"
  12. )
  13. func main() {
  14. if len(os.Args) != 2 {
  15. println("usage: backport-locales <to-ref>")
  16. println("eg: backport-locales release/v1.19")
  17. os.Exit(1)
  18. }
  19. mustNoErr := func(err error) {
  20. if err != nil {
  21. panic(err)
  22. }
  23. }
  24. collectInis := func(ref string) map[string]setting.ConfigProvider {
  25. inis := map[string]setting.ConfigProvider{}
  26. err := filepath.WalkDir("options/locale", func(path string, d os.DirEntry, err error) error {
  27. if err != nil {
  28. return err
  29. }
  30. if d.IsDir() || !strings.HasSuffix(d.Name(), ".ini") {
  31. return nil
  32. }
  33. cfg, err := setting.NewConfigProviderForLocale(path)
  34. mustNoErr(err)
  35. inis[path] = cfg
  36. fmt.Printf("collecting: %s @ %s\n", path, ref)
  37. return nil
  38. })
  39. mustNoErr(err)
  40. return inis
  41. }
  42. // collect new locales from current working directory
  43. inisNew := collectInis("HEAD")
  44. // switch to the target ref, and collect the old locales
  45. cmd := exec.Command("git", "checkout", os.Args[1])
  46. cmd.Stdout = os.Stdout
  47. cmd.Stderr = os.Stderr
  48. mustNoErr(cmd.Run())
  49. inisOld := collectInis(os.Args[1])
  50. // use old en-US as the base, and copy the new translations to the old locales
  51. enUsOld := inisOld["options/locale/locale_en-US.ini"]
  52. brokenWarned := map[string]bool{}
  53. for path, iniOld := range inisOld {
  54. if iniOld == enUsOld {
  55. continue
  56. }
  57. iniNew := inisNew[path]
  58. if iniNew == nil {
  59. continue
  60. }
  61. for _, secEnUS := range enUsOld.Sections() {
  62. secOld := iniOld.Section(secEnUS.Name())
  63. secNew := iniNew.Section(secEnUS.Name())
  64. for _, keyEnUs := range secEnUS.Keys() {
  65. if secNew.HasKey(keyEnUs.Name()) {
  66. oldStr := secOld.Key(keyEnUs.Name()).String()
  67. newStr := secNew.Key(keyEnUs.Name()).String()
  68. broken := oldStr != "" && strings.Count(oldStr, "%") != strings.Count(newStr, "%")
  69. broken = broken || strings.Contains(oldStr, "\n") || strings.Contains(oldStr, "\n")
  70. if broken {
  71. brokenWarned[secOld.Name()+"."+keyEnUs.Name()] = true
  72. fmt.Println("----")
  73. fmt.Printf("WARNING: skip broken locale: %s , [%s] %s\n", path, secEnUS.Name(), keyEnUs.Name())
  74. fmt.Printf("\told: %s\n", strings.ReplaceAll(oldStr, "\n", "\\n"))
  75. fmt.Printf("\tnew: %s\n", strings.ReplaceAll(newStr, "\n", "\\n"))
  76. continue
  77. }
  78. secOld.Key(keyEnUs.Name()).SetValue(newStr)
  79. }
  80. }
  81. }
  82. mustNoErr(iniOld.SaveTo(path))
  83. }
  84. fmt.Println("========")
  85. for path, iniNew := range inisNew {
  86. for _, sec := range iniNew.Sections() {
  87. for _, key := range sec.Keys() {
  88. str := sec.Key(key.Name()).String()
  89. broken := strings.Contains(str, "\n")
  90. broken = broken || strings.HasPrefix(str, "`") != strings.HasSuffix(str, "`")
  91. broken = broken || strings.HasPrefix(str, "\"`")
  92. broken = broken || strings.HasPrefix(str, "`\"")
  93. broken = broken || strings.Count(str, `"`)%2 == 1
  94. broken = broken || strings.Count(str, "`")%2 == 1
  95. if broken && !brokenWarned[sec.Name()+"."+key.Name()] {
  96. fmt.Printf("WARNING: found broken locale: %s , [%s] %s\n", path, sec.Name(), key.Name())
  97. fmt.Printf("\tstr: %s\n", strings.ReplaceAll(str, "\n", "\\n"))
  98. fmt.Println("----")
  99. }
  100. }
  101. }
  102. }
  103. }