-
Notifications
You must be signed in to change notification settings - Fork 18
/
vcf.go
65 lines (58 loc) · 1.6 KB
/
vcf.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
package vcfgo
import (
"fmt"
"strings"
)
// VCFError satisfies the error interface and allows multiple errors.
// This is useful because, for example, on a single line, every sample may have
// a field that doesn't match the description in the header. We want to keep parsing
// but also let the caller know about the error.
type VCFError struct {
Msgs []string
Lines []int64
}
// Error returns a string with all errors delimited by newlines.
func (e *VCFError) Error() string {
var msgs []string
seen := make(map[string]struct{})
for i, m := range e.Msgs {
// remove duplicates
if _, ok := seen[m]; !ok {
seen[m] = struct{}{}
msgs = append(msgs, fmt.Sprintf("%s. [line: %d]", m, e.Lines[i]))
}
}
return strings.Join(msgs, "\n")
}
// NewVCFError allocates the needed ingredients.
func NewVCFError() *VCFError {
e := VCFError{Msgs: make([]string, 0), Lines: make([]int64, 0)}
return &e
}
// Add adds an error and the line number within the vcf where the error took place.
func (e *VCFError) Add(err error, line int64) {
if err != nil {
if ierr := err.Error(); ierr != "" {
if len(e.Msgs) == 5000 {
// only keep at most 5K errors.
m := make([]string, 0, 5000)
l := make([]int64, 0, 5000)
m = append(m, e.Msgs[3000:]...)
l = append(l, e.Lines[3000:]...)
e.Msgs = m
e.Lines = l
}
e.Msgs = append(e.Msgs, ierr)
e.Lines = append(e.Lines, line)
}
}
}
// IsEmpty returns true if there no errors stored.
func (e *VCFError) IsEmpty() bool {
return len(e.Msgs) == 0
}
// Clear empties the Messages
func (e *VCFError) Clear() {
e.Msgs = e.Msgs[:0]
e.Lines = e.Lines[:0]
}