diff --git a/context.go b/context.go index 6a82432..e055602 100644 --- a/context.go +++ b/context.go @@ -53,3 +53,17 @@ func (data *ContextData) AddVariable(name string, value string) { data.Variables[name] = value } + +func (data *ContextData) AddVariables(vars map[string]string) { + if len(vars) == 0 { + return + } + + if data.Variables == nil { + data.Variables = make(map[string]string) + } + + for k, v := range vars { + data.Variables[k] = v + } +} diff --git a/errgoengine.go b/errgoengine.go index 22e2440..8c2cd9c 100644 --- a/errgoengine.go +++ b/errgoengine.go @@ -4,7 +4,6 @@ import ( "bytes" "fmt" "io/fs" - "path/filepath" sitter "github.com/smacker/go-tree-sitter" ) @@ -58,52 +57,11 @@ func (e *ErrgoEngine) Analyze(workingPath, msg string) (*CompiledErrorTemplate, contextData.Analyzer = template.Language.AnalyzerFactory(contextData) contextData.AddVariable("message", msg) - groupNames := template.Pattern.SubexpNames() - for _, submatches := range template.Pattern.FindAllStringSubmatch(msg, -1) { - for idx, matchedContent := range submatches { - if len(groupNames[idx]) == 0 { - continue - } - - contextData.AddVariable(groupNames[idx], matchedContent) - } - } + // extract variables from the error message + contextData.AddVariables(template.ExtractVariables(msg)) // extract stack trace - rawStackTraceItem := contextData.Variables["stacktrace"] - symbolGroupIdx := template.StackTraceRegex().SubexpIndex("symbol") - pathGroupIdx := template.StackTraceRegex().SubexpIndex("path") - posGroupIdx := template.StackTraceRegex().SubexpIndex("position") - stackTraceMatches := template.StackTraceRegex().FindAllStringSubmatch(rawStackTraceItem, -1) - - for _, submatches := range stackTraceMatches { - if len(submatches) == 0 { - continue - } - - rawSymbolName := "" - if symbolGroupIdx != -1 { - rawSymbolName = submatches[symbolGroupIdx] - } - rawPath := submatches[pathGroupIdx] - rawPos := submatches[posGroupIdx] - - // convert relative paths to absolute for parsing - if len(workingPath) != 0 && !filepath.IsAbs(rawPath) { - rawPath = filepath.Clean(filepath.Join(workingPath, rawPath)) - } - - stLoc := template.Language.LocationConverter(LocationConverterContext{ - Path: rawPath, - Pos: rawPos, - ContextData: contextData, - }) - if contextData.TraceStack == nil { - contextData.TraceStack = TraceStack{} - } - - contextData.TraceStack.Add(rawSymbolName, stLoc) - } + contextData.TraceStack = template.ExtractStackTrace(contextData) // open contents of the extracted stack file locations parser := sitter.NewParser() diff --git a/error_template.go b/error_template.go index 1eef406..b25b57f 100644 --- a/error_template.go +++ b/error_template.go @@ -2,6 +2,7 @@ package errgoengine import ( "fmt" + "path/filepath" "regexp" "strings" ) @@ -37,6 +38,65 @@ func (tmp *CompiledErrorTemplate) StackTraceRegex() *regexp.Regexp { return tmp.Language.stackTraceRegex } +func (tmp *CompiledErrorTemplate) ExtractVariables(msg string) map[string]string { + variables := map[string]string{} + groupNames := tmp.Pattern.SubexpNames() + for _, submatches := range tmp.Pattern.FindAllStringSubmatch(msg, -1) { + for idx, matchedContent := range submatches { + if len(groupNames[idx]) == 0 { + continue + } + + variables[groupNames[idx]] = matchedContent + } + } + return variables +} + +func (tmp *CompiledErrorTemplate) ExtractStackTrace(cd *ContextData) TraceStack { + traceStack := TraceStack{} + workingPath := cd.WorkingPath + + rawStackTraceItem := cd.Variables["stacktrace"] + stackTraceRegex := tmp.StackTraceRegex() + symbolGroupIdx := stackTraceRegex.SubexpIndex("symbol") + pathGroupIdx := stackTraceRegex.SubexpIndex("path") + posGroupIdx := stackTraceRegex.SubexpIndex("position") + stackTraceMatches := stackTraceRegex.FindAllStringSubmatch(rawStackTraceItem, -1) + + for _, submatches := range stackTraceMatches { + if len(submatches) == 0 { + continue + } + + rawSymbolName := "" + if symbolGroupIdx != -1 { + rawSymbolName = submatches[symbolGroupIdx] + } + rawPath := submatches[pathGroupIdx] + rawPos := submatches[posGroupIdx] + + // convert relative paths to absolute for parsing + if len(workingPath) != 0 && !filepath.IsAbs(rawPath) { + rawPath = filepath.Clean(filepath.Join(workingPath, rawPath)) + } + + stLoc := tmp.Language.LocationConverter(LocationConverterContext{ + Path: rawPath, + Pos: rawPos, + ContextData: cd, + }) + + traceStack.Add(rawSymbolName, stLoc) + } + + return traceStack +} + +func (tmp *CompiledErrorTemplate) Match(str string) bool { + return tmp.Pattern.MatchString(str) +} + type ErrorTemplates map[string]*CompiledErrorTemplate const defaultStackTraceRegex = `(?P(?:.|\s)*)` @@ -110,7 +170,7 @@ func (tmps ErrorTemplates) MustAdd(language *Language, template ErrorTemplate) * func (tmps ErrorTemplates) Match(msg string) *CompiledErrorTemplate { for _, tmp := range tmps { - if tmp.Pattern.MatchString(msg) { + if tmp.Match(msg) { return tmp } } diff --git a/error_template_test.go b/error_template_test.go index 2ab23e6..faa8fae 100644 --- a/error_template_test.go +++ b/error_template_test.go @@ -1,57 +1,28 @@ package errgoengine_test import ( - "context" + "encoding/json" + "fmt" + "reflect" + "strings" "testing" lib "github.com/nedpals/errgoengine" testutils "github.com/nedpals/errgoengine/test_utils" ) -type TestAnalyzer struct{} - -func (TestAnalyzer) FallbackSymbol() lib.Symbol { - return nil -} - -func (TestAnalyzer) FindSymbol(string) lib.Symbol { return nil } - -func (TestAnalyzer) AnalyzeNode(context.Context, lib.SyntaxNode) lib.Symbol { - return nil -} - -func (TestAnalyzer) AnalyzeImport(lib.ImportParams) lib.ResolvedImport { - return lib.ResolvedImport{} -} - -var testLanguage = &lib.Language{ - Name: "TestLang", - FilePatterns: []string{".test"}, - StackTracePattern: `\sin (?P\S+) at (?P\S+):(?P\d+)`, - LocationConverter: func(ctx lib.LocationConverterContext) lib.Location { - return lib.Location{ - DocumentPath: ctx.Path, - StartPos: lib.Position{0, 0, 0}, - EndPos: lib.Position{0, 0, 0}, - } - }, - AnalyzerFactory: func(cd *lib.ContextData) lib.LanguageAnalyzer { - return TestAnalyzer{} - }, -} - func emptyExplainFn(cd *lib.ContextData, gen *lib.ExplainGenerator) {} func emptyBugFixFn(cd *lib.ContextData, gen *lib.BugFixGenerator) {} func setupTemplate(template lib.ErrorTemplate) (*lib.CompiledErrorTemplate, error) { errorTemplates := lib.ErrorTemplates{} - return errorTemplates.Add(testLanguage, template) + return errorTemplates.Add(lib.TestLanguage, template) } func TestErrorTemplate(t *testing.T) { t.Run("Simple", func(t *testing.T) { tmp, err := setupTemplate(lib.ErrorTemplate{ - Name: "SampleError", + Name: "ErrorA", Pattern: "This is a sample error", OnGenExplainFn: emptyExplainFn, OnGenBugFixFn: emptyBugFixFn, @@ -61,8 +32,8 @@ func TestErrorTemplate(t *testing.T) { t.Fatal(err) } - testutils.Equals(t, tmp.Name, "SampleError") - testutils.Equals(t, tmp.Language, testLanguage) + testutils.Equals(t, tmp.Name, "ErrorA") + testutils.Equals(t, tmp.Language, lib.TestLanguage) testutils.Equals(t, tmp.Pattern.String(), `(?m)^This is a sample error(?P(?:.|\s)*)$`) testutils.Equals(t, tmp.StackTraceRegex().String(), `(?m)\sin (?P\S+) at (?P\S+):(?P\d+)`) testutils.ExpectNil(t, tmp.StackTracePattern) @@ -70,7 +41,7 @@ func TestErrorTemplate(t *testing.T) { t.Run("With custom stack trace", func(t *testing.T) { tmp, err := setupTemplate(lib.ErrorTemplate{ - Name: "SampleError2", + Name: "ErrorB", Pattern: "This is a sample error with stack trace", OnGenExplainFn: emptyExplainFn, OnGenBugFixFn: emptyBugFixFn, @@ -81,15 +52,15 @@ func TestErrorTemplate(t *testing.T) { t.Fatal(err) } - testutils.Equals(t, tmp.Name, "SampleError2") - testutils.Equals(t, tmp.Language, testLanguage) + testutils.Equals(t, tmp.Name, "ErrorB") + testutils.Equals(t, tmp.Language, lib.TestLanguage) testutils.Equals(t, tmp.Pattern.String(), `(?m)^This is a sample error with stack trace(?P(?:.|\s)*)$`) testutils.Equals(t, tmp.StackTraceRegex().String(), `(?P\S+):(?P\S+):(?P\d+)`) }) t.Run("With custom error pattern", func(t *testing.T) { tmp, err := setupTemplate(lib.ErrorTemplate{ - Name: "SampleError3", + Name: "ErrorC", Pattern: lib.CustomErrorPattern("Stack trace in middle $stacktracetest"), OnGenExplainFn: emptyExplainFn, OnGenBugFixFn: emptyBugFixFn, @@ -99,8 +70,8 @@ func TestErrorTemplate(t *testing.T) { t.Fatal(err) } - testutils.Equals(t, tmp.Name, "SampleError3") - testutils.Equals(t, tmp.Language, testLanguage) + testutils.Equals(t, tmp.Name, "ErrorC") + testutils.Equals(t, tmp.Language, lib.TestLanguage) testutils.Equals(t, tmp.Pattern.String(), `(?m)^Stack trace in middle (?P(?:.|\s)*)test$`) testutils.ExpectNil(t, tmp.StackTracePattern) }) @@ -118,7 +89,7 @@ func TestStackTraceRegex(t *testing.T) { t.Fatal(err) } - testutils.Equals(t, tmp.StackTraceRegex().String(), "(?m)"+testLanguage.StackTracePattern) + testutils.Equals(t, tmp.StackTraceRegex().String(), "(?m)"+lib.TestLanguage.StackTracePattern) }) t.Run("With custom stack trace", func(t *testing.T) { @@ -166,4 +137,394 @@ func TestStackTraceRegex(t *testing.T) { t.Fatal("expected panic, got successful execution instead") }) + +} + +func TestExtractVariables(t *testing.T) { + tmp, err := setupTemplate(lib.ErrorTemplate{ + Name: "WithVarError", + Pattern: "invalid input '(?P.*)'", + OnGenExplainFn: emptyExplainFn, + OnGenBugFixFn: emptyBugFixFn, + }) + if err != nil { + t.Fatal(err) + } + + tmp2, err := setupTemplate(lib.ErrorTemplate{ + Name: "WithoutVarError", + Pattern: "invalid input '.*'", + OnGenExplainFn: emptyExplainFn, + OnGenBugFixFn: emptyBugFixFn, + }) + if err != nil { + t.Fatal(err) + } + + t.Run("Simple", func(t *testing.T) { + input := "invalid input '123abc'\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1" + if !tmp.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp.ExtractVariables(input) + exp := map[string]string{ + "stacktrace": "\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + "input": "123abc", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) + + t.Run("No stack trace", func(t *testing.T) { + input := "invalid input 'wxyz88@'" + if !tmp.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp.ExtractVariables(input) + exp := map[string]string{ + "input": "wxyz88@", + "stacktrace": "", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) + + t.Run("No variables", func(t *testing.T) { + input := "invalid input '123abc'\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1" + if !tmp2.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp2.ExtractVariables(input) + exp := map[string]string{ + "stacktrace": "\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) + + t.Run("No variables + no stack trace", func(t *testing.T) { + input := "invalid input '123abc'" + if !tmp2.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp2.ExtractVariables(input) + exp := map[string]string{ + "stacktrace": "", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) + + t.Run("Multiple", func(t *testing.T) { + inputs := []string{ + "invalid input '123abc'\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + "invalid input 'shouldNotBeIncluded'\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + } + input := strings.Join(inputs, "\n\n") + if !tmp.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp.ExtractVariables(input) + exp := map[string]string{ + "stacktrace": "\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + "input": "123abc", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) + + t.Run("Multiple no stack trace", func(t *testing.T) { + inputs := []string{ + "invalid input '123abc'", + "invalid input 'shouldNotBeIncluded'", + } + input := strings.Join(inputs, "\n\n") + if !tmp.Match(input) { + t.Fatalf("expected template to match input, got false instead") + } + + variables := tmp.ExtractVariables(input) + exp := map[string]string{ + "stacktrace": "\nin main at /home/user/main.py:123\nin main at /home/user/main.py:1", + "input": "123abc", + } + + fmt.Printf("%q\n", variables) + if !reflect.DeepEqual(variables, exp) { + t.Fatalf("expected %v, got %v", exp, variables) + } + }) +} + +func printStackTraceJson(stackTrace lib.TraceStack) { + b, _ := json.MarshalIndent(stackTrace, "", " ") + fmt.Println(string(b)) +} + +func TestExtractStackTrace(t *testing.T) { + tmp, err := setupTemplate(lib.ErrorTemplate{ + Name: "A", + Pattern: "invalid input '(?P.*)'", + OnGenExplainFn: emptyExplainFn, + OnGenBugFixFn: emptyBugFixFn, + }) + if err != nil { + t.Fatal(err) + } + + t.Run("Simple", func(t *testing.T) { + cd := lib.NewContextData(lib.NewEmptyStore(), "/home/user") + cd.Variables = map[string]string{ + "stacktrace": "\nin funcA at /home/user/main.py:123\nin main at /home/user/main.py:1", + } + + stackTrace := tmp.ExtractStackTrace(cd) + exp := lib.TraceStack{ + lib.StackTraceEntry{ + SymbolName: "funcA", + Location: lib.Location{ + DocumentPath: "/home/user/main.py", + StartPos: lib.Position{ + Line: 123, + Column: 0, + Index: 0, + }, + EndPos: lib.Position{ + Line: 123, + Column: 0, + Index: 0, + }, + }, + }, + lib.StackTraceEntry{ + SymbolName: "main", + Location: lib.Location{ + DocumentPath: "/home/user/main.py", + StartPos: lib.Position{ + Line: 1, + Column: 0, + Index: 0, + }, + EndPos: lib.Position{ + Line: 1, + Column: 0, + Index: 0, + }, + }, + }, + } + + printStackTraceJson(exp) + printStackTraceJson(stackTrace) + if !reflect.DeepEqual(stackTrace, exp) { + t.Fatalf("expected %v, got %v", exp, stackTrace) + } + }) + + t.Run("Empty", func(t *testing.T) { + cd := lib.NewContextData(lib.NewEmptyStore(), "/home/user") + cd.Variables = map[string]string{ + "stacktrace": "", + } + + stackTrace := tmp.ExtractStackTrace(cd) + exp := lib.TraceStack{} + + printStackTraceJson(exp) + printStackTraceJson(stackTrace) + if !reflect.DeepEqual(stackTrace, exp) { + t.Fatalf("expected %v, got %v", exp, stackTrace) + } + }) + + t.Run("Invalid stack trace", func(t *testing.T) { + cd := lib.NewContextData(lib.NewEmptyStore(), "/home/user") + cd.Variables = map[string]string{ + "stacktrace": "\nin main at 123\nin main at 1", + } + + stackTrace := tmp.ExtractStackTrace(cd) + exp := lib.TraceStack{} + + printStackTraceJson(exp) + printStackTraceJson(stackTrace) + if !reflect.DeepEqual(stackTrace, exp) { + t.Fatalf("expected %v, got %v", exp, stackTrace) + } + }) + + t.Run("Invalid in the middle", func(t *testing.T) { + cd := lib.NewContextData(lib.NewEmptyStore(), "/home/user") + cd.Variables = map[string]string{ + "stacktrace": "\nin funcB at /home/user/main.py:123\nin funcA at 123\nin main at /home/user/main.py:1", + } + + stackTrace := tmp.ExtractStackTrace(cd) + exp := lib.TraceStack{ + lib.StackTraceEntry{ + SymbolName: "funcB", + Location: lib.Location{ + DocumentPath: "/home/user/main.py", + StartPos: lib.Position{ + Line: 123, + Column: 0, + Index: 0, + }, + EndPos: lib.Position{ + Line: 123, + Column: 0, + Index: 0, + }, + }, + }, + lib.StackTraceEntry{ + SymbolName: "main", + Location: lib.Location{ + DocumentPath: "/home/user/main.py", + StartPos: lib.Position{ + Line: 1, + Column: 0, + Index: 0, + }, + EndPos: lib.Position{ + Line: 1, + Column: 0, + Index: 0, + }, + }, + }, + } + + printStackTraceJson(exp) + printStackTraceJson(stackTrace) + if !reflect.DeepEqual(stackTrace, exp) { + t.Fatalf("expected %v, got %v", exp, stackTrace) + } + }) +} + +func TestErrorTemplates(t *testing.T) { + errorTemplates := lib.ErrorTemplates{} + tmp := errorTemplates.MustAdd(lib.TestLanguage, lib.ErrorTemplate{ + Name: "ErrorA", + Pattern: `This is a sample error\n`, + OnGenExplainFn: emptyExplainFn, + OnGenBugFixFn: emptyBugFixFn, + }) + + tmp2 := errorTemplates.MustAdd(lib.TestLanguage, lib.ErrorTemplate{ + Name: "ErrorB", + Pattern: `Another exmaple error\n`, + OnGenExplainFn: emptyExplainFn, + OnGenBugFixFn: emptyBugFixFn, + }) + + fmt.Println(tmp.Pattern.String()) + fmt.Println(tmp2.Pattern.String()) + + t.Run("Simple", func(t *testing.T) { + inputs := []string{ + "This is a sample error", + "Another exmaple error", + } + + expected := []*lib.CompiledErrorTemplate{ + tmp, + tmp2, + } + + for i, input := range inputs { + matched := errorTemplates.Match(input + "\n" + lib.TestLanguage.StackTracePattern) + + if !reflect.DeepEqual(matched, expected[i]) { + t.Fatalf("expected %s, got %s", expected[i].Name, matched.Name) + } + } + }) + + t.Run("SimpleReverse", func(t *testing.T) { + inputs := []string{ + "Another exmaple error", + "This is a sample error", + } + + expected := []*lib.CompiledErrorTemplate{ + tmp2, + tmp, + } + + for i, input := range inputs { + matched := errorTemplates.Match(input + "\n" + lib.TestLanguage.StackTracePattern) + + if !reflect.DeepEqual(matched, expected[i]) { + t.Fatalf("expected %s, got %s", expected[i].Name, matched.Name) + } + } + }) + + t.Run("Should be nil", func(t *testing.T) { + inputs := []string{ + "This is a sample errorz\n", + "AAnother exmaple error\n", + "Another eaaxmaple error\n" + lib.TestLanguage.StackTracePattern, + "This is a sample erroar\n" + lib.TestLanguage.StackTracePattern, + } + + for _, input := range inputs { + matched := errorTemplates.Match(input) + + if matched != nil { + t.Fatalf("expected nil, got %s", matched.Name) + } + } + }) + + t.Run("Stacked", func(t *testing.T) { + inputs := []string{ + "This is a sample error", + "Another exmaple error", + } + + input := strings.Join(inputs, "\nin main at /home/user/main.py:1\n\n") + matched := errorTemplates.Match(input) + + if !reflect.DeepEqual(matched, tmp) { + t.Fatalf("expected %s, got %s", tmp.Name, matched.Name) + } + }) + + t.Run("StackedReverse", func(t *testing.T) { + // reverse + inputs := []string{ + "Another exmaple error", + "This is a sample error", + } + + input := strings.Join(inputs, "\nin main at /home/user/main.py:1\n\n") + matched := errorTemplates.Match(input) + + if !reflect.DeepEqual(matched, tmp2) { + t.Fatalf("expected %s, got %s", tmp2.Name, matched.Name) + } + }) } diff --git a/language.go b/language.go index 81cf193..eba9b1d 100644 --- a/language.go +++ b/language.go @@ -28,6 +28,8 @@ type LocationConverterContext struct { ContextData *ContextData } +type LocationConverterFunc func(ctx LocationConverterContext) Position + type Language struct { isCompiled bool stackTraceRegex *regexp.Regexp @@ -126,6 +128,7 @@ func DefaultLocationConverter(ctx LocationConverterContext) Location { if _, err := fmt.Sscanf(ctx.Pos, "%d", &trueLine); err != nil { panic(err) } + return Location{ DocumentPath: ctx.Path, StartPos: Position{Line: trueLine}, diff --git a/source_test.go b/source_test.go index 78496ef..45fa3b7 100644 --- a/source_test.go +++ b/source_test.go @@ -1,60 +1,16 @@ package errgoengine import ( - "context" "strings" "testing" sitter "github.com/smacker/go-tree-sitter" - "github.com/smacker/go-tree-sitter/python" ) -var testLanguage = &Language{ - Name: "Python", - FilePatterns: []string{".py"}, - SitterLanguage: python.GetLanguage(), - StackTracePattern: `\s+File "(?P\S+)", line (?P\d+), in (?P\S+)`, - ErrorPattern: `Traceback \(most recent call last\):$stacktrace$message`, - AnalyzerFactory: func(cd *ContextData) LanguageAnalyzer { - return &testAnalyzer{cd} - }, - SymbolsToCapture: ` -(expression_statement - (assignment - left: (identifier) @assignment.name - right: (identifier) @assignment.content) @assignment) -`, -} - -type testAnalyzer struct { - *ContextData -} - -func (an *testAnalyzer) FallbackSymbol() Symbol { - return Builtin("any") -} - -func (an *testAnalyzer) FindSymbol(name string) Symbol { - return nil -} - -func (an *testAnalyzer) AnalyzeNode(_ context.Context, n SyntaxNode) Symbol { - // TODO: - return Builtin("void") -} - -func (an *testAnalyzer) AnalyzeImport(params ImportParams) ResolvedImport { - // TODO: - - return ResolvedImport{ - Path: "", - } -} - func TestParseDocument(t *testing.T) { parser := sitter.NewParser() - doc, err := ParseDocument("test", strings.NewReader(`hello = 1`), parser, testLanguage, nil) + doc, err := ParseDocument("test", strings.NewReader(`hello = 1`), parser, TestLanguage, nil) if err != nil { t.Error(err) } @@ -67,7 +23,7 @@ func TestParseDocument(t *testing.T) { func TestEditableDocument(t *testing.T) { parser := sitter.NewParser() - doc, err := ParseDocument("test", strings.NewReader(`hello = 1`), parser, testLanguage, nil) + doc, err := ParseDocument("test", strings.NewReader(`hello = 1`), parser, TestLanguage, nil) if err != nil { t.Error(err) } else if doc.TotalLines() < 1 { @@ -371,7 +327,7 @@ func TestEditableDocument(t *testing.T) { }) t.Run("EditableDocument.ReplaceWithPadding", func(t *testing.T) { - doc, err := ParseDocument("test", strings.NewReader(` hello = 1`), parser, testLanguage, nil) + doc, err := ParseDocument("test", strings.NewReader(` hello = 1`), parser, TestLanguage, nil) if err != nil { t.Error(err) } else if doc.TotalLines() < 1 { diff --git a/testlang.go b/testlang.go new file mode 100644 index 0000000..07cf02d --- /dev/null +++ b/testlang.go @@ -0,0 +1,45 @@ +package errgoengine + +import ( + "context" +) + +var TestLanguage = &Language{ + Name: "TestLang", + FilePatterns: []string{".test"}, + StackTracePattern: `\sin (?P\S+) at (?P\S+):(?P\d+)`, + AnalyzerFactory: func(cd *ContextData) LanguageAnalyzer { + return &testAnalyzer{} + }, + SymbolsToCapture: ` +(expression_statement + (assignment + left: (identifier) @assignment.name + right: (identifier) @assignment.content) @assignment) + `, +} + +type testAnalyzer struct { + *ContextData +} + +func (an *testAnalyzer) FallbackSymbol() Symbol { + return Builtin("any") +} + +func (an *testAnalyzer) FindSymbol(name string) Symbol { + return nil +} + +func (an *testAnalyzer) AnalyzeNode(_ context.Context, n SyntaxNode) Symbol { + // TODO: + return Builtin("void") +} + +func (an *testAnalyzer) AnalyzeImport(params ImportParams) ResolvedImport { + // TODO: + + return ResolvedImport{ + Path: "", + } +}