@@ -359,22 +359,35 @@ func (d *Decoder) parseFieldLiteral(n uint8, it indexType) error {
359
359
360
360
var hf HeaderField
361
361
wantStr := d .emitEnabled || it .indexed ()
362
+ var undecodedName undecodedString
362
363
if nameIdx > 0 {
363
364
ihf , ok := d .at (nameIdx )
364
365
if ! ok {
365
366
return DecodingError {InvalidIndexError (nameIdx )}
366
367
}
367
368
hf .Name = ihf .Name
368
369
} else {
369
- hf . Name , buf , err = d .readString (buf , wantStr )
370
+ undecodedName , buf , err = d .readString (buf )
370
371
if err != nil {
371
372
return err
372
373
}
373
374
}
374
- hf . Value , buf , err = d .readString (buf , wantStr )
375
+ undecodedValue , buf , err : = d .readString (buf )
375
376
if err != nil {
376
377
return err
377
378
}
379
+ if wantStr {
380
+ if nameIdx <= 0 {
381
+ hf .Name , err = d .decodeString (undecodedName )
382
+ if err != nil {
383
+ return err
384
+ }
385
+ }
386
+ hf .Value , err = d .decodeString (undecodedValue )
387
+ if err != nil {
388
+ return err
389
+ }
390
+ }
378
391
d .buf = buf
379
392
if it .indexed () {
380
393
d .dynTab .add (hf )
@@ -459,46 +472,52 @@ func readVarInt(n byte, p []byte) (i uint64, remain []byte, err error) {
459
472
return 0 , origP , errNeedMore
460
473
}
461
474
462
- // readString decodes an hpack string from p.
475
+ // readString reads an hpack string from p.
463
476
//
464
- // wantStr is whether s will be used. If false, decompression and
465
- // []byte->string garbage are skipped if s will be ignored
466
- // anyway. This does mean that huffman decoding errors for non-indexed
467
- // strings past the MAX_HEADER_LIST_SIZE are ignored, but the server
468
- // is returning an error anyway, and because they're not indexed, the error
469
- // won't affect the decoding state.
470
- func (d * Decoder ) readString (p []byte , wantStr bool ) (s string , remain []byte , err error ) {
477
+ // It returns a reference to the encoded string data to permit deferring decode costs
478
+ // until after the caller verifies all data is present.
479
+ func (d * Decoder ) readString (p []byte ) (u undecodedString , remain []byte , err error ) {
471
480
if len (p ) == 0 {
472
- return "" , p , errNeedMore
481
+ return u , p , errNeedMore
473
482
}
474
483
isHuff := p [0 ]& 128 != 0
475
484
strLen , p , err := readVarInt (7 , p )
476
485
if err != nil {
477
- return "" , p , err
486
+ return u , p , err
478
487
}
479
488
if d .maxStrLen != 0 && strLen > uint64 (d .maxStrLen ) {
480
- return "" , nil , ErrStringLength
489
+ // Returning an error here means Huffman decoding errors
490
+ // for non-indexed strings past the maximum string length
491
+ // are ignored, but the server is returning an error anyway
492
+ // and because the string is not indexed the error will not
493
+ // affect the decoding state.
494
+ return u , nil , ErrStringLength
481
495
}
482
496
if uint64 (len (p )) < strLen {
483
- return "" , p , errNeedMore
484
- }
485
- if ! isHuff {
486
- if wantStr {
487
- s = string (p [:strLen ])
488
- }
489
- return s , p [strLen :], nil
497
+ return u , p , errNeedMore
490
498
}
499
+ u .isHuff = isHuff
500
+ u .b = p [:strLen ]
501
+ return u , p [strLen :], nil
502
+ }
491
503
492
- if wantStr {
493
- buf := bufPool .Get ().(* bytes.Buffer )
494
- buf .Reset () // don't trust others
495
- defer bufPool .Put (buf )
496
- if err := huffmanDecode (buf , d .maxStrLen , p [:strLen ]); err != nil {
497
- buf .Reset ()
498
- return "" , nil , err
499
- }
504
+ type undecodedString struct {
505
+ isHuff bool
506
+ b []byte
507
+ }
508
+
509
+ func (d * Decoder ) decodeString (u undecodedString ) (string , error ) {
510
+ if ! u .isHuff {
511
+ return string (u .b ), nil
512
+ }
513
+ buf := bufPool .Get ().(* bytes.Buffer )
514
+ buf .Reset () // don't trust others
515
+ var s string
516
+ err := huffmanDecode (buf , d .maxStrLen , u .b )
517
+ if err == nil {
500
518
s = buf .String ()
501
- buf .Reset () // be nice to GC
502
519
}
503
- return s , p [strLen :], nil
520
+ buf .Reset () // be nice to GC
521
+ bufPool .Put (buf )
522
+ return s , err
504
523
}
0 commit comments