i18n: Allow custom language codes
authorKevin Gimbel <hallo@kevingimbel.com>
Sun, 15 Oct 2017 10:35:25 +0000 (12:35 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 4 Nov 2017 09:48:29 +0000 (10:48 +0100)
Use the new `RegisterPluralSpec` function to register all defined
languages. This allows the usage of language identifiers which are not
part of the Unicode CLDR standard.

Closes #3564

i18n/i18n_test.go
i18n/translationProvider.go

index 6a9d362b01fe1c7293de15cf157f8461a822fc86..9e7aed0398be15c09e8bbbec6613c01464e56f5d 100644 (file)
@@ -23,9 +23,11 @@ import (
 
        "github.com/gohugoio/hugo/config"
        "github.com/nicksnyder/go-i18n/i18n/bundle"
+       "github.com/nicksnyder/go-i18n/i18n/language"
        jww "github.com/spf13/jwalterweatherman"
        "github.com/spf13/viper"
        "github.com/stretchr/testify/require"
+       "strings"
 )
 
 var logger = jww.NewNotepad(jww.LevelError, jww.LevelError, os.Stdout, ioutil.Discard, "", log.Ldate|log.Ltime)
@@ -137,10 +139,29 @@ var i18nTests = []i18nTest{
                expected:     "hello",
                expectedFlag: "[i18n] hello",
        },
+       // Non Unicode CLDR language code
+       {
+               data: map[string][]byte{
+                       "dk.toml": []byte("[hello]\nother = \"hej\""),
+               },
+               args:         nil,
+               lang:         "dk",
+               id:           "hello",
+               expected:     "hej",
+               expectedFlag: "hej",
+       },
 }
 
 func doTestI18nTranslate(t *testing.T, test i18nTest, cfg config.Provider) string {
        i18nBundle := bundle.New()
+       ids := []string{}
+
+       for file := range test.data {
+               id := strings.TrimSuffix(file, ".toml")
+               ids = append(ids, id)
+       }
+
+       language.RegisterPluralSpec(ids, &language.PluralSpec{})
 
        for file, content := range test.data {
                err := i18nBundle.ParseTranslationFileBytes(file, content)
index 9947d3ce52156969c5abde3cc1d07fcf587aae6d..663a8dd5e0d15dff82af971e4047f96c4e4d78e3 100644 (file)
@@ -15,12 +15,18 @@ package i18n
 
 import (
        "fmt"
+       "sync"
 
        "github.com/gohugoio/hugo/deps"
        "github.com/gohugoio/hugo/source"
        "github.com/nicksnyder/go-i18n/i18n/bundle"
+       "github.com/nicksnyder/go-i18n/i18n/language"
 )
 
+// Unfortunately this needs to be global, see
+// https://github.com/nicksnyder/go-i18n/issues/82
+var tpMu sync.Mutex
+
 // TranslationProvider provides translation handling, i.e. loading
 // of bundles etc.
 type TranslationProvider struct {
@@ -34,6 +40,9 @@ func NewTranslationProvider() *TranslationProvider {
 
 // Update updates the i18n func in the provided Deps.
 func (tp *TranslationProvider) Update(d *deps.Deps) error {
+       tpMu.Lock()
+       defer tpMu.Unlock()
+
        dir := d.PathSpec.AbsPathify(d.Cfg.GetString("i18nDir"))
        sp := source.NewSourceSpec(d.Cfg, d.Fs)
        sources := []source.Input{sp.NewFilesystem(dir)}
@@ -48,6 +57,17 @@ func (tp *TranslationProvider) Update(d *deps.Deps) error {
 
        i18nBundle := bundle.New()
 
+       langs := []string{}
+       for _, currentSource := range sources {
+               for _, r := range currentSource.Files() {
+                       langs = append(langs, r.BaseFileName())
+               }
+       }
+       // We need to register all language codes as "plural spec" to prevent errors with unknown language codes.
+       // see https://github.com/gohugoio/hugo/issues/3564
+       ps := &language.PluralSpec{}
+       language.RegisterPluralSpec(langs, ps)
+
        for _, currentSource := range sources {
                for _, r := range currentSource.Files() {
                        err := i18nBundle.ParseTranslationFileBytes(r.LogicalName(), r.Bytes())