Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
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.
242 lines
6.7 KiB
242 lines
6.7 KiB
// Copyright 2014 The Macaron Authors |
|
// |
|
// Licensed under the Apache License, Version 2.0 (the "License"): you may |
|
// not use this file except in compliance with the License. You may obtain |
|
// a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
|
// License for the specific language governing permissions and limitations |
|
// under the License. |
|
|
|
// Package i18n provides an Internationalization and Localization middleware for Macaron applications. |
|
package i18n |
|
|
|
import ( |
|
"fmt" |
|
"log" |
|
"os" |
|
"path" |
|
"strings" |
|
|
|
"gitea.com/macaron/macaron" |
|
"github.com/unknwon/i18n" |
|
"golang.org/x/text/language" |
|
) |
|
|
|
const _VERSION = "0.4.0" |
|
|
|
func Version() string { |
|
return _VERSION |
|
} |
|
|
|
// isFile returns true if given path is a file, |
|
// or returns false when it's a directory or does not exist. |
|
func isFile(filePath string) bool { |
|
f, e := os.Stat(filePath) |
|
if e != nil { |
|
return false |
|
} |
|
return !f.IsDir() |
|
} |
|
|
|
// initLocales initializes language type list and Accept-Language header matcher. |
|
func initLocales(opt Options) language.Matcher { |
|
tags := make([]language.Tag, len(opt.Langs)) |
|
for i, lang := range opt.Langs { |
|
tags[i] = language.Raw.Make(lang) |
|
fname := fmt.Sprintf(opt.Format, lang) |
|
// Append custom locale file. |
|
custom := []interface{}{} |
|
customPath := path.Join(opt.CustomDirectory, fname) |
|
if isFile(customPath) { |
|
custom = append(custom, customPath) |
|
} |
|
|
|
var locale interface{} |
|
if data, ok := opt.Files[fname]; ok { |
|
locale = data |
|
} else { |
|
locale = path.Join(opt.Directory, fname) |
|
} |
|
|
|
err := i18n.SetMessageWithDesc(lang, opt.Names[i], locale, custom...) |
|
if err != nil && err != i18n.ErrLangAlreadyExist { |
|
log.Printf("ERROR: failed to set message file(%s) for language %s: %v", lang, opt.Names[i], err) |
|
} |
|
} |
|
return language.NewMatcher(tags) |
|
} |
|
|
|
// A Locale describles the information of localization. |
|
type Locale struct { |
|
i18n.Locale |
|
} |
|
|
|
// Language returns language current locale represents. |
|
func (l Locale) Language() string { |
|
return l.Lang |
|
} |
|
|
|
// Options represents a struct for specifying configuration options for the i18n middleware. |
|
type Options struct { |
|
// Suburl of path. Default is empty. |
|
SubURL string |
|
// Directory to load locale files. Default is "conf/locale" |
|
Directory string |
|
// File stores actual data of locale files. Used for in-memory purpose. |
|
Files map[string][]byte |
|
// Custom directory to overload locale files. Default is "custom/conf/locale" |
|
CustomDirectory string |
|
// Langauges that will be supported, order is meaningful. |
|
Langs []string |
|
// Human friendly names corresponding to Langs list. |
|
Names []string |
|
// Default language locale, leave empty to remain unset. |
|
DefaultLang string |
|
// Locale file naming style. Default is "locale_%s.ini". |
|
Format string |
|
// Name of language parameter name in URL. Default is "lang". |
|
Parameter string |
|
// Redirect when user uses get parameter to specify language. |
|
Redirect bool |
|
// Name that maps into template variable. Default is "i18n". |
|
TmplName string |
|
// Configuration section name. Default is "i18n". |
|
Section string |
|
// Domain used for `lang` cookie. Default is "" |
|
CookieDomain string |
|
// Set the Secure flag on the `lang` cookie. Default is disabled. |
|
Secure bool |
|
// Set the HTTP Only flag on the `lang` cookie. Default is disabled. |
|
CookieHttpOnly bool |
|
} |
|
|
|
func prepareOptions(options []Options) Options { |
|
var opt Options |
|
if len(options) > 0 { |
|
opt = options[0] |
|
} |
|
|
|
if len(opt.Section) == 0 { |
|
opt.Section = "i18n" |
|
} |
|
sec := macaron.Config().Section(opt.Section) |
|
|
|
opt.SubURL = strings.TrimSuffix(opt.SubURL, "/") |
|
|
|
if len(opt.Langs) == 0 { |
|
opt.Langs = sec.Key("LANGS").Strings(",") |
|
} |
|
if len(opt.Names) == 0 { |
|
opt.Names = sec.Key("NAMES").Strings(",") |
|
} |
|
if len(opt.Langs) == 0 { |
|
panic("no language is specified") |
|
} else if len(opt.Langs) != len(opt.Names) { |
|
panic("length of langs is not same as length of names") |
|
} |
|
i18n.SetDefaultLang(opt.DefaultLang) |
|
|
|
if len(opt.Directory) == 0 { |
|
opt.Directory = sec.Key("DIRECTORY").MustString("conf/locale") |
|
} |
|
if len(opt.CustomDirectory) == 0 { |
|
opt.CustomDirectory = sec.Key("CUSTOM_DIRECTORY").MustString("custom/conf/locale") |
|
} |
|
if len(opt.Format) == 0 { |
|
opt.Format = sec.Key("FORMAT").MustString("locale_%s.ini") |
|
} |
|
if len(opt.Parameter) == 0 { |
|
opt.Parameter = sec.Key("PARAMETER").MustString("lang") |
|
} |
|
if !opt.Redirect { |
|
opt.Redirect = sec.Key("REDIRECT").MustBool() |
|
} |
|
if len(opt.TmplName) == 0 { |
|
opt.TmplName = sec.Key("TMPL_NAME").MustString("i18n") |
|
} |
|
|
|
return opt |
|
} |
|
|
|
type LangType struct { |
|
Lang, Name string |
|
} |
|
|
|
// I18n is a middleware provides localization layer for your application. |
|
// Paramenter langs must be in the form of "en-US", "zh-CN", etc. |
|
// Otherwise it may not recognize browser input. |
|
func I18n(options ...Options) macaron.Handler { |
|
opt := prepareOptions(options) |
|
m := initLocales(opt) |
|
return func(ctx *macaron.Context) { |
|
isNeedRedir := false |
|
hasCookie := false |
|
|
|
// 1. Check URL arguments. |
|
lang := ctx.Query(opt.Parameter) |
|
|
|
// 2. Get language information from cookies. |
|
if len(lang) == 0 { |
|
lang = ctx.GetCookie("lang") |
|
hasCookie = true |
|
} else { |
|
isNeedRedir = true |
|
} |
|
|
|
// Check again in case someone modify by purpose. |
|
if !i18n.IsExist(lang) { |
|
lang = "" |
|
isNeedRedir = false |
|
hasCookie = false |
|
} |
|
|
|
// 3. Get language information from 'Accept-Language'. |
|
// The first element in the list is chosen to be the default language automatically. |
|
if len(lang) == 0 { |
|
tags, _, _ := language.ParseAcceptLanguage(ctx.Req.Header.Get("Accept-Language")) |
|
tag, _, _ := m.Match(tags...) |
|
lang = tag.String() |
|
isNeedRedir = false |
|
} |
|
|
|
curLang := LangType{ |
|
Lang: lang, |
|
} |
|
|
|
// Save language information in cookies. |
|
if !hasCookie { |
|
ctx.SetCookie("lang", curLang.Lang, 1<<31-1, "/"+strings.TrimPrefix(opt.SubURL, "/"), opt.CookieDomain, opt.Secure, opt.CookieHttpOnly) |
|
} |
|
|
|
restLangs := make([]LangType, 0, i18n.Count()-1) |
|
langs := i18n.ListLangs() |
|
names := i18n.ListLangDescs() |
|
for i, v := range langs { |
|
if lang != v { |
|
restLangs = append(restLangs, LangType{v, names[i]}) |
|
} else { |
|
curLang.Name = names[i] |
|
} |
|
} |
|
|
|
// Set language properties. |
|
locale := Locale{Locale: i18n.Locale{Lang: lang}} |
|
ctx.Map(locale) |
|
ctx.Locale = locale |
|
ctx.Data[opt.TmplName] = locale |
|
ctx.Data["Tr"] = i18n.Tr |
|
ctx.Data["Lang"] = locale.Lang |
|
ctx.Data["LangName"] = curLang.Name |
|
ctx.Data["AllLangs"] = append([]LangType{curLang}, restLangs...) |
|
ctx.Data["RestLangs"] = restLangs |
|
|
|
if opt.Redirect && isNeedRedir { |
|
ctx.Redirect(opt.SubURL + path.Clean(ctx.Req.RequestURI[:strings.Index(ctx.Req.RequestURI, "?")])) |
|
} |
|
} |
|
}
|
|
|