2020-06-05 07:50:04 +02:00
|
|
|
package i18n
|
|
|
|
|
|
|
|
import (
|
2020-06-22 13:51:44 +02:00
|
|
|
"context"
|
2020-06-05 07:50:04 +02:00
|
|
|
"net/http"
|
|
|
|
|
2021-07-05 15:10:49 +02:00
|
|
|
"github.com/grpc-ecosystem/go-grpc-middleware/util/metautils"
|
2020-06-05 07:50:04 +02:00
|
|
|
"github.com/nicksnyder/go-i18n/v2/i18n"
|
2022-04-27 01:01:45 +02:00
|
|
|
"github.com/zitadel/logging"
|
2020-06-05 07:50:04 +02:00
|
|
|
"golang.org/x/text/language"
|
2021-07-05 15:10:49 +02:00
|
|
|
|
2022-04-27 01:01:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/api/authz"
|
|
|
|
http_util "github.com/zitadel/zitadel/internal/api/http"
|
2020-06-05 07:50:04 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Translator struct {
|
2021-07-09 09:45:31 +02:00
|
|
|
bundle *i18n.Bundle
|
|
|
|
cookieName string
|
|
|
|
cookieHandler *http_util.CookieHandler
|
|
|
|
preferredLanguages []string
|
2023-12-05 12:12:01 +01:00
|
|
|
allowedLanguages []language.Tag
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type TranslatorConfig struct {
|
|
|
|
DefaultLanguage language.Tag
|
|
|
|
CookieName string
|
|
|
|
}
|
|
|
|
|
2021-07-05 15:10:49 +02:00
|
|
|
type Message struct {
|
|
|
|
ID string
|
|
|
|
Text string
|
|
|
|
}
|
|
|
|
|
2023-12-05 12:12:01 +01:00
|
|
|
// NewZitadelTranslator translates to all supported languages, as the ZITADEL texts are not customizable.
|
|
|
|
func NewZitadelTranslator(defaultLanguage language.Tag) (*Translator, error) {
|
|
|
|
return newTranslator(ZITADEL, defaultLanguage, SupportedLanguages(), "")
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
2023-12-05 12:12:01 +01:00
|
|
|
func NewNotificationTranslator(defaultLanguage language.Tag, allowedLanguages []language.Tag) (*Translator, error) {
|
|
|
|
return newTranslator(NOTIFICATION, defaultLanguage, allowedLanguages, "")
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
2023-12-05 12:12:01 +01:00
|
|
|
func NewLoginTranslator(defaultLanguage language.Tag, allowedLanguages []language.Tag, cookieName string) (*Translator, error) {
|
|
|
|
return newTranslator(LOGIN, defaultLanguage, allowedLanguages, cookieName)
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
2023-12-05 12:12:01 +01:00
|
|
|
func newTranslator(ns Namespace, defaultLanguage language.Tag, allowedLanguages []language.Tag, cookieName string) (*Translator, error) {
|
|
|
|
t := new(Translator)
|
|
|
|
var err error
|
|
|
|
t.allowedLanguages = allowedLanguages
|
|
|
|
if len(t.allowedLanguages) == 0 {
|
|
|
|
t.allowedLanguages = SupportedLanguages()
|
2021-07-09 09:45:31 +02:00
|
|
|
}
|
2024-04-16 14:08:18 +02:00
|
|
|
t.bundle, err = newBundle(ns, defaultLanguage, t.allowedLanguages)
|
2021-07-09 09:45:31 +02:00
|
|
|
if err != nil {
|
2023-12-05 12:12:01 +01:00
|
|
|
return nil, err
|
2021-07-09 09:45:31 +02:00
|
|
|
}
|
2023-12-05 12:12:01 +01:00
|
|
|
t.cookieHandler = http_util.NewCookieHandler()
|
|
|
|
t.cookieName = cookieName
|
|
|
|
return t, nil
|
2021-07-09 09:45:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Translator) SupportedLanguages() []language.Tag {
|
2023-12-05 12:12:01 +01:00
|
|
|
return t.allowedLanguages
|
2021-07-09 09:45:31 +02:00
|
|
|
}
|
|
|
|
|
2024-12-20 11:31:03 +01:00
|
|
|
// AddMessages adds messages to the translator for the given language tag.
|
|
|
|
// If the tag is not in the allowed languages, the messages are not added.
|
2021-07-05 15:10:49 +02:00
|
|
|
func (t *Translator) AddMessages(tag language.Tag, messages ...Message) error {
|
|
|
|
if len(messages) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2024-12-20 11:31:03 +01:00
|
|
|
var isAllowed bool
|
|
|
|
for _, allowed := range t.allowedLanguages {
|
|
|
|
if allowed == tag {
|
|
|
|
isAllowed = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !isAllowed {
|
|
|
|
return nil
|
|
|
|
}
|
2021-07-05 15:10:49 +02:00
|
|
|
i18nMessages := make([]*i18n.Message, len(messages))
|
|
|
|
for i, message := range messages {
|
|
|
|
i18nMessages[i] = &i18n.Message{
|
|
|
|
ID: message.ID,
|
|
|
|
Other: message.Text,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return t.bundle.AddMessages(tag, i18nMessages...)
|
|
|
|
}
|
|
|
|
|
2020-06-05 07:50:04 +02:00
|
|
|
func (t *Translator) LocalizeFromRequest(r *http.Request, id string, args map[string]interface{}) string {
|
2020-07-08 13:56:37 +02:00
|
|
|
return localize(t.localizerFromRequest(r), id, args)
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
2020-06-22 13:51:44 +02:00
|
|
|
func (t *Translator) LocalizeFromCtx(ctx context.Context, id string, args map[string]interface{}) string {
|
2020-07-08 13:56:37 +02:00
|
|
|
return localize(t.localizerFromCtx(ctx), id, args)
|
2020-06-22 13:51:44 +02:00
|
|
|
}
|
|
|
|
|
2020-06-09 15:11:42 +02:00
|
|
|
func (t *Translator) Localize(id string, args map[string]interface{}, langs ...string) string {
|
2020-07-08 13:56:37 +02:00
|
|
|
return localize(t.localizer(langs...), id, args)
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
2022-11-07 09:55:12 +01:00
|
|
|
func (t *Translator) LocalizeWithoutArgs(id string, langs ...string) string {
|
|
|
|
return localize(t.localizer(langs...), id, map[string]interface{}{})
|
|
|
|
}
|
|
|
|
|
2020-06-05 07:50:04 +02:00
|
|
|
func (t *Translator) Lang(r *http.Request) language.Tag {
|
2023-12-05 12:12:01 +01:00
|
|
|
matcher := language.NewMatcher(t.allowedLanguages)
|
2020-06-05 07:50:04 +02:00
|
|
|
tag, _ := language.MatchStrings(matcher, t.langsFromRequest(r)...)
|
|
|
|
return tag
|
|
|
|
}
|
|
|
|
|
2022-04-25 10:01:17 +02:00
|
|
|
func (t *Translator) SetLangCookie(w http.ResponseWriter, r *http.Request, lang language.Tag) {
|
|
|
|
t.cookieHandler.SetCookie(w, t.cookieName, r.Host, lang.String())
|
2020-06-05 07:50:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Translator) localizerFromRequest(r *http.Request) *i18n.Localizer {
|
|
|
|
return t.localizer(t.langsFromRequest(r)...)
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:51:44 +02:00
|
|
|
func (t *Translator) localizerFromCtx(ctx context.Context) *i18n.Localizer {
|
|
|
|
return t.localizer(t.langsFromCtx(ctx)...)
|
|
|
|
}
|
|
|
|
|
2020-06-05 07:50:04 +02:00
|
|
|
func (t *Translator) localizer(langs ...string) *i18n.Localizer {
|
|
|
|
return i18n.NewLocalizer(t.bundle, langs...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Translator) langsFromRequest(r *http.Request) []string {
|
2021-07-09 09:45:31 +02:00
|
|
|
langs := t.preferredLanguages
|
2020-06-05 07:50:04 +02:00
|
|
|
if r != nil {
|
|
|
|
lang, err := t.cookieHandler.GetCookieValue(r, t.cookieName)
|
|
|
|
if err == nil {
|
|
|
|
langs = append(langs, lang)
|
|
|
|
}
|
|
|
|
langs = append(langs, r.Header.Get("Accept-Language"))
|
|
|
|
}
|
|
|
|
return langs
|
|
|
|
}
|
2020-06-22 13:51:44 +02:00
|
|
|
|
|
|
|
func (t *Translator) langsFromCtx(ctx context.Context) []string {
|
2021-07-09 09:45:31 +02:00
|
|
|
langs := t.preferredLanguages
|
2020-06-22 13:51:44 +02:00
|
|
|
if ctx != nil {
|
2020-08-28 09:44:43 +02:00
|
|
|
ctxData := authz.GetCtxData(ctx)
|
2022-04-25 10:01:17 +02:00
|
|
|
if ctxData.PreferredLanguage != language.Und.String() {
|
2020-08-28 09:44:43 +02:00
|
|
|
langs = append(langs, authz.GetCtxData(ctx).PreferredLanguage)
|
|
|
|
}
|
2020-06-22 13:51:44 +02:00
|
|
|
langs = append(langs, getAcceptLanguageHeader(ctx))
|
|
|
|
}
|
|
|
|
return langs
|
|
|
|
}
|
|
|
|
|
2021-07-09 09:45:31 +02:00
|
|
|
func (t *Translator) SetPreferredLanguages(langs ...string) {
|
|
|
|
t.preferredLanguages = langs
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:51:44 +02:00
|
|
|
func getAcceptLanguageHeader(ctx context.Context) string {
|
2022-04-25 10:01:17 +02:00
|
|
|
acceptLanguage := metautils.ExtractIncoming(ctx).Get("accept-language")
|
|
|
|
if acceptLanguage != "" {
|
|
|
|
return acceptLanguage
|
|
|
|
}
|
2020-06-22 13:51:44 +02:00
|
|
|
return metautils.ExtractIncoming(ctx).Get("grpcgateway-accept-language")
|
|
|
|
}
|
2020-07-08 13:56:37 +02:00
|
|
|
|
|
|
|
func localize(localizer *i18n.Localizer, id string, args map[string]interface{}) string {
|
|
|
|
s, err := localizer.Localize(&i18n.LocalizeConfig{
|
|
|
|
MessageID: id,
|
|
|
|
TemplateData: args,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2022-04-25 10:01:17 +02:00
|
|
|
logging.WithFields("id", id, "args", args).WithError(err).Warnf("missing translation")
|
2020-07-08 13:56:37 +02:00
|
|
|
return id
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|