Phrases are now stored in JSON files.

We now use a sync.Map for language packs for better concurrency.
This commit is contained in:
Azareal 2017-09-11 01:56:09 +01:00
parent b289b52210
commit ab0776c52b
4 changed files with 121 additions and 61 deletions

View File

@ -52,6 +52,11 @@ func gloinit() error {
return err return err
} }
err = initPhrases()
if err != nil {
log.Fatal(err)
}
if config.CacheTopicUser == CACHE_STATIC { if config.CacheTopicUser == CACHE_STATIC {
users = NewMemoryUserStore(config.UserCacheCapacity) users = NewMemoryUserStore(config.UserCacheCapacity)
topics = NewMemoryTopicStore(config.TopicCacheCapacity) topics = NewMemoryTopicStore(config.TopicCacheCapacity)

43
langs/english.json Normal file
View File

@ -0,0 +1,43 @@
{
"Name": "english",
"Levels": {
"Level": "Level {0}",
"LevelMax": ""
},
"GlobalPerms": {
"BanUsers": "Can ban users",
"ActivateUsers": "Can activate users",
"EditUser": "Can edit users",
"EditUserEmail": "Can change a user's email",
"EditUserPassword": "Can change a user's password",
"EditUserGroup": "Can change a user's group",
"EditUserGroupSuperMod": "Can edit super-mods",
"EditUserGroupAdmin": "Can edit admins",
"EditGroup": "Can edit groups",
"EditGroupLocalPerms": "Can edit a group's minor perms",
"EditGroupGlobalPerms": "Can edit a group's global perms",
"EditGroupSuperMod": "Can edit super-mod groups",
"EditGroupAdmin": "Can edit admin groups",
"ManageForums": "Can manage forums",
"EditSettings": "Can edit settings",
"ManageThemes": "Can manage themes",
"ManagePlugins": "Can manage plugins",
"ViewAdminLogs": "Can view the administrator action logs",
"ViewIPs": "Can view IP addresses"
},
"LocalPerms": {
"ViewTopic": "Can view topics",
"LikeItem": "Can like items",
"CreateTopic": "Can create topics",
"EditTopic": "Can edit topics",
"DeleteTopic": "Can delete topics",
"CreateReply": "Can create replies",
"EditReply": "Can edit replies",
"DeleteReply": "Can delete replies",
"PinTopic": "Can pin topics",
"CloseTopic": "Can lock topics"
},
"SettingLabels": {
"activation_type": "Activate All,Email Activation,Admin Approval"
}
}

View File

@ -31,6 +31,8 @@ var enableWebsockets = false // Don't change this, the value is overwritten by a
var router *GenRouter var router *GenRouter
var startTime time.Time var startTime time.Time
// ? - Make this more customisable?
var externalSites = map[string]string{ var externalSites = map[string]string{
"YT": "https://www.youtube.com/", "YT": "https://www.youtube.com/",
} }
@ -133,6 +135,11 @@ func main() {
log.Fatal(err) log.Fatal(err)
} }
err = initPhrases()
if err != nil {
log.Fatal(err)
}
if config.CacheTopicUser == CACHE_STATIC { if config.CacheTopicUser == CACHE_STATIC {
users = NewMemoryUserStore(config.UserCacheCapacity) users = NewMemoryUserStore(config.UserCacheCapacity)
topics = NewMemoryTopicStore(config.TopicCacheCapacity) topics = NewMemoryTopicStore(config.TopicCacheCapacity)

View File

@ -7,88 +7,96 @@
package main package main
import ( import (
"encoding/json"
"errors"
"io/ioutil"
"log"
"os"
"path/filepath"
"sync" "sync"
"sync/atomic" "sync/atomic"
) )
// TODO: Let the admin edit phrases from inside the Control Panel? How should we persist these? Should we create a copy of the langpack or edit the primaries? Use the changeLangpack mutex for this? // TODO: Let the admin edit phrases from inside the Control Panel? How should we persist these? Should we create a copy of the langpack or edit the primaries? Use the changeLangpack mutex for this?
// nolint Be quiet megacheck, this *is* used // nolint Be quiet megacheck, this *is* used
var changeLangpackMutex sync.Mutex
var currentLanguage = "english" var currentLanguage = "english"
var currentLangPack atomic.Value var currentLangPack atomic.Value
var langpackCount int // TODO: Use atomics for this
// We'll be implementing the level phrases in the software proper very very soon!
type LevelPhrases struct { type LevelPhrases struct {
Level string Level string
LevelMax string LevelMax string // ? Add a max level setting?
// Override the phrase for individual levels, if the phrases exist // Override the phrase for individual levels, if the phrases exist
Levels []string // index = level Levels []string // index = level
} }
type LanguagePack struct { type LanguagePack struct {
Name string Name string
Phrases map[string]string // Should we use a sync map or a struct for these? It would be nice, if we could keep all the phrases consistent. Phrases map[string]string // Should we use a sync map or a struct for these? It would be nice, if we could keep all the phrases consistent.
LevelPhrases LevelPhrases Levels LevelPhrases
GlobalPermPhrases map[string]string GlobalPerms map[string]string
LocalPermPhrases map[string]string LocalPerms map[string]string
SettingLabels map[string]string SettingLabels map[string]string
} }
// TODO: Add the ability to edit language JSON files from the Control Panel and automatically scan the files for changes // TODO: Add the ability to edit language JSON files from the Control Panel and automatically scan the files for changes
// TODO: Move the english language pack into a JSON file and load that on start-up // TODO: Move the english language pack into a JSON file and load that on start-up
var langpacks = map[string]*LanguagePack{ ////var langpacks = map[string]*LanguagePack
"english": &LanguagePack{ var langpacks sync.Map // nolint it is used
Name: "english",
// We'll be implementing the level phrases in the software proper very very soon! func initPhrases() error {
LevelPhrases: LevelPhrases{ log.Print("Loading the language packs")
Level: "Level {0}", err := filepath.Walk("./langs", func(path string, f os.FileInfo, err error) error {
LevelMax: "", // Add a max level setting? if f.IsDir() {
}, return nil
}
GlobalPermPhrases: map[string]string{ data, err := ioutil.ReadFile(path)
"BanUsers": "Can ban users", if err != nil {
"ActivateUsers": "Can activate users", return err
"EditUser": "Can edit users", }
"EditUserEmail": "Can change a user's email",
"EditUserPassword": "Can change a user's password",
"EditUserGroup": "Can change a user's group",
"EditUserGroupSuperMod": "Can edit super-mods",
"EditUserGroupAdmin": "Can edit admins",
"EditGroup": "Can edit groups",
"EditGroupLocalPerms": "Can edit a group's minor perms",
"EditGroupGlobalPerms": "Can edit a group's global perms",
"EditGroupSuperMod": "Can edit super-mod groups",
"EditGroupAdmin": "Can edit admin groups",
"ManageForums": "Can manage forums",
"EditSettings": "Can edit settings",
"ManageThemes": "Can manage themes",
"ManagePlugins": "Can manage plugins",
"ViewAdminLogs": "Can view the administrator action logs",
"ViewIPs": "Can view IP addresses",
},
LocalPermPhrases: map[string]string{ var ext = filepath.Ext("/langs/" + path)
"ViewTopic": "Can view topics", if ext != ".json" {
"LikeItem": "Can like items", if dev.DebugMode {
"CreateTopic": "Can create topics", log.Print("Found a " + ext + "in /langs/")
"EditTopic": "Can edit topics", }
"DeleteTopic": "Can delete topics", return nil
"CreateReply": "Can create replies", }
"EditReply": "Can edit replies",
"DeleteReply": "Can delete replies",
"PinTopic": "Can pin topics",
"CloseTopic": "Can lock topics",
},
SettingLabels: map[string]string{ var langPack LanguagePack
"activation_type": "Activate All,Email Activation,Admin Approval", err = json.Unmarshal(data, &langPack)
}, if err != nil {
}, return err
}
log.Print("Adding the '" + langPack.Name + "' language pack")
langpacks.Store(langPack.Name, &langPack)
langpackCount++
return nil
})
if err != nil {
return err
}
if langpackCount == 0 {
return errors.New("You don't have any language packs")
}
langPack, ok := langpacks.Load(currentLanguage)
if !ok {
return errors.New("Couldn't find the " + currentLanguage + " language pack")
}
currentLangPack.Store(langPack)
return nil
} }
func init() { func LoadLangPack(name string) error {
currentLangPack.Store(langpacks[currentLanguage]) _ = name
return nil
} }
// We might not need to use a mutex for this, we shouldn't need to change the phrases after start-up, and when we do we could overwrite the entire map // We might not need to use a mutex for this, we shouldn't need to change the phrases after start-up, and when we do we could overwrite the entire map
@ -98,7 +106,7 @@ func GetPhrase(name string) (string, bool) {
} }
func GetGlobalPermPhrase(name string) string { func GetGlobalPermPhrase(name string) string {
res, ok := currentLangPack.Load().(*LanguagePack).GlobalPermPhrases[name] res, ok := currentLangPack.Load().(*LanguagePack).GlobalPerms[name]
if !ok { if !ok {
return "{name}" return "{name}"
} }
@ -106,7 +114,7 @@ func GetGlobalPermPhrase(name string) string {
} }
func GetLocalPermPhrase(name string) string { func GetLocalPermPhrase(name string) string {
res, ok := currentLangPack.Load().(*LanguagePack).LocalPermPhrases[name] res, ok := currentLangPack.Load().(*LanguagePack).LocalPerms[name]
if !ok { if !ok {
return "{name}" return "{name}"
} }
@ -140,13 +148,10 @@ func DeletePhrase() {
// TODO: Use atomics to store the pointer of the current active langpack? // TODO: Use atomics to store the pointer of the current active langpack?
// nolint // nolint
func ChangeLanguagePack(name string) (exists bool) { func ChangeLanguagePack(name string) (exists bool) {
changeLangpackMutex.Lock() pack, ok := langpacks.Load(name)
pack, ok := langpacks[name]
if !ok { if !ok {
changeLangpackMutex.Unlock()
return false return false
} }
currentLangPack.Store(pack) currentLangPack.Store(pack)
changeLangpackMutex.Unlock()
return true return true
} }