/vendor/github.com/docker/distribution/registry/api/v2/headerparser.go

https://github.com/dotcloud/docker · Go · 161 lines · 118 code · 13 blank · 30 comment · 26 complexity · d06f71c8ae98450addb5b577426955f4 MD5 · raw file

  1. package v2
  2. import (
  3. "fmt"
  4. "regexp"
  5. "strings"
  6. "unicode"
  7. )
  8. var (
  9. // according to rfc7230
  10. reToken = regexp.MustCompile(`^[^"(),/:;<=>?@[\]{}[:space:][:cntrl:]]+`)
  11. reQuotedValue = regexp.MustCompile(`^[^\\"]+`)
  12. reEscapedCharacter = regexp.MustCompile(`^[[:blank:][:graph:]]`)
  13. )
  14. // parseForwardedHeader is a benevolent parser of Forwarded header defined in rfc7239. The header contains
  15. // a comma-separated list of forwarding key-value pairs. Each list element is set by single proxy. The
  16. // function parses only the first element of the list, which is set by the very first proxy. It returns a map
  17. // of corresponding key-value pairs and an unparsed slice of the input string.
  18. //
  19. // Examples of Forwarded header values:
  20. //
  21. // 1. Forwarded: For=192.0.2.43; Proto=https,For="[2001:db8:cafe::17]",For=unknown
  22. // 2. Forwarded: for="192.0.2.43:443"; host="registry.example.org", for="10.10.05.40:80"
  23. //
  24. // The first will be parsed into {"for": "192.0.2.43", "proto": "https"} while the second into
  25. // {"for": "192.0.2.43:443", "host": "registry.example.org"}.
  26. func parseForwardedHeader(forwarded string) (map[string]string, string, error) {
  27. // Following are states of forwarded header parser. Any state could transition to a failure.
  28. const (
  29. // terminating state; can transition to Parameter
  30. stateElement = iota
  31. // terminating state; can transition to KeyValueDelimiter
  32. stateParameter
  33. // can transition to Value
  34. stateKeyValueDelimiter
  35. // can transition to one of { QuotedValue, PairEnd }
  36. stateValue
  37. // can transition to one of { EscapedCharacter, PairEnd }
  38. stateQuotedValue
  39. // can transition to one of { QuotedValue }
  40. stateEscapedCharacter
  41. // terminating state; can transition to one of { Parameter, Element }
  42. statePairEnd
  43. )
  44. var (
  45. parameter string
  46. value string
  47. parse = forwarded[:]
  48. res = map[string]string{}
  49. state = stateElement
  50. )
  51. Loop:
  52. for {
  53. // skip spaces unless in quoted value
  54. if state != stateQuotedValue && state != stateEscapedCharacter {
  55. parse = strings.TrimLeftFunc(parse, unicode.IsSpace)
  56. }
  57. if len(parse) == 0 {
  58. if state != stateElement && state != statePairEnd && state != stateParameter {
  59. return nil, parse, fmt.Errorf("unexpected end of input")
  60. }
  61. // terminating
  62. break
  63. }
  64. switch state {
  65. // terminate at list element delimiter
  66. case stateElement:
  67. if parse[0] == ',' {
  68. parse = parse[1:]
  69. break Loop
  70. }
  71. state = stateParameter
  72. // parse parameter (the key of key-value pair)
  73. case stateParameter:
  74. match := reToken.FindString(parse)
  75. if len(match) == 0 {
  76. return nil, parse, fmt.Errorf("failed to parse token at position %d", len(forwarded)-len(parse))
  77. }
  78. parameter = strings.ToLower(match)
  79. parse = parse[len(match):]
  80. state = stateKeyValueDelimiter
  81. // parse '='
  82. case stateKeyValueDelimiter:
  83. if parse[0] != '=' {
  84. return nil, parse, fmt.Errorf("expected '=', not '%c' at position %d", parse[0], len(forwarded)-len(parse))
  85. }
  86. parse = parse[1:]
  87. state = stateValue
  88. // parse value or quoted value
  89. case stateValue:
  90. if parse[0] == '"' {
  91. parse = parse[1:]
  92. state = stateQuotedValue
  93. } else {
  94. value = reToken.FindString(parse)
  95. if len(value) == 0 {
  96. return nil, parse, fmt.Errorf("failed to parse value at position %d", len(forwarded)-len(parse))
  97. }
  98. if _, exists := res[parameter]; exists {
  99. return nil, parse, fmt.Errorf("duplicate parameter %q at position %d", parameter, len(forwarded)-len(parse))
  100. }
  101. res[parameter] = value
  102. parse = parse[len(value):]
  103. value = ""
  104. state = statePairEnd
  105. }
  106. // parse a part of quoted value until the first backslash
  107. case stateQuotedValue:
  108. match := reQuotedValue.FindString(parse)
  109. value += match
  110. parse = parse[len(match):]
  111. switch {
  112. case len(parse) == 0:
  113. return nil, parse, fmt.Errorf("unterminated quoted string")
  114. case parse[0] == '"':
  115. res[parameter] = value
  116. value = ""
  117. parse = parse[1:]
  118. state = statePairEnd
  119. case parse[0] == '\\':
  120. parse = parse[1:]
  121. state = stateEscapedCharacter
  122. }
  123. // parse escaped character in a quoted string, ignore the backslash
  124. // transition back to QuotedValue state
  125. case stateEscapedCharacter:
  126. c := reEscapedCharacter.FindString(parse)
  127. if len(c) == 0 {
  128. return nil, parse, fmt.Errorf("invalid escape sequence at position %d", len(forwarded)-len(parse)-1)
  129. }
  130. value += c
  131. parse = parse[1:]
  132. state = stateQuotedValue
  133. // expect either a new key-value pair, new list or end of input
  134. case statePairEnd:
  135. switch parse[0] {
  136. case ';':
  137. parse = parse[1:]
  138. state = stateParameter
  139. case ',':
  140. state = stateElement
  141. default:
  142. return nil, parse, fmt.Errorf("expected ',' or ';', not %c at position %d", parse[0], len(forwarded)-len(parse))
  143. }
  144. }
  145. }
  146. return res, parse, nil
  147. }