go-i18n/grammar_test.go
Virgil 92b2f2ce94 feat(i18n): localize failure prefix
Co-Authored-By: Virgil <virgil@lethean.io>
2026-04-02 08:23:19 +00:00

1336 lines
30 KiB
Go

package i18n
import (
"strings"
"testing"
"text/template"
"time"
)
type regionFallbackLoader struct{}
func (regionFallbackLoader) Languages() []string {
return []string{"en-GB"}
}
func (regionFallbackLoader) Load(lang string) (map[string]Message, *GrammarData, error) {
return map[string]Message{}, nil, nil
}
func TestPastTense(t *testing.T) {
// Ensure grammar data is loaded from embedded JSON
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
verb string
want string
}{
// Irregular verbs (from JSON)
{"be", "was"},
{"go", "went"},
{"run", "ran"},
{"write", "wrote"},
{"build", "built"},
{"find", "found"},
{"set", "set"},
{"put", "put"},
{"cut", "cut"},
{"commit", "committed"},
// Irregular verbs (from Go map only)
{"break", "broke"},
{"speak", "spoke"},
{"steal", "stole"},
{"freeze", "froze"},
// Compound irregular verbs
{"undo", "undid"},
{"redo", "redid"},
{"rerun", "reran"},
{"rewrite", "rewrote"},
{"rebuild", "rebuilt"},
{"resend", "resent"},
{"override", "overrode"},
{"rethink", "rethought"},
{"remake", "remade"},
{"undergo", "underwent"},
{"overcome", "overcame"},
{"withdraw", "withdrew"},
{"uphold", "upheld"},
{"withhold", "withheld"},
{"outgrow", "outgrew"},
{"outrun", "outran"},
{"overshoot", "overshot"},
// Simple irregular verbs (dev/ops)
{"become", "became"},
{"come", "came"},
{"give", "gave"},
{"fall", "fell"},
{"understand", "understood"},
{"arise", "arose"},
{"bind", "bound"},
{"spin", "spun"},
{"quit", "quit"},
{"cast", "cast"},
{"broadcast", "broadcast"},
{"burst", "burst"},
{"cost", "cost"},
{"shed", "shed"},
{"rid", "rid"},
{"shrink", "shrank"},
{"shoot", "shot"},
{"forbid", "forbade"},
{"offset", "offset"},
{"upset", "upset"},
{"input", "input"},
{"output", "output"},
// CVC doubling failures (stressed final syllable)
{"debug", "debugged"},
{"embed", "embedded"},
{"unzip", "unzipped"},
{"remap", "remapped"},
{"unpin", "unpinned"},
{"unwrap", "unwrapped"},
// Regular verbs
{"delete", "deleted"},
{"update", "updated"},
{"push", "pushed"},
{"pull", "pulled"},
{"start", "started"},
{"copy", "copied"},
{"apply", "applied"},
// Edge cases
{"", ""},
{" delete ", "deleted"},
{"DELETE", "deleted"},
}
for _, tt := range tests {
t.Run(tt.verb, func(t *testing.T) {
got := PastTense(tt.verb)
if got != tt.want {
t.Errorf("PastTense(%q) = %q, want %q", tt.verb, got, tt.want)
}
})
}
}
func TestGerund(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
verb string
want string
}{
// Irregular verbs (from JSON)
{"be", "being"},
{"go", "going"},
{"run", "running"},
{"build", "building"},
{"write", "writing"},
{"commit", "committing"},
// Compound irregular verbs
{"undo", "undoing"},
{"rerun", "rerunning"},
{"override", "overriding"},
{"rebuild", "rebuilding"},
// Simple irregular (dev/ops)
{"become", "becoming"},
{"give", "giving"},
{"bind", "binding"},
{"spin", "spinning"},
{"quit", "quitting"},
{"cast", "casting"},
{"broadcast", "broadcasting"},
// CVC doubling failures
{"debug", "debugging"},
{"embed", "embedding"},
{"unzip", "unzipping"},
{"remap", "remapping"},
{"unpin", "unpinning"},
{"unwrap", "unwrapping"},
// Regular verbs
{"delete", "deleting"},
{"push", "pushing"},
{"pull", "pulling"},
{"start", "starting"},
{"die", "dying"},
// Edge cases
{"", ""},
}
for _, tt := range tests {
t.Run(tt.verb, func(t *testing.T) {
got := Gerund(tt.verb)
if got != tt.want {
t.Errorf("Gerund(%q) = %q, want %q", tt.verb, got, tt.want)
}
})
}
}
func TestPluralize(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
noun string
count int
want string
}{
// Singular (count=1 always returns original)
{"file", 1, "file"},
{"child", 1, "child"},
// From JSON grammar data
{"file", 5, "files"},
{"repo", 3, "repos"},
{"branch", 2, "branches"},
{"repository", 2, "repositories"},
{"vulnerability", 2, "vulnerabilities"},
{"person", 2, "people"},
{"child", 3, "children"},
// From irregular nouns map
{"mouse", 2, "mice"},
{"sheep", 5, "sheep"},
{"knife", 3, "knives"},
// Regular plurals
{"server", 2, "servers"},
{"box", 2, "boxes"},
// Count 0
{"file", 0, "files"},
}
for _, tt := range tests {
t.Run(tt.noun, func(t *testing.T) {
got := Pluralize(tt.noun, tt.count)
if got != tt.want {
t.Errorf("Pluralize(%q, %d) = %q, want %q", tt.noun, tt.count, got, tt.want)
}
})
}
}
func TestPluralize_UsesLocaleSingularOverride(t *testing.T) {
const lang = "en-x-singular"
prev := Default()
t.Cleanup(func() {
SetDefault(prev)
SetGrammarData(lang, nil)
})
svc, err := NewWithLoader(pluralizeOverrideLoader{})
if err != nil {
t.Fatalf("NewWithLoader() failed: %v", err)
}
SetDefault(svc)
if err := SetLanguage(lang); err != nil {
t.Fatalf("SetLanguage(%s) failed: %v", lang, err)
}
if got, want := Pluralize("person", 1), "human"; got != want {
t.Fatalf("Pluralize(%q, 1) = %q, want %q", "person", got, want)
}
if got, want := Pluralize("Person", 1), "Human"; got != want {
t.Fatalf("Pluralize(%q, 1) = %q, want %q", "Person", got, want)
}
if got, want := Pluralize("person", 2), "people"; got != want {
t.Fatalf("Pluralize(%q, 2) = %q, want %q", "person", got, want)
}
}
func TestPluralize_PreservesUnicodeCapitalization(t *testing.T) {
prev := Default()
t.Cleanup(func() {
SetDefault(prev)
})
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
if got, want := Pluralize("Élément", 1), "Élément"; got != want {
t.Fatalf("Pluralize(%q, 1) = %q, want %q", "Élément", got, want)
}
if got, want := Pluralize("Élément", 2), "Éléments"; got != want {
t.Fatalf("Pluralize(%q, 2) = %q, want %q", "Élément", got, want)
}
if got, want := PluralForm("Élément"), "Éléments"; got != want {
t.Fatalf("PluralForm(%q) = %q, want %q", "Élément", got, want)
}
}
func TestPluralForm(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
noun string
want string
}{
{"", ""},
// Capitalisation preserved
{"File", "Files"},
{"Child", "Children"},
// Regular rules
{"bus", "buses"},
{"fox", "foxes"},
{"city", "cities"},
{"key", "keys"},
}
for _, tt := range tests {
t.Run(tt.noun, func(t *testing.T) {
got := PluralForm(tt.noun)
if got != tt.want {
t.Errorf("PluralForm(%q) = %q, want %q", tt.noun, got, tt.want)
}
})
}
}
func TestArticle(t *testing.T) {
tests := []struct {
word string
want string
}{
{"file", "a"},
{"error", "an"},
{"apple", "an"},
{"user", "a"}, // Consonant sound: "yoo-zer"
{"hour", "an"}, // Vowel sound: silent h
{"honest", "an"}, // Vowel sound
{"university", "a"}, // Consonant sound
{"one", "a"}, // Consonant sound
{"SSH", "an"}, // Initialism: "ess-ess-aitch"
{"URL", "a"}, // Initialism: "you-are-ell"
{"", ""},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := Article(tt.word)
if got != tt.want {
t.Errorf("Article(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestArticleFrenchLocale(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
tests := []struct {
word string
want string
}{
{"branche", "la"},
{"branches", "les"},
{"amis", "des"},
{"enfant", "l'"},
{"fichier", "le"},
{"inconnu", "un"},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := Article(tt.word)
if got != tt.want {
t.Errorf("Article(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestArticleFrenchElisionKeepsLeadingConsonant(t *testing.T) {
prevData := GetGrammarData("fr")
t.Cleanup(func() {
SetGrammarData("fr", prevData)
})
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
SetGrammarData("fr", &GrammarData{
Nouns: map[string]NounForms{
"amie": {One: "amie", Other: "amies", Gender: "f"},
"accord": {One: "accord", Other: "accords", Gender: "d"},
"homme": {One: "homme", Other: "hommes", Gender: "m"},
"héros": {One: "héros", Other: "héros", Gender: "m"},
"idole": {One: "idole", Other: "idoles", Gender: "j"},
},
Articles: ArticleForms{
IndefiniteDefault: "un",
IndefiniteVowel: "un",
Definite: "le",
ByGender: map[string]string{
"d": "de",
"f": "la",
"j": "je",
"m": "le",
},
},
})
tests := []struct {
word string
want string
}{
{"homme", "l'"},
{"héros", "le"},
{"amie", "l'"},
{"accord", "d'"},
{"idole", "j'"},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := Article(tt.word)
if got != tt.want {
t.Errorf("Article(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
phraseTests := []struct {
word string
want string
}{
{"accord", "d'accord"},
{"idole", "j'idole"},
}
for _, tt := range phraseTests {
t.Run(tt.word+"_phrase", func(t *testing.T) {
got := ArticlePhrase(tt.word)
if got != tt.want {
t.Errorf("ArticlePhrase(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
type pluralizeOverrideLoader struct{}
func (pluralizeOverrideLoader) Languages() []string {
return []string{"en-x-singular"}
}
func (pluralizeOverrideLoader) Load(lang string) (map[string]Message, *GrammarData, error) {
grammar := &GrammarData{
Nouns: map[string]NounForms{
"person": {One: "human", Other: "people"},
},
}
SetGrammarData(lang, grammar)
return map[string]Message{}, grammar, nil
}
func TestTitle(t *testing.T) {
tests := []struct {
input string
want string
}{
{"hello world", "Hello World"},
{"hello", "Hello"},
{"", ""},
{"HELLO", "HELLO"},
{"hello-world", "Hello-World"},
{"config.yaml", "Config.yaml"},
}
for _, tt := range tests {
t.Run(tt.input, func(t *testing.T) {
got := Title(tt.input)
if got != tt.want {
t.Errorf("Title(%q) = %q, want %q", tt.input, got, tt.want)
}
})
}
}
func TestQuote(t *testing.T) {
if got := Quote("hello"); got != `"hello"` {
t.Errorf("Quote(%q) = %q, want %q", "hello", got, `"hello"`)
}
if got := Quote(`a "quoted" path\name`); got != `"a \"quoted\" path\\name"` {
t.Errorf("Quote(%q) = %q, want %q", `a "quoted" path\name`, got, `"a \"quoted\" path\\name"`)
}
}
func TestCaseHelpers(t *testing.T) {
if got := Lower("HELLO"); got != "hello" {
t.Fatalf("Lower(%q) = %q, want %q", "HELLO", got, "hello")
}
if got := Upper("hello"); got != "HELLO" {
t.Fatalf("Upper(%q) = %q, want %q", "hello", got, "HELLO")
}
}
func TestArticlePhrase(t *testing.T) {
tests := []struct {
word string
want string
}{
{"file", "a file"},
{"error", "an error"},
{"", ""},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := ArticlePhrase(tt.word)
if got != tt.want {
t.Errorf("ArticlePhrase(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestArticlePhrase_RespectsWordMap(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
data := GetGrammarData("en")
if data == nil {
t.Fatal("GetGrammarData(\"en\") returned nil")
}
original, existed := data.Words["go_mod"]
data.Words["go_mod"] = "go.mod"
t.Cleanup(func() {
if existed {
data.Words["go_mod"] = original
return
}
delete(data.Words, "go_mod")
})
if got, want := ArticlePhrase("go_mod"), "a go.mod"; got != want {
t.Fatalf("ArticlePhrase(%q) = %q, want %q", "go_mod", got, want)
}
}
func TestArticlePhrase_UsesRenderedWordForArticleSelection(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
data := GetGrammarData("en")
if data == nil {
t.Fatal("GetGrammarData(\"en\") returned nil")
}
original, existed := data.Words["ssh"]
data.Words["ssh"] = "SSH"
t.Cleanup(func() {
if existed {
data.Words["ssh"] = original
return
}
delete(data.Words, "ssh")
})
if got, want := ArticlePhrase("ssh"), "an SSH"; got != want {
t.Fatalf("ArticlePhrase(%q) = %q, want %q", "ssh", got, want)
}
}
func TestArticlePhraseFrenchLocale(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
tests := []struct {
word string
want string
}{
{"branche", "la branche"},
{"branches", "les branches"},
{"amis", "des amis"},
{"enfant", "l'enfant"},
{"fichier", "le fichier"},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := ArticlePhrase(tt.word)
if got != tt.want {
t.Errorf("ArticlePhrase(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestDefiniteArticle(t *testing.T) {
tests := []struct {
word string
want string
}{
{"file", "the"},
{"error", "the"},
{"", ""},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := DefiniteArticle(tt.word)
if got != tt.want {
t.Errorf("DefiniteArticle(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestDefinitePhraseFrenchLocale(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
tests := []struct {
word string
want string
}{
{"branche", "la branche"},
{"branches", "les branches"},
{"amis", "les amis"},
{"enfant", "l'enfant"},
{"fichier", "le fichier"},
{"héros", "le héros"},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := DefinitePhrase(tt.word)
if got != tt.want {
t.Errorf("DefinitePhrase(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestLabel(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
word string
want string
}{
{"status", "Status:"},
{"", ""},
}
for _, tt := range tests {
t.Run(tt.word, func(t *testing.T) {
got := Label(tt.word)
if got != tt.want {
t.Errorf("Label(%q) = %q, want %q", tt.word, got, tt.want)
}
})
}
}
func TestProgress(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
verb string
want string
}{
{"build", "Building..."},
{"delete", "Deleting..."},
{"scan", "Scanning..."},
}
for _, tt := range tests {
t.Run(tt.verb, func(t *testing.T) {
got := Progress(tt.verb)
if got != tt.want {
t.Errorf("Progress(%q) = %q, want %q", tt.verb, got, tt.want)
}
})
}
}
func TestProgressSubject(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
got := ProgressSubject("build", "project")
want := "Building project..."
if got != want {
t.Errorf("ProgressSubject(%q, %q) = %q, want %q", "build", "project", got, want)
}
}
func TestActionResult(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tests := []struct {
verb, subject string
want string
}{
{"delete", "config.yaml", "Config.yaml deleted"},
{"build", "project", "Project built"},
{"", "file", ""},
{"delete", "", "Deleted"},
}
for _, tt := range tests {
t.Run(tt.verb+"_"+tt.subject, func(t *testing.T) {
got := ActionResult(tt.verb, tt.subject)
if got != tt.want {
t.Errorf("ActionResult(%q, %q) = %q, want %q", tt.verb, tt.subject, got, tt.want)
}
})
}
}
func TestActionFailed(t *testing.T) {
tests := []struct {
verb, subject string
want string
}{
{"delete", "config.yaml", "Failed to delete config.yaml"},
{"push", "commits", "Failed to push commits"},
{"push", "", "Failed to push"},
{"", "", ""},
}
for _, tt := range tests {
t.Run(tt.verb+"_"+tt.subject, func(t *testing.T) {
got := ActionFailed(tt.verb, tt.subject)
if got != tt.want {
t.Errorf("ActionFailed(%q, %q) = %q, want %q", tt.verb, tt.subject, got, tt.want)
}
})
}
}
func TestActionFailed_RespectsWordMap(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
data := GetGrammarData("en")
if data == nil {
t.Fatal("GetGrammarData(\"en\") returned nil")
}
original, existed := data.Words["push"]
data.Words["push"] = "submit"
t.Cleanup(func() {
if existed {
data.Words["push"] = original
return
}
delete(data.Words, "push")
})
if got, want := ActionFailed("push", "commits"), "Failed to submit commits"; got != want {
t.Fatalf("ActionFailed(%q, %q) = %q, want %q", "push", "commits", got, want)
}
}
func TestActionFailedFrenchLocale(t *testing.T) {
prev := Default()
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
t.Cleanup(func() {
SetDefault(prev)
})
if err := SetLanguage("fr"); err != nil {
t.Fatalf("SetLanguage(fr) failed: %v", err)
}
if got, want := ActionFailed("supprimer", ""), "Impossible de supprimer"; got != want {
t.Fatalf("ActionFailed(%q, %q) = %q, want %q", "supprimer", "", got, want)
}
}
func TestGrammarData_Signals(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
data := GetGrammarData("en")
if data == nil {
t.Fatal("GetGrammarData(\"en\") returned nil")
}
if got := len(data.Signals.NounDeterminers); got < 20 {
t.Errorf("NounDeterminers: got %d entries, want >= 20", got)
}
if got := len(data.Signals.VerbAuxiliaries); got < 19 {
t.Errorf("VerbAuxiliaries: got %d entries, want >= 19", got)
}
if len(data.Signals.VerbInfinitive) != 1 || data.Signals.VerbInfinitive[0] != "to" {
t.Errorf("VerbInfinitive: got %v, want [\"to\"]", data.Signals.VerbInfinitive)
}
// Spot-check known values
detFound := false
for _, d := range data.Signals.NounDeterminers {
if d == "the" {
detFound = true
}
}
if !detFound {
t.Error("NounDeterminers missing 'the'")
}
auxFound := false
for _, a := range data.Signals.VerbAuxiliaries {
if a == "will" {
auxFound = true
}
}
if !auxFound {
t.Error("VerbAuxiliaries missing 'will'")
}
}
func TestGrammarData_DualClassEntries(t *testing.T) {
svc, _ := New()
SetDefault(svc)
data := GetGrammarData("en")
dualClass := []string{"commit", "run", "test", "check", "file", "build"}
for _, word := range dualClass {
if _, ok := data.Verbs[word]; !ok {
t.Errorf("gram.verb missing dual-class word %q", word)
}
if _, ok := data.Nouns[word]; !ok {
t.Errorf("gram.noun missing dual-class word %q", word)
}
}
}
func TestFrenchGrammarData(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
data := GetGrammarData("fr")
if data == nil {
t.Fatal("GetGrammarData(\"fr\") returned nil")
}
// Verbs loaded
if got := len(data.Verbs); got < 40 {
t.Errorf("French verbs: got %d, want >= 40", got)
}
// Spot-check irregular verb forms
verbTests := []struct {
verb, past, gerund string
}{
{"être", "été", "étant"},
{"avoir", "eu", "ayant"},
{"faire", "fait", "faisant"},
{"supprimer", "supprimé", "supprimant"},
{"construire", "construit", "construisant"},
{"écrire", "écrit", "écrivant"},
{"prendre", "pris", "prenant"},
}
for _, tt := range verbTests {
forms, ok := data.Verbs[tt.verb]
if !ok {
t.Errorf("French verb %q not found", tt.verb)
continue
}
if forms.Past != tt.past {
t.Errorf("French verb %q past = %q, want %q", tt.verb, forms.Past, tt.past)
}
if forms.Gerund != tt.gerund {
t.Errorf("French verb %q gerund = %q, want %q", tt.verb, forms.Gerund, tt.gerund)
}
}
// Nouns loaded with gender
if got := len(data.Nouns); got < 20 {
t.Errorf("French nouns: got %d, want >= 20", got)
}
nounTests := []struct {
noun, one, other, gender string
}{
{"fichier", "fichier", "fichiers", "m"},
{"branche", "branche", "branches", "f"},
{"vulnérabilité", "vulnérabilité", "vulnérabilités", "f"},
{"serveur", "serveur", "serveurs", "m"},
}
for _, tt := range nounTests {
forms, ok := data.Nouns[tt.noun]
if !ok {
t.Errorf("French noun %q not found", tt.noun)
continue
}
if forms.One != tt.one {
t.Errorf("French noun %q one = %q, want %q", tt.noun, forms.One, tt.one)
}
if forms.Other != tt.other {
t.Errorf("French noun %q other = %q, want %q", tt.noun, forms.Other, tt.other)
}
if forms.Gender != tt.gender {
t.Errorf("French noun %q gender = %q, want %q", tt.noun, forms.Gender, tt.gender)
}
}
// Articles with gender
if data.Articles.Definite != "le" {
t.Errorf("French definite article = %q, want \"le\"", data.Articles.Definite)
}
if data.Articles.IndefiniteDefault != "un" {
t.Errorf("French indefinite article = %q, want \"un\"", data.Articles.IndefiniteDefault)
}
if data.Articles.ByGender == nil {
t.Fatal("French articles.ByGender is nil")
}
if got := data.Articles.ByGender["m"]; got != "le" {
t.Errorf("French article by_gender[m] = %q, want \"le\"", got)
}
if got := data.Articles.ByGender["f"]; got != "la" {
t.Errorf("French article by_gender[f] = %q, want \"la\"", got)
}
// Punctuation — French space before colon
if data.Punct.LabelSuffix != " :" {
t.Errorf("French label suffix = %q, want \" :\"", data.Punct.LabelSuffix)
}
// Signals loaded
if got := len(data.Signals.NounDeterminers); got < 25 {
t.Errorf("French NounDeterminers: got %d, want >= 25", got)
}
if got := len(data.Signals.VerbAuxiliaries); got < 15 {
t.Errorf("French VerbAuxiliaries: got %d, want >= 15", got)
}
}
func TestGrammarFallbackToBaseLanguageTag(t *testing.T) {
prevDefault := Default()
prevGrammar := GetGrammarData("en")
t.Cleanup(func() {
SetGrammarData("en", prevGrammar)
SetDefault(prevDefault)
})
SetGrammarData("en", &GrammarData{
Verbs: map[string]VerbForms{
"delete": {Past: "deleted", Gerund: "deleting"},
},
Nouns: map[string]NounForms{
"file": {One: "file", Other: "files"},
},
Articles: ArticleForms{
IndefiniteDefault: "a",
IndefiniteVowel: "an",
Definite: "the",
},
Punct: PunctuationRules{
LabelSuffix: ":",
ProgressSuffix: "...",
},
Words: map[string]string{
"status": "Status",
},
})
svc, err := NewWithLoader(regionFallbackLoader{})
if err != nil {
t.Fatalf("NewWithLoader() failed: %v", err)
}
SetDefault(svc)
if err := svc.SetLanguage("en-GB"); err != nil {
t.Fatalf("SetLanguage(en-GB) failed: %v", err)
}
if got := PastTense("delete"); got != "deleted" {
t.Fatalf("PastTense(delete) = %q, want %q", got, "deleted")
}
if got := Pluralize("file", 2); got != "files" {
t.Fatalf("Pluralize(file, 2) = %q, want %q", got, "files")
}
if got := Article("apple"); got != "an" {
t.Fatalf("Article(apple) = %q, want %q", got, "an")
}
if got := Label("status"); got != "Status:" {
t.Fatalf("Label(status) = %q, want %q", got, "Status:")
}
}
func TestTemplateFuncs(t *testing.T) {
funcs := TemplateFuncs()
expected := []string{
"title",
"lower",
"upper",
"n",
"past",
"gerund",
"plural",
"pluralForm",
"article",
"quote",
"label",
"progress",
"progressSubject",
"actionResult",
"actionFailed",
"prompt",
"lang",
"timeAgo",
"formatAgo",
}
for _, name := range expected {
if _, ok := funcs[name]; !ok {
t.Errorf("TemplateFuncs() missing %q", name)
}
}
}
func TestTemplateFuncs_Article(t *testing.T) {
tmpl, err := template.New("").Funcs(TemplateFuncs()).Parse(`{{article "apple"}}`)
if err != nil {
t.Fatalf("Parse() failed: %v", err)
}
var buf strings.Builder
if err := tmpl.Execute(&buf, nil); err != nil {
t.Fatalf("Execute() failed: %v", err)
}
if got, want := buf.String(), "an apple"; got != want {
t.Fatalf("template article = %q, want %q", got, want)
}
}
func TestTemplateFuncs_CompositeHelpers(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tmpl, err := template.New("").Funcs(TemplateFuncs()).Parse(
`{{label "status"}}|{{progress "build"}}|{{progressSubject "build" "project"}}|{{actionResult "delete" "file"}}|{{actionFailed "delete" "file"}}`,
)
if err != nil {
t.Fatalf("Parse() failed: %v", err)
}
var buf strings.Builder
if err := tmpl.Execute(&buf, nil); err != nil {
t.Fatalf("Execute() failed: %v", err)
}
want := "Status:|Building...|Building project...|File deleted|Failed to delete file"
if got := buf.String(); got != want {
t.Fatalf("template composite helpers = %q, want %q", got, want)
}
}
func TestTemplateFuncs_PromptAndLang(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tmpl, err := template.New("").Funcs(TemplateFuncs()).Parse(
`{{prompt "confirm"}}|{{lang "de"}}`,
)
if err != nil {
t.Fatalf("Parse() failed: %v", err)
}
var buf strings.Builder
if err := tmpl.Execute(&buf, nil); err != nil {
t.Fatalf("Execute() failed: %v", err)
}
if got, want := buf.String(), "Are you sure?|German"; got != want {
t.Fatalf("template prompt/lang = %q, want %q", got, want)
}
}
func TestTemplateFuncs_NumericAlias(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tmpl, err := template.New("").Funcs(TemplateFuncs()).Parse(
`{{n "number" 1234567}}|{{n "ago" 3 "hours"}}`,
)
if err != nil {
t.Fatalf("Parse() failed: %v", err)
}
var buf strings.Builder
if err := tmpl.Execute(&buf, nil); err != nil {
t.Fatalf("Execute() failed: %v", err)
}
got := buf.String()
if !strings.HasPrefix(got, "1,234,567|3 hours ago") {
t.Fatalf("template numeric alias = %q, want prefix %q", got, "1,234,567|3 hours ago")
}
}
func TestTemplateFuncs_TimeHelpers(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
tmpl, err := template.New("").Funcs(TemplateFuncs()).Parse(
`{{formatAgo 3 "hour"}}|{{timeAgo .}}`,
)
if err != nil {
t.Fatalf("Parse() failed: %v", err)
}
var buf strings.Builder
if err := tmpl.Execute(&buf, time.Now().Add(-5*time.Minute)); err != nil {
t.Fatalf("Execute() failed: %v", err)
}
got := buf.String()
if !strings.HasPrefix(got, "3 hours ago|") {
t.Fatalf("template time helpers prefix = %q, want %q", got, "3 hours ago|")
}
if !strings.Contains(got, "minutes ago") && !strings.Contains(got, "just now") {
t.Fatalf("template time helpers suffix = %q, want relative time output", got)
}
}
func TestCompositeHelpersRespectWordMap(t *testing.T) {
svc, err := New()
if err != nil {
t.Fatalf("New() failed: %v", err)
}
SetDefault(svc)
data := GetGrammarData("en")
if data == nil {
t.Fatal("GetGrammarData(\"en\") returned nil")
}
original, existed := data.Words["go_mod"]
data.Words["go_mod"] = "go.mod"
t.Cleanup(func() {
if existed {
data.Words["go_mod"] = original
return
}
delete(data.Words, "go_mod")
})
if got, want := Label("go_mod"), "go.mod:"; got != want {
t.Fatalf("Label(%q) = %q, want %q", "go_mod", got, want)
}
if got, want := ProgressSubject("build", "go_mod"), "Building go.mod..."; got != want {
t.Fatalf("ProgressSubject(%q, %q) = %q, want %q", "build", "go_mod", got, want)
}
if got, want := ProgressSubject("build", ""), "Building..."; got != want {
t.Fatalf("ProgressSubject(%q, %q) = %q, want %q", "build", "", got, want)
}
if got, want := ActionResult("delete", "go_mod"), "go.mod deleted"; got != want {
t.Fatalf("ActionResult(%q, %q) = %q, want %q", "delete", "go_mod", got, want)
}
if got, want := ActionResult("delete", ""), "Deleted"; got != want {
t.Fatalf("ActionResult(%q, %q) = %q, want %q", "delete", "", got, want)
}
if got, want := ActionFailed("delete", "go_mod"), "Failed to delete go.mod"; got != want {
t.Fatalf("ActionFailed(%q, %q) = %q, want %q", "delete", "go_mod", got, want)
}
}
// --- Benchmarks ---
func BenchmarkPastTense_Irregular(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
PastTense("build")
}
}
func BenchmarkPastTense_Regular(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
PastTense("delete")
}
}
func BenchmarkPastTense_Compound(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
PastTense("rebuild")
}
}
func BenchmarkGerund(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
Gerund("commit")
}
}
func BenchmarkPluralize(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
Pluralize("repository", 5)
}
}
func BenchmarkArticle(b *testing.B) {
for i := 0; i < b.N; i++ {
Article("error")
}
}
func BenchmarkProgress(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
Progress("build")
}
}
func BenchmarkActionResult(b *testing.B) {
svc, _ := New()
SetDefault(svc)
b.ResetTimer()
for i := 0; i < b.N; i++ {
ActionResult("delete", "config.yaml")
}
}