forked from square/grange
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcompress.go
114 lines (99 loc) · 2.39 KB
/
compress.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
package grange
import (
"fmt"
"regexp"
"sort"
"strconv"
"strings"
)
var (
_ = fmt.Println
)
// Normalizes a result set into a minimal range expression, such as
// +{foo,bar}.example.com+.
func Compress(nodes *Result) string {
noDomain := []string{}
domains := map[string][]string{}
for node := range nodes.Iter() {
tokens := strings.SplitN(node.(string), ".", 2)
if len(tokens) == 2 {
domains[tokens[1]] = append(domains[tokens[1]], tokens[0])
} else {
noDomain = append(noDomain, node.(string))
}
}
sort.Strings(noDomain)
result := compressNumeric(noDomain)
var domainKeys = []string{}
for domain, _ := range domains {
domainKeys = append(domainKeys, domain)
}
sort.Strings(domainKeys)
for _, domain := range domainKeys {
domainNodes := domains[domain]
sort.Strings(domainNodes)
domainNodes = compressNumeric(domainNodes)
joined := strings.Join(domainNodes, ",")
if len(domainNodes) > 1 {
joined = "{" + joined + "}"
}
result = append(result, joined+"."+domain)
}
return strings.Join(result, ",")
}
func numericExpansionFor(prefix string, start int, end string, suffix string) string {
endN, _ := strconv.Atoi(end)
if start == endN {
return fmt.Sprintf("%s%s%s", prefix, end, suffix)
} else {
leadingZeros := strings.Repeat("0", len(end)-len(strconv.Itoa(start)))
return fmt.Sprintf("%s%s%d..%d%s", prefix, leadingZeros, start, endN, suffix)
}
}
func compressNumeric(nodes []string) []string {
r := regexp.MustCompile("^(.*?)(\\d+)([^\\d]*)$")
result := []string{}
currentPrefix := ""
currentSuffix := ""
currentNstr := ""
start := -1
currentN := -1
flush := func() {
if start > -1 {
result = append(result, numericExpansionFor(currentPrefix, start, currentNstr, currentSuffix))
start = -1
currentPrefix = ""
currentSuffix = ""
currentN = -1
currentNstr = ""
}
}
for _, node := range nodes {
match := r.FindStringSubmatch(node)
if match == nil {
flush()
result = append(result, node)
} else {
prefix := match[1]
n := match[2]
suffix := match[3]
if prefix != currentPrefix || suffix != currentSuffix {
flush()
}
//if len(n) != len(currentNstr) {
//flush
newN, _ := strconv.Atoi(n)
if start < 0 || newN != currentN+1 {
// first in run
flush()
start = newN
}
currentNstr = n
currentN = newN
currentPrefix = prefix
currentSuffix = suffix
}
}
flush()
return result
}