diff --git a/services/gitdiff/gitdiff.go b/services/gitdiff/gitdiff.go
index 91105399db..164e1c0ca5 100644
--- a/services/gitdiff/gitdiff.go
+++ b/services/gitdiff/gitdiff.go
@@ -181,64 +181,60 @@ var (
removedCodePrefix = []byte(``)
codeTagSuffix = []byte(``)
)
-var addSpanRegex = regexp.MustCompile(`]?$`)
+
+// shouldWriteInline represents combinations where we manually write inline changes
+func shouldWriteInline(diff diffmatchpatch.Diff, lineType DiffLineType) bool {
+ if true &&
+ diff.Type == diffmatchpatch.DiffEqual ||
+ diff.Type == diffmatchpatch.DiffInsert && lineType == DiffLineAdd ||
+ diff.Type == diffmatchpatch.DiffDelete && lineType == DiffLineDel {
+ return true
+ }
+ return false
+}
func diffToHTML(fileName string, diffs []diffmatchpatch.Diff, lineType DiffLineType) template.HTML {
buf := bytes.NewBuffer(nil)
- var addSpan string
- for i := range diffs {
+ match := ""
+
+ for _, diff := range diffs {
+ if shouldWriteInline(diff, lineType) {
+ if len(match) > 0 {
+ diff.Text = match + diff.Text
+ match = ""
+ }
+ // Chroma HTML syntax highlighting is done before diffing individual lines in order to maintain consistency.
+ // Since inline changes might split in the middle of a chroma span tag, make we manually put it back together
+ // before writing so we don't try insert added/removed code spans in the middle of an existing chroma span
+ // and create broken HTML.
+ m := trailingSpanRegex.FindStringSubmatchIndex(diff.Text)
+ if m != nil {
+ match = diff.Text[m[0]:m[1]]
+ diff.Text = strings.TrimSuffix(diff.Text, match)
+ }
+ // Print an existing closing span first before opening added/remove-code span so it doesn't unintentionally close it
+ if strings.HasPrefix(diff.Text, "") {
+ buf.WriteString("")
+ diff.Text = strings.TrimPrefix(diff.Text, "")
+ }
+ // If we weren't able to fix it then this should avoid broken HTML by not inserting more spans below
+ // The previous/next diff section will contain the rest of the tag that is missing here
+ if strings.Count(diff.Text, "<") != strings.Count(diff.Text, ">") {
+ buf.WriteString(diff.Text)
+ continue
+ }
+ }
switch {
- case diffs[i].Type == diffmatchpatch.DiffEqual:
- // Looking for the case where our 3rd party diff library previously detected a string difference
- // in the middle of a span class because we highlight them first. This happens when added/deleted code
- // also changes the chroma class name, either partially or fully. If found, just move the openining span code forward into the next section
- // see TestDiffToHTML for examples
- if len(addSpan) > 0 {
- diffs[i].Text = addSpan + diffs[i].Text
- addSpan = ""
- }
- m := addSpanRegex.FindStringSubmatchIndex(diffs[i].Text)
- if m != nil {
- addSpan = diffs[i].Text[m[0]:m[1]]
- buf.WriteString(strings.TrimSuffix(diffs[i].Text, addSpan))
- } else {
- addSpan = ""
- buf.WriteString(getLineContent(diffs[i].Text))
- }
- case diffs[i].Type == diffmatchpatch.DiffInsert && lineType == DiffLineAdd:
- if len(addSpan) > 0 {
- diffs[i].Text = addSpan + diffs[i].Text
- addSpan = ""
- }
- // Print existing closing span first before opening added-code span so it doesn't unintentionally close it
- if strings.HasPrefix(diffs[i].Text, "") {
- buf.WriteString("")
- diffs[i].Text = strings.TrimPrefix(diffs[i].Text, "")
- }
- m := addSpanRegex.FindStringSubmatchIndex(diffs[i].Text)
- if m != nil {
- addSpan = diffs[i].Text[m[0]:m[1]]
- diffs[i].Text = strings.TrimSuffix(diffs[i].Text, addSpan)
- }
+ case diff.Type == diffmatchpatch.DiffEqual:
+ buf.WriteString(diff.Text)
+ case diff.Type == diffmatchpatch.DiffInsert && lineType == DiffLineAdd:
buf.Write(addedCodePrefix)
- buf.WriteString(getLineContent(diffs[i].Text))
+ buf.WriteString(diff.Text)
buf.Write(codeTagSuffix)
- case diffs[i].Type == diffmatchpatch.DiffDelete && lineType == DiffLineDel:
- if len(addSpan) > 0 {
- diffs[i].Text = addSpan + diffs[i].Text
- addSpan = ""
- }
- if strings.HasPrefix(diffs[i].Text, "") {
- buf.WriteString("")
- diffs[i].Text = strings.TrimPrefix(diffs[i].Text, "")
- }
- m := addSpanRegex.FindStringSubmatchIndex(diffs[i].Text)
- if m != nil {
- addSpan = diffs[i].Text[m[0]:m[1]]
- diffs[i].Text = strings.TrimSuffix(diffs[i].Text, addSpan)
- }
+ case diff.Type == diffmatchpatch.DiffDelete && lineType == DiffLineDel:
buf.Write(removedCodePrefix)
- buf.WriteString(getLineContent(diffs[i].Text))
+ buf.WriteString(diff.Text)
buf.Write(codeTagSuffix)
}
}
diff --git a/services/gitdiff/gitdiff_test.go b/services/gitdiff/gitdiff_test.go
index 64cd4f1c21..e7eeca7004 100644
--- a/services/gitdiff/gitdiff_test.go
+++ b/services/gitdiff/gitdiff_test.go
@@ -50,7 +50,7 @@ func TestDiffToHTML(t *testing.T) {
{Type: dmp.DiffEqual, Text: " {"},
}, DiffLineAdd))
- assertEqual(t, "tagURL := fmt.Sprintf("## [%s](%s/%s/%s/%s?q=&type=all&state=closed&milestone=%d) - %s", ge.Milestone\", ge.BaseURL, ge.Owner, ge.Repo, from, milestoneID, time.Now().Format("2006-01-02"))", diffToHTML("", []dmp.Diff{
+ assertEqual(t, "tagURL := fmt.Sprintf("## [%s](%s/%s/%s/%s?q=&type=all&state=closed&milestone=%d) - %s", ge.Milestone\", ge.BaseURL, ge.Owner, ge.Repo, from, milestoneID, time.Now().Format("2006-01-02"))", diffToHTML("", []dmp.Diff{
{Type: dmp.DiffEqual, Text: "tagURL := fmt.Sprintf("## [%s](%s/%s/%s/%s?q=&type=all&state=closed&milestone=%d) - %s", ge.Milestone\""},
{Type: dmp.DiffInsert, Text: "f\">getGiteaTagURL(client"},
@@ -60,7 +60,7 @@ func TestDiffToHTML(t *testing.T) {
{Type: dmp.DiffEqual, Text: ")"},
}, DiffLineDel))
- assertEqual(t, "r.WrapperRenderer(w, language, true, attrs, false)", diffToHTML("", []dmp.Diff{
+ assertEqual(t, "r.WrapperRenderer(w, language, true, attrs, false)", diffToHTML("", []dmp.Diff{
{Type: dmp.DiffEqual, Text: "r.WrapperRenderer(w, "},
{Type: dmp.DiffDelete, Text: "language, true, attrs"},
{Type: dmp.DiffEqual, Text: ", false)"},
}, DiffLineAdd))
+
+ assertEqual(t, "print("// ", sys.argv)", diffToHTML("", []dmp.Diff{
+ {Type: dmp.DiffEqual, Text: "print"},
+ {Type: dmp.DiffInsert, Text: "("},
+ {Type: dmp.DiffEqual, Text: ""// ", sys.argv"},
+ {Type: dmp.DiffInsert, Text: ")"},
+ }, DiffLineAdd))
}
func TestParsePatch_singlefile(t *testing.T) {