1
2
3
4
5
6 package diff
7
8 import (
9 "fmt"
10 "slices"
11 "sort"
12 "strings"
13 )
14
15
16 type Edit struct {
17 Start, End int
18 New string
19 }
20
21 func (e Edit) String() string {
22 return fmt.Sprintf("{Start:%d,End:%d,New:%q}", e.Start, e.End, e.New)
23 }
24
25
26
27
28
29
30
31 func Apply(src string, edits []Edit) (string, error) {
32 edits, size, err := validate(src, edits)
33 if err != nil {
34 return "", err
35 }
36
37
38 out := make([]byte, 0, size)
39 lastEnd := 0
40 for _, edit := range edits {
41 if lastEnd < edit.Start {
42 out = append(out, src[lastEnd:edit.Start]...)
43 }
44 out = append(out, edit.New...)
45 lastEnd = edit.End
46 }
47 out = append(out, src[lastEnd:]...)
48
49 if len(out) != size {
50 panic("wrong size")
51 }
52
53 return string(out), nil
54 }
55
56
57
58 func ApplyBytes(src []byte, edits []Edit) ([]byte, error) {
59 res, err := Apply(string(src), edits)
60 return []byte(res), err
61 }
62
63
64
65
66 func validate(src string, edits []Edit) ([]Edit, int, error) {
67 if !sort.IsSorted(editsSort(edits)) {
68 edits = slices.Clone(edits)
69 SortEdits(edits)
70 }
71
72
73 size := len(src)
74 lastEnd := 0
75 for _, edit := range edits {
76 if !(0 <= edit.Start && edit.Start <= edit.End && edit.End <= len(src)) {
77 return nil, 0, fmt.Errorf("diff has out-of-bounds edits")
78 }
79 if edit.Start < lastEnd {
80 return nil, 0, fmt.Errorf("diff has overlapping edits")
81 }
82 size += len(edit.New) + edit.Start - edit.End
83 lastEnd = edit.End
84 }
85
86 return edits, size, nil
87 }
88
89
90
91
92
93
94 func SortEdits(edits []Edit) {
95 sort.Stable(editsSort(edits))
96 }
97
98 type editsSort []Edit
99
100 func (a editsSort) Len() int { return len(a) }
101 func (a editsSort) Less(i, j int) bool {
102 if cmp := a[i].Start - a[j].Start; cmp != 0 {
103 return cmp < 0
104 }
105 return a[i].End < a[j].End
106 }
107 func (a editsSort) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
108
109
110
111
112 func lineEdits(src string, edits []Edit) ([]Edit, error) {
113 edits, _, err := validate(src, edits)
114 if err != nil {
115 return nil, err
116 }
117
118
119
120
121 for _, edit := range edits {
122 if edit.Start >= len(src) ||
123 edit.Start > 0 && src[edit.Start-1] != '\n' ||
124 edit.End > 0 && src[edit.End-1] != '\n' ||
125 edit.New != "" && edit.New[len(edit.New)-1] != '\n' {
126 goto expand
127 }
128 }
129 return edits, nil
130
131 expand:
132 if len(edits) == 0 {
133 return edits, nil
134 }
135 expanded := make([]Edit, 0, len(edits))
136 prev := edits[0]
137
138
139 for _, edit := range edits[1:] {
140 between := src[prev.End:edit.Start]
141 if !strings.Contains(between, "\n") {
142
143 prev.New += between + edit.New
144 prev.End = edit.End
145 } else {
146
147 expanded = append(expanded, expandEdit(prev, src))
148 prev = edit
149 }
150 }
151 return append(expanded, expandEdit(prev, src)), nil
152 }
153
154
155 func expandEdit(edit Edit, src string) Edit {
156
157
158 start := edit.Start
159 if delta := start - 1 - strings.LastIndex(src[:start], "\n"); delta > 0 {
160 edit.Start -= delta
161 edit.New = src[start-delta:start] + edit.New
162 }
163
164
165 end := edit.End
166 if end > 0 && src[end-1] != '\n' ||
167 edit.New != "" && edit.New[len(edit.New)-1] != '\n' {
168 if nl := strings.IndexByte(src[end:], '\n'); nl < 0 {
169 edit.End = len(src)
170 } else {
171 edit.End = end + nl + 1
172 }
173 }
174 edit.New += src[end:edit.End]
175
176 return edit
177 }
178
View as plain text