Commit 3748d226 authored by Rob Pike's avatar Rob Pike

fmt.Scanf: improve error message when input does not match format

R=rsc
CC=golang-dev
https://golang.org/cl/1693043
parent 4fd7880d
...@@ -885,6 +885,7 @@ func (s *ss) doScan(a []interface{}) (numProcessed int, err os.Error) { ...@@ -885,6 +885,7 @@ func (s *ss) doScan(a []interface{}) (numProcessed int, err os.Error) {
// either input or format behave as a single space. This routine also // either input or format behave as a single space. This routine also
// handles the %% case. If the return value is zero, either format // handles the %% case. If the return value is zero, either format
// starts with a % (with no following %) or the input is empty. // starts with a % (with no following %) or the input is empty.
// If it is negative, the input did not match the string.
func (s *ss) advance(format string) (i int) { func (s *ss) advance(format string) (i int) {
for i < len(format) { for i < len(format) {
fmtc, w := utf8.DecodeRuneInString(format[i:]) fmtc, w := utf8.DecodeRuneInString(format[i:])
...@@ -919,7 +920,7 @@ func (s *ss) advance(format string) (i int) { ...@@ -919,7 +920,7 @@ func (s *ss) advance(format string) (i int) {
inputc := s.mustGetRune() inputc := s.mustGetRune()
if fmtc != inputc { if fmtc != inputc {
s.UngetRune(inputc) s.UngetRune(inputc)
return return -1
} }
i += w i += w
} }
...@@ -940,10 +941,11 @@ func (s *ss) doScanf(format string, a []interface{}) (numProcessed int, err os.E ...@@ -940,10 +941,11 @@ func (s *ss) doScanf(format string, a []interface{}) (numProcessed int, err os.E
} }
// Either we failed to advance, we have a percent character, or we ran out of input. // Either we failed to advance, we have a percent character, or we ran out of input.
if format[i] != '%' { if format[i] != '%' {
// Can't advance format. Do we have arguments still to process? // Can't advance format. Why not?
if i < len(a) { if w < 0 {
s.errorString("too many arguments for format") s.errorString("input does not match format")
} }
// Otherwise at EOF; "too many operands" error handled below
break break
} }
i++ // % is one byte i++ // % is one byte
......
...@@ -303,6 +303,7 @@ var multiTests = []ScanfMultiTest{ ...@@ -303,6 +303,7 @@ var multiTests = []ScanfMultiTest{
ScanfMultiTest{"%d %d %d", "23 18", args(&i, &j), args(23, 18), "too few operands"}, ScanfMultiTest{"%d %d %d", "23 18", args(&i, &j), args(23, 18), "too few operands"},
ScanfMultiTest{"%d %d", "23 18 27", args(&i, &j, &k), args(23, 18), "too many operands"}, ScanfMultiTest{"%d %d", "23 18 27", args(&i, &j, &k), args(23, 18), "too many operands"},
ScanfMultiTest{"%c", "\u0100", args(&int8Val), nil, "overflow"}, ScanfMultiTest{"%c", "\u0100", args(&int8Val), nil, "overflow"},
ScanfMultiTest{"X%d", "10X", args(&intVal), nil, "input does not match format"},
// Bad UTF-8: should see every byte. // Bad UTF-8: should see every byte.
ScanfMultiTest{"%c%c%c", "\xc2X\xc2", args(&i, &j, &k), args(utf8.RuneError, 'X', utf8.RuneError), ""}, ScanfMultiTest{"%c%c%c", "\xc2X\xc2", args(&i, &j, &k), args(utf8.RuneError, 'X', utf8.RuneError), ""},
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment