-
Notifications
You must be signed in to change notification settings - Fork 131
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
formatting: Return diff hunks instead of whole file content
- Loading branch information
1 parent
27afcd2
commit e2d489a
Showing
13 changed files
with
1,169 additions
and
80 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
package hcl | ||
|
||
import ( | ||
"github.com/hashicorp/hcl/v2" | ||
"github.com/hashicorp/terraform-ls/internal/filesystem" | ||
"github.com/hashicorp/terraform-ls/internal/source" | ||
"github.com/pmezard/go-difflib/difflib" | ||
) | ||
|
||
type fileChange struct { | ||
newText string | ||
rng hcl.Range | ||
opCode difflib.OpCode | ||
} | ||
|
||
func (ch *fileChange) Text() string { | ||
return ch.newText | ||
} | ||
|
||
func (ch *fileChange) Range() hcl.Range { | ||
return ch.rng | ||
} | ||
|
||
const ( | ||
OpReplace = 'r' | ||
OpDelete = 'd' | ||
OpInsert = 'i' | ||
OpEqual = 'e' | ||
) | ||
|
||
// DiffBytes calculates difference between two byte sequences | ||
// and returns them as filesystem.FileChanges | ||
func DiffLines(filename string, beforeLines, afterLines source.Lines) filesystem.FileChanges { | ||
context := 3 | ||
|
||
m := difflib.NewMatcher( | ||
source.StringLines(beforeLines), | ||
source.StringLines(afterLines)) | ||
|
||
changes := make(filesystem.FileChanges, 0) | ||
|
||
for _, group := range m.GetGroupedOpCodes(context) { | ||
for _, c := range group { | ||
beforeStart, beforeEnd := c.I1, c.I2 | ||
afterStart, afterEnd := c.J1, c.J2 | ||
|
||
if c.Tag == OpEqual { | ||
continue | ||
} | ||
|
||
if c.Tag == OpReplace { | ||
var rng hcl.Range | ||
var newBytes []byte | ||
|
||
for i, line := range beforeLines[beforeStart:beforeEnd] { | ||
if i == 0 { | ||
rng = line.Range() | ||
continue | ||
} | ||
rng.End = line.Range().End | ||
} | ||
|
||
for _, line := range afterLines[afterStart:afterEnd] { | ||
newBytes = append(newBytes, line.Bytes()...) | ||
} | ||
|
||
changes = append(changes, &fileChange{ | ||
newText: string(newBytes), | ||
rng: rng, | ||
}) | ||
continue | ||
} | ||
|
||
if c.Tag == OpDelete { | ||
var deleteRng hcl.Range | ||
for i, line := range beforeLines[beforeStart:beforeEnd] { | ||
if i == 0 { | ||
deleteRng = line.Range() | ||
continue | ||
} | ||
deleteRng.End = line.Range().End | ||
} | ||
changes = append(changes, &fileChange{ | ||
newText: "", | ||
rng: deleteRng, | ||
opCode: c, | ||
}) | ||
continue | ||
} | ||
|
||
if c.Tag == OpInsert { | ||
var insertRng hcl.Range | ||
insertRng.Start = beforeLines[beforeStart-1].Range().End | ||
insertRng.End = beforeLines[beforeStart-1].Range().End | ||
var newBytes []byte | ||
|
||
for _, line := range afterLines[afterStart:afterEnd] { | ||
newBytes = append(newBytes, line.Bytes()...) | ||
} | ||
|
||
changes = append(changes, &fileChange{ | ||
newText: string(newBytes), | ||
rng: insertRng, | ||
}) | ||
continue | ||
} | ||
|
||
} | ||
} | ||
|
||
return changes | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,201 @@ | ||
package hcl | ||
|
||
import ( | ||
"fmt" | ||
"testing" | ||
|
||
"github.com/google/go-cmp/cmp" | ||
"github.com/google/go-cmp/cmp/cmpopts" | ||
"github.com/hashicorp/hcl/v2" | ||
"github.com/hashicorp/terraform-ls/internal/filesystem" | ||
"github.com/hashicorp/terraform-ls/internal/source" | ||
"github.com/pmezard/go-difflib/difflib" | ||
) | ||
|
||
func TestDiff(t *testing.T) { | ||
testCases := []struct { | ||
name string | ||
beforeCfg, afterCfg string | ||
expectedChanges filesystem.FileChanges | ||
}{ | ||
{ | ||
"no-op", | ||
`aaa | ||
bbb | ||
ccc`, | ||
`aaa | ||
bbb | ||
ccc`, | ||
filesystem.FileChanges{}, | ||
}, | ||
{ | ||
"two separate lines replaced", | ||
`resource "aws_vpc" "name" { | ||
cidr_block = "sdf" | ||
tags = { | ||
"key" = "value" | ||
sdfasd = 1 | ||
s = 3 | ||
} | ||
}`, | ||
`resource "aws_vpc" "name" { | ||
cidr_block = "sdf" | ||
tags = { | ||
"key" = "value" | ||
sdfasd = 1 | ||
s = 3 | ||
} | ||
}`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: ` "key" = "value" | ||
`, | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 4, Column: 1, Byte: 60}, | ||
End: hcl.Pos{Line: 5, Column: 1, Byte: 80}, | ||
}, | ||
}, | ||
&fileChange{ | ||
newText: ` s = 3 | ||
`, | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 6, Column: 1, Byte: 95}, | ||
End: hcl.Pos{Line: 7, Column: 1, Byte: 105}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
{ | ||
"whitespace shrinking", | ||
`resource "aws_vpc" "name" { | ||
cidr_block = "sdf" | ||
tags = { | ||
"key" = "value" | ||
sdfasd = 1 | ||
s = 3 | ||
} | ||
}`, | ||
`resource "aws_vpc" "name" { | ||
cidr_block = "sdf" | ||
tags = { | ||
"key" = "value" | ||
sdfasd = 1 | ||
s = 3 | ||
} | ||
}`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: "", | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 7, Column: 1, Byte: 111}, | ||
End: hcl.Pos{Line: 9, Column: 1, Byte: 113}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
{ | ||
"trailing whitespace removal", | ||
`resource "aws_vpc" "name" { | ||
}`, | ||
`resource "aws_vpc" "name" { | ||
}`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: "\n", | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 2, Column: 1, Byte: 28}, | ||
End: hcl.Pos{Line: 3, Column: 1, Byte: 31}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
{ | ||
"new line insertion", | ||
`resource "aws_vpc" "name" {}`, | ||
`resource "aws_vpc" "name" { | ||
}`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: `resource "aws_vpc" "name" { | ||
}`, | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 1, Column: 1, Byte: 0}, | ||
End: hcl.Pos{Line: 1, Column: 29, Byte: 28}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
{ | ||
"new line insertion at EOF", | ||
`resource "aws_vpc" "name" { | ||
} | ||
`, | ||
`resource "aws_vpc" "name" { | ||
} | ||
`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: "\n", | ||
rng: hcl.Range{ | ||
Start: hcl.Pos{Line: 3, Column: 1, Byte: 30}, | ||
End: hcl.Pos{Line: 3, Column: 1, Byte: 30}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
{ | ||
"line insertion", | ||
`resource "aws_vpc" "name" { | ||
attr1 = "one" | ||
attr3 = "three" | ||
}`, | ||
`resource "aws_vpc" "name" { | ||
attr1 = "one" | ||
attr2 = "two" | ||
attr3 = "three" | ||
}`, | ||
filesystem.FileChanges{ | ||
&fileChange{ | ||
newText: ` attr2 = "two" | ||
`, | ||
rng: hcl.Range{ | ||
Filename: "test.tf", | ||
Start: hcl.Pos{Line: 3, Column: 1, Byte: 44}, | ||
End: hcl.Pos{Line: 4, Column: 1, Byte: 45}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
|
||
for i, tc := range testCases { | ||
t.Run(fmt.Sprintf("%d-%s", i, tc.name), func(t *testing.T) { | ||
linesBefore := source.MakeSourceLines("test.tf", | ||
[]byte(tc.beforeCfg)) | ||
linesAfter := source.MakeSourceLines("test.tf", | ||
[]byte(tc.afterCfg)) | ||
|
||
changes := DiffLines("test.tf", linesBefore, linesAfter) | ||
|
||
opts := cmp.Options{ | ||
cmp.AllowUnexported(fileChange{}), | ||
cmpopts.IgnoreTypes(difflib.OpCode{}), | ||
} | ||
|
||
if diff := cmp.Diff(tc.expectedChanges, changes, opts...); diff != "" { | ||
t.Fatalf("Changes don't match: %s", diff) | ||
} | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.