-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathreplace_lines.go
236 lines (225 loc) · 6.11 KB
/
replace_lines.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
// -----------------------------------------------------------------------------
// CMDX Utilities Suite cmdx/[replace_lines.go]
// (c) balarabe@protonmail.com License: GPLv3
// -----------------------------------------------------------------------------
package main
import (
"sort"
"strings"
"github.com/balacode/zr"
)
// replaceLines(
// lines []string,
// finds [][]string,
// repls [][]string,
// caseMode zr.CaseMode,
// ) (
// changedLines []string,
// changes int,
// )
//
// # Subfunctions
// (M replaceLinesM) getBatches(
// finds, repls []Lines,
// ) (
// []int, map[int]*FindReplLinesBatch,
// )
// (M replaceLinesM) getTree(
// finds, repls []Lines,
// caseMode zr.CaseMode,
// ) (
// ret FindReplLinesTree,
// )
// (M replaceLinesM) replaceMany(
// lines []string,
// finds []Lines,
// repls []Lines,
// caseMode zr.CaseMode,
// ) (
// changedLines Lines,
// changes int,
// )
// replaceLinesM joins all subfunctions used by replaceLines(),
// so that their names don't clutter the project's namespace.
type replaceLinesM struct{}
// replaceLines replaces multiple blocks of lines.
func replaceLines(
lines []string,
finds [][]string,
repls [][]string,
caseMode zr.CaseMode,
) (
changedLines []string,
changes int,
) {
if len(finds) == 0 {
return lines, 0 // avoid allocation
}
if DebugReplaceLines {
zr.DV("--------------------------------------------------")
zr.DV("replaceLines() args:")
zr.DV("lines", lines)
zr.DV("finds", finds)
zr.DV("repls", repls)
zr.DV("caseMode", caseMode)
zr.DV("--------------------------------------------------")
}
// validate arguments
if len(finds) != len(repls) {
zr.Error(zr.EInvalidArg, ": lineCounts don't match:",
len(finds), "and", len(repls))
return Lines{}, 0 // erv
}
if caseMode != zr.IgnoreCase && caseMode != zr.MatchCase {
zr.Error(zr.EInvalidArg,
"^caseMode", ":", caseMode, "defaulting to 'MatchCase'")
caseMode = zr.MatchCase
}
// copy [][]string to []Lines. Lis there a way to cast?
var (
findLines = make([]Lines, len(finds))
replLines = make([]Lines, len(repls))
)
for i, find := range finds {
findLines = append(findLines, find)
replLines = append(replLines, repls[i])
}
// make replacements by batches with largest number of lines first
var M replaceLinesM
descLineCounts, batches := M.getBatches(findLines, replLines)
for _, batchSize := range descLineCounts {
b := batches[batchSize]
n := 0
lines, n = M.replaceMany(lines, b.FindLines, b.ReplLines, caseMode)
changes += n
}
if DebugReplaceLines {
zr.DV("--------------------------------------------------")
zr.DV("replaceLines() returns:")
zr.DV("lines", lines)
zr.DV("changes", changes)
zr.DV("--------------------------------------------------")
}
return lines, changes
}
// -----------------------------------------------------------------------------
// # Subfunctions
// getBatches _ _
func (M replaceLinesM) getBatches(
finds, repls []Lines,
) (
[]int, map[int]*FindReplLinesBatch,
) {
var lineCounts []int
batches := map[int]*FindReplLinesBatch{}
for i, find := range finds {
n := len(find)
_, has := batches[n]
if !has {
batches[n] = &FindReplLinesBatch{}
lineCounts = append(lineCounts, n)
}
b := batches[n]
b.FindLines = append(b.FindLines, find)
b.ReplLines = append(b.ReplLines, repls[i])
}
sort.Sort(sort.Reverse(sort.IntSlice(lineCounts)))
return lineCounts, batches
}
// getTree builds a tree from a slice of strings to find and to replace.
// The branches of the tree are made from each line in batches of repls.
func (M replaceLinesM) getTree(
finds, repls []Lines,
caseMode zr.CaseMode,
) (
ret FindReplLinesTree,
) {
ret.Sub = make(map[string]*FindReplLinesTree)
for f, find := range finds {
node := &ret
last := len(find) - 1
for i, line := range find {
line = strings.TrimSpace(line)
if caseMode == zr.IgnoreCase {
line = strings.ToLower(line)
}
_, exist := node.Sub[line]
var sub *FindReplLinesTree
if exist {
sub = node.Sub[line]
} else {
sub = &FindReplLinesTree{
Sub: make(map[string]*FindReplLinesTree),
}
}
if i == last {
sub.FindLines = find
sub.ReplLines = repls[f]
}
node.Sub[line] = sub
node = sub
}
}
return ret
}
// replaceMany _ _
func (M replaceLinesM) replaceMany(
lines []string,
finds []Lines,
repls []Lines,
caseMode zr.CaseMode,
) (
changedLines Lines,
changes int,
) {
var (
linesLen = len(lines)
root = M.getTree(finds, repls, caseMode)
node = &root // *tree pointing to current branch
match = 0 // <- number of matching characters
prev = 0
)
var ret []string
for i := 0; i < linesLen; i++ {
line := strings.TrimSpace(lines[i])
if caseMode == zr.IgnoreCase {
line = strings.ToLower(line)
}
// check if the tree's branch has a key matching the current line
// if not, reset matching count and start over from root
{
sub, found := node.Sub[line]
if !found {
node = &root
i -= match
match = 0
continue
}
match++
node = sub
}
// even if found, keep matching if there are more matches to make.
// when node.FindLines is not empty, it's a leaf node, then proceed
findLen := len(node.FindLines)
if findLen == 0 || findLen != match {
continue
}
// replacement starts:
changes++
// concatenate preceding lines that remain unchanged
if prev < i-findLen+1 {
ret = append(ret, lines[prev:i-findLen+1]...)
}
// append the replacement lines
ret = append(ret, node.ReplLines...)
prev = i + 1
node = &root // restart matching from beginning
match = 0
}
// write any remaining unchanged lines
if prev < linesLen {
ret = append(ret, lines[prev:]...)
}
return ret, changes
}
// end