···27272828// Fragment describes changed lines starting at a specific line in a text file.
2929type Fragment struct {
3030+ Comment string
3131+3032 OldPosition int64
3133 OldLines int64
3234
+35-27
gitdiff/parser.go
···44 "bufio"
55 "fmt"
66 "io"
77- "regexp"
87 "strconv"
98 "strings"
109)
···5453 newFilePrefix = "+++ "
55545655 devNull = "/dev/null"
5757-)
5858-5959-var (
6060- // TODO(bkeyes): are the boundary conditions necessary?
6161- fragmentHeaderRegexp = regexp.MustCompile(`^@@ -(\d+),(\d+) \+(\d+)(?:,(\d+))? @@.*\n`)
6256)
63576458// ParseNextFileHeader finds and parses the next file header in the stream. It
···166160 return len(line) >= len(shortestValidHeader) && strings.HasPrefix(line, fragmentHeaderPrefix)
167161}
168162169169-func parseFragmentHeader(f *Fragment, header string) error {
170170- // TODO(bkeyes): use strings.FieldsFunc instead of regexp
171171- match := fragmentHeaderRegexp.FindStringSubmatch(header)
172172- if len(match) < 5 {
163163+// TODO(bkeyes): fix duplication with isMaybeFragmentHeader
164164+func parseFragmentHeader(f *Fragment, header string) (err error) {
165165+ const startMark = "@@ "
166166+ const endMark = " @@"
167167+168168+ parts := strings.SplitAfterN(header, endMark, 2)
169169+ if len(parts) < 2 || !strings.HasPrefix(parts[0], startMark) || !strings.HasSuffix(parts[0], endMark) {
173170 return fmt.Errorf("invalid fragment header")
174171 }
175172176176- parseInt := func(s string, v *int64) (err error) {
177177- if *v, err = strconv.ParseInt(s, 10, 64); err != nil {
178178- nerr := err.(*strconv.NumError)
179179- return fmt.Errorf("invalid fragment header value: %s: %v", s, nerr.Err)
180180- }
181181- return
173173+ header = parts[0][len(startMark) : len(parts[0])-len(endMark)]
174174+ f.Comment = strings.TrimSpace(parts[1])
175175+176176+ ranges := strings.Split(header, " ")
177177+ if len(ranges) != 2 {
178178+ return fmt.Errorf("invalid fragment header")
182179 }
183180184184- if err := parseInt(match[1], &f.OldPosition); err != nil {
185185- return err
181181+ if !strings.HasPrefix(ranges[0], "-") || !strings.HasPrefix(ranges[1], "+") {
182182+ return fmt.Errorf("invalid fragment header: bad range marker")
186183 }
187187- if err := parseInt(match[2], &f.OldLines); err != nil {
188188- return err
184184+ if f.OldPosition, f.OldLines, err = parseRange(ranges[0][1:]); err != nil {
185185+ return fmt.Errorf("invalid fragment header: %v", err)
186186+ }
187187+ if f.NewPosition, f.NewLines, err = parseRange(ranges[1][1:]); err != nil {
188188+ return fmt.Errorf("invalid fragment header: %v", err)
189189 }
190190+ return nil
191191+}
192192+193193+func parseRange(s string) (start int64, end int64, err error) {
194194+ parts := strings.SplitN(s, ",", 2)
190195191191- if err := parseInt(match[3], &f.NewPosition); err != nil {
192192- return err
196196+ if start, err = strconv.ParseInt(parts[0], 10, 64); err != nil {
197197+ nerr := err.(*strconv.NumError)
198198+ return 0, 0, fmt.Errorf("bad start of range: %s: %v", parts[0], nerr.Err)
193199 }
194200195195- f.NewLines = 1
196196- if match[4] != "" {
197197- if err := parseInt(match[4], &f.NewLines); err != nil {
198198- return err
201201+ if len(parts) > 1 {
202202+ if end, err = strconv.ParseInt(parts[1], 10, 64); err != nil {
203203+ nerr := err.(*strconv.NumError)
204204+ return 0, 0, fmt.Errorf("bad end of range: %s: %v", parts[1], nerr.Err)
199205 }
206206+ } else {
207207+ end = 1
200208 }
201209202202- return nil
210210+ return
203211}