-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathfuzz_test.go
149 lines (143 loc) · 4.95 KB
/
fuzz_test.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
//go:build go1.18
// +build go1.18
package shaping
import (
"reflect"
"testing"
"github.com/go-text/typesetting/di"
"github.com/go-text/typesetting/language"
"github.com/go-text/typesetting/segmenter"
"golang.org/x/image/math/fixed"
)
// FuzzE2E shapes and wraps large strings looking for unshapable text or failures
// in rune accounting.
func FuzzE2E(f *testing.F) {
face := loadOpentypeFont(f, "../font/testdata/Amiri-Regular.ttf")
f.Add(benchParagraphLatin)
f.Add(benchParagraphArabic)
f.Fuzz(func(t *testing.T, input string) {
textInput := []rune(input)
var shaper HarfbuzzShaper
out := []Output{shaper.Shape(Input{
Text: textInput,
RunStart: 0,
RunEnd: len(textInput),
Direction: di.DirectionRTL,
Face: face,
Size: 16 * 72,
Script: language.Arabic,
Language: language.NewLanguage("AR"),
})}
if out[0].Runes.Count != len(textInput) {
t.Errorf("expected %d shaped runes, got %d", len(textInput), out[0].Runes.Count)
}
var l LineWrapper
outs, _ := l.WrapParagraph(WrapConfig{}, 100, textInput, NewSliceIterator(out))
totalRunes := 0
for _, l := range outs {
for _, run := range l {
if run.Runes.Offset != totalRunes {
t.Errorf("expected rune offset %d, got %d", totalRunes, run.Runes.Offset)
}
totalRunes += run.Runes.Count
}
}
if totalRunes != len(textInput) {
t.Errorf("mismatched runes! expected %d, but wrapped output only contains %d", len(textInput), totalRunes)
}
_ = outs
})
}
// FuzzE2EVariableSize shapes and wraps large strings at varying text sizes and line widths.
func FuzzE2EVariableSize(f *testing.F) {
face := loadOpentypeFont(f, "../font/testdata/Amiri-Regular.ttf")
f.Add(benchParagraphLatin, byte(16), byte(100), byte(0))
f.Add(benchParagraphArabic, byte(16), byte(100), byte(0))
f.Fuzz(func(t *testing.T, input string, textSize byte, lineWidth byte, truncateBreakCont byte) {
// We pack the wrapconfig fields into a byte in order to ensure that the fuzzer spends more time
// exploring interesting parts of the space instead of varying bits that don't matter.
wc := WrapConfig{
TruncateAfterLines: int(truncateBreakCont & 0b11111),
TextContinues: truncateBreakCont&0b100000 != 0,
BreakPolicy: LineBreakPolicy((truncateBreakCont & 0b11000000 >> 5) % 3),
}
textInput := []rune(input)
var shaper HarfbuzzShaper
out := []Output{shaper.Shape(Input{
Text: textInput,
RunStart: 0,
RunEnd: len(textInput),
Direction: di.DirectionRTL,
Face: face,
Size: fixed.I(int(textSize)),
Script: language.Arabic,
Language: language.NewLanguage("AR"),
})}
if out[0].Runes.Count != len(textInput) {
t.Errorf("expected %d shaped runes, got %d", len(textInput), out[0].Runes.Count)
}
var l LineWrapper
outs, truncated := l.WrapParagraph(wc, int(lineWidth), textInput, NewSliceIterator(out))
totalRunes := 0
for _, l := range outs {
for _, run := range l {
if run.Runes.Offset != totalRunes {
if reflect.DeepEqual(run, wc.Truncator) {
continue
}
t.Errorf("expected rune offset %d, got %d", totalRunes, run.Runes.Offset)
}
totalRunes += run.Runes.Count
}
}
totalRunes += truncated
if totalRunes != len(textInput) {
t.Errorf("mismatched runes! expected %d, but wrapped output only contains %d", len(textInput), totalRunes)
}
_ = outs
})
}
func FuzzBreakOptions(f *testing.F) {
f.Add(string([]rune{183067, 318808839, 476266048}))
f.Add(benchParagraphArabic)
f.Add(benchParagraphLatin)
f.Fuzz(func(t *testing.T, input string) {
runes := []rune(input)
breaker := newBreaker(&segmenter.Segmenter{}, runes)
var wordOptions []breakOption
for b, ok := breaker.nextWordBreak(); ok; b, ok = breaker.nextWordBreak() {
prevRuneIndex := 0
if len(wordOptions) > 0 {
prevRuneIndex = wordOptions[len(wordOptions)-1].breakAtRune + 1
}
segmentRunes := runes[prevRuneIndex : b.breakAtRune+1]
segmentGraphemes := []breakOption{}
for b, ok := breaker.nextGraphemeBreak(); ok; b, ok = breaker.nextGraphemeBreak() {
// Adjust break offset to be relative to the start of the segment.
b.breakAtRune -= prevRuneIndex
segmentGraphemes = append(segmentGraphemes, b)
}
seg := segmenter.Segmenter{}
seg.Init(segmentRunes)
correctGraphemes := []int{}
count := 0
for iter := seg.GraphemeIterator(); iter.Next(); {
g := iter.Grapheme()
breakAt := g.Offset + len(g.Text) - 1
firstGraphemeInText := prevRuneIndex == 0
if !firstGraphemeInText {
continue
}
count++
correctGraphemes = append(correctGraphemes, breakAt)
}
if count > 0 && len(segmentGraphemes) != count {
t.Errorf("runes[%d:%d] expected %d graphemes, got %d", prevRuneIndex, b.breakAtRune+1, count, len(segmentGraphemes))
t.Errorf("correct graphemes: %v\ngot graphemes: %v", correctGraphemes, segmentGraphemes)
}
checkOptions(t, segmentRunes, segmentGraphemes)
wordOptions = append(wordOptions, b)
}
checkOptions(t, runes, wordOptions)
})
}