Improved indexing, hover documentation and implemente renaming
This commit is contained in:
58
test/index_cleanup_test.go
Normal file
58
test/index_cleanup_test.go
Normal file
@@ -0,0 +1,58 @@
|
||||
package integration
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/marte-community/marte-dev-tools/internal/index"
|
||||
"github.com/marte-community/marte-dev-tools/internal/parser"
|
||||
)
|
||||
|
||||
func TestIndexCleanup(t *testing.T) {
|
||||
idx := index.NewProjectTree()
|
||||
file := "cleanup.marte"
|
||||
content := `
|
||||
#package Pkg
|
||||
+Node = { Class = Type }
|
||||
`
|
||||
p := parser.NewParser(content)
|
||||
cfg, err := p.Parse()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
idx.AddFile(file, cfg)
|
||||
|
||||
// Check node exists
|
||||
// Root -> Pkg -> Node
|
||||
pkgNode := idx.Root.Children["Pkg"]
|
||||
if pkgNode == nil {
|
||||
t.Fatal("Pkg node should exist")
|
||||
}
|
||||
if pkgNode.Children["Node"] == nil {
|
||||
t.Fatal("Node should exist")
|
||||
}
|
||||
|
||||
// Update file: remove +Node
|
||||
content2 := `
|
||||
#package Pkg
|
||||
// Removed node
|
||||
`
|
||||
p2 := parser.NewParser(content2)
|
||||
cfg2, _ := p2.Parse()
|
||||
idx.AddFile(file, cfg2)
|
||||
|
||||
// Check Node is gone
|
||||
pkgNode = idx.Root.Children["Pkg"]
|
||||
if pkgNode == nil {
|
||||
// Pkg should exist because of #package Pkg
|
||||
t.Fatal("Pkg node should exist after update")
|
||||
}
|
||||
if pkgNode.Children["Node"] != nil {
|
||||
t.Error("Node should be gone")
|
||||
}
|
||||
|
||||
// Test removing file completely
|
||||
idx.RemoveFile(file)
|
||||
if len(idx.Root.Children) != 0 {
|
||||
t.Errorf("Root should be empty after removing file, got %d children", len(idx.Root.Children))
|
||||
}
|
||||
}
|
||||
75
test/lsp_hover_gam_usage_test.go
Normal file
75
test/lsp_hover_gam_usage_test.go
Normal file
@@ -0,0 +1,75 @@
|
||||
package integration
|
||||
|
||||
import (
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/marte-community/marte-dev-tools/internal/index"
|
||||
"github.com/marte-community/marte-dev-tools/internal/lsp"
|
||||
"github.com/marte-community/marte-dev-tools/internal/parser"
|
||||
)
|
||||
|
||||
func TestHoverGAMUsage(t *testing.T) {
|
||||
// Setup
|
||||
lsp.Tree = index.NewProjectTree()
|
||||
lsp.Documents = make(map[string]string)
|
||||
|
||||
content := `
|
||||
+DS1 = {
|
||||
Class = FileReader
|
||||
+Signals = {
|
||||
Sig1 = { Type = uint32 }
|
||||
}
|
||||
}
|
||||
+GAM1 = {
|
||||
Class = IOGAM
|
||||
+InputSignals = {
|
||||
S1 = {
|
||||
DataSource = DS1
|
||||
Alias = Sig1
|
||||
}
|
||||
}
|
||||
}
|
||||
+GAM2 = {
|
||||
Class = IOGAM
|
||||
+OutputSignals = {
|
||||
S2 = {
|
||||
DataSource = DS1
|
||||
Alias = Sig1
|
||||
}
|
||||
}
|
||||
}
|
||||
`
|
||||
uri := "file://test_gam_usage.marte"
|
||||
lsp.Documents[uri] = content
|
||||
p := parser.NewParser(content)
|
||||
cfg, err := p.Parse()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
lsp.Tree.AddFile("test_gam_usage.marte", cfg)
|
||||
lsp.Tree.ResolveReferences()
|
||||
|
||||
// Query hover for Sig1 (Line 5)
|
||||
// Line 4: Sig1... (0-based)
|
||||
params := lsp.HoverParams{
|
||||
TextDocument: lsp.TextDocumentIdentifier{URI: uri},
|
||||
Position: lsp.Position{Line: 4, Character: 9},
|
||||
}
|
||||
|
||||
hover := lsp.HandleHover(params)
|
||||
if hover == nil {
|
||||
t.Fatal("Expected hover")
|
||||
}
|
||||
|
||||
contentHover := hover.Contents.(lsp.MarkupContent).Value
|
||||
if !strings.Contains(contentHover, "**Used in GAMs**") {
|
||||
t.Errorf("Expected 'Used in GAMs' section, got:\n%s", contentHover)
|
||||
}
|
||||
if !strings.Contains(contentHover, "- +GAM1") {
|
||||
t.Error("Expected +GAM1 in usage list")
|
||||
}
|
||||
if !strings.Contains(contentHover, "- +GAM2") {
|
||||
t.Error("Expected +GAM2 in usage list")
|
||||
}
|
||||
}
|
||||
110
test/lsp_rename_signal_test.go
Normal file
110
test/lsp_rename_signal_test.go
Normal file
@@ -0,0 +1,110 @@
|
||||
package integration
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/marte-community/marte-dev-tools/internal/index"
|
||||
"github.com/marte-community/marte-dev-tools/internal/lsp"
|
||||
"github.com/marte-community/marte-dev-tools/internal/parser"
|
||||
"github.com/marte-community/marte-dev-tools/internal/validator"
|
||||
)
|
||||
|
||||
func TestRenameSignalInGAM(t *testing.T) {
|
||||
// Setup
|
||||
lsp.Tree = index.NewProjectTree()
|
||||
lsp.Documents = make(map[string]string)
|
||||
|
||||
content := `
|
||||
+DS = {
|
||||
Class = FileReader
|
||||
+Signals = {
|
||||
Sig1 = { Type = uint32 }
|
||||
}
|
||||
}
|
||||
+GAM = {
|
||||
Class = IOGAM
|
||||
+InputSignals = {
|
||||
// Implicit match
|
||||
Sig1 = { DataSource = DS }
|
||||
// Explicit Alias
|
||||
S2 = { DataSource = DS Alias = Sig1 }
|
||||
}
|
||||
}
|
||||
`
|
||||
uri := "file://rename_sig.marte"
|
||||
lsp.Documents[uri] = content
|
||||
p := parser.NewParser(content)
|
||||
cfg, err := p.Parse()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
lsp.Tree.AddFile("rename_sig.marte", cfg)
|
||||
lsp.Tree.ResolveReferences()
|
||||
|
||||
// Run validator to populate Targets
|
||||
v := validator.NewValidator(lsp.Tree, ".")
|
||||
v.ValidateProject()
|
||||
|
||||
// Rename DS.Sig1 to NewSig
|
||||
// Sig1 is at Line 5.
|
||||
// Line 0: empty
|
||||
// Line 1: +DS
|
||||
// Line 2: Class
|
||||
// Line 3: +Signals
|
||||
// Line 4: Sig1
|
||||
params := lsp.RenameParams{
|
||||
TextDocument: lsp.TextDocumentIdentifier{URI: uri},
|
||||
Position: lsp.Position{Line: 4, Character: 9}, // Sig1
|
||||
NewName: "NewSig",
|
||||
}
|
||||
|
||||
edit := lsp.HandleRename(params)
|
||||
if edit == nil {
|
||||
t.Fatal("Expected edits")
|
||||
}
|
||||
|
||||
edits := edit.Changes[uri]
|
||||
|
||||
// Expect:
|
||||
// 1. Definition of Sig1 in DS (Line 5) -> NewSig
|
||||
// 2. Definition of Sig1 in GAM (Line 10) -> NewSig (Implicit match)
|
||||
// 3. Alias reference in S2 (Line 12) -> NewSig
|
||||
|
||||
// Line 10: Sig1 = ... (0-based 9)
|
||||
// Line 12: S2 = ... Alias = Sig1 (0-based 11)
|
||||
|
||||
expectedCount := 3
|
||||
if len(edits) != expectedCount {
|
||||
t.Errorf("Expected %d edits, got %d", expectedCount, len(edits))
|
||||
for _, e := range edits {
|
||||
t.Logf("Edit: %s at %d", e.NewText, e.Range.Start.Line)
|
||||
}
|
||||
}
|
||||
|
||||
// Check Implicit Signal Rename
|
||||
foundImplicit := false
|
||||
for _, e := range edits {
|
||||
if e.Range.Start.Line == 11 {
|
||||
if e.NewText == "NewSig" {
|
||||
foundImplicit = true
|
||||
}
|
||||
}
|
||||
}
|
||||
if !foundImplicit {
|
||||
t.Error("Did not find implicit signal rename")
|
||||
}
|
||||
|
||||
// Check Alias Rename
|
||||
foundAlias := false
|
||||
for _, e := range edits {
|
||||
if e.Range.Start.Line == 13 {
|
||||
// Alias = Sig1. Range should cover Sig1.
|
||||
if e.NewText == "NewSig" {
|
||||
foundAlias = true
|
||||
}
|
||||
}
|
||||
}
|
||||
if !foundAlias {
|
||||
t.Error("Did not find alias reference rename")
|
||||
}
|
||||
}
|
||||
92
test/lsp_rename_test.go
Normal file
92
test/lsp_rename_test.go
Normal file
@@ -0,0 +1,92 @@
|
||||
package integration
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/marte-community/marte-dev-tools/internal/index"
|
||||
"github.com/marte-community/marte-dev-tools/internal/lsp"
|
||||
"github.com/marte-community/marte-dev-tools/internal/parser"
|
||||
)
|
||||
|
||||
func TestHandleRename(t *testing.T) {
|
||||
// Setup
|
||||
lsp.Tree = index.NewProjectTree()
|
||||
lsp.Documents = make(map[string]string)
|
||||
|
||||
content := `
|
||||
#package Some
|
||||
+MyNode = {
|
||||
Class = Type
|
||||
}
|
||||
+Consumer = {
|
||||
Link = MyNode
|
||||
PkgLink = Some.MyNode
|
||||
}
|
||||
`
|
||||
uri := "file://rename.marte"
|
||||
lsp.Documents[uri] = content
|
||||
p := parser.NewParser(content)
|
||||
cfg, err := p.Parse()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
lsp.Tree.AddFile("rename.marte", cfg)
|
||||
lsp.Tree.ResolveReferences()
|
||||
|
||||
// Rename +MyNode to NewNode
|
||||
// +MyNode is at Line 2 (after #package)
|
||||
// Line 0: empty
|
||||
// Line 1: #package
|
||||
// Line 2: +MyNode
|
||||
params := lsp.RenameParams{
|
||||
TextDocument: lsp.TextDocumentIdentifier{URI: uri},
|
||||
Position: lsp.Position{Line: 2, Character: 4}, // +MyNode
|
||||
NewName: "NewNode",
|
||||
}
|
||||
|
||||
edit := lsp.HandleRename(params)
|
||||
if edit == nil {
|
||||
t.Fatal("Expected edits")
|
||||
}
|
||||
|
||||
edits := edit.Changes[uri]
|
||||
if len(edits) != 3 {
|
||||
t.Errorf("Expected 3 edits (Def, Link, PkgLink), got %d", len(edits))
|
||||
}
|
||||
|
||||
// Verify Definition change (+MyNode -> +NewNode)
|
||||
foundDef := false
|
||||
for _, e := range edits {
|
||||
if e.NewText == "+NewNode" {
|
||||
foundDef = true
|
||||
if e.Range.Start.Line != 2 {
|
||||
t.Errorf("Definition edit line wrong: %d", e.Range.Start.Line)
|
||||
}
|
||||
}
|
||||
}
|
||||
if !foundDef {
|
||||
t.Error("Did not find definition edit +NewNode")
|
||||
}
|
||||
|
||||
// Verify Link change (MyNode -> NewNode)
|
||||
foundLink := false
|
||||
for _, e := range edits {
|
||||
if e.NewText == "NewNode" && e.Range.Start.Line == 6 { // Link = MyNode
|
||||
foundLink = true
|
||||
}
|
||||
}
|
||||
if !foundLink {
|
||||
t.Error("Did not find Link edit")
|
||||
}
|
||||
|
||||
// Verify PkgLink change (Some.MyNode -> Some.NewNode)
|
||||
foundPkg := false
|
||||
for _, e := range edits {
|
||||
if e.NewText == "NewNode" && e.Range.Start.Line == 7 { // PkgLink = Some.MyNode
|
||||
foundPkg = true
|
||||
}
|
||||
}
|
||||
if !foundPkg {
|
||||
t.Error("Did not find PkgLink edit")
|
||||
}
|
||||
}
|
||||
@@ -107,7 +107,11 @@ func TestHierarchicalPackageMerge(t *testing.T) {
|
||||
}
|
||||
|
||||
// We can also inspect the tree to verify FieldX is there (optional, but good for confidence)
|
||||
baseNode := idx.Root.Children["Base"]
|
||||
projNode := idx.Root.Children["Proj"]
|
||||
if projNode == nil {
|
||||
t.Fatal("Proj node not found")
|
||||
}
|
||||
baseNode := projNode.Children["Base"]
|
||||
if baseNode == nil {
|
||||
t.Fatal("Base node not found")
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user