| // Copyright 2017 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package cldrtree |
| |
| import ( |
| "bytes" |
| "flag" |
| "log" |
| "math/rand" |
| "os" |
| "path/filepath" |
| "reflect" |
| "regexp" |
| "strconv" |
| "strings" |
| "testing" |
| |
| "golang.org/x/text/internal/gen" |
| "golang.org/x/text/internal/language/compact" |
| "golang.org/x/text/language" |
| "golang.org/x/text/unicode/cldr" |
| ) |
| |
| var genOutput = flag.Bool("gen", false, "generate output files") |
| |
| func TestAliasRegexp(t *testing.T) { |
| testCases := []struct { |
| alias string |
| want []string |
| }{{ |
| alias: "miscPatterns[@numberSystem='latn']", |
| want: []string{ |
| "miscPatterns[@numberSystem='latn']", |
| "miscPatterns", |
| "[@numberSystem='latn']", |
| "numberSystem", |
| "latn", |
| }, |
| }, { |
| alias: `calendar[@type='greg-foo']/days/`, |
| want: []string{ |
| "calendar[@type='greg-foo']", |
| "calendar", |
| "[@type='greg-foo']", |
| "type", |
| "greg-foo", |
| }, |
| }, { |
| alias: "eraAbbr", |
| want: []string{ |
| "eraAbbr", |
| "eraAbbr", |
| "", |
| "", |
| "", |
| }, |
| }, { |
| // match must be anchored at beginning. |
| alias: `../calendar[@type='gregorian']/days/`, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.alias, func(t *testing.T) { |
| got := aliasRe.FindStringSubmatch(tc.alias) |
| if !reflect.DeepEqual(got, tc.want) { |
| t.Errorf("got %v; want %v", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func TestBuild(t *testing.T) { |
| tree1, _ := loadTestdata(t, "test1") |
| tree2, _ := loadTestdata(t, "test2") |
| |
| // Constants for second test |
| const ( |
| calendar = iota |
| field |
| ) |
| const ( |
| month = iota |
| era |
| filler |
| cyclicNameSet |
| ) |
| const ( |
| abbreviated = iota |
| narrow |
| wide |
| ) |
| |
| testCases := []struct { |
| desc string |
| tree *Tree |
| locale string |
| path []uint16 |
| isFeature bool |
| result string |
| }{{ |
| desc: "und/chinese month format wide m1", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 0, month, 0, wide, 1), |
| result: "cM01", |
| }, { |
| desc: "und/chinese month format wide m12", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 0, month, 0, wide, 12), |
| result: "cM12", |
| }, { |
| desc: "und/non-existing value", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 0, month, 0, wide, 13), |
| result: "", |
| }, { |
| desc: "und/dangi:chinese month format wide", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 1, month, 0, wide, 1), |
| result: "cM01", |
| }, { |
| desc: "und/chinese month format abbreviated:wide", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 0, month, 0, abbreviated, 1), |
| result: "cM01", |
| }, { |
| desc: "und/chinese month format narrow:wide", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 0, month, 0, narrow, 1), |
| result: "cM01", |
| }, { |
| desc: "und/gregorian month format wide", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 2, month, 0, wide, 2), |
| result: "gM02", |
| }, { |
| desc: "und/gregorian month format:stand-alone narrow", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 2, month, 0, narrow, 1), |
| result: "1", |
| }, { |
| desc: "und/gregorian month stand-alone:format abbreviated", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 2, month, 1, abbreviated, 1), |
| result: "gM01", |
| }, { |
| desc: "und/gregorian month stand-alone:format wide ", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 2, month, 1, abbreviated, 1), |
| result: "gM01", |
| }, { |
| desc: "und/dangi:chinese month format narrow:wide ", |
| tree: tree1, |
| locale: "und", |
| path: path(calendar, 1, month, 0, narrow, 4), |
| result: "cM04", |
| }, { |
| desc: "und/field era displayname 0", |
| tree: tree2, |
| locale: "und", |
| path: path(field, 0, 0, 0), |
| result: "Era", |
| }, { |
| desc: "en/field era displayname 0", |
| tree: tree2, |
| locale: "en", |
| path: path(field, 0, 0, 0), |
| result: "era", |
| }, { |
| desc: "und/calendar hebrew format wide 7-leap", |
| tree: tree2, |
| locale: "und", |
| path: path(calendar, 7, month, 0, wide, 0), |
| result: "Adar II", |
| }, { |
| desc: "en-GB:en-001:en:und/calendar hebrew format wide 7-leap", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 7, month, 0, wide, 0), |
| result: "Adar II", |
| }, { |
| desc: "und/buddhist month format wide 11", |
| tree: tree2, |
| locale: "und", |
| path: path(calendar, 0, month, 0, wide, 12), |
| result: "genWideM12", |
| }, { |
| desc: "en-GB/gregorian month stand-alone narrow 2", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, month, 1, narrow, 3), |
| result: "gbNarrowM3", |
| }, { |
| desc: "en-GB/gregorian month format narrow 3/missing in en-GB", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, month, 0, narrow, 4), |
| result: "enNarrowM4", |
| }, { |
| desc: "en-GB/gregorian month format narrow 3/missing in en and en-GB", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, month, 0, narrow, 7), |
| result: "gregNarrowM7", |
| }, { |
| desc: "en-GB/gregorian month format narrow 3/missing in en and en-GB", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, month, 0, narrow, 7), |
| result: "gregNarrowM7", |
| }, { |
| desc: "en-GB/gregorian era narrow", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, era, abbreviated, 0, 1), |
| isFeature: true, |
| result: "AD", |
| }, { |
| desc: "en-GB/gregorian era narrow", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, era, narrow, 0, 0), |
| isFeature: true, |
| result: "BC", |
| }, { |
| desc: "en-GB/gregorian era narrow", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 6, era, wide, 1, 0), |
| isFeature: true, |
| result: "Before Common Era", |
| }, { |
| desc: "en-GB/dangi:chinese cyclicName, months, format, narrow:abbreviated 2", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(calendar, 1, cyclicNameSet, 3, 0, 1, 2), |
| isFeature: true, |
| result: "year2", |
| }, { |
| desc: "en-GB/field era-narrow ", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(field, 2, 0, 0), |
| result: "era", |
| }, { |
| desc: "en-GB/field month-narrow relativeTime future one", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(field, 5, 2, 0, 1), |
| isFeature: true, |
| result: "001NarrowFutMOne", |
| }, { |
| // Don't fall back to the one of "en". |
| desc: "en-GB/field month-short relativeTime past one:other", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(field, 4, 2, 1, 1), |
| isFeature: true, |
| result: "001ShortPastMOther", |
| }, { |
| desc: "en-GB/field month relativeTime future two:other", |
| tree: tree2, |
| locale: "en-GB", |
| path: path(field, 3, 2, 0, 2), |
| isFeature: true, |
| result: "enFutMOther", |
| }} |
| |
| for _, tc := range testCases { |
| t.Run(tc.desc, func(t *testing.T) { |
| tag, _ := compact.RegionalID(compact.Tag(language.MustParse(tc.locale))) |
| s := tc.tree.lookup(tag, tc.isFeature, tc.path...) |
| if s != tc.result { |
| t.Errorf("got %q; want %q", s, tc.result) |
| } |
| }) |
| } |
| } |
| |
| func path(e ...uint16) []uint16 { return e } |
| |
| func TestGen(t *testing.T) { |
| testCases := []string{"test1", "test2"} |
| for _, tc := range testCases { |
| t.Run(tc, func(t *testing.T) { |
| _, got := loadTestdata(t, tc) |
| |
| // Remove sizes that may vary per architecture. |
| re := regexp.MustCompile("// Size: [0-9]*") |
| got = re.ReplaceAllLiteral(got, []byte("// Size: xxxx")) |
| re = regexp.MustCompile("// Total table size [0-9]*") |
| got = re.ReplaceAllLiteral(got, []byte("// Total table size: xxxx")) |
| |
| file := filepath.Join("testdata", tc, "output.go") |
| if *genOutput { |
| os.WriteFile(file, got, 0700) |
| t.SkipNow() |
| } |
| |
| b, err := os.ReadFile(file) |
| if err != nil { |
| t.Fatalf("failed to open file: %v", err) |
| } |
| if want := string(b); string(got) != want { |
| t.Log(string(got)) |
| t.Errorf("files differ") |
| } |
| }) |
| } |
| } |
| |
| func loadTestdata(t *testing.T, test string) (tree *Tree, file []byte) { |
| b := New("test") |
| |
| var d cldr.Decoder |
| |
| data, err := d.DecodePath(filepath.Join("testdata", test)) |
| if err != nil { |
| t.Fatalf("error decoding testdata: %v", err) |
| } |
| |
| context := Enum("context") |
| widthMap := func(s string) string { |
| // Align era with width values. |
| if r, ok := map[string]string{ |
| "eraAbbr": "abbreviated", |
| "eraNarrow": "narrow", |
| "eraNames": "wide", |
| }[s]; ok { |
| s = r |
| } |
| return "w" + strings.Title(s) |
| } |
| width := EnumFunc("width", widthMap, "abbreviated", "narrow", "wide") |
| month := Enum("month", "leap7") |
| relative := EnumFunc("relative", func(s string) string { |
| x, err := strconv.ParseInt(s, 10, 8) |
| if err != nil { |
| log.Fatal("Invalid number:", err) |
| } |
| return []string{ |
| "before1", |
| "current", |
| "after1", |
| }[x+1] |
| }) |
| cycleType := EnumFunc("cycleType", func(s string) string { |
| return "cyc" + strings.Title(s) |
| }) |
| r := rand.New(rand.NewSource(0)) |
| |
| for _, loc := range data.Locales() { |
| ldml := data.RawLDML(loc) |
| x := b.Locale(language.Make(loc)) |
| |
| if x := x.Index(ldml.Dates.Calendars); x != nil { |
| for _, cal := range ldml.Dates.Calendars.Calendar { |
| x := x.IndexFromType(cal) |
| if x := x.Index(cal.Months); x != nil { |
| for _, mc := range cal.Months.MonthContext { |
| x := x.IndexFromType(mc, context) |
| for _, mw := range mc.MonthWidth { |
| x := x.IndexFromType(mw, width) |
| for _, m := range mw.Month { |
| x.SetValue(m.Yeartype+m.Type, m, month) |
| } |
| } |
| } |
| } |
| if x := x.Index(cal.CyclicNameSets); x != nil { |
| for _, cns := range cal.CyclicNameSets.CyclicNameSet { |
| x := x.IndexFromType(cns, cycleType) |
| for _, cc := range cns.CyclicNameContext { |
| x := x.IndexFromType(cc, context) |
| for _, cw := range cc.CyclicNameWidth { |
| x := x.IndexFromType(cw, width) |
| for _, c := range cw.CyclicName { |
| x.SetValue(c.Type, c) |
| } |
| } |
| } |
| } |
| } |
| if x := x.Index(cal.Eras); x != nil { |
| opts := []Option{width, SharedType()} |
| if x := x.Index(cal.Eras.EraNames, opts...); x != nil { |
| for _, e := range cal.Eras.EraNames.Era { |
| x.IndexFromAlt(e).SetValue(e.Type, e) |
| } |
| } |
| if x := x.Index(cal.Eras.EraAbbr, opts...); x != nil { |
| for _, e := range cal.Eras.EraAbbr.Era { |
| x.IndexFromAlt(e).SetValue(e.Type, e) |
| } |
| } |
| if x := x.Index(cal.Eras.EraNarrow, opts...); x != nil { |
| for _, e := range cal.Eras.EraNarrow.Era { |
| x.IndexFromAlt(e).SetValue(e.Type, e) |
| } |
| } |
| } |
| { |
| // Ensure having more than 2 buckets. |
| f := x.IndexWithName("filler") |
| b := make([]byte, maxStrlen) |
| opt := &options{parent: x} |
| r.Read(b) |
| f.setValue("0", string(b), opt) |
| } |
| } |
| } |
| if x := x.Index(ldml.Dates.Fields); x != nil { |
| for _, f := range ldml.Dates.Fields.Field { |
| x := x.IndexFromType(f) |
| for _, d := range f.DisplayName { |
| x.Index(d).SetValue("", d) |
| } |
| for _, r := range f.Relative { |
| x.Index(r).SetValue(r.Type, r, relative) |
| } |
| for _, rt := range f.RelativeTime { |
| x := x.Index(rt).IndexFromType(rt) |
| for _, p := range rt.RelativeTimePattern { |
| x.SetValue(p.Count, p) |
| } |
| } |
| for _, rp := range f.RelativePeriod { |
| x.Index(rp).SetValue("", rp) |
| } |
| } |
| } |
| } |
| |
| tree, err = build(b) |
| if err != nil { |
| t.Fatal("error building tree:", err) |
| } |
| w := gen.NewCodeWriter() |
| generate(b, tree, w) |
| generateTestData(b, w) |
| buf := &bytes.Buffer{} |
| if _, err = w.WriteGo(buf, "test", ""); err != nil { |
| t.Log(buf.String()) |
| t.Fatal("error generating code:", err) |
| } |
| return tree, buf.Bytes() |
| } |