forked from hound-search/hound
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcoalesce.go
99 lines (80 loc) · 1.71 KB
/
coalesce.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
package client
import (
"github.com/gitgrep-com/gitgrep/index"
)
type Block struct {
Lines []string
Matches []bool
Start int
}
func endOfBlock(b *Block) int {
return b.Start + len(b.Lines) - 1
}
func startOfMatch(m *index.Match) int {
return m.LineNumber - len(m.Before)
}
func matchIsInBlock(m *index.Match, b *Block) bool {
return startOfMatch(m) <= endOfBlock(b)
}
func matchToBlock(m *index.Match) *Block {
b, a := len(m.Before), len(m.After)
n := 1 + b + a
l := make([]string, 0, n)
v := make([]bool, n)
v[b] = true
for _, line := range m.Before {
l = append(l, line)
}
l = append(l, m.Line)
for _, line := range m.After {
l = append(l, line)
}
return &Block{
Lines: l,
Matches: v,
Start: m.LineNumber - len(m.Before),
}
}
func clampZero(n int) int {
if n < 0 {
return 0
}
return n
}
func mergeMatchIntoBlock(m *index.Match, b *Block) {
off := endOfBlock(b) - startOfMatch(m) + 1
idx := len(b.Lines) - off
nb := len(m.Before)
for i := off; i < nb; i++ {
b.Lines = append(b.Lines, m.Before[i])
b.Matches = append(b.Matches, false)
}
if off < nb+1 {
b.Lines = append(b.Lines, m.Line)
b.Matches = append(b.Matches, true)
} else {
b.Matches[idx+nb] = true
}
for i, n := clampZero(off-nb-1), len(m.After); i < n; i++ {
b.Lines = append(b.Lines, m.After[i])
b.Matches = append(b.Matches, false)
}
}
func coalesceMatches(matches []*index.Match) []*Block {
var res []*Block
var curr *Block
for _, match := range matches {
if curr != nil && matchIsInBlock(match, curr) {
mergeMatchIntoBlock(match, curr)
} else {
if curr != nil {
res = append(res, curr)
}
curr = matchToBlock(match)
}
}
if curr != nil {
res = append(res, curr)
}
return res
}