zitadel/internal/domain/language.go
Tim Möhlmann f680dd934d
refactor: rename package errors to zerrors (#7039)
* chore: rename package errors to zerrors

* rename package errors to gerrors

* fix error related linting issues

* fix zitadel error assertion

* fix gosimple linting issues

* fix deprecated linting issues

* resolve gci linting issues

* fix import structure

---------

Co-authored-by: Elio Bischof <elio@zitadel.com>
2023-12-08 15:30:55 +01:00

131 lines
3.4 KiB
Go

package domain
import (
"errors"
"golang.org/x/text/language"
"github.com/zitadel/zitadel/internal/zerrors"
)
func StringsToLanguages(langs []string) []language.Tag {
return GenericMapSlice(langs, language.Make)
}
func LanguagesToStrings(langs []language.Tag) []string {
return GenericMapSlice(langs, func(lang language.Tag) string { return lang.String() })
}
func GenericMapSlice[T any, U any](from []T, mapTo func(T) U) []U {
if from == nil {
return nil
}
result := make([]U, len(from))
for i, lang := range from {
result[i] = mapTo(lang)
}
return result
}
// LanguagesDiffer returns true if the languages differ.
func LanguagesDiffer(left, right []language.Tag) bool {
if left == nil && right == nil {
return false
}
if left == nil || right == nil || len(left) != len(right) {
return true
}
return !languagesAreContained(left, right)
}
func LanguageIsAllowed(allowUndefined bool, allowedLanguages []language.Tag, lang language.Tag) error {
err := LanguageIsDefined(lang)
if err != nil && allowUndefined {
return nil
}
if err != nil {
return err
}
if len(allowedLanguages) > 0 && !languageIsContained(allowedLanguages, lang) {
return zerrors.ThrowPreconditionFailed(nil, "LANG-2M9fs", "Errors.Language.NotAllowed")
}
return nil
}
func LanguagesAreSupported(supportedLanguages []language.Tag, lang ...language.Tag) error {
unsupported := make([]language.Tag, 0)
for _, l := range lang {
if l.IsRoot() {
continue
}
if !languageIsContained(supportedLanguages, l) {
unsupported = append(unsupported, l)
}
}
if len(unsupported) == 0 {
return nil
}
if len(unsupported) == 1 {
return zerrors.ThrowInvalidArgument(nil, "LANG-lg4DP", "Errors.Language.NotSupported")
}
return zerrors.ThrowInvalidArgumentf(nil, "LANG-XHiK5", "Errors.Languages.NotSupported: %s", LanguagesToStrings(unsupported))
}
func LanguageIsDefined(lang language.Tag) error {
if lang.IsRoot() {
return zerrors.ThrowInvalidArgument(nil, "LANG-3M9f2", "Errors.Language.Undefined")
}
return nil
}
// LanguagesHaveDuplicates returns an error if the passed slices contains duplicates.
// The error lists the duplicates.
func LanguagesHaveDuplicates(langs []language.Tag) error {
unique := make(map[language.Tag]struct{})
duplicates := make([]language.Tag, 0)
for _, lang := range langs {
if _, ok := unique[lang]; ok {
duplicates = append(duplicates, lang)
}
unique[lang] = struct{}{}
}
if len(duplicates) == 0 {
return nil
}
if len(duplicates) > 1 {
return zerrors.ThrowInvalidArgument(nil, "LANG-3M9f2", "Errors.Language.Duplicate")
}
return zerrors.ThrowInvalidArgumentf(nil, "LANG-XHiK5", "Errors.Languages.Duplicate: %s", LanguagesToStrings(duplicates))
}
func ParseLanguage(lang ...string) (tags []language.Tag, err error) {
tags = make([]language.Tag, len(lang))
for i := range lang {
var parseErr error
tags[i], parseErr = language.Parse(lang[i])
err = errors.Join(err, parseErr)
}
if err != nil {
err = zerrors.ThrowInvalidArgument(err, "LANG-jc8Sq", "Errors.Language.NotParsed")
}
return tags, err
}
func languagesAreContained(languages, search []language.Tag) bool {
for _, s := range search {
if !languageIsContained(languages, s) {
return false
}
}
return true
}
func languageIsContained(languages []language.Tag, search language.Tag) bool {
for _, lang := range languages {
if lang == search {
return true
}
}
return false
}