Skip to content

Commit b5e28b4

Browse files
committed
Delay parsing of translations until they're used
While doing some profiling for miniflux#2900, I noticed that `miniflux.app/v2/internal/locale.LoadCatalogMessages` is responsible for more than 10% of the consumed memory. As most miniflux instances won't have enough diverse users to use all the available translations at the same time, it makes sense to load them on demand. The overhead is a single function call and a check in a map, per call to translation-related functions. This should close miniflux#2975
1 parent c1ef986 commit b5e28b4

File tree

3 files changed

+30
-21
lines changed

3 files changed

+30
-21
lines changed

internal/cli/cli.go

-5
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@ import (
1313

1414
"miniflux.app/v2/internal/config"
1515
"miniflux.app/v2/internal/database"
16-
"miniflux.app/v2/internal/locale"
1716
"miniflux.app/v2/internal/storage"
1817
"miniflux.app/v2/internal/ui/static"
1918
"miniflux.app/v2/internal/version"
@@ -153,10 +152,6 @@ func Parse() {
153152
slog.Info("The default value for DATABASE_URL is used")
154153
}
155154

156-
if err := locale.LoadCatalogMessages(); err != nil {
157-
printErrorAndExit(fmt.Errorf("unable to load translations: %v", err))
158-
}
159-
160155
if err := static.CalculateBinaryFileChecksums(); err != nil {
161156
printErrorAndExit(fmt.Errorf("unable to calculate binary file checksums: %v", err))
162157
}

internal/locale/catalog.go

+11-2
Original file line numberDiff line numberDiff line change
@@ -12,15 +12,24 @@ import (
1212
type translationDict map[string]interface{}
1313
type catalog map[string]translationDict
1414

15-
var defaultCatalog catalog
15+
var defaultCatalog = make(catalog, len(AvailableLanguages()))
1616

1717
//go:embed translations/*.json
1818
var translationFiles embed.FS
1919

20+
func GetTranslationDict(language string) (translationDict, error) {
21+
if _, ok := defaultCatalog[language]; !ok {
22+
var err error
23+
if defaultCatalog[language], err = loadTranslationFile(language); err != nil {
24+
return nil, err
25+
}
26+
}
27+
return defaultCatalog[language], nil
28+
}
29+
2030
// LoadCatalogMessages loads and parses all translations encoded in JSON.
2131
func LoadCatalogMessages() error {
2232
var err error
23-
defaultCatalog = make(catalog, len(AvailableLanguages()))
2433

2534
for language := range AvailableLanguages() {
2635
defaultCatalog[language], err = loadTranslationFile(language)

internal/locale/printer.go

+19-14
Original file line numberDiff line numberDiff line change
@@ -11,26 +11,28 @@ type Printer struct {
1111
}
1212

1313
func (p *Printer) Print(key string) string {
14-
if str, ok := defaultCatalog[p.language][key]; ok {
15-
if translation, ok := str.(string); ok {
16-
return translation
14+
if dict, err := GetTranslationDict(p.language); err == nil {
15+
if str, ok := dict[key]; ok {
16+
if translation, ok := str.(string); ok {
17+
return translation
18+
}
1719
}
1820
}
1921
return key
2022
}
2123

2224
// Printf is like fmt.Printf, but using language-specific formatting.
2325
func (p *Printer) Printf(key string, args ...interface{}) string {
24-
var translation string
26+
translation := key
2527

26-
str, found := defaultCatalog[p.language][key]
27-
if !found {
28-
translation = key
29-
} else {
30-
var valid bool
31-
translation, valid = str.(string)
32-
if !valid {
33-
translation = key
28+
if dict, err := GetTranslationDict(p.language); err == nil {
29+
str, found := dict[key]
30+
if found {
31+
var valid bool
32+
translation, valid = str.(string)
33+
if !valid {
34+
translation = key
35+
}
3436
}
3537
}
3638

@@ -39,9 +41,12 @@ func (p *Printer) Printf(key string, args ...interface{}) string {
3941

4042
// Plural returns the translation of the given key by using the language plural form.
4143
func (p *Printer) Plural(key string, n int, args ...interface{}) string {
42-
choices, found := defaultCatalog[p.language][key]
44+
dict, err := GetTranslationDict(p.language)
45+
if err != nil {
46+
return key
47+
}
4348

44-
if found {
49+
if choices, found := dict[key]; found {
4550
var plurals []string
4651

4752
switch v := choices.(type) {

0 commit comments

Comments
 (0)