<!DOCTYPE html>
<html lang="en">
	<head>
		<title>httpencoding - encoding.go</title>
		<link type="text/css" rel="stylesheet" href="/style/src.css">
	</head>
	<body>
		<h1><a href="/httpencoding">httpencoding</a> - encoding.go</h1>
		<pre>
<span class="hidden"><a id="L1" href="#L1">     1</a>  </span><span class="text">// Package httpencoding provides a function to deal with the Accept-Encoding</span>
<span class="hidden"><a id="L2" href="#L2">     2</a>  </span><span class="text">// header.</span>
<span class="hidden"><a id="L3" href="#L3">     3</a>  </span><span>package httpencoding </span><span class="text">// import &#34;vimagination.zapto.org/httpencoding&#34;</span>
<span class="hidden"><a id="L4" href="#L4">     4</a>  </span>
<span class="hidden"><a id="L5" href="#L5">     5</a>  </span><span>import (</span>
<span class="hidden"><a id="L6" href="#L6">     6</a>  </span><span>	&#34;net</span><span>/http&#34;</span>
<span class="hidden"><a id="L7" href="#L7">     7</a>  </span><span>	&#34;slices&#34;</span>
<span class="hidden"><a id="L8" href="#L8">     8</a>  </span><span>	&#34;sort&#34;</span>
<span class="hidden"><a id="L9" href="#L9">     9</a>  </span><span>	&#34;strings&#34;</span>
<span class="hidden"><a id="L10" href="#L10">    10</a>  </span>
<span class="hidden"><a id="L11" href="#L11">    11</a>  </span><span>	&#34;vimagination.zapto.org</span><span>/parser&#34;</span>
<span class="hidden"><a id="L12" href="#L12">    12</a>  </span><span>)</span>
<span class="hidden"><a id="L13" href="#L13">    13</a>  </span>
<span class="hidden"><a id="L14" href="#L14">    14</a>  </span><span>const (</span>
<span class="hidden"><a id="L15" href="#L15">    15</a>  </span><span>	acceptEncoding   = &#34;Accept-Encoding&#34;</span>
<span class="hidden"><a id="L16" href="#L16">    16</a>  </span><span>	anyEncoding      = &#34;*&#34;</span>
<span class="hidden"><a id="L17" href="#L17">    17</a>  </span><span>	identityEncoding = &#34;identity&#34;</span>
<span class="hidden"><a id="L18" href="#L18">    18</a>  </span><span>	acceptSplit      = &#34;,&#34;</span>
<span class="hidden"><a id="L19" href="#L19">    19</a>  </span><span>	partSplit        = &#39;;&#39;</span>
<span class="hidden"><a id="L20" href="#L20">    20</a>  </span><span>	weightPrefix     = &#34;q=&#34;</span>
<span class="hidden"><a id="L21" href="#L21">    21</a>  </span><span>)</span>
<span class="hidden"><a id="L22" href="#L22">    22</a>  </span>
<span class="hidden"><a id="L23" href="#L23">    23</a>  </span><span>type encodings []encoding</span>
<span class="hidden"><a id="L24" href="#L24">    24</a>  </span>
<span class="hidden"><a id="L25" href="#L25">    25</a>  </span><span>func (e encodings) Len() int {</span>
<span class="hidden"><a id="L26" href="#L26">    26</a>  </span><span>	return len(e)</span>
<span class="hidden"><a id="L27" href="#L27">    27</a>  </span><span>}</span>
<span class="hidden"><a id="L28" href="#L28">    28</a>  </span>
<span class="hidden"><a id="L29" href="#L29">    29</a>  </span><span>func (e encodings) Less(i, j int) bool {</span>
<span class="hidden"><a id="L30" href="#L30">    30</a>  </span><span>	return e[j].weight &lt; e[i].weight</span>
<span class="hidden"><a id="L31" href="#L31">    31</a>  </span><span>}</span>
<span class="hidden"><a id="L32" href="#L32">    32</a>  </span>
<span class="hidden"><a id="L33" href="#L33">    33</a>  </span><span>func (e encodings) Swap(i, j int) {</span>
<span class="hidden"><a id="L34" href="#L34">    34</a>  </span><span>	e[i], e[j] = e[j], e[i]</span>
<span class="hidden"><a id="L35" href="#L35">    35</a>  </span><span>}</span>
<span class="hidden"><a id="L36" href="#L36">    36</a>  </span>
<span class="hidden"><a id="L37" href="#L37">    37</a>  </span><span>type encoding struct {</span>
<span class="hidden"><a id="L38" href="#L38">    38</a>  </span><span>	encoding Encoding</span>
<span class="hidden"><a id="L39" href="#L39">    39</a>  </span><span>	weight   int16</span>
<span class="hidden"><a id="L40" href="#L40">    40</a>  </span><span>}</span>
<span class="hidden"><a id="L41" href="#L41">    41</a>  </span>
<span class="hidden"><a id="L42" href="#L42">    42</a>  </span><span class="text">// Encoding represents an encoding string as used by the client. Examples are</span>
<span class="hidden"><a id="L43" href="#L43">    43</a>  </span><span class="text">// gzip, br and deflate.</span>
<span class="hidden"><a id="L44" href="#L44">    44</a>  </span><span>type Encoding string</span>
<span class="hidden"><a id="L45" href="#L45">    45</a>  </span>
<span class="hidden"><a id="L46" href="#L46">    46</a>  </span><span class="text">// Handler provides an interface to handle an encoding.</span>
<span class="hidden"><a id="L47" href="#L47">    47</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L48" href="#L48">    48</a>  </span><span class="text">// The encoding string (e.g. gzip, br, deflate) is passed to the handler, which</span>
<span class="hidden"><a id="L49" href="#L49">    49</a>  </span><span class="text">// is expected to return true if no more encodings are required and false</span>
<span class="hidden"><a id="L50" href="#L50">    50</a>  </span><span class="text">// otherwise.</span>
<span class="hidden"><a id="L51" href="#L51">    51</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L52" href="#L52">    52</a>  </span><span class="text">// The empty string &#34;&#34; is used to signify the identity encoding, or plain text.</span>
<span class="hidden"><a id="L53" href="#L53">    53</a>  </span><span>type Handler interface {</span>
<span class="hidden"><a id="L54" href="#L54">    54</a>  </span><span>	Handle(encoding Encoding) bool</span>
<span class="hidden"><a id="L55" href="#L55">    55</a>  </span><span>}</span>
<span class="hidden"><a id="L56" href="#L56">    56</a>  </span>
<span class="hidden"><a id="L57" href="#L57">    57</a>  </span><span class="text">// HandlerFunc wraps a func to make it satisfy the Handler interface.</span>
<span class="hidden"><a id="L58" href="#L58">    58</a>  </span><span>type HandlerFunc func(Encoding) bool</span>
<span class="hidden"><a id="L59" href="#L59">    59</a>  </span>
<span class="hidden"><a id="L60" href="#L60">    60</a>  </span><span class="text">// Handle calls the underlying func.</span>
<span class="hidden"><a id="L61" href="#L61">    61</a>  </span><span>func (h HandlerFunc) Handle(e Encoding) bool {</span>
<span class="hidden"><a id="L62" href="#L62">    62</a>  </span><span>	return h(e)</span>
<span class="hidden"><a id="L63" href="#L63">    63</a>  </span><span>}</span>
<span class="hidden"><a id="L64" href="#L64">    64</a>  </span>
<span class="hidden"><a id="L65" href="#L65">    65</a>  </span><span class="text">// InvalidEncoding writes the 406 header.</span>
<span class="hidden"><a id="L66" href="#L66">    66</a>  </span><span>func InvalidEncoding(w http.ResponseWriter) {</span>
<span class="hidden"><a id="L67" href="#L67">    67</a>  </span><span>	w.WriteHeader(http.StatusNotAcceptable)</span>
<span class="hidden"><a id="L68" href="#L68">    68</a>  </span><span>}</span>
<span class="hidden"><a id="L69" href="#L69">    69</a>  </span>
<span class="hidden"><a id="L70" href="#L70">    70</a>  </span><span class="text">// HandleEncoding will process the Accept-Encoding header and calls the given</span>
<span class="hidden"><a id="L71" href="#L71">    71</a>  </span><span class="text">// handler for each encoding until the handler returns true.</span>
<span class="hidden"><a id="L72" href="#L72">    72</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L73" href="#L73">    73</a>  </span><span class="text">// This function returns true when the Handler returns true, false otherwise.</span>
<span class="hidden"><a id="L74" href="#L74">    74</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L75" href="#L75">    75</a>  </span><span class="text">// For the identity (plain text) encoding the encoding string will be the</span>
<span class="hidden"><a id="L76" href="#L76">    76</a>  </span><span class="text">// empty string.</span>
<span class="hidden"><a id="L77" href="#L77">    77</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L78" href="#L78">    78</a>  </span><span class="text">// The wildcard encoding (*) will, after the &#39;*&#39;, contain a semi-colon separated</span>
<span class="hidden"><a id="L79" href="#L79">    79</a>  </span><span class="text">// list of all disallowed encodings (q=0).</span>
<span class="hidden"><a id="L80" href="#L80">    80</a>  </span><span>func HandleEncoding(r *http.Request, h Handler) bool {</span>
<span class="hidden"><a id="L81" href="#L81">    81</a>  </span><span>	acceptHeader := strings.TrimSpace(r.Header.Get(acceptEncoding))</span>
<span class="hidden"><a id="L82" href="#L82">    82</a>  </span>
<span class="hidden"><a id="L83" href="#L83">    83</a>  </span><span>	if len(acceptHeader) == 0 {</span>
<span class="hidden"><a id="L84" href="#L84">    84</a>  </span><span>		acceptHeader = anyEncoding</span>
<span class="hidden"><a id="L85" href="#L85">    85</a>  </span><span>	}</span>
<span class="hidden"><a id="L86" href="#L86">    86</a>  </span>
<span class="hidden"><a id="L87" href="#L87">    87</a>  </span><span>	for _, accept := range parseAccepts(acceptHeader) {</span>
<span class="hidden"><a id="L88" href="#L88">    88</a>  </span><span>		if accept.weight != 0 &amp;&amp; h.Handle(accept.encoding) {</span>
<span class="hidden"><a id="L89" href="#L89">    89</a>  </span><span>			return true</span>
<span class="hidden"><a id="L90" href="#L90">    90</a>  </span><span>		}</span>
<span class="hidden"><a id="L91" href="#L91">    91</a>  </span><span>	}</span>
<span class="hidden"><a id="L92" href="#L92">    92</a>  </span>
<span class="hidden"><a id="L93" href="#L93">    93</a>  </span><span>	return false</span>
<span class="hidden"><a id="L94" href="#L94">    94</a>  </span><span>}</span>
<span class="hidden"><a id="L95" href="#L95">    95</a>  </span>
<span class="hidden"><a id="L96" href="#L96">    96</a>  </span><span>func parseAccepts(acceptHeader string) []encoding {</span>
<span class="hidden"><a id="L97" href="#L97">    97</a>  </span><span>	accepts := make(encodings, 0, strings.Count(acceptHeader, acceptSplit)&#43;2)</span>
<span class="hidden"><a id="L98" href="#L98">    98</a>  </span><span>	hasIdentity := false</span>
<span class="hidden"><a id="L99" href="#L99">    99</a>  </span><span>	hasNoAny := false</span>
<span class="hidden"><a id="L100" href="#L100">   100</a>  </span><span>	anyPos := -1</span>
<span class="hidden"><a id="L101" href="#L101">   101</a>  </span>
<span class="hidden"><a id="L102" href="#L102">   102</a>  </span><span>	var nots strings.Builder</span>
<span class="hidden"><a id="L103" href="#L103">   103</a>  </span>
<span class="hidden"><a id="L104" href="#L104">   104</a>  </span><span>	nots.WriteString(&#34;*&#34;)</span>
<span class="hidden"><a id="L105" href="#L105">   105</a>  </span>
<span class="hidden"><a id="L106" href="#L106">   106</a>  </span><span>	p := parseAccept(acceptHeader)</span>
<span class="hidden"><a id="L107" href="#L107">   107</a>  </span>
<span class="hidden"><a id="L108" href="#L108">   108</a>  </span><span>	for {</span>
<span class="hidden"><a id="L109" href="#L109">   109</a>  </span><span>		coding := p.Next()</span>
<span class="hidden"><a id="L110" href="#L110">   110</a>  </span><span>		if coding.Type == parser.TokenDone {</span>
<span class="hidden"><a id="L111" href="#L111">   111</a>  </span><span>			break</span>
<span class="hidden"><a id="L112" href="#L112">   112</a>  </span><span>		}</span>
<span class="hidden"><a id="L113" href="#L113">   113</a>  </span>
<span class="hidden"><a id="L114" href="#L114">   114</a>  </span><span>		name := coding.Data</span>
<span class="hidden"><a id="L115" href="#L115">   115</a>  </span>
<span class="hidden"><a id="L116" href="#L116">   116</a>  </span><span>		if p.Accept(tokenInvalidWeight) {</span>
<span class="hidden"><a id="L117" href="#L117">   117</a>  </span><span>			continue</span>
<span class="hidden"><a id="L118" href="#L118">   118</a>  </span><span>		}</span>
<span class="hidden"><a id="L119" href="#L119">   119</a>  </span>
<span class="hidden"><a id="L120" href="#L120">   120</a>  </span><span>		weight := int16(1000)</span>
<span class="hidden"><a id="L121" href="#L121">   121</a>  </span>
<span class="hidden"><a id="L122" href="#L122">   122</a>  </span><span>		if p.Peek().Type == tokenWeight {</span>
<span class="hidden"><a id="L123" href="#L123">   123</a>  </span><span>			weight = parseQ(p.Next().Data)</span>
<span class="hidden"><a id="L124" href="#L124">   124</a>  </span><span>		}</span>
<span class="hidden"><a id="L125" href="#L125">   125</a>  </span>
<span class="hidden"><a id="L126" href="#L126">   126</a>  </span><span>		if name == identityEncoding {</span>
<span class="hidden"><a id="L127" href="#L127">   127</a>  </span><span>			hasIdentity = true</span>
<span class="hidden"><a id="L128" href="#L128">   128</a>  </span><span>			name = &#34;&#34;</span>
<span class="hidden"><a id="L129" href="#L129">   129</a>  </span><span>		} else if name == anyEncoding &amp;&amp; weight == 0 {</span>
<span class="hidden"><a id="L130" href="#L130">   130</a>  </span><span>			hasNoAny = true</span>
<span class="hidden"><a id="L131" href="#L131">   131</a>  </span><span>		}</span>
<span class="hidden"><a id="L132" href="#L132">   132</a>  </span>
<span class="hidden"><a id="L133" href="#L133">   133</a>  </span><span>		if slices.ContainsFunc(accepts, func(e encoding) bool { return e.encoding == Encoding(name) }) {</span>
<span class="hidden"><a id="L134" href="#L134">   134</a>  </span><span>			continue</span>
<span class="hidden"><a id="L135" href="#L135">   135</a>  </span><span>		}</span>
<span class="hidden"><a id="L136" href="#L136">   136</a>  </span>
<span class="hidden"><a id="L137" href="#L137">   137</a>  </span><span>		if weight == 0 {</span>
<span class="hidden"><a id="L138" href="#L138">   138</a>  </span><span>			nots.WriteByte(&#39;;&#39;)</span>
<span class="hidden"><a id="L139" href="#L139">   139</a>  </span><span>			nots.WriteString(name)</span>
<span class="hidden"><a id="L140" href="#L140">   140</a>  </span><span>		}</span>
<span class="hidden"><a id="L141" href="#L141">   141</a>  </span>
<span class="hidden"><a id="L142" href="#L142">   142</a>  </span><span>		if name == anyEncoding {</span>
<span class="hidden"><a id="L143" href="#L143">   143</a>  </span><span>			if anyPos != -1 {</span>
<span class="hidden"><a id="L144" href="#L144">   144</a>  </span><span>				continue</span>
<span class="hidden"><a id="L145" href="#L145">   145</a>  </span><span>			}</span>
<span class="hidden"><a id="L146" href="#L146">   146</a>  </span>
<span class="hidden"><a id="L147" href="#L147">   147</a>  </span><span>			anyPos = len(accepts)</span>
<span class="hidden"><a id="L148" href="#L148">   148</a>  </span><span>		}</span>
<span class="hidden"><a id="L149" href="#L149">   149</a>  </span>
<span class="hidden"><a id="L150" href="#L150">   150</a>  </span><span>		accepts = append(accepts, encoding{</span>
<span class="hidden"><a id="L151" href="#L151">   151</a>  </span><span>			encoding: Encoding(name),</span>
<span class="hidden"><a id="L152" href="#L152">   152</a>  </span><span>			weight:   weight,</span>
<span class="hidden"><a id="L153" href="#L153">   153</a>  </span><span>		})</span>
<span class="hidden"><a id="L154" href="#L154">   154</a>  </span><span>	}</span>
<span class="hidden"><a id="L155" href="#L155">   155</a>  </span>
<span class="hidden"><a id="L156" href="#L156">   156</a>  </span><span>	if anyPos != -1 {</span>
<span class="hidden"><a id="L157" href="#L157">   157</a>  </span><span>		accepts[anyPos].encoding = Encoding(nots.String())</span>
<span class="hidden"><a id="L158" href="#L158">   158</a>  </span><span>	}</span>
<span class="hidden"><a id="L159" href="#L159">   159</a>  </span>
<span class="hidden"><a id="L160" href="#L160">   160</a>  </span><span>	sort.Stable(accepts)</span>
<span class="hidden"><a id="L161" href="#L161">   161</a>  </span>
<span class="hidden"><a id="L162" href="#L162">   162</a>  </span><span>	if !hasIdentity &amp;&amp; !hasNoAny {</span>
<span class="hidden"><a id="L163" href="#L163">   163</a>  </span><span>		accepts = append(accepts, encoding{</span>
<span class="hidden"><a id="L164" href="#L164">   164</a>  </span><span>			encoding: &#34;&#34;,</span>
<span class="hidden"><a id="L165" href="#L165">   165</a>  </span><span>			weight:   1,</span>
<span class="hidden"><a id="L166" href="#L166">   166</a>  </span><span>		})</span>
<span class="hidden"><a id="L167" href="#L167">   167</a>  </span><span>	}</span>
<span class="hidden"><a id="L168" href="#L168">   168</a>  </span>
<span class="hidden"><a id="L169" href="#L169">   169</a>  </span><span>	return accepts</span>
<span class="hidden"><a id="L170" href="#L170">   170</a>  </span><span>}</span>
<span class="hidden"><a id="L171" href="#L171">   171</a>  </span>
<span class="hidden"><a id="L172" href="#L172">   172</a>  </span><span>var multiplies = [...]int16{100, 10, 1}</span>
<span class="hidden"><a id="L173" href="#L173">   173</a>  </span>
<span class="hidden"><a id="L174" href="#L174">   174</a>  </span><span>func parseQ(q string) int16 {</span>
<span class="hidden"><a id="L175" href="#L175">   175</a>  </span><span>	if q[0] == &#39;1&#39; {</span>
<span class="hidden"><a id="L176" href="#L176">   176</a>  </span><span>		return 1000</span>
<span class="hidden"><a id="L177" href="#L177">   177</a>  </span><span>	}</span>
<span class="hidden"><a id="L178" href="#L178">   178</a>  </span>
<span class="hidden"><a id="L179" href="#L179">   179</a>  </span><span>	if len(q) &lt; 2 {</span>
<span class="hidden"><a id="L180" href="#L180">   180</a>  </span><span>		return 0</span>
<span class="hidden"><a id="L181" href="#L181">   181</a>  </span><span>	}</span>
<span class="hidden"><a id="L182" href="#L182">   182</a>  </span>
<span class="hidden"><a id="L183" href="#L183">   183</a>  </span><span>	var qv int16</span>
<span class="hidden"><a id="L184" href="#L184">   184</a>  </span>
<span class="hidden"><a id="L185" href="#L185">   185</a>  </span><span>	for n, v := range q[2:] {</span>
<span class="hidden"><a id="L186" href="#L186">   186</a>  </span><span>		qv &#43;= int16(v-&#39;0&#39;) * multiplies[n]</span>
<span class="hidden"><a id="L187" href="#L187">   187</a>  </span><span>	}</span>
<span class="hidden"><a id="L188" href="#L188">   188</a>  </span>
<span class="hidden"><a id="L189" href="#L189">   189</a>  </span><span>	return qv</span>
<span class="hidden"><a id="L190" href="#L190">   190</a>  </span><span>}</span>
<span class="hidden"><a id="L191" href="#L191">   191</a>  </span>
<span class="hidden"><a id="L192" href="#L192">   192</a>  </span><span class="text">// ClearEncoding removes the Accept-Encoding header so that any further</span>
<span class="hidden"><a id="L193" href="#L193">   193</a>  </span><span class="text">// attempts to establish an encoding will simply used the default, plain text,</span>
<span class="hidden"><a id="L194" href="#L194">   194</a>  </span><span class="text">// encoding.</span>
<span class="hidden"><a id="L195" href="#L195">   195</a>  </span><span class="text">//</span>
<span class="hidden"><a id="L196" href="#L196">   196</a>  </span><span class="text">// Useful when you don&#39;t want a handler down the chain to also handle encoding.</span>
<span class="hidden"><a id="L197" href="#L197">   197</a>  </span><span>func ClearEncoding(r *http.Request) {</span>
<span class="hidden"><a id="L198" href="#L198">   198</a>  </span><span>	r.Header.Del(acceptEncoding)</span>
<span class="hidden"><a id="L199" href="#L199">   199</a>  </span><span>}</span>
<span class="hidden"><a id="L200" href="#L200">   200</a>  </span>
<span class="hidden"><a id="L201" href="#L201">   201</a>  </span><span class="text">// IsDisallowedInWildcard will return true if the given encoding is disallowed</span>
<span class="hidden"><a id="L202" href="#L202">   202</a>  </span><span class="text">// in the given accept string.</span>
<span class="hidden"><a id="L203" href="#L203">   203</a>  </span><span>func IsDisallowedInWildcard(accept, encoding Encoding) bool {</span>
<span class="hidden"><a id="L204" href="#L204">   204</a>  </span><span>	if !strings.HasPrefix(string(accept), &#34;*;&#34;) {</span>
<span class="hidden"><a id="L205" href="#L205">   205</a>  </span><span>		return false</span>
<span class="hidden"><a id="L206" href="#L206">   206</a>  </span><span>	}</span>
<span class="hidden"><a id="L207" href="#L207">   207</a>  </span>
<span class="hidden"><a id="L208" href="#L208">   208</a>  </span><span>	for enc := range strings.SplitSeq(string(accept[2:]), &#34;;&#34;) {</span>
<span class="hidden"><a id="L209" href="#L209">   209</a>  </span><span>		if enc == string(encoding) {</span>
<span class="hidden"><a id="L210" href="#L210">   210</a>  </span><span>			return true</span>
<span class="hidden"><a id="L211" href="#L211">   211</a>  </span><span>		}</span>
<span class="hidden"><a id="L212" href="#L212">   212</a>  </span><span>	}</span>
<span class="hidden"><a id="L213" href="#L213">   213</a>  </span>
<span class="hidden"><a id="L214" href="#L214">   214</a>  </span><span>	return false</span>
<span class="hidden"><a id="L215" href="#L215">   215</a>  </span><span>}</span>
<span class="hidden"><a id="L216" href="#L216">   216</a>  </span>
<span class="hidden"><a id="L217" href="#L217">   217</a>  </span><span class="text">// IsWildcard returns true when the given accept string is a wildcard match.</span>
<span class="hidden"><a id="L218" href="#L218">   218</a>  </span><span>func IsWildcard(accept Encoding) bool {</span>
<span class="hidden"><a id="L219" href="#L219">   219</a>  </span><span>	return accept == &#34;*&#34; || strings.HasPrefix(string(accept), &#34;*;&#34;)</span>
<span class="hidden"><a id="L220" href="#L220">   220</a>  </span><span>}</span>
<span class="hidden"><a id="L221" href="#L221">   221</a>  </span></pre>
	</body>
</html>
