package handlebars import ( "os" "path" "regexp" "strings" "testing" "gopkg.in/yaml.v3" ) // // Note, as the JS implementation, the divergences from mustache spec: // - we don't support alternative delimeters // - the mustache lambda spec differs // type mustacheTest struct { Name string Desc string Data interface{} Template string Expected string Partials map[string]string } type mustacheTestFile struct { Overview string Tests []mustacheTest } var ( rAltDelim = regexp.MustCompile(regexp.QuoteMeta("{{=")) ) var ( musTestLambdaInterMult = 0 ) func TestMustache(t *testing.T) { t.Skip("Does not work") skipFiles := map[string]bool{ // mustache lambdas differ from handlebars lambdas "~lambdas.yml": true, } for _, fileName := range mustacheTestFiles() { if skipFiles[fileName] { // fmt.Printf("Skipped file: %s\n", fileName) continue } launchTests(t, testsFromMustacheFile(fileName)) } } func testsFromMustacheFile(fileName string) []Test { result := []Test{} fileData, err := os.ReadFile(path.Join("mustache", "specs", fileName)) if err != nil { panic(err) } var testFile mustacheTestFile if err := yaml.Unmarshal(fileData, &testFile); err != nil { panic(err) } for _, mustacheTest := range testFile.Tests { if mustBeSkipped(mustacheTest, fileName) { // fmt.Printf("Skipped test: %s\n", mustacheTest.Name) continue } test := Test{ name: mustacheTest.Name, input: mustacheTest.Template, data: mustacheTest.Data, partials: mustacheTest.Partials, output: mustacheTest.Expected, } result = append(result, test) } return result } // returns true if test must be skipped func mustBeSkipped(test mustacheTest, fileName string) bool { // handlebars does not support alternative delimiters return haveAltDelimiter(test) || // the JS implementation skips those tests fileName == "partials.yml" && (test.Name == "Failed Lookup" || test.Name == "Standalone Indentation") } // returns true if test have alternative delimeter in template or in partials func haveAltDelimiter(test mustacheTest) bool { // check template if rAltDelim.MatchString(test.Template) { return true } // check partials for _, partial := range test.Partials { if rAltDelim.MatchString(partial) { return true } } return false } func mustacheTestFiles() []string { var result []string files, err := os.ReadDir(path.Join("mustache", "specs")) if err != nil { panic(err) } for _, file := range files { fileName := file.Name() if !file.IsDir() && strings.HasSuffix(fileName, ".yml") { result = append(result, fileName) } } return result } // // Following tests come fron ~lambdas.yml // var mustacheLambdasTests = []Test{ { "Interpolation", "Hello, {{lambda}}!", map[string]interface{}{"lambda": func() string { return "world" }}, nil, nil, nil, "Hello, world!", }, // // SKIP: lambda return value is not parsed // { // "Interpolation - Expansion", // "Hello, {{lambda}}!", // map[string]interface{}{"lambda": func() string { return "{{planet}}" }}, // nil, nil, nil, // "Hello, world!", // }, // SKIP "Interpolation - Alternate Delimiters" { "Interpolation - Multiple Calls", "{{lambda}} == {{{lambda}}} == {{lambda}}", map[string]interface{}{"lambda": func() string { musTestLambdaInterMult++ return Str(musTestLambdaInterMult) }}, nil, nil, nil, "1 == 2 == 3", }, { "Escaping", "<{{lambda}}{{{lambda}}}", map[string]interface{}{"lambda": func() string { return ">" }}, nil, nil, nil, "<>>", }, // // SKIP: "Lambdas used for sections should receive the raw section string." // { // "Section", // "<{{#lambda}}{{x}}{{/lambda}}>", // map[string]interface{}{"lambda": func(param string) string { // if param == "{{x}}" { // return "yes" // } // return "false" // }, "x": "Error!"}, // nil, nil, nil, // "", // }, // // SKIP: lambda return value is not parsed // { // "Section - Expansion", // "<{{#lambda}}-{{/lambda}}>", // map[string]interface{}{"lambda": func(param string) string { // return param + "{{planet}}" + param // }, "planet": "Earth"}, // nil, nil, nil, // "<-Earth->", // }, // SKIP: "Section - Alternate Delimiters" { "Section - Multiple Calls", "{{#lambda}}FILE{{/lambda}} != {{#lambda}}LINE{{/lambda}}", map[string]interface{}{"lambda": func(options *Options) string { return "__" + options.Fn() + "__" }}, nil, nil, nil, "__FILE__ != __LINE__", }, // // SKIP: "Lambdas used for inverted sections should be considered truthy." // { // "Inverted Section", // "<{{^lambda}}{{static}}{{/lambda}}>", // map[string]interface{}{ // "lambda": func() interface{} { // return false // }, // "static": "static", // }, // nil, nil, nil, // "<>", // }, } func TestMustacheLambdas(t *testing.T) { t.Parallel() launchTests(t, mustacheLambdasTests) }